diff --git a/.buildinfo b/.buildinfo new file mode 100644 index 0000000..4583538 --- /dev/null +++ b/.buildinfo @@ -0,0 +1,4 @@ +# Sphinx build info version 1 +# This file hashes the configuration used when building these files. When it is not found, a full rebuild will be done. +config: ac22fac5056599433d86ad9f568f9662 +tags: 645f666f9bcd5a90fca523b33c5a78b7 diff --git a/.nojekyll b/.nojekyll new file mode 100644 index 0000000..e69de29 diff --git a/README.md b/README.md new file mode 100644 index 0000000..aa660b3 --- /dev/null +++ b/README.md @@ -0,0 +1,3 @@ +#GitHub Pages + +Last update of sphinx html documentation from [e1aea05](https://github.com/NSLS-II-BMM/BeamlineManual/tree/e1aea05f3418772f749f127e15ecdb8b794efdbe) diff --git a/_downloads/1dc54fa3966864a6d9a6482fafe73e28/HutchSearch.docx b/_downloads/1dc54fa3966864a6d9a6482fafe73e28/HutchSearch.docx new file mode 100644 index 0000000..a5e2f8c Binary files /dev/null and b/_downloads/1dc54fa3966864a6d9a6482fafe73e28/HutchSearch.docx differ diff --git a/_downloads/55576a1ae002c3af9e018e120e040c09/Binder-instructions.pdf b/_downloads/55576a1ae002c3af9e018e120e040c09/Binder-instructions.pdf new file mode 100644 index 0000000..6475354 Binary files /dev/null and b/_downloads/55576a1ae002c3af9e018e120e040c09/Binder-instructions.pdf differ diff --git a/_downloads/cc3bb6345510248846628dfa6ac1c77b/5203_10--32263_131839.pdf b/_downloads/cc3bb6345510248846628dfa6ac1c77b/5203_10--32263_131839.pdf new file mode 100644 index 0000000..d1a0e5b Binary files /dev/null and b/_downloads/cc3bb6345510248846628dfa6ac1c77b/5203_10--32263_131839.pdf differ diff --git a/_downloads/e50b981fbdda59942b7c7b1b1337a8a2/RT-2-11-DIMENSIONS.JPG b/_downloads/e50b981fbdda59942b7c7b1b1337a8a2/RT-2-11-DIMENSIONS.JPG new file mode 100644 index 0000000..1a0f5e0 Binary files /dev/null and b/_downloads/e50b981fbdda59942b7c7b1b1337a8a2/RT-2-11-DIMENSIONS.JPG differ diff --git a/_images/1element.jpg b/_images/1element.jpg new file mode 100644 index 0000000..ce845b5 Binary files /dev/null and b/_images/1element.jpg differ diff --git a/_images/25ldewar.jpg b/_images/25ldewar.jpg new file mode 100644 index 0000000..27e266f Binary files /dev/null and b/_images/25ldewar.jpg differ diff --git a/_images/4element.jpg b/_images/4element.jpg new file mode 100644 index 0000000..d69ed0b Binary files /dev/null and b/_images/4element.jpg differ diff --git a/_images/743lobby.jpg b/_images/743lobby.jpg new file mode 100644 index 0000000..d8861dd Binary files /dev/null and b/_images/743lobby.jpg differ diff --git a/_images/7element.jpg b/_images/7element.jpg new file mode 100644 index 0000000..521aa12 Binary files /dev/null and b/_images/7element.jpg differ diff --git a/_images/BMM_utilities.png b/_images/BMM_utilities.png new file mode 100644 index 0000000..589041e Binary files /dev/null and b/_images/BMM_utilities.png differ diff --git a/_images/BMMcontrolstation.jpg b/_images/BMMcontrolstation.jpg new file mode 100644 index 0000000..60398a9 Binary files /dev/null and b/_images/BMMcontrolstation.jpg differ diff --git a/_images/Beamline-bananagrams.jpg b/_images/Beamline-bananagrams.jpg new file mode 100644 index 0000000..23e75a0 Binary files /dev/null and b/_images/Beamline-bananagrams.jpg differ diff --git a/_images/Bnc_map.png b/_images/Bnc_map.png new file mode 100644 index 0000000..66ea4bb Binary files /dev/null and b/_images/Bnc_map.png differ diff --git a/_images/Calibration_111.png b/_images/Calibration_111.png new file mode 100644 index 0000000..df885c5 Binary files /dev/null and b/_images/Calibration_111.png differ diff --git a/_images/DM3_first_cat6.jpg b/_images/DM3_first_cat6.jpg new file mode 100644 index 0000000..cdf9e1f Binary files /dev/null and b/_images/DM3_first_cat6.jpg differ diff --git a/_images/DM3_patch_panel.jpg b/_images/DM3_patch_panel.jpg new file mode 100644 index 0000000..1db2a03 Binary files /dev/null and b/_images/DM3_patch_panel.jpg differ diff --git a/_images/Gas_handling.png b/_images/Gas_handling.png new file mode 100644 index 0000000..2b44019 Binary files /dev/null and b/_images/Gas_handling.png differ diff --git a/_images/Kill_switch_connector.png b/_images/Kill_switch_connector.png new file mode 100644 index 0000000..f7d123d Binary files /dev/null and b/_images/Kill_switch_connector.png differ diff --git a/_images/Kill_switches.jpg b/_images/Kill_switches.jpg new file mode 100644 index 0000000..bdc9014 Binary files /dev/null and b/_images/Kill_switches.jpg differ diff --git a/_images/LOB-3.png b/_images/LOB-3.png new file mode 100644 index 0000000..68f2853 Binary files /dev/null and b/_images/LOB-3.png differ diff --git a/_images/Linkam_info.png b/_images/Linkam_info.png new file mode 100644 index 0000000..782c89f Binary files /dev/null and b/_images/Linkam_info.png differ diff --git a/_images/Linkam_main.png b/_images/Linkam_main.png new file mode 100644 index 0000000..7c39f5e Binary files /dev/null and b/_images/Linkam_main.png differ diff --git a/_images/Linkam_reboot.png b/_images/Linkam_reboot.png new file mode 100644 index 0000000..93c49fe Binary files /dev/null and b/_images/Linkam_reboot.png differ diff --git a/_images/Logitech.png b/_images/Logitech.png new file mode 100644 index 0000000..bac06ac Binary files /dev/null and b/_images/Logitech.png differ diff --git a/_images/Nd-6510.png b/_images/Nd-6510.png new file mode 100644 index 0000000..f0baac9 Binary files /dev/null and b/_images/Nd-6510.png differ diff --git a/_images/Nd-6560.png b/_images/Nd-6560.png new file mode 100644 index 0000000..45197c0 Binary files /dev/null and b/_images/Nd-6560.png differ diff --git a/_images/Nd-compare.png b/_images/Nd-compare.png new file mode 100644 index 0000000..034e0a2 Binary files /dev/null and b/_images/Nd-compare.png differ diff --git a/_images/Ref_wheel.jpg b/_images/Ref_wheel.jpg new file mode 100644 index 0000000..037cdbf Binary files /dev/null and b/_images/Ref_wheel.jpg differ diff --git a/_images/Samplewheel.jpg b/_images/Samplewheel.jpg new file mode 100644 index 0000000..a4484ca Binary files /dev/null and b/_images/Samplewheel.jpg differ diff --git a/_images/Vortex_pressure.jpeg b/_images/Vortex_pressure.jpeg new file mode 100644 index 0000000..3ee658d Binary files /dev/null and b/_images/Vortex_pressure.jpeg differ diff --git a/_images/Water_flow_CSS.png b/_images/Water_flow_CSS.png new file mode 100644 index 0000000..0d8ff16 Binary files /dev/null and b/_images/Water_flow_CSS.png differ diff --git a/_images/Water_flow_valves_1.jpg b/_images/Water_flow_valves_1.jpg new file mode 100644 index 0000000..400fe61 Binary files /dev/null and b/_images/Water_flow_valves_1.jpg differ diff --git a/_images/Water_flow_valves_2.jpg b/_images/Water_flow_valves_2.jpg new file mode 100644 index 0000000..72394a4 Binary files /dev/null and b/_images/Water_flow_valves_2.jpg differ diff --git a/_images/Windows_assets_folder.png b/_images/Windows_assets_folder.png new file mode 100644 index 0000000..3c5f743 Binary files /dev/null and b/_images/Windows_assets_folder.png differ diff --git a/_images/Windows_disconnect.png b/_images/Windows_disconnect.png new file mode 100644 index 0000000..95983ca Binary files /dev/null and b/_images/Windows_disconnect.png differ diff --git a/_images/Windows_map_new_drive.png b/_images/Windows_map_new_drive.png new file mode 100644 index 0000000..4ceade5 Binary files /dev/null and b/_images/Windows_map_new_drive.png differ diff --git a/_images/Windows_specify_new_drive.png b/_images/Windows_specify_new_drive.png new file mode 100644 index 0000000..b812a59 Binary files /dev/null and b/_images/Windows_specify_new_drive.png differ diff --git a/_images/Windows_stale_folder.png b/_images/Windows_stale_folder.png new file mode 100644 index 0000000..99ea4d6 Binary files /dev/null and b/_images/Windows_stale_folder.png differ diff --git a/_images/Winvm_startup.png b/_images/Winvm_startup.png new file mode 100644 index 0000000..60da217 Binary files /dev/null and b/_images/Winvm_startup.png differ diff --git a/_images/XAFS_live_view.png b/_images/XAFS_live_view.png new file mode 100644 index 0000000..ac17d83 Binary files /dev/null and b/_images/XAFS_live_view.png differ diff --git a/_images/XASwebcam.jpg b/_images/XASwebcam.jpg new file mode 100644 index 0000000..6fbf0bb Binary files /dev/null and b/_images/XASwebcam.jpg differ diff --git a/_images/Xf06bm-ws5.jpg b/_images/Xf06bm-ws5.jpg new file mode 100644 index 0000000..6066ea2 Binary files /dev/null and b/_images/Xf06bm-ws5.jpg differ diff --git a/_images/analog.jpg b/_images/analog.jpg new file mode 100644 index 0000000..5f04de2 Binary files /dev/null and b/_images/analog.jpg differ diff --git a/_images/bad_evaluation.png b/_images/bad_evaluation.png new file mode 100644 index 0000000..4560676 Binary files /dev/null and b/_images/bad_evaluation.png differ diff --git a/_images/biologic.png b/_images/biologic.png new file mode 100644 index 0000000..be7ce5b Binary files /dev/null and b/_images/biologic.png differ diff --git a/_images/bsui_startup.png b/_images/bsui_startup.png new file mode 100644 index 0000000..b112726 Binary files /dev/null and b/_images/bsui_startup.png differ diff --git a/_images/bsui_startup2.png b/_images/bsui_startup2.png new file mode 100644 index 0000000..2d5f2ed Binary files /dev/null and b/_images/bsui_startup2.png differ diff --git a/_images/cadashboard.png b/_images/cadashboard.png new file mode 100644 index 0000000..231907a Binary files /dev/null and b/_images/cadashboard.png differ diff --git a/_images/consumer_startup.png b/_images/consumer_startup.png new file mode 100644 index 0000000..fe929ad Binary files /dev/null and b/_images/consumer_startup.png differ diff --git a/_images/corridor.jpg b/_images/corridor.jpg new file mode 100644 index 0000000..1a6a820 Binary files /dev/null and b/_images/corridor.jpg differ diff --git a/_images/cryostat.jpg b/_images/cryostat.jpg new file mode 100644 index 0000000..12de205 Binary files /dev/null and b/_images/cryostat.jpg differ diff --git a/_images/ctrl-z.jpg b/_images/ctrl-z.jpg new file mode 100644 index 0000000..18f0567 Binary files /dev/null and b/_images/ctrl-z.jpg differ diff --git a/_images/dewar.jpg b/_images/dewar.jpg new file mode 100644 index 0000000..4e90ddd Binary files /dev/null and b/_images/dewar.jpg differ diff --git a/_images/dossier.png b/_images/dossier.png new file mode 100644 index 0000000..bd912cf Binary files /dev/null and b/_images/dossier.png differ diff --git a/_images/double_wheel_sm.jpg b/_images/double_wheel_sm.jpg new file mode 100644 index 0000000..193b911 Binary files /dev/null and b/_images/double_wheel_sm.jpg differ diff --git a/_images/doublewheel_spreadsheet.png b/_images/doublewheel_spreadsheet.png new file mode 100644 index 0000000..3aec911 Binary files /dev/null and b/_images/doublewheel_spreadsheet.png differ diff --git a/_images/filezilla_connect.png b/_images/filezilla_connect.png new file mode 100644 index 0000000..9ed0a55 Binary files /dev/null and b/_images/filezilla_connect.png differ diff --git a/_images/filezilla_folder.png b/_images/filezilla_folder.png new file mode 100644 index 0000000..8d8a3b5 Binary files /dev/null and b/_images/filezilla_folder.png differ diff --git a/_images/filezilla_password.png b/_images/filezilla_password.png new file mode 100644 index 0000000..fb99dea Binary files /dev/null and b/_images/filezilla_password.png differ diff --git a/_images/filezilla_queue.png b/_images/filezilla_queue.png new file mode 100644 index 0000000..1317bd0 Binary files /dev/null and b/_images/filezilla_queue.png differ diff --git a/_images/filezilla_remote.png b/_images/filezilla_remote.png new file mode 100644 index 0000000..03d13eb Binary files /dev/null and b/_images/filezilla_remote.png differ diff --git a/_images/filezilla_site_manager.png b/_images/filezilla_site_manager.png new file mode 100644 index 0000000..abf1a2b Binary files /dev/null and b/_images/filezilla_site_manager.png differ diff --git a/_images/filezilla_startup.png b/_images/filezilla_startup.png new file mode 100644 index 0000000..b0a054b Binary files /dev/null and b/_images/filezilla_startup.png differ diff --git a/_images/filezilla_transfer.png b/_images/filezilla_transfer.png new file mode 100644 index 0000000..89d8f9c Binary files /dev/null and b/_images/filezilla_transfer.png differ diff --git a/_images/find_slot.png b/_images/find_slot.png new file mode 100644 index 0000000..ded91e8 Binary files /dev/null and b/_images/find_slot.png differ diff --git a/_images/glancing_angle_stage.jpg b/_images/glancing_angle_stage.jpg new file mode 100644 index 0000000..638c460 Binary files /dev/null and b/_images/glancing_angle_stage.jpg differ diff --git a/_images/glancingangle_spreadsheet.png b/_images/glancingangle_spreadsheet.png new file mode 100644 index 0000000..b2933b9 Binary files /dev/null and b/_images/glancingangle_spreadsheet.png differ diff --git a/_images/good_evaluation.png b/_images/good_evaluation.png new file mode 100644 index 0000000..626b1cf Binary files /dev/null and b/_images/good_evaluation.png differ diff --git a/_images/grid_spreadsheet.png b/_images/grid_spreadsheet.png new file mode 100644 index 0000000..3cbbd72 Binary files /dev/null and b/_images/grid_spreadsheet.png differ diff --git a/_images/halfwheel.jpg b/_images/halfwheel.jpg new file mode 100644 index 0000000..e72d4dd Binary files /dev/null and b/_images/halfwheel.jpg differ diff --git a/_images/hdmi.jpg b/_images/hdmi.jpg new file mode 100644 index 0000000..202e115 Binary files /dev/null and b/_images/hdmi.jpg differ diff --git a/_images/lakeshore331.png b/_images/lakeshore331.png new file mode 100644 index 0000000..a15262d Binary files /dev/null and b/_images/lakeshore331.png differ diff --git a/_images/lakeshore_spreadsheet.png b/_images/lakeshore_spreadsheet.png new file mode 100644 index 0000000..7cf3b51 Binary files /dev/null and b/_images/lakeshore_spreadsheet.png differ diff --git a/_images/linkam.jpg b/_images/linkam.jpg new file mode 100644 index 0000000..f9d15ca Binary files /dev/null and b/_images/linkam.jpg differ diff --git a/_images/linkam_spreadsheet.png b/_images/linkam_spreadsheet.png new file mode 100644 index 0000000..12272ec Binary files /dev/null and b/_images/linkam_spreadsheet.png differ diff --git a/_images/map.png b/_images/map.png new file mode 100644 index 0000000..9574f91 Binary files /dev/null and b/_images/map.png differ diff --git a/_images/monitoring.png b/_images/monitoring.png new file mode 100644 index 0000000..159a1b5 Binary files /dev/null and b/_images/monitoring.png differ diff --git a/_images/new_IC.jpg b/_images/new_IC.jpg new file mode 100644 index 0000000..09c32b7 Binary files /dev/null and b/_images/new_IC.jpg differ diff --git a/_images/old_IC.jpg b/_images/old_IC.jpg new file mode 100644 index 0000000..8a15cf7 Binary files /dev/null and b/_images/old_IC.jpg differ diff --git a/_images/pilatus.jpg b/_images/pilatus.jpg new file mode 100644 index 0000000..fcb2a1f Binary files /dev/null and b/_images/pilatus.jpg differ diff --git a/_images/prompt.png b/_images/prompt.png new file mode 100644 index 0000000..c8df43a Binary files /dev/null and b/_images/prompt.png differ diff --git a/_images/pup-btr.png b/_images/pup-btr.png new file mode 100644 index 0000000..b40b41b Binary files /dev/null and b/_images/pup-btr.png differ diff --git a/_images/pup-cycle.png b/_images/pup-cycle.png new file mode 100644 index 0000000..fe11534 Binary files /dev/null and b/_images/pup-cycle.png differ diff --git a/_images/pup-select.png b/_images/pup-select.png new file mode 100644 index 0000000..c13a8e6 Binary files /dev/null and b/_images/pup-select.png differ diff --git a/_images/pup-start.png b/_images/pup-start.png new file mode 100644 index 0000000..b4e5f57 Binary files /dev/null and b/_images/pup-start.png differ diff --git a/_images/quadem.jpg b/_images/quadem.jpg new file mode 100644 index 0000000..645d24d Binary files /dev/null and b/_images/quadem.jpg differ diff --git a/_images/ref_wheel.jpg b/_images/ref_wheel.jpg new file mode 100644 index 0000000..037cdbf Binary files /dev/null and b/_images/ref_wheel.jpg differ diff --git a/_images/rocking_curve_333_E=3716.png b/_images/rocking_curve_333_E=3716.png new file mode 100644 index 0000000..5066fc5 Binary files /dev/null and b/_images/rocking_curve_333_E=3716.png differ diff --git a/_images/segfault.png b/_images/segfault.png new file mode 100644 index 0000000..657aa3e Binary files /dev/null and b/_images/segfault.png differ diff --git a/_images/slack.png b/_images/slack.png new file mode 100644 index 0000000..a79c73a Binary files /dev/null and b/_images/slack.png differ diff --git a/_images/small_rotation_stage.jpg b/_images/small_rotation_stage.jpg new file mode 100644 index 0000000..79a3a8f Binary files /dev/null and b/_images/small_rotation_stage.jpg differ diff --git a/_images/spinner-alignment.png b/_images/spinner-alignment.png new file mode 100644 index 0000000..808e625 Binary files /dev/null and b/_images/spinner-alignment.png differ diff --git a/_images/tilt_stage.jpg b/_images/tilt_stage.jpg new file mode 100644 index 0000000..391dfa5 Binary files /dev/null and b/_images/tilt_stage.jpg differ diff --git a/_images/triplot.png b/_images/triplot.png new file mode 100644 index 0000000..212b03b Binary files /dev/null and b/_images/triplot.png differ diff --git a/_images/usbcam1.jpg b/_images/usbcam1.jpg new file mode 100644 index 0000000..8ea12a5 Binary files /dev/null and b/_images/usbcam1.jpg differ diff --git a/_images/usbcam2.jpg b/_images/usbcam2.jpg new file mode 100644 index 0000000..3fd6c41 Binary files /dev/null and b/_images/usbcam2.jpg differ diff --git a/_images/wheel_spreadsheet.png b/_images/wheel_spreadsheet.png new file mode 100644 index 0000000..0b6d189 Binary files /dev/null and b/_images/wheel_spreadsheet.png differ diff --git a/_images/wheel_stage.jpg b/_images/wheel_stage.jpg new file mode 100644 index 0000000..e8c55ef Binary files /dev/null and b/_images/wheel_stage.jpg differ diff --git a/_sources/automation.rst b/_sources/automation.rst new file mode 100644 index 0000000..0d1bc27 --- /dev/null +++ b/_sources/automation.rst @@ -0,0 +1,594 @@ +.. + This document was developed primarily by a NIST employee. Pursuant + to title 17 United States Code Section 105, works of NIST employees + are not subject to copyright protection in the United States. Thus + this repository may not be licensed under the same terms as Bluesky + itself. + + See the LICENSE file for details. + +.. _automation: + +Beamline automation +=================== + +.. caution:: Spreadsheets with version number earlier than 13 **will not + work** as of 1 March, 2024. + + + +BMM currently supports five categories of spreadsheet-based automation: + +#. Sample wheels (`spreadsheet + `__) +#. Linkam stage temperature control (`spreadsheet + `__) +#. LakeShore 331 controller for Displex cryostat (`spreadsheet `__) +#. Glancing angle stage (`spreadsheet + `__) +#. Generic XY and XYZ grids (`spreadsheet + `__) + +The latest spreadsheets for each of these can always be found at +https://github.com/NSLS-II-BMM/profile_collection/tree/master/startup/xlsx + + +Each of the spreadsheets looks much like this, although there are some +differences in the columns corresponding to the different instruments. + +.. _fig-wheel-spreadsheet: +.. figure:: _images/wheel_spreadsheet.png + :target: _images/wheel_spreadsheet.png + :width: 70% + :align: center + + Example spreadsheet for running an experiment from a wheel with a + single sample ring. + +Make sure you are using the most up-to-date version of the spreadsheet. + +.. note:: The current spreadsheet version is **14**, as of 14 + November, 2024. You should *always* use a current + spreadsheet. + +.. caution:: Spreadsheets with version number earlier 13 **will not + work** as of 1 March, 2024. + + +Common features +--------------- + +Default information +~~~~~~~~~~~~~~~~~~~ + +All the spreadsheets use the concept of "default" scan information, +that is, information that is expected to be used for most or all of the +indicated measurements. In :numref:`Figure %s +`, the defults are entered into the row with the +green background. All rows underneath the green line are used to +describe individual measurements. + +For each individual measurement: + ++ If a white cell is left blank, the default value from the + corresponding green cell will be used. ++ If a white cell is filled in, that value will be used for that + measurement. + +Experimenters +~~~~~~~~~~~~~ + +.. note:: + + As of summer 2024, with the implementation of data security, the + beamline now has access to some information about the proposal and + SAF. It is no longer necessary to specify the names of the + experimenters. All names on the proposal will be put in the + metadata of every scan. + +.. + The other green part of the spreadsheet is a cell for entering the + names of all the experimenters involved in the measurement. + + This should **always** be filled in. Doing so allows for the + possibility of searching BMM's master database for data associated + with a particular user. + +.. _spreadsheet_options: + +Measurement options +~~~~~~~~~~~~~~~~~~~ + +Beneath the experimenter cell, there are three drop-down menus for +setting aspects of the sequence of measurements described on the +spreadsheet tab. + +#. A yes/no menu for telling Bluesky to close the shutter at the end + of the measurement sequence. + +#. A menu of options for modifying filenames to contain information + about things like absorber element, edge symbol, Linkam stage + temperature, and so on. This simplifies data entry into the + ``filename`` column of the spreadsheet. + +#. A place for specifying the number of repetitions of the entire + spreadsheet. This is different from the column labeled + "repetitions", which specifies the number of repeated XAS scans of + the sample in that row of the spreadsheet. + +Detector position +~~~~~~~~~~~~~~~~~ + +On the right hand side of each spreadsheet, there is a column for +specifying the position of the fluorescence detector. A smaller value +is closer to the sample. + +The detector position is set on a sample-by-sample basis, allowing the +best possible measurement |nd| not saturating the detector while +maximizing the signal for samples of different absorber concentrations +|nd| for each sample. For many experiments, most of the set up work +involves moving from sample to sample and setting the values of this +column. + + +Fine tuning sample position and slits +~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~ + +On the right hand side of each spreadsheet, there are columns for +specifying specific positions for sample X and Y and for slit width +and height. This allows you to fine tune the sample position and beam +size on a per-sample basis. + +The motor grid spreadsheet offers three columns for specifying motor +positions. The motors associated with those columns are +user-select-able. In this way, a grid over any beamline motor can +programmed. + +The glancing angle spreadsheet has the columns for specifying slit +width and height. It also has columns for specifying sample Y and +pitch (X and pitch when in perpendicular mode) when manual alignment +rather than automated alignment is selected. There is no column for +specifying the X position (or Y when in perpendicular mode) as use of +the glancing angle stage presumes that all the samples are mounted at +the centers of the spinners. + + +Selecting a spreadsheet +----------------------- + +All spreadsheets are imported using the :file:`xlsx()` command. The +spreadsheets are self-identifying. Every spreadsheet has an +identifying string spanning cells B1:C1. This is the cell with the +pink background. + +.. caution:: **Never** change the text in the pink cell or your + spreadsheet will likely be interpreted incorrectly. + +Import a spreadsheet +~~~~~~~~~~~~~~~~~~~~ + +To convert a spreadsheet into a macro then run the macro, do the +following: + +.. sourcecode:: python + + xlsx() + +This will show a numbered list of all :file:`.xlsx` files in your data +folder, something like this: + +.. sourcecode:: text + + Select your xlsx file: + + 1: 20210127-KB1.xlsx + 2: 20210127-KB3.xlsx + 3: 20210128-KB2.xlsx + 4: 20210128-KB4.xlsx + 5: 20210128-KB5.xlsx + 6: wheel_template.xlsx + + r: return + + Select a file > + +Select the :file:`.xlsx` file you want to import. Based on the +content of the pink identifying cell, your spreadsheet will be +interpreted appropriately. + +You may have multiple tabs in the spreadsheet file. If the file you +selected from the menu shown above has multiple tabs, you will be +presented with a menu of tabs, something like this: + +.. sourcecode:: text + + Select a sheet from yourfile.xlsx: + + 1: tab1 + 2: tab2 + 3: tab3 + + r: return + + Select a file > + +Enter the number corresponding to the tab to be measured. + +The menu of tab selections will only be presented if there is more +than one tab in the spreadsheet file. + +You may organize your experiment in a single file with multiple tabs +or in multiple files (each with one or more tabs). That is enturely +up to you. + +Generating Bluesky instructions +~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~ + +The tab on the selected spreadsheet file will be parsed, then a macro +file generated called :file:`_macro.py` and an INI file called +:file:`.ini`, where :file:`` is the name of the tab from +which the instructions were read. + +It is, therefor, a very good idea to give your tabs names that +indicate something about the experiment being described on that tab. + +The INI file (:numref:`Section %s `) contains the default values +from the green line (see :numref:`Figure %s `). +The macro file is imported into the BlueSky session, providing a new +with the name of the spreadsheet file. If the tab in the spreadsheet +was called :file:`mysamples`, the new BlueSky command is called +``mysamples_macro()``. + +.. admonition:: Future Tech! + + Convert spreadsheets to Bluesky |qs| input. + + +.. _sample_wheel_automation: + +Sample wheel automation +----------------------- + +The standard *ex situ* sample holder at BMM is a plastic wheel mounted +on a rotation stage. Examples are shown in figures +:numref:`fig-doublewheel`. The rotation stage is mounted on an XY +stage, so when one slot on the sample wheel is aligned, all the slots +are aligned. + + +.. _fig-doublewheel: +.. figure:: _images/double_wheel_sm.jpg + :target: _images/double_wheel_sm.jpg + :width: 50% + :align: center + + Double-ring sample wheels with 48 sample positions. There + are options for both wheel styles with 13mm x 3 mm slots or 13mm + diameter holes. The rings on the double wheel are 26 mm apart + (center to center of slots/holes). + +The automation concept is that a measurement at an edge on a slot on +the sample wheel is described by a row in the spreadsheet. Each +column of the spreadsheet carries one parameter of the XAFS scan. + + +.. _fig-doublewheel-spreadsheet: +.. figure:: _images/doublewheel_spreadsheet.png + :target: _images/doublewheel_spreadsheet.png + :width: 70% + :align: center + + Example spreadsheet for running an experiment from a wheel with a + two sample rings. Links: `single wheel spreadsheet + `_ + and `double wheel spreadsheet `_. + + +If you have read :numref:`Section %s ` about the INI file, then +most of the columns in this spreadsheet will be quite familiar. Most +of the columns are used to specify the same set of parameters as in +the INI file |nd| file name, element, edge, and so on. + +The green cell in the first row is used to input the names of all the +people involved in the experiment, as explained above. + +As explained above, row 6, the row with the lime-green background, +is used to specify the default values for all the parameters. The +concept here is to try to avoid having to input repetitive +information. For instance, in this case, all measurements will be +made at the Fe K edge. The element and edge are all specified in the +green row. Those cells are left blank for all subsequent rows, so the +default values will be used. + +In short, any cell that is left blank will use the value from the +green, default row. Any cell for which a value is specified will be +used in the macro that gets generated. + +The first column is used to specify the slot number for each sample on +the sample wheel. + +The second column is a simple way of excluding the slot from +measurement simply by specifying *No*. + +The next several columns correspond to lines in the INI file as +explained in :numref:`Section %s `. + +Energy changes can be included in the macro by specifying different +values for element and/or edge in a row. When specified +and different from the previous row, a call to the ``change_edge()`` +command (:numref:`Section {number} `) is inserted into the macro. + +Not shown in :numref:`Figure %s ` are columns +for tweaking the ``xafs_x`` and ``xafs_y`` positions, adjusting the +horizontal and vertical size of :numref:`slits3 (see Section %s) +`, and adjusting the fluorescence detector position. + + +Again, assuming the tab in the spreadsheet was called ``mysamples``, +you can then run the macro generated from the spreadsheet by:: + + RE(mysamples_macro()) + + +Here are the first few lines of the macro generated from this +spreadsheet. Note that for each sample, the macro first moves using +the ``slot()`` command, then measures XAS using the ``xafs()`` +command. The ``xafs()`` command uses the INI file (:numref:`Section +{number} `) generated from the green default line and has +explicit arguments for the filled-in spreadsheet cells. + +.. sourcecode:: python + :linenos: + + yield from slot(1) + yield from xafs('MnFewheel.ini', filename='Fe-Rhodonite', sample='MnSiO3', comment='ID:93 Russia') + close_last_plot() + + yield from slot(2) + yield from xafs('MnFewheel.ini', filename='Fe-Johannsonite', sample='CaMnSi2O6 - LT', comment='B –Iron Cap Mine; Graham Country, Arizona') + close_last_plot() + + yield from slot(3) + yield from xafs('MnFewheel.ini', filename='Fe-Spessartine', sample='Mn3Al2(SiO4)3', comment='Grants Mining District; New Mexico') + close_last_plot() + + ## and so on.... + +.. _auto_linkam: + +Linkam stage automation +----------------------- + +One of the temperature control options at BMM is a `Linkam stage +`_. Ours is the kind that can cool +using liquid nitrogen flow or heat up to 600 C using a resistive +heater. The linkam stage is typically mounted upright on top fo the +XY stage. + + +.. subfigure:: AB + :layout-sm: AB + :subcaptions: above + :gap: 8px + :name: fig-linkamstage + :class-grid: outline + + .. image:: _images/linkam.jpg + + .. image:: _images/dewar.jpg + + (Left) The Linkham stage mounted for transmission on the sample + stage. (Right) The 25 L dewar used for cooling the Linkam stage. + + +The automation concept for the Linkam stage is quite similar to the +*ex situ* sample holder. Instead of specifying the slot position of the +sample, you will specify the target temperature for the measurement. +There is also a column for specifying the holding time after arriving +at temperature before beginning the XAFS measurement. + +The feature described in :numref:`Section %s ` +for modifying filenames is particularly useful in this context. It +can be used to put the measurement temperature in the filename, +allowing you to simply specify a default filename, leaving that cell +in each row blank. The generated data files will then have sensible +names. + + +.. _fig-linkam-spreadsheet: +.. figure:: _images/linkam_spreadsheet.png + :target: _images/linkam_spreadsheet.png + :width: 70% + :align: center + + Example spreadsheet for running a temperature-dependent experiment + using the Linkam stage. Link to the `Linkam spreadsheet + `_ + +.. _auto_lakeshore: + +LakeShore/Displex automation +---------------------------- + +For extremely low temperature experiments, BMM has a Displex crystat +which uses a two-stage helium compressor to cool the cold head down as +low as 10K with temperature control between 10K and 500K using a +resistive heater and a `LakeShore temperature controller +`__. + +This is a somewhat unusual version of the Displex system in that it is +suitable for low-vibration applications. The compressor is +mechanically decoupled from the cold head, reducing the motion of the +sample. As a result of this cooling system, it is somewhat +time-consuming to temperature cycle and replace samples. Expect that +cooling from room temperature to 10K will take about 2 hours and +budget up to an an hour for returning to room temperature and changing +samples. + +.. subfigure:: AB + :layout-sm: AB + :gap: 8px + :subcaptions: above + :name: fig-displex + :class-grid: outline + + .. image:: _images/cryostat.jpg + + .. image:: _images/lakeshore331.png + + (Left) The Displex cryostat and it's compressor. (Right) The + LakeShore 331 controller, used to control temperature for the + cryostat shown to the left. + + +The automation for the LakeShore 331 works much the same as for the +Linkam stage. Again, you will specify the target temperature for the +measurement. And there is a column for specifying the holding time +after arriving at temperature before beginning the XAFS measurement. + +There is a column for specifying the power level of the heater in the +cryostat. There are three power settings. You probably want to use +the high power setting. The controller is pretty well tuned for the +cryostat. It is unlikely to overshoot the when raising temperature. + +.. _fig-lakeshore-spreadsheet: +.. figure:: _images/lakeshore_spreadsheet.png + :target: _images/lakeshore_spreadsheet.png + :width: 70% + :align: center + + Example spreadsheet for running a temperature-dependent experiment + using the Displex cryostat and the LakeShore 331. Link to the + `LakeShore spreadsheet `_. + + +.. + .. _fig-lakeshoreCSS: + .. figure:: _images/lakeshoreCSS.png + :target: _images/lakeshoreCSS.png + :width: 30% + :align: center + + The CSS screen for the LakeShore 331. + + +.. _auto_ga: + +Glancing angle stage automation +------------------------------- + +This stage is used to automate measurement at glancing angle, usually +on thin film samples. The stage can be mounted horizontally or +vertically, allowing measurement of in- or out-of-plane strain in thin +films. + +.. _fig-glancinganglestage: +.. figure:: _images/glancing_angle_stage.jpg + :target: _images/glancing_angle_stage.jpg + :width: 50% + :align: center + + The glancing angle stage with 8 sample positions. + +This stage is mounted on a rotation stage to move between samples. +The rotation stage is mounted on a tilt stage to set the incident +angle of the beam relative to the sample surface. This entire set up +is mounted on the XY stage for alignment on the beam. + +Each sample is affixed to a sample spinner (which is simply a cheap, 24 VDC +CPU fan). The 8 spinners are independently controlled via slip ring +electrical connection that runs through the axis of the rotation +stage. In practice, only the sample that is being measured is spinning. + +Again, the automation concept is very similar to the *ex situ* sample +wheel. Instead of specifying slot number, the spinner number is +specified on each row. There is also a yes/no menu for specifying +whether the sample spins during measurement. + +.. _fig-glancingangle-spreadsheet: +.. figure:: _images/glancingangle_spreadsheet.png + :target: _images/glancingangle_spreadsheet.png + :width: 70% + :align: center + + Example spreadsheet for running an experiment using the glancing + angle stage. Link to the `glancing angle spreadsheet + `_. + +Not shown in :numref:`Figure %s ` are +columns for specifying how sample alignment is handled. The default +is to do automated alignment. This works by following this script: + +#. Move the stage to an incident angle that is close to flat for the + sample and start the sample spinning. +#. Do a scan in the vertical direction, measuring the signal in the + transmission chamber. Fit an error function to the transmission + signal. The centroid of that function is the position with the + sample half-way in the beam. +#. Do a scan in pitch, measuring the signal in the transmission + chamber. The peak of that measurement is the position where the + sample is flat relative to the beam direction. +#. Repeat steps 2 and 3. +#. Move the sample tilt to the angle specified by the user in the + spreadsheet. +#. Do a scan in the vertical direction, measuring the signal in the + fluorescence detector. The center of mass of that measurement is + the position where the beam is well-centered on the sample. + +The result of this fully automated sequence is shown in +:numref:`Figure %s `. + +.. _fig-spinner_alignment: +.. figure:: _images/spinner-alignment.png + :target: _images/spinner-alignment.png + :width: 50% + :align: center + + This visual representation of the automated glancing angle + alignment is posted to Slack and presented in the measurement + :numref:`dossier (Section %s) `. + +For some samples, the automated alignment is unreliable, so there is +an option in the spreadsheet for manual alignment. In that case, find +the ``xafs_y`` and ``xafs_pitch`` positions for the sample at its +measurement angle and well-aligned in the beam. Enter those numbers +and they will be used by the macro rather than performing the +automated alignment. + +Motor grid automation +--------------------- + +The final kind of automation-via-spreadsheet available is BMM is for a +generic motor grid. The most common motor grid used for measurement +is the sample XY stage, ``xafs_x`` and ``xafs_y``. However, any two +motors on the beamline can be used for the grid. + +There are columns (to the left of the view shown in :numref:`Figure %s +`) for specifying the axes in the grid. + +In all other ways |nd| except for the ``slot`` column |nd| this +spreadsheet is identical to the *ex situ* sample wheel spreadsheet. + +.. _fig-grid-spreadsheet: +.. figure:: _images/grid_spreadsheet.png + :target: _images/grid_spreadsheet.png + :width: 70% + :align: center + + Example spreadsheet for running an experiment on an XY grid. Link + to the `motor grid spreadsheet + `_. + + +.. admonition:: Future Tech! + + Spreadsheets for: + + + Electrochemistry experiments using the BioLogic potentiostat + + Chemistry experiments using the gas cart, including its mass flow + controllers, valves, temperature controller, and mass spectrometer. + + + +.. caution:: Spreadsheets with version number earlier than 13 **will + not work** as of 1 March, 2024. diff --git a/_sources/before.rst b/_sources/before.rst new file mode 100644 index 0000000..02e0a58 --- /dev/null +++ b/_sources/before.rst @@ -0,0 +1,270 @@ +.. + This document was developed primarily by a NIST employee. Pursuant + to title 17 United States Code Section 105, works of NIST employees + are not subject to copyright protection in the United States. Thus + this repository may not be licensed under the same terms as Bluesky + itself. + + See the LICENSE file for details. + +.. _before: + +Before your Beamtime +==================== + +This page is a check list of things each user must do before beamtime +at BMM. Several tasks on this page are called out in caution boxes +that look like this: + +.. caution:: + + This is a caution box. + +Failure to do the tasks in caution boxes will result in delay or +cancellation of your experiment. + +:TL;DR: *read all the caution boxes.* + +Register for BNL Site Access +---------------------------- + +Upon receipt of notification of your beamtime, verify that all +the team members attending your experiment have a valid BNL Guest +registration. + +An experimenter who is not listed on the original proposal (for +example, a student who recently joined the research group), will not +receive reminder emails about the need for a BNL appointment. So, be +sure **everyone** coming to the beamline has a valid appointment. + +.. note:: + + + New registration requests as well as extension requests for + expired registrations should be submitted as soon as possible. + + Time required for approval of a guest registrations: + + + **US citizens**: 7 days + + **Citizens of non-sensitive countries**: 14 days + + **Citizens of sensitive countries**: 45 days + + + `Definition of sensitive country `__ + +Apply for a BNL Guest appointment: https://www.bnl.gov/ps/userguide/beforearrival.php + +.. caution:: Without an active appointment, you will not be allowed + on site and will be turned away at the main gate. + +BNL Domain Account +------------------ + +When your BNL Guest registration becomes active, you should receive an +email with instructions for enabling your BNL domain account. You must +do this, including setting your password and signing up for DUO +Federal two-factor authentication. + +If you need to request an account, go to https://www.bnl.gov/accounts/ +and click the button that says "Submit account registration form". + +Enabling your BNL account is non-optional. You must have an active +account in order to unlock the computers at the beamline. + +.. caution:: Without an active account, a known password, and DUO + Federal two-factor authentication, you will be unable to + perform your experiment or to access the data from the + experiment. + +Safety Approval Form +-------------------- + +Submit your Safety Approval Form (SAF) **at least** two weeks before +the start date for your experiment. Your SAF should provide a +complete and accurate description of the experiment, including all +samples, chemicals, solvents, reference materials, and equipment that +will be used during the experiment and for sample preparation. In +addition, any additional equipment not provided by the beamline has to +be identified and described in the SAF. Provide detailed information +on sample processing and any safety measures necessary. + +.. caution:: An approved SAF is required on the first day of beamtime. + Without enough time for approval, you may lose your + beamtime. + +To submit a SAF: https://pass.bnl.gov + +If your experiment requires chemicals, compressed gases, lab access, +or other equipment, please contact `Bruce Ravel +`__ well ahead of your expriment. +Transportation of hazardous materials related to your experiment to +the BNL site and back to the home institution is your responsibility. + + +Support Laboratories +-------------------- + +Currently, four labs support user activities at BMM. They are: + ++ Lab 3L-L06: XAS Sample Prep Lab ++ Lab 3L-L07: Environmental Science Wet Lab ++ Lab 3L-L09: XAS Chemistry Lab ++ Lab 4L-L09: Electrochemistry Lab + +See https://www.bnl.gov/nsls2/labs/ for details of what equipment is +available in each lab. + +If you prepare or treat samples on-site, you **must** request use of +the appropriate NSLS-II User Laboratory when you submit your Safety +Approval Form. + +You cannot request a laboratory after an SAF has been approved. + +Here is some advice on selecting the correct lab: + +Unbound nanoparticles + Please note that supported nanoparticles or nanoparticles in + solution are not considered “unbound”. If your samples are indeed + unbound, please contact BMM beamline scientist Bruce Ravel + (bravel@bnl.gov) as soon as possible. + +Solid sample preparation (pellet pressing, spreading on tape, etc) + Request Lab 3L-L06. + +Liquid samples (organic solvents, acids, bases) and simple chemistry + Request Lab 3L-L09. + +Hydrofluoric acid treatment + Request Lab 3L-L09 and notify the Cognizant Space Manager, Syed Khalid + (khalid@bnl.gov), as soon as possible. + +Environmental and earth science sample preparation and treatment + Request Lab 3L-L07. + +Electrochemistry and battery science + Request Lab 4L-L09. + +.. admonition:: Radiological samples + :class: important + + Radionuclide samples must be prepared and properly contained at the + home institution. NSLS-II has no lab available for work on + radiological materials. Samples with depleted uranium and uranium + at low or environmentally relevant concentrations **are** + considered to be radiological samples at NSLS-II and must be + treated as such. + +Requests for using gloveboxes should be discussed prior to submitting +the SAF, contact Bruce Ravel (bravel@bnl.gov) and/or Syed Khalid +(khalid@bnl.gov) + +.. important:: Each lab you select on your safety approval form makes work + for the lab stewards. Only request the lab(s) that you + actually intend to use. + +Training +-------- + +Please complete all required online user training at least one week +before the starting date. + +To complete your training, follow the training instructions at: +https://www.bnl.gov/nsls2/userguide/before-arrival.php#step2 + +Links to specific course: + ++ `Cyber Security Training (GE-CYBERSEC) `__ ++ `Computer Use Agreement (GE-COMPUSE-AGREE) `__ ++ `General Employee Radiation Training (TQ-GERT) `__ ++ `NSLS-II Safety Module (PS-NSLS-II-USER-MOD) `__ ++ `Guest Site Orientation (TQ-GSO) `__ + +Returning users can `check training status online `__. + + +.. caution:: + + Failure to complete training will result in a delay starting your + experiment. + + +Electrical Equipment +-------------------- + +Any equipment with an electrical cable including low-voltage +instruments and sample holderss are considered electrical equipment +(laptop computers are excluded) and are subject to an electrical +equipment inspection (EEI). + +If you plan to bring electrical equipment for your experiment, go to +http://bit.ly/m/eei to fill out the EEI request form. + +The EEI request form includes questions describing the user supplied +equipment, thus is a chore better suited for the user who has +knowledge of the equipment coming to the beamline. + +.. caution:: Failure to schedule an EEI will result in a delay to your + experiment. Equipment that cannot pass EEI may not be used + at NSLS-II. + +Note that some already-approved electrical equipment (e.g., +potentiostat, heaters, cryostats, temperature controllers, sample +spinners, and more) is available at the beamline. + + +Research Equipment Pool +----------------------- + +Some experimental tools are available from `the NSLS-II Research +Equipment Pool +`__. If you +wish to use any items from equipment pool, contact `Bruce Ravel +`__ well ahead of your experiment. + +If you wish to use a potentiostat in lab 4L-L09 to pre-condition +battery or other electrochemical samples, contact `Bruce +`__ to have a potentiostat from the equipment +pool reserved for you. + + +Shipping things to the beamline +------------------------------- + +Packages should be shipped to + +.. code-block:: none + + Bruce Ravel / Beamline 6BM + Building 740 + Brookhaven National Laboratory + Upton NY 11973 + +If shipping radiological materials, **do not ship samples directly to +the beamline**. Follow the `instructions for radiological materials +transport +`__. + + +The Day of your Experiment +-------------------------- + +.. note:: Unless told otherwise, your experiment starts at 8 am. + +You should consider having all team members arrive the afternoon prior +to the first day of the beamtime to check in at `the GUV Center +`__ and to complete beamline specific +training. + +The GUV Center opens at `7 a.m. on weekdays +`__, so it is possible to obtain +badges and access cards prior to the 8 a.m. start date of your +experiment. + +Please consult `NSLS-II user guide `__ +or contact BMM beamline staff for more information. + +.. ADMONITION:: **NSLS-II Attire** + :class: caution + + At a minimum, all users are required to wear long pants or + skirt/dress that extends to the ankle, short-sleeve or long sleeve + shirts, and shoes that enclose the entire foot for entry to NSLS-II + non-office areas, which includes the experimental floor and sample + preparation laboratories. diff --git a/_sources/cheatsheet.rst b/_sources/cheatsheet.rst new file mode 100644 index 0000000..932a618 --- /dev/null +++ b/_sources/cheatsheet.rst @@ -0,0 +1,364 @@ +.. + This document was developed primarily by a NIST employee. Pursuant + to title 17 United States Code Section 105, works of NIST employees + are not subject to copyright protection in the United States. Thus + this repository may not be licensed under the same terms as Bluesky + itself. + + See the LICENSE file for details. + +.. role:: key + :class: key + +.. _cheatsheet: + +Command cheatsheet +================== + +This section provides a quick summary of the data acquisition commands +discussed in the previous sections. + +.. admonition:: Stopping a measurement + + To stop any command do :key:`Ctrl`-:key:`C` :key:`Ctrl`-:key:`C` + + That is, hit :key:`Ctrl`-:key:`C` twice! + + This will **pause** the running command and return you to the command line. + + Do ``RE.resume()`` to continue the command from where it stopped. + + Do ``RE.stop()`` to fully stop the command. + + +The most essential information +------------------------------ + +.. note:: Parentheses |nd| ``()`` |nd| are an essential part of the + syntax. Same for ``RE()`` and ``%`` when specified. + + +Most common commands +~~~~~~~~~~~~~~~~~~~~ + + +``shb.open()`` / ``shb.close()`` + Open / close the photon shutter + +``RE(mv(dcm.energy, ))`` + Move TO an energy value + +``RE(mv(xafs_y, ))`` + Move any named motor TO a position (``xafs_y`` is an example,see :numref:`Section %s `) + +``RE(mvr(xafs_y, ))`` + Move any named motor BY an amount (``xafs_y`` is an example, see :numref:`Section %s `) + +``RE(change_edge('Xx'))`` + setup beamline for an absorption edge, ``Xx`` = 1- or 2-letter symbol, + e.g. ``Cu`` or ``Y``. + +``RE(slot())`` + Move sample wheel to slot #N (1 |le| N |le| 24) + +``RE(xafs_wheel.outer())`` / ``RE(xafs_wheel.inner())`` + move the *ex situ* wheel sample holder to the outer/inner ring + +``%xrf`` + measure and display an XRF spectrum + +``RE(linescan(, , , , ))`` + move a motor, plot a signal + +``RE(pluck())`` + Select a point from a plot on screen and move that motor to that position + +``xlsx()`` + import a spreadsheet + +``RE(xanes())`` + quick-n-dirty XANES scan at the current element and edge + + +Most common motors +~~~~~~~~~~~~~~~~~~ + +See :numref:`Section %s ` for the full list of +``xafs_`` motors. + + ============== =========== ========= ======================= ================================== + motor type units notes directions + ============== =========== ========= ======================= ================================== + ``xafs_x`` linear mm main sample stage |plus| outboard, - inboard + ``xafs_y`` linear mm main sample stage |plus| up, - down + ``xafs_det`` linear mm detector mount |plus| away from sample, - closer + ============== =========== ========= ======================= ================================== + + + +Full list of commands +--------------------- + +Note that some command must be run through the run engine, other do +not. The use of ``RE()`` is called explicitly in :numref:`Table %s +`. Any command that should be run through the run +engine will complain with this hint: + +.. code-block:: text + + BMM D.111 [48] > mvr(xafs_x, 2) + Out[48] Hint: enclose bsui commands in RE() + + + +.. table:: Main BlueSky commands used at BMM (don't type the ``<`` or + ``>``, those symbols indicate that the argument is numeric.) + :name: command-list + :align: left + + +------------------------------------------+----------------------------------------------------------------------------+ + | |mquad| |mquad| |mquad| |mquad| |mquad| **Shutter commands** | + +------------------------------------------+----------------------------------------------------------------------------+ + | ``shb.open()`` / ``shb.close()`` | Open / close the photon shutter | + +------------------------------------------+----------------------------------------------------------------------------+ + | |mquad| |mquad| |mquad| |mquad| |mquad| **Mono tuning commands** | + +------------------------------------------+----------------------------------------------------------------------------+ + | ``RE(rocking_curve())`` | Measure the 2nd crystal rocking curve | + +------------------------------------------+----------------------------------------------------------------------------+ + | ``tu()`` / ``td()`` | Tune the mono 2nd crystal by hand | + +------------------------------------------+----------------------------------------------------------------------------+ + | ``TUNE_STEP=0.004`` | Tuning step size – 0.004 is good for Si(111), 0.002 for Si(311) | + +------------------------------------------+----------------------------------------------------------------------------+ + | |mquad| |mquad| |mquad| |mquad| |mquad| **Import an automation spreadsheet** | + +------------------------------------------+----------------------------------------------------------------------------+ + | ``xlsx()`` | Select an existing spreadsheet from a list | + +------------------------------------------+----------------------------------------------------------------------------+ + | |mquad| |mquad| |mquad| |mquad| |mquad| **Mono movement commands** | + +------------------------------------------+----------------------------------------------------------------------------+ + | ``RE(mv(dcm.energy, ))`` | Move TO an energy value | + +------------------------------------------+----------------------------------------------------------------------------+ + | ``RE(mv(dcm.energy, + 50))`` | Move 50 eV **above** the edge | + +------------------------------------------+----------------------------------------------------------------------------+ + | ``RE(mv(dcm.energy, - 50))`` | Move 50 eV **below** the edge | + +------------------------------------------+----------------------------------------------------------------------------+ + | ``RE(mvr(dcm.energy, ))`` | Move BY an energy step | + +------------------------------------------+----------------------------------------------------------------------------+ + | |mquad| |mquad| |mquad| |mquad| |mquad| **Motor movement commands** | + +------------------------------------------+----------------------------------------------------------------------------+ + | ``RE(mv(xafs_y, ))`` | Move any named motor TO a position (``xafs_y`` is an example) | + +------------------------------------------+----------------------------------------------------------------------------+ + | ``RE(mvr(xafs_y, ))`` | Move any named motor BY an amount (``xafs_y`` is an example) | + +------------------------------------------+----------------------------------------------------------------------------+ + | |mquad| |mquad| |mquad| |mquad| |mquad| **Where are things?** | + +------------------------------------------+----------------------------------------------------------------------------+ + | ``%w dcm`` | where's the mono? | + +------------------------------------------+----------------------------------------------------------------------------+ + | ``%w slits3`` | where are the slits? | + +------------------------------------------+----------------------------------------------------------------------------+ + | ``%w m2`` | where's mirror 2? (toroidal focusing mirror) | + +------------------------------------------+----------------------------------------------------------------------------+ + | ``%w m3`` | where's mirror 3? (flat, harmonic rejection mirror) | + +------------------------------------------+----------------------------------------------------------------------------+ + | ``%w xafs_table`` | where's the XAFS table? | + +------------------------------------------+----------------------------------------------------------------------------+ + | |mquad| |mquad| |mquad| |mquad| |mquad| **Change edge and mono crystal** | + +------------------------------------------+----------------------------------------------------------------------------+ + | ``RE(change_edge('Xx'))`` | setup beamline for an absorption edge, 1- or 2-letter symbol | + +------------------------------------------+----------------------------------------------------------------------------+ + | ``RE(change_xtals('h11'))`` | set monochromator, Si(111) or Si(311), h=1 or h=3 | + +------------------------------------------+----------------------------------------------------------------------------+ + | |mquad| |mquad| |mquad| |mquad| |mquad| **Experiment setup** | + +------------------------------------------+----------------------------------------------------------------------------+ + | ``RE(slit_height())`` | explore position of slits3 (then pluck to move dm3_bct) | + +------------------------------------------+----------------------------------------------------------------------------+ + | ``RE(dark_current())`` | measure electrometer dark currents | + +------------------------------------------+----------------------------------------------------------------------------+ + | |mquad| |mquad| |mquad| |mquad| |mquad| **ex situ sample stage** | + +------------------------------------------+----------------------------------------------------------------------------+ + | ``RE(slot())`` | Move sample wheel to slot #N (1 |le| N |le| 24) | + +------------------------------------------+----------------------------------------------------------------------------+ + | ``RE(find_slot())`` | align the *ex situ* wheel sample holder | + +------------------------------------------+----------------------------------------------------------------------------+ + | ``RE(xafs_wheel.outer())`` | move the *ex situ* wheel sample holder to the outer ring | + +------------------------------------------+----------------------------------------------------------------------------+ + | ``RE(xafs_wheel.inner())`` | move the *ex situ* wheel sample holder to the inner ring | + +------------------------------------------+----------------------------------------------------------------------------+ + | |mquad| |mquad| |mquad| |mquad| |mquad| **Glancing angle stage** | + +------------------------------------------+----------------------------------------------------------------------------+ + | ``RE(ga.to(N))`` | move stage to sample N (1 |le| N |le| 8) + start spinner | + +------------------------------------------+----------------------------------------------------------------------------+ + | ``RE(ga.auto_align(pitch))`` | automatically align glancing angle stage and move to pitch | + +------------------------------------------+----------------------------------------------------------------------------+ + | ``RE(ga.flatten())`` | return to the flat position found during ``auto_align()`` | + +------------------------------------------+----------------------------------------------------------------------------+ + | |mquad| |mquad| |mquad| |mquad| |mquad| **Linkam stage** | + +------------------------------------------+----------------------------------------------------------------------------+ + | ``RE(mv(linkam, T))`` | move Linkam stage to temperature T | + +------------------------------------------+----------------------------------------------------------------------------+ + | ``linkam.settle_time = 120`` | set Linkam settling time (in seconds) | + +------------------------------------------+----------------------------------------------------------------------------+ + | ``linkam.status()`` | display Linkam status | + +------------------------------------------+----------------------------------------------------------------------------+ + | ``linkam.on()`` / ``linkam.off()`` | turn Linkam on or off | + +------------------------------------------+----------------------------------------------------------------------------+ + | |mquad| |mquad| |mquad| |mquad| |mquad| **LakeShore temperature controller** | + +------------------------------------------+----------------------------------------------------------------------------+ + | ``RE(lakeshore.to(T, power))`` | move cryostat to temperature T with heater at ``power`` | + +------------------------------------------+----------------------------------------------------------------------------+ + | ``lakeshore.settle_time = 120`` | set Linkam settling time (in seconds) | + +------------------------------------------+----------------------------------------------------------------------------+ + | ``lakeshore.status()`` | display Linkam status | + +------------------------------------------+----------------------------------------------------------------------------+ + | ``RE(mv(lakeshore.power,3))`` | turn heater to full power | + +------------------------------------------+----------------------------------------------------------------------------+ + | ``RE(mv(lakeshore.power,0))`` | turn heater off | + +------------------------------------------+----------------------------------------------------------------------------+ + | |mquad| |mquad| |mquad| |mquad| |mquad| **Reference wheel** | + +------------------------------------------+----------------------------------------------------------------------------+ + | ``RE(reference('Xx'))`` | Move to reference element 'Xx' | + +------------------------------------------+----------------------------------------------------------------------------+ + | |mquad| |mquad| |mquad| |mquad| |mquad| **Get help** | + +------------------------------------------+----------------------------------------------------------------------------+ + | ``%m`` | Show motors | + +------------------------------------------+----------------------------------------------------------------------------+ + | ``%h`` | Show help | + +------------------------------------------+----------------------------------------------------------------------------+ + | ``%k`` | Show keyboard shortcuts | + +------------------------------------------+----------------------------------------------------------------------------+ + | ``%se`` | Show reference foil and ROI configurations | + +------------------------------------------+----------------------------------------------------------------------------+ + + + +Photon delivery system modes +---------------------------- + +.. table:: Photon delivery system modes + :name: pds-mode-table + + ======= ============= ================= + Mode focused energy range + ======= ============= ================= + A |checkmark| 8 keV and up + B |checkmark| below 6 keV + C |checkmark| 6 keV to 8 keV + D |xmark| 8 keV and up + E |xmark| 6 keV to 8 keV + F |xmark| below 6 keV + XRD |checkmark| 8 keV and up + ======= ============= ================= + + +Slits3 coordinated motions +-------------------------- + +These coordinated motions behave just like single motors and can be +used with the motor movement commands in :numref:`Table %s `. + +Horizontal size + ``slits3.hsize`` (nominally 8 mm) + +Horizontal center + ``slits3.hcenter`` (nominally 0 mm) + + +Vertical size + ``slits3.vsize`` (nominally 1 mm) + +Vertical center + ``slits3.vcenter`` (nominally 0 mm) + + The vertical center should never be changed. Instead, align the + slits by moving ``dm3_bct`` or running the ``slit_height()`` plan + (:numref:`Section %s `) + + +Example movement: ``RE(mv(slits3.vsize, 0.5))`` + +Individual slits are named ``slits3.top``, ``slits3.bottom``, +``slits3.inboard``, ``slits3.outboard``. + + +Motor positions and limits +-------------------------- + +These commands work on any named motor (:numref:`Table %s `). + +Where is a sample motor? + ``%w xafs_x`` + +What are the soft limits? + ``xafs_x.hlm.value`` / ``xafs_x.llm.value`` + +Set a soft limit: + ``xafs_x.hlm.put(-95)`` / ``xafs_x.llm.put(-157)`` + + +Line scans +---------- + +.. code-block:: python + + RE(linescan(, , , , )) + +where + ++ ```` is a named motor, see :numref:`Table %s ` ++ ```` is one of: ``'it'``, ``'if'``, ``'ir'``, or ``'i0'`` ++ ````, ````, ```` are the boundaries relative to the + current position and the number of steps. + +The plot will be determined from the values of ```` and +```` + +This is a relative scan. + +After the scan, you will prompted to select a position to move to. +Single click the left button after a linescan to move to a position. + +.. code-block:: python + + RE(pluck()) + +to repeat that on the current plot. + + +Energy scans +------------ + +Start an XAFS scan, prompting for an :numref:`INI file (section %s) ` + +.. code-block:: python + + RE(xafs()) + +Start an XAFS scan using a specified :numref:`INI file (section %s) ` + +.. code-block:: python + + RE(xafs('myscan.ini')) + +In the INI file, set ``mode`` to transmission, fluorescence, +reference, or both to control the in-scan plotting display (both = +show transmission and fluorescence) + +Import a spreadsheet to perform automated XAFS measurements: + +.. code-block:: python + + xlsx() + +You will be prompted first for the name of a spreadsheet file, then +for the tab to be read. + +.. + Experiment log + -------------- + + Log entries are made for each scan. System and beamtime specific logs + are maintained. To insert a comment in the log, do: + + .. code-block:: text + + BMM_log_info(“This is my log entry”) + diff --git a/_sources/commonchores.rst b/_sources/commonchores.rst new file mode 100644 index 0000000..4f3fbf0 --- /dev/null +++ b/_sources/commonchores.rst @@ -0,0 +1,342 @@ +.. + This document was developed primarily by a NIST employee. Pursuant + to title 17 United States Code Section 105, works of NIST employees + are not subject to copyright protection in the United States. Thus + this repository may not be licensed under the same terms as Bluesky + itself. + + See the LICENSE file for details. + +.. role:: key + :class: key + +.. _commonchores: + +Common chores +============= + +In this section, a variety of procedures around the beamline are +explained in detail. + + +Search the Hutch +---------------- + +The buttons on the hutch interlock panel are labeled with black +stickers with white numbers. Those labels correspond to the numbering +in these lists. Note that there is no step #4 when opening the hutch +and no step #2 when closing. + +To OPEN the hutch +~~~~~~~~~~~~~~~~~ + ++ 6. Close the shutter ++ 5. Press the "SBE" button ++ 4. ∙∙∙ ++ 3. Make sure the “Front Left Maglock” button is not green ++ 2. Press the green “Open Door” button, wait for the slow door to open ++ 1. Enter the hutch + + +To CLOSE the hutch +~~~~~~~~~~~~~~~~~~ + +Everyone but the person performing the search should leave the hutch. + +The search is done by and only one person. Only one person pushes all +the buttons. The serach **is not** a team sport! + +1. Complete the hutch search + + + Wait for others to exit the hutch + + Look in the mirror to see that no one is behind the table + + Press search button 1 + + Visually inspect the length of the hutch, verifying that it is empty of people + + Press search button 2 + + Exit the hutch + +2. ∙∙∙ +3. Push the “Front Left Maglock” button. It should be lit green +4. Push and hold the yellow “Close Door” button. Listen for the magnetic latch +5. Push the "SBE" button, wait for the loud noise to finish +6. You can now open the shutter + +Hutch search sign +~~~~~~~~~~~~~~~~~ + +These instructions are printed, laminated, mounted on a magnetic +sheet, and hung near the search station. I would like to make these +signs in various languages. If you would like to help by translating +this sign, download the Word document below. Make a translation into +your language and mail the Word doc back to Bruce. He will print and +mount the new translation! I am most eager for Chinese, Russian, and +German translations. + +:download:`Word document with hutch search instructions <_static/HutchSearch.docx>`. + + + +Change energy and prepare for fluorescence measurements +------------------------------------------------------- + +In this example, we are using Fe as the example. That is, we assume +you are moving the beamline to the state in which it is ready to +measure at the Fe K edge. For any other edge, simply change ``Fe`` to +the appropriate element in the following example. + +When you change samples, don’t forget to put the detector in a safe +state before swapping out the sample wheel. Move the detector all the +way back – ``RE(mv(xafs_det, 205))`` – then put the cap back on the +detector. Once you have secured the new sample wheel, remove the cap +before searching the hutch. + +Once the hutch is secured and you are back at the computer: + +#. Change edge with the ``RE(change_edge(‘Fe’))`` command. Wait for it + to finish and return to the command line. (See :numref:`{name}, + Section {number} ` for details.) + +#. Move the detector to middle position, say, 100: ``RE(mv(xafs_det, + 100))`` The range of positions for that motor is 205 (farthest back) + to about 20 (very close to sample). You can check the lower limit + with ``xafs_det.limits``. + +#. Measure an XRF spectra: ``%xrf``. Remember that you want the OCR + (total count rate) to be 200,000 or less on each of the seven + channels. + +#. Adjust the detector position ``RE(mv(xafs_det, ))``. + Remeasure ``%xrf``. Iterate on the detector position until you get + a sensible signal on the detector. + +#. Record the detector position (``xafs_det.position`` or the number + next to ``det:`` in the :numref:`display (Section %s) + ` at the top of the top screen) for the current sample + in the spreadsheet. + +#. Move to a new sample position. To change slots on the *ex situ* + sample wheel: ``RE(slot(##))``, where the number (``##``) is + between 1 and 24. To change between inner and outer rings of + samples: ``RE(xafs_wheel.inner())`` and ``RE(xafs_wheel.outer())``. + For other sample holders, move the appropriate sample stages. + +#. To import the spreadsheet into |bsui| (the data acquisition program): + ``xlsx()`` You will be presented with a list of .xlsx file choices, + pick the correct one by number and hit :key:`Enter`. You will then be + presented with a list of tabs in the spreadsheet file, pick the + correct tab by number and hit :key:`Enter`. (See :numref:`{name}, + Section {number} ` for details.) + +#. The output of the ``xlsx()`` command will display the command for + reviewing the file (the one with ``??``) and the command for + running the measurement sequence (the one with ``RE( + ... )``). Check to make sure it looks like |bsui| correctly + imported the information from the spreadsheet (you did remember to + save the spreadsheet, yes...?), then start the measurements. + +Align the *ex situ* sample wheel +-------------------------------- + +The attractive feature of the *ex situ* sample wheel is that it makes +it easy to move from sample to sample. Once the wheel is properly +aligned in the beam, the commands ``RE(slot(#))``, +``RE(xafs_wheel.inner())``, and ``RE(xafs_wheel.outer())`` are all +that's needed to move from sample to sample. + +For this to work, the wheel has to be properly aligned in the frist +place. + +Once the rotation stage is in place and a wheel is mounted on the +stage, place a phosphor screen in front of any slot *on the outer +ring*. The alignment procedure assumes that the initial alignment be +done of the outer ring. + +#. Rotate the wheel such that the phosphor screen is in the beam path + using ``RE(slot(#))``, where # is between 1 and 24. + +#. Make sure that at least one camera is looking at the phosphor screen. + +#. Secure the hutch and open the shutter. + +#. Move the sample stage in X and Y until you can see the spot of the + beam on the phosphor screen when looking through one of the + cameras. Use ``RE(mvr(xafs_x, ))`` and ``RE(mvr(xafs_y, + ))`` to move the stage in X and Y. ```` is a + *number* |nd| some sensible distance in millimeters. + +#. Using ``RE(mvr(xafs_x, ))`` and ``RE(mvr(xafs_y, + ))``, move the stage such that the beam is approximately in + position to go through the middle of a slot on the outer ring. + +#. Rotate the stage to an empty slot using ``RE(slot(#))``. + +#. Run the alignment procedure ``RE(find_slot())``. This will perform + linescans in X and Y to find the position such that the beam is + passing through the middle of the slot. The resulting plot should + look like something like this: + + .. _fig-find_slot_chore: + .. figure:: _images/find_slot.png + :target: _images/find_slot.png + :width: 50% + :align: center + + An example of the final plot for an alignment of the *ex situ* + sample wheel. The green X marks shows the aligned positions in + ``xafs_x`` and ``xafs_y``. + + +This procedure sets a parameter specifying the ``xafs_x`` position of +the outer ring. The inner ring is known to be 26 mm away. Thus, the +positions of both rings are set. This is why it is important to run +``RE(find_slot())`` on an outer ring slot. + +Restart the Linkam stage +------------------------ + +When it comes time to change samples in the Linkam stage, it is +usually much easier to dismount the stage so that it can be loaded +while flat. + +Dismount +~~~~~~~~ + +#. Turn off the Linkam controller, the black box sitting on the half + rack underneath the XAS table. Reach around behind the unit and + press the power button. If using the LN2 siphon, you do not need + to power down the siphon box. + +#. Remove the control cable from the stage. This is the black cable + with the barrel connector seen in :numref:`Figure %s + `. If using the LN2 siphon, also remove the + two LN2 hoses, the thick gray and thinner white hoses in that + photo. + + .. _fig-linkam_chore: + .. figure:: _images/linkam.jpg + :target: _images/linkam.jpg + :width: 50% + :align: center + +#. Unbolt the metal bracket supporting the Linkam stage from the + sample stage. + +Remount +~~~~~~~ + +#. Affix the metal bracket to the same position on the sample stage. + +#. Reattach the control cable and the LN2 hoses (if using the LN2 + siphon). + +#. If using the LN2 siphon, you may need to blow dry nitrogen gas + through the various parts of the system to avoid obstruction of LN2 + flow by water ice. 30 seconds of flowing dry nitrogen should be + enough to clean out all the lines. + +#. Power up the main Llinkam controller by pressing the power button + on the backside of the black box. + +#. At any computer with CSS, reboot the Linkam control software, by + following these steps: + + #. Click the "Info" button on the Linkam screen: + + .. _fig-linkam_main: + .. figure:: _images/Linkam_main.png + :target: _images/Linkam_main.png + :width: 50% + :align: center + + #. Click the "Reboot" button in the middle of the "IOC Diagnostics" + screen + + .. _fig-linkam_info: + .. figure:: _images/Linkam_info.png + :target: _images/Linkam_info.png + :width: 50% + :align: center + + #. Click the "Reboot now!" button on the "IOC Reboot Confirmation" + page. + + .. _fig-linkam_reboot: + .. figure:: _images/Linkam_reboot.png + :target: _images/Linkam_reboot.png + :width: 50% + :align: center + + The "Reboot Now!" button will be surrounded by a dashed pink + outline while rebooting. When the dashed pink outline goes + away, the reboot is finished. + + You can close the reboot screen by clicking the little x on its + tab. You can then return to the main Linkam screen by clicking + the yellow arrow at the top of the "IOC Diagnostics" screen. + +You are now ready to begin using the Linkam stage. + + +Filling the 25 liter LN2 dewar +------------------------------ + +#. Open both hutch doors. To open the right (manual) door, press the + :green:`Front Right Maglock` button (the green light should go out) on + the HMI panel. Push the door open manually. + +#. Open main LN2 valve. Use the button on the "06BM utilities" CSS + screen or at the |bsui| command line do ``ln2.open()``. + + .. _fig-utilities: + .. figure:: _images/BMM_utilities.png + :target: _images/BMM_utilities.png + :width: 60% + :align: center + +#. Lower the dewar to the lowest possible position on the blue lift + cart. + + .. _fig-25ldewar: + .. figure:: _images/25ldewar.jpg + :target: _images/25ldewar.jpg + :width: 60% + :align: center + +#. Place the dewar close to the LN2 tap, but some distance away from + the O2 sensor on the ODH monitor. If using the 25 L dewar, move it + on the cart at all times. (If using the 2 L dewar, secure the top + while moving the dewar.) + +#. Don safety glasses, face shield, lab coat, and insulated gloves. + All of those things can be found in the hutch. + +#. With gloved hand, place feed line securely in the dewar with the + outlet at the bottom of the dewar. + +#. With gloved hand, open the manual valve. Listen to make sure the + flow sounds OK. The noise should quiet down once liquid starts + flowing. That should take a minute or less. + +#. Monitor the dewar **AT ALL TIMES** while filling. + +#. Look for the styrofoam float to know when the LN2 level is + approaching the top. **DO NOT OVERFILL**. + +#. When full, using a gloved hand, close the valve in the hutch to + stop LN2 flow. + +#. With a gloved hand, remove the feed line from dewar. Slide it into + its resting place against the back wall, behind the cable trays. + +#. Move the blue cart back to its position near the experiment. Lift + the dewar to its operating height such that the siphon reaches the + Linkam stage. + +#. Do not remove glasses, face shield, lab coat, or gloves from the + hutch. + +#. Close main LN2 valve. Use the button on the "06BM utilities" CSS + screen or at the |bsui| command line do ``ln2.close()``. + +#. Press the :green:`Front Right Maglock` button. The green light should be + on. Manually close the right hutch door. diff --git a/_sources/data.rst b/_sources/data.rst new file mode 100644 index 0000000..d8a5f9c --- /dev/null +++ b/_sources/data.rst @@ -0,0 +1,306 @@ +.. + This document was developed primarily by a NIST employee. Pursuant + to title 17 United States Code Section 105, works of NIST employees + are not subject to copyright protection in the United States. Thus + this repository may not be licensed under the same terms as Bluesky + itself. + + See the LICENSE file for details. + +.. _data: + +Data access +=========== + +In May 2024, data security policies were implemented at BMM. This has +impact on user operations and data access. + +Data is now written to a secure location on central storage. Access +to your data now requires authentication using your BNL domain +account, your password, and two-factor authentication with DUO or a +Yubi key. + +The beamline operator account does not have access to user's data. + +This section of the beamline manual explains how to access your data +during and after your experiment. + +Downloading data +---------------- + +SFTP Client +~~~~~~~~~~~ + +You will need an sftp client. + ++ **Cross-platform:** `FileZilla `__. + This is a free program available for Windows, Apple, and Linux. The + explanation below will be made using FileZilla. + ++ **Windows users:** A popular option is `WinSCP + `__. Be careful at the WinSCP + website. You will see multiple pop-up adds with download links to + other software packages. Be careful to click on the link to the + WinSCP package. + ++ **Mac users:** Other options are `Termius + `__ + and `Flow `__. + ++ **Linux users:** Your desktop file manager likely has an sftp client + built in. Try typing + ``sftp://@sftp.nsls2.bnl.gov`` into your file + manager or creat a new network drive using ``ssh`` and + ``sftp.nsls2.bnl.gov``. + + `sshfs `__ is an + excellent solution. ``sshfs`` allows you to easily + mount the remote sftp site to a local mount point, + allowing you to browse the remote site as if it were a + local folder. For example, I do the following to mount + the data folder locally on my laptop: + + .. code-block:: bash + + sshfs bravel@sftp.nsls2.bnl.gov:/nsls2/data/bmm ~/mnt/bmm -o follow_symlinks + + and this to unmount the data folder + + .. code-block:: bash + + fusermount -u mnt/bmm + + + +The Short Version +~~~~~~~~~~~~~~~~~ + + +.. admonition:: Executive Summary + + #. Connect to sftp.nsls2.bnl.gov in your sftp client + #. Authenticate using your BNL username/password and DUO two-factor + authentication + #. Navigate to ``/nsls2/data/bmm/proposals/``, then to the cycle + folder corresponding to the date of your experiment, then to the + folder with your proposal number. So, something like + ``/nsls2/data/bmm/proposals/2024-2/pass-333333``. + #. Transfer your data to your local computer. + +If you preserve the folder structure from the remote host, the +:numref:`the dossier files (Section %s) ` will work as +expected. + +The ``assets`` folder contains raw image and HDF5 from your +experiment. Those files will have database-friendly but +user-unfriendly names. The HDF5 files are rather large and will take +some time to download. The can skip downloading the ``assets`` folder +if you do not plan on using the HDF5 files directly. + + +A Guide with Pictures +~~~~~~~~~~~~~~~~~~~~~ + +What follows is a guide with screenshots of using FileZilla to connect +to the SFTP download site and begin downloading data. + +To begin, open your sftp client. Here is FileZilla at startup. For +FileZilla, click on the File menu, then click on Site Manager. + +.. _fig-fz-startup: +.. figure:: _images/filezilla_startup.png + :target: _images/filezilla_startup.png + :width: 50% + :align: center + + FileZilla startup. Go to the Site Manager to establish a new + location. + + +In the site manager, click on the "New site" button and fill in the +details as shown below. Select the SFTP protocol, enter +``sftp.nsls2.bnl.gov`` as the Host. The correct port number is 22, +but you can usually leave that blank. 22 is the default port for the +sftp protocol. + +Finally, select "Interactive" as the logon type. That will tell +FileZilla to prompt you for both user name and two-factor authentication. + +.. _fig-fz-site_manager: +.. figure:: _images/filezilla_site_manager.png + :target: _images/filezilla_site_manager.png + :width: 50% + :align: center + + Fill in the site manager with the location and logon type for the + NSLS2 data center. + +Click OK to finish this configuration, then connect to the host. + + +.. _fig-fz-connect: +.. figure:: _images/filezilla_connect.png + :target: _images/filezilla_connect.png + :width: 50% + :align: center + + Select the NSLS2 host from the drop-down list and click to connect. + + +Connecting to the NSLS2 SFTP host will open up the password entry dialog. + + +.. _fig-fz-password: +.. figure:: _images/filezilla_password.png + :target: _images/filezilla_password.png + :width: 50% + :align: center + + Enter your BNL password and click OK. + +After entering your password, you will be prompted for two factor +authentication. In the "Password" box, type ``1`` and hit OK. Then +go to your phone and accept the DUO push. + +If you use a Yubikey, click on the "Password" box and touch the button +on your Yubikey. + +Once you have completed the DUO push, you will be able to navigate on +the remote site. Click your way to ``/nsls2/data/bmm/`` as shown below. + +.. _fig-fz-remote: +.. figure:: _images/filezilla_remote.png + :target: _images/filezilla_remote.png + :width: 50% + :align: center + + Navigate down to the BMM proposals area on the SFTP server. + +Click into ``proposals`` then into the folder for the cycle in which +your experiment happened, then into the folder for your proposal +number: + + +.. _fig-fz-folder: +.. figure:: _images/filezilla_folder.png + :target: _images/filezilla_folder.png + :width: 50% + :align: center + + Navigate into the folder for your proposal and the cycle in which + it ran. + +Now select the data files you want to transfer. You may select +multiple files or even entire folders. + +.. _fig-fz-queue: +.. figure:: _images/filezilla_queue.png + :target: _images/filezilla_queue.png + :width: 50% + :align: center + + Select some or all of your data and add it to the queue. + + +Click on the transfer button at the top of the screen to initiate the +transfer. At the beginning of the transfer, you will have to +re-authenticate yourself. + +.. _fig-fz-transfer: +.. figure:: _images/filezilla_transfer.png + :target: _images/filezilla_transfer.png + :width: 50% + :align: center + + Click the transfer button to download your data. You may need to + re-authenticate at the start of transfer. + +Your data is now on your computer. Yay! + + +Using the VDI virtual Desktop +----------------------------- + +.. todo:: Details needed + + +Accessing data from the beamline computers +------------------------------------------ + +Under the new data security regime, the beamline computer does not +have normal access to your data. This is because all users run their +experiment as the beamline operator. If the beamline operator |nd| +``xf06bm`` |nd| could see data, than any user could look at any other +user's data. + +Instead, data are stored on central storage with read permission +granted to everyone named on the user proposal. In this way, data are +secured from other users and access to the data requires +authentication. + +To look at your data while at the beamline, do the following + ++ Open a terminal window. Normally a terminal window with a white + background is open on screen and intended for this purpose. + |bsui| is typically run from a window with a black background, so + the white background is meant as a visual cue indicating that it is + the place for data access. + ++ In that terminal window type + + .. code-block:: bash + + ``su - `` + + replacing ```` with your actual user name. Enter your + password and respond to DUO push. + ++ ``cd`` to ``/nsls2/data3/bmm/proposals/2024-2/pass-123456``, replacing + ``2024-2`` with the cycle of your visit and ``123456`` with your + proposal number. + +|athena| can be launched from the command line. The best way to do this +is to type + +.. code-block:: bash + + dathena 2&>1 & + +at the command line. That incantation will suppress spurious screen +messages and put |athena| into the background so you can continue using +the command line. From there, simply use |athena|'s `File menu +`__ +to load data from your proposal folder. + + + +Accessing data via Tiled +------------------------ + +.. todo:: Details needed + +Accessing data via Jupyter +-------------------------- + +.. todo:: Details needed + +Why is data security important? +------------------------------- + +For those who have been coming to NSLS-II over the last decade, this +new emphasis on data security might be a bit surprising. In short, +the new data security model is consistent with Department of Energy +data policies. + +Recently, there were a pair of incidents involving accidental leaks of +sensitive synchrotron data to unauthorized parties. This sort of +violation of DOE policy can have an impact on the authorization to +operate NSLS-II as a user facility. Safe operations of the facility +includes data security. + +As a result, our Data Science and Systems Integration team at NSLS-II +has been begun moving the beamlines to a data acquisition model that +includes sounds data security practices. BMM volunteered to be an +early adopter of the new data security practices. We now provide an +excellent user experience at BM that includes secure data management. + diff --git a/_sources/desktop.rst b/_sources/desktop.rst new file mode 100644 index 0000000..46e41ae --- /dev/null +++ b/_sources/desktop.rst @@ -0,0 +1,136 @@ +.. + This document was developed primarily by a NIST employee. Pursuant + to title 17 United States Code Section 105, works of NIST employees + are not subject to copyright protection in the United States. Thus + this repository may not be licensed under the same terms as Bluesky + itself. + + See the LICENSE file for details. + +.. role:: key + :class: key + +.. _desktop: + +BMM's Desktop +============= + +This section provides an explanation of how to recover BMM's data +collection workspace after restarting the computer. + +This is made a bit easier if a couple of symbolic links are made in +``~/bin/``. If not already done, do this at the command line: + +.. code-block:: sh + + ln -s ~/.ipython/profile_collection/startup/consumer/run-consumer ~/bin/run-consumer + ln -s ~/git/BMM-beamline-configuration/tools/run-cadashboard ~/bin/run-cadashboard + +Monitoring +---------- + +To have eyeballs on the operation of the beamline, we want ``Phoebus`` +for engineering screens and the ``cadashboard`` application for the +heads-up overview of the state of the beamline. + +Phoebus (engineering screens) + There are two ways to start Phoebus. + + The more hands-on way is to open a new terminal window or tab, the + type ``run-phoebus`` at the command line. This will open the + Phoebus window and it should remember to place it on the top screen. + + It is not necessary to keep open the terminal window used to launch + Phoebus. Phoebus is **very noisy**, spewing en endless stream of + logging messages to the screen. Happily, it forks itself upon + launch, so it is not necessary to keep that terminal window open. + If you first opened a new window or tab, you can safely close it. + + There should be an icon for Phoebus on the icon bar at the bottom of + the lower, left screen. It looks like a red eyeball. Click on it. + + Once Phoebus is open, if the layout was not restored, click on the + "Window" menu, then on "Load Layout", then select "Two cameras". + +cadashboard + Open a new terminal window. Resize the window so it is a few lines + tall, and the full width of the screen. + + At the command line run ``run-cadashboard``. Make the font much + bigger by hitting :key:`Ctrl`-\ :key:`Shift`-\ :key:`+` several + times |nd| maybe as many as 10 |nd| until it takes up the full width + of the screen. + + Resize the height to just cover the three lines of the dashboard. + Move the terminal to the very top of the top screen. You can hide + the window decoration by using :key:`Windows`-\ :mark:`leftclick,.` + to position the terminal window. + + + +.. _fig-monitoring: +.. figure:: _images/monitoring.png + :target: _images/monitoring.png + :width: 80% + :align: center + + Once Phoebus and cadashboard are open, they should look something + like this. + + + +Data collection and visualization +--------------------------------- + +Data collection and visualization are separate things and should be +managed separately. + +On the main virtual desktop, open a terminal window for |bsui|. At the +command line type ``bsui``. |bsui| startup at BMM is rather +time-consuming, but after a couple minutes it is ready to go. + + +On the right-most virtual desktop, you will setup the Kafka file and +plot workers as explained in :numref:`the section on Kafka workers +(Section %s) `. + + +.. figure:: _images/bsui_startup2.png + :target: _images/bsui_startup2.png + :width: 80% + :align: center + + The tail end of the |bsui| startup messages and the + command prompt. + + +All the rest +------------ + +Some other things that are handy to have on the desktop during an +experiment: + ++ Firefox, with tabs open to :numref:`the command cheatsheet (Section + %s) ` and the XAS webcam. ++ Slack ++ A dolphin (file browser) window with a tab open to the current + Workspace folder + +Note that there is a button on the icon bar at the bottom of the +screen for |hephaestus|. + +In the new data security regime, launching |athena| as the beamline +account (``xf06bm``) is not very helpful given that it cannot access +the data. + +If data are accessed by opening a terminal window and doing ``su - +`` followed by authentication with password and DUO (see +:numref:`section %s `), first ``cd`` to the proposal directory, +then start |athena| with this command: + +.. code-block:: bash + + dathena > /dev/null 2>&1 & + +That will open |athena| and suppress the large stream of uniteresting +warning messages from the graphics tool kit. diff --git a/_sources/details.rst b/_sources/details.rst new file mode 100644 index 0000000..cb243df --- /dev/null +++ b/_sources/details.rst @@ -0,0 +1,906 @@ +.. + This document was developed primarily by a NIST employee. Pursuant + to title 17 United States Code Section 105, works of NIST employees + are not subject to copyright protection in the United States. Thus + this repository may not be licensed under the same terms as Bluesky + itself. + + See the LICENSE file for details. + +.. _details: + +Instrumentation Details +======================= + +This section is a scattershot assortment of details about things at +the beamline. This is basically an attempt to capture institutional +knowledge ... someplace. + +Network configuration +--------------------- + +When introducing a new device to the beamline network: + ++ Open a Jira ticket to request a new IP address and device name for + DNS. As an example, a new Moxa terminal server was introduced. In + the Jira ticket, we requested an IP address of 10.68.42.83 and a DNS + entry of ``xf06bm-tsrv13`` (full name: + ``xf06bm-tsrv13.nsls2.bnl.local``). + ++ In the case of a device that one in s series of similar devices, be + sure that the name matches the similar devices and that the number + is incremented. In the case of the new Moxa server, there are 12 + others at the beamline. The numbers were incremented correctly for + both the name and the IP address + ++ Identify a port on one of the servers that is available and + configured for the correct network. The networks are: + + + **SCI** (650): 10.68.40.xxx + + **CAM** (651): 10.68.41.xxx + + **INST** (652): 10.68.42.xxx + + **EPICS** (653): 10.68.43.xxx + +When configuring an individual device, here are the network configurations: + ++ The gateway address on each network is ``xz.yy.zz.2``. For example, + the gateway on the SCI network is 10.68.40.2. + +* The netmask is ``255.255.255.0``. + ++ There are two DNS servers: + + + DNS1: 10.65.2.25 + + DNS2: 10.65.2.26 + + +xf06bm-ws5, in-hutch Zoom calls +------------------------------- + +``xf06bm-ws5`` (``10.68.40.225``) is the System 76 Meerkat mounted on +the upstream wall of the end station. This machine is intended to +allow the users in the hutch to join a Zoom chat from within the +hutch. This helps Bruce provide user support from home. + +.. _fig-ws5: +.. figure:: _images/Xf06bm-ws5.jpg + :target: _images/Xf06bm-ws5.jpg + :width: 70% + :align: center + + +There are a number of peripherals attached to ``xf06bm-ws5``: + ++ A wireless mouse and keyboard clearly labeled as being for this + computer. These are normally tucked away on the ledge formed by the + panel seam on the upstream wall. I usually place these on the long + table when I want to use them + ++ A screen. This is the rather large screen mounted in the corner of + the hutch. It can be moved around somewhat for better viewing. + ++ A reasonably loud speaker. This is the black ball-shaped item + sitting on the frame of the upstream portion of the diagnostic + module. + ++ A good microphone. This is the Blue Yeti on stand above the I\ + :sub:`t` chamber on the XAS table. It has good noise cancellation so + the din from the XSpress3 should not effect voice quality. It is, + however, important that the speaker face the microphone rather than + the screen. + ++ A decent camera. This is the Nexigo mounted overhead next to the + Axis webcam and looking at the table. + + +While these devices are all connected to ``xf06bm-ws5`` and powered +on, there are no long running processes that connect to the camera or +microphone. You are not being spied upon while in the hutch -- unless +you are on a Zoom call, in which case the Zoom session will be on +screen. + +``xf06bm-ws5`` is available via Guacamole. When needed for remote +support, Bruce will usually initiate the Zoom call and have the hutch +computer join in. + + + + + +BNC Cable Map +------------- + +Here is an explanation of the BNC and SHV patch panels going between +rack D at the control station, Rack C on the roof of the hutch, and +the in-hutch patch panel. + + +.. _fig-bncpatch: +.. figure:: _images/Bnc_map.png + :target: _images/Bnc_map.png + :width: 100% + :align: center + +Inert Gas Plumbing +------------------ + +Needle valves are mounted on the outboard side of DM3. Quick connect +outlets for the gases are mounted on the upstream/inboard corner of +the XAFS table. + +.. admonition:: Gaseous nitrogen supply + :class: note + + BMM no longer uses a nitrogen cylinder as the supply of N\ :sub:`2` + for the ion chambers. The house GN2 supplies N\ :sub:`2` to the + needle valves. + +.. _fig-inertgas: +.. figure:: _images/Gas_handling.png + :target: _images/Gas_handling.png + :width: 100% + :align: center + +Vendor link for quick-disconnect fixture: https://www.mcmaster.com/5012K122/ + +In practice, the H\ :sub:`2`/N\ :sub:`2` and N\ :sub:`2`/Ar mixing +channels are not much used. Unless measuring with the incident beam +below 5 keV or above 21 keV, it is a poor use of time to make changes +to the gas content of the ion chambers. This is because it takes +quite some time for the volume of the ion chamber to equillibrate. + +N\ :sub:`2` is adequate for almost all experiments at BMM. For Tc or +Ru, it is helpful to use about 20% Ar. For Sc or lower, 50% He might +be helpful. But remember that purging the ion chambers takes +**hours**. + + + +Analog Video Capture +-------------------- + +Implementing `this USB video adapter +`__ +to capture video from the small analog cameras in the hutch took a bit +of doing. + +First, the adapter must be plugged directly into the computer. Using +a USB hub makes for an unreliable interface to the camera. + +Second, the file ``/etc/udev/rules.d/99-usb-camera-capture.rules`` is +needed to set permissions on ``/dev/video0`` correctly when the adapter is +plugged in. + +.. code-block:: none + + ACTION!="add|change", GOTO="webcam_capture_end" + SUBSYSTEM=="usb", ATTRS{idVendor}=="534d", ATTRS{idProduct}=="0021", MODE="0666" + KERNEL=="video*", ATTRS{idVendor}=="534d", ATTRS{idProduct}=="0021", GROUP="video", MODE="0666" + KERNEL=="video*", ATTRS{idVendor}=="534d", ATTRS{idProduct}=="0021", ATTRS{avoid_reset_quirk}=1 + KERNEL=="video*", ATTRS{idVendor}=="534d", ATTRS{idProduct}=="0021", ATTRS{quirks}=0x100 + LABEL="webcam_capture_end" + +Putting this file in place will require assistance from DSSI. Beamline +staff do not have permission to make a file in that folder. See `this +Jira ticket `__ for an +example of what to ask for. + +This recognizes the vendor and product IDs of the specific adapter +that I bought. When inserted, it sets the device to RW for all users +and sets a couple of possibly relevant attributes. (This udev rules +file was adapted from the rules file that comes with the easycap dc60 +package – info and links `here +`__). + +Next a small function was written as a wrapper around `fswebcam +`__ to grab frames from the +camera. The function is basically a wrapper around a call to +``fswebcam`` like so: + +.. code-block:: sh + + fswebcam -d /dev/video0 -r 640x480 -S 30 -F 5 foo.jpg + +along with some image processing using python's ``wand`` package. + +Required packages: + ++ ``fswebcam`` ++ ``python-wand`` ++ ``imagemagick`` + +This whole setup is filled with quirk. There is a delay accessing the +video capture. The ``-S`` switch builds in a 1 second delay, giving the +capture device enough time to begin displaying the image. The ``-F`` +switch tells the script how many frames to accumulate for good signal. +5 is probably overkill. + +In any case, it is now possible to grab screen shots of the currently +displayed analog video while collecting data. + +All of this is implemented in ``BMM/camera_device.py`` for use in +Bluesky. The heart of the implementation is a system call to +``fswebcam``. From there, the image is saved as an asset and correctly +pointed to in databroker. See: + ++ `BMM/user_ns/detectors.py `__ ++ `BMM/camera_device.py `__ + + +Pilatus 100K +------------ + +.. todo:: Need to flesh this out with explanatory text and screenshots + +How files saving works +~~~~~~~~~~~~~~~~~~~~~~ + ++ tiff files to /disk2 ++ /disk2 is mounted on xf06bm-ioc1 ++ tiff and hdf5 AD plugins write files to proposal directories ++ in |bsui|, there are pilatus and pilatus_tiff objects. normally use + pilatus, puilatus_tiff is helpful for testing tiff file writing, + which is used by IBM + + + +Moving the detector between end stations +~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~ + ++ power cables (strip and detector) ++ ethernet cables ++ GN2 line ++ grounding line + +The NFS server might need to be restarted after rebooting. As root on +xf06bm-pilatus100k, do + +.. code:: bash + + /etc/init.d/nfsserver restart + + + + +DI Water Flow +------------- + +The DI water is controled by manual valves, which should only be +operated by the utilities group, and by solenoid valves in the FOE. +The solenoid valves are triggered by a water-sensing strip along the +floor of the FOE. They are also actuated by switches on the CSS +utilities screen. These toggles are the ones circled in pink inthe +screenshot on the left. + +The valves themselves are the large yellow and black boxes mounted +high on the back wall of the FOE. The valve indicators are the rods +with orange markings. When the valves are open, the orange marks are +facing downstream. When closed, the orange marks are rotated towards +the wall. Opening and closing those valves is managed through CSS. +They must be open for the utilities group to do their work on the DI +delivery to the mono and the filter assemblies. + +.. subfigure:: ABC + :layout-sm: ABC + :gap: 8px + :subcaptions: above + :name: fig-diwater + :class-grid: outline + + .. image:: _images/Water_flow_CSS.png + + .. image:: _images/Water_flow_valves_1.jpg + + .. image:: _images/Water_flow_valves_2.jpg + + (Left) The CSS utilities screen where the water valve controls are + found. (Middle) A view into the FOE. (Right) The inboard wall + where the physical valve is found. + +Disabling an MCS8 axis after a move +----------------------------------- + +From Adam Young at FMBO +~~~~~~~~~~~~~~~~~~~~~~~ + +.. code-block:: none + + The motors can be disabled after a movement and this can be set at the + Delta Tau level. + + First you will need to connect to each MCS8+ with the beamline laptop + and start PEWin. + + Then please do the following: + + + Click on the 'View' menu at the top of the window. Then click + 'Program/PLC Status (and upload)'. + + Select PLC1 and click 'Upload'. An editor showing PLC1 will appear. + + Scroll down to find the variables P105 to P805. The '1' to '8' part + of these variables represent axis 1 to 8 on the MCS8+. The value of + these variables determines whether or not the motors will be + disabled after a move. They are likely all set to '0' meaning power + stays on. The lateral motors are on axis 4 and 5 so P405 and P505 + should be set to '1'. + + Click on the yellow downwards pointing arrow on the toolbar in the + editor. This downloads the modified PLC1 from the editor to the + Delta Tau. Close the editor. + + In the terminal window issue a 'save' to save the modified + configuration to the Delta Tau non-volatile memory and issue '$$$' + to refresh the controller. + +A follow up from Graeme Elliner, FMBO +~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~ + +.. code-block:: none + + Just done a fast scan of the config file and I think it is probably + because P302=1. + + Px02 and Px05 (where x is the motor number) are special Pvars for + setting the final state of the motor once it has stopped moving, they + are used in PLC1x and set as you know in PLC1 + + If Px02=1 the PLC to check if the motor is in position and its + desired velocity is zero, if these two conditions are set a Flag is + set, If the conditions are still met 1second later then the motor + is put into OPEN LOOP. This means the motor is still enabled but + will ignore the encoder and the motor will hold its current rotary + location. This is useful for the motors that have DPTs pushing + against them in flexures (trapezoidal roll and pitch assy on the + DCMs), it gives a firm base for the DPT to push against but will + not try to hold position (as it would in closed loop) when the DPT + pushes the top part of the stage and moves the encoder. If Px05=1 + then the PLC checks to see if the motor is in position and has zero + velocity, then 1second later it will kill that motor + + Due to the way the code is ordered (it looks for thePx02 first) it + will enter Px02 check first, when the conditions are met it will + set the first Flag After that check it then see the Px05 check and + kills the motor. However on the next pass through the PLC it will + again enter the Px02 check, see that the first flag has been set + then trigger the open loop command, re-enabling the motor. + + Hence by setting P302=0 in PLC1, it will not go into the check and + not accidentally enable the motor. If this does not fix it then + the issue is in EPICS + +Conclusion +~~~~~~~~~~ + +The above suggestions were done for ``dm3_bct``, a motor that was +showing the re-enable behavior. This made that motor tricky to +operate in bluesky. Setting ``P302=0`` and ``P305=1`` did the trick. + + +Vortex pressure +--------------- + +Using a probe to measure the voltage on the IP port of the Vortex ME4. +This reading will tell you the internal pressure according to the +table in the snapshot below. + +.. _fig-votexpressure: +.. figure:: _images/Vortex_pressure.jpeg + :target: _images/Vortex_pressure.jpeg + :width: 40% + :align: center + + +====================== ========== + IP reading (Voltage) Pressure +====================== ========== + -0.01 5E-9 + -0.1 5E-8 + -1 5E-7 + -10 5E-6 +====================== ========== + + +Temperature reading should be 1.5 V when the TEC is at proper +temperature. + +`Vortex SDD manual +`__ +(link to copy at APS detector pool). + +There is a copy of the Vortex manual at BMM. Look in ``/nsls2/data3/bmm/legacy/products/ME7/``, +the file is called ``Vtx-Multi-El User Manual Rev 15.0_Oct 16, 2023.pdf``. + + +DM3 CAT6 Patch Panel +-------------------- + +13 more CAT6 ports for use in the hutch. Note that ports listed as +SCI/EPICS are tagged ports on both subnets. + +This is needed by workstations (like ``xf06bm-ws5``), display machines +running CSS (like ``xf06bm-disp1``), and machines running IOCs (like +``xf06bm-xspress3``). + +Note that ``xf06bm-em1`` needs to be on an INST port while the ion +chambers are on EPICS ports. The difference is that the ion chambers +are running their own on-board IOCs, making them more like IOC servers +than instruments. + + + + ++-----------+----------+--------------------+----------------+---------------------+-------------------+ +| **Patch** | **Port** | **xf06bm-a port** | **Network** | **Role** | **Cable number** | ++-----------+----------+--------------------+----------------+---------------------+-------------------+ +| **DM3-A** | 1 | 44 | EPICS | xf06bm-ic1 | 200235 | ++-----------+----------+--------------------+----------------+---------------------+-------------------+ +| | 2 | 45 | EPICS | xf06bm-ic2 | 200236 | ++-----------+----------+--------------------+----------------+---------------------+-------------------+ +| | 3 | 46 | EPICS | xf06bm-ic3 | 200237 | ++-----------+----------+--------------------+----------------+---------------------+-------------------+ +| | 4 | 06bm-agg 36 | INST | xf06bm-em1 | 200238 | ++-----------+----------+--------------------+----------------+---------------------+-------------------+ +| **DM3-B** | 1 | 17 | SCI/EPICS | xf06bm-ws5 | 200239 | ++-----------+----------+--------------------+----------------+---------------------+-------------------+ +| | 2 | 18 | SCI/EPICS | xf06bm-disp1 | 200240 | ++-----------+----------+--------------------+----------------+---------------------+-------------------+ +| | 3 | 19 | SCI/EPICS | xf06bm-xspress3 | 200241 | ++-----------+----------+--------------------+----------------+---------------------+-------------------+ +| | 4 | | SCI/EPICS | | 200242 | ++-----------+----------+--------------------+----------------+---------------------+-------------------+ +| **DM3-C** | 1 | | | | 200243 | ++-----------+----------+--------------------+----------------+---------------------+-------------------+ +| | 2 | | | | 200244 | ++-----------+----------+--------------------+----------------+---------------------+-------------------+ +| | 3 | | | | 200245 | ++-----------+----------+--------------------+----------------+---------------------+-------------------+ +| | 4 | | | | 200246 | ++-----------+----------+--------------------+----------------+---------------------+-------------------+ +| **DM3-D** | 1 | | | | 200247 | ++-----------+----------+--------------------+----------------+---------------------+-------------------+ +| | 2 | | | unused | | ++-----------+----------+--------------------+----------------+---------------------+-------------------+ +| | 3 | | | unused | | ++-----------+----------+--------------------+----------------+---------------------+-------------------+ +| | 4 | | | unused | | ++-----------+----------+--------------------+----------------+---------------------+-------------------+ + + +Some photos of the patch panel: + +.. subfigure:: AB + :layout-sm: AB + :gap: 8px + :subcaptions: above + :name: fig-dm3cat6 + :class-grid: outline + + .. image:: _images/DM3_patch_panel.jpg + + .. image:: _images/DM3_first_cat6.jpg + + (Left) CAT6 patch panel at DM3. (Right) Lowest numbered label on + the CAT6 cables in the DM3 patch panel + + +Logitech controller +------------------- + +.. _fig-logitech: +.. figure:: _images/Logitech.png + :target: _images/Logitech.png + :width: 100% + :align: center + + +.. todo:: + + Explain how to configure buttons in CSS + +.. todo:: Left joystick will be used for detector YZ. Not X! + + +Motor controllers +----------------- + +This section is a big, long list of all the motor PV names at BMM. + +Most motors have aliases. The alias is an alternate, easier-to-type +name for the axis. These are equivalent: + +.. code-block:: sh + + caget XF:06BMA-OP{Mono:DCM1-Ax:Bragg}Mtr + caget xafs_bragg + +Aliases work with most motor record fields, as well. The following are +also equivalent: + +.. code-block:: sh + + caget XF:06BMA-OP{Mono:DCM1-Ax:Bragg}Mtr.VELO + caget xafs_bragg.VELO + + +The following tables give PV name and alias, a brief description of +the purpose of the motor, the controller and location of that +controller, and the channel number in the controller. A few +abbreviations are used: + +:us: upstream +:ds: downsteam +:ib: inboard +:ob: outboard +:para: parallel +:perp: perpendicular + + +Collimating mirror, M1 +~~~~~~~~~~~~~~~~~~~~~~ + +================================ ========= ========================= ====================== ============== +PV alias Motor Description controller motor number +================================ ========= ========================= ====================== ============== +XF:06BM-OP{Mir:M1-Ax:YU}Mtr m1_yu us jack MC01 (mezzanine) 1 +XF:06BM-OP{Mir:M1-Ax:YDO}Mtr m1_ydo ds, outboard jack MC01 (mezzanine) 2 +XF:06BM-OP{Mir:M1-Ax:YDI}Mtr m1_ydi ds, inboard jack MC01 (mezzanine) 3 +XF:06BM-OP{Mir:M1-Ax:XU}Mtr m1_xu us lateral MC01 (mezzanine) 4 +XF:06BM-OP{Mir:M1-Ax:XD}Mtr m1_xd ds lateral MC01 (mezzanine) 5 +================================ ========= ========================= ====================== ============== + +Filters, DM1 +~~~~~~~~~~~~ + +================================ ============ ========================= ====================== ============== +PV alias Motor Description controller motor number +================================ ============ ========================= ====================== ============== +XF:06BMA-BI{Fltr:01-Ax:Y1}Mtr dm1_filters1 assembly #1 MC05 (RGA) 6 +XF:06BMA-BI{Fltr:01-Ax:Y2}Mtr dm1_filters2 assembly #2 MC05 (RGA) 7 +================================ ============ ========================= ====================== ============== + +DCM +~~~ + +=================================== ============ ==================== ====================== ============== +PV alias Motor Description controller motor number +=================================== ============ ==================== ====================== ============== +XF:06BMA-OP{Mono:DCM1-Ax:Bragg}Mtr dcm_bragg DCM Bragg MC02 (RGA) 1 +XF:06BMA-OP{Mono:DCM1-Ax:Bragg2}Mtr dcm_bragg2 Bragg 2nd encoder MC02 (RGA) +XF:06BMA-OP{Mono:DCM1-Ax:P2}Mtr dcm_pitch 2nd xtal pitch MC02 (RGA) 3 +XF:06BMA-OP{Mono:DCM1-Ax:R2}Mtr dcm_roll 2nd xtal roll MC02 (RGA) 4 +XF:06BMA-OP{Mono:DCM1-Ax:Per2}Mtr dcm_para 2nd xtal perp MC02 (RGA) 5 +XF:06BMA-OP{Mono:DCM1-Ax:Par2}Mtr dcm_perp 2nd xtal para MC02 (RGA) 6 +XF:06BMA-OP{Mono:DCM1-Ax:X}Mtr dcm_x lateral MC02 (RGA) 7 +XF:06BMA-OP{Mono:DCM1-Ax:Y}Mtr dcm_y vertical MC02 (RGA) 8 +=================================== ============ ==================== ====================== ============== + +Slits 2, DM2 +~~~~~~~~~~~~ + +============================== ============ ==================== ====================== ============== +PV alias Motor Description controller motor number +============================== ============ ==================== ====================== ============== +XF:06BMA-OP{Slt:01-Ax:O}Mtr dm2_slits_o outboard MC03 (RGA) 1 +XF:06BMA-OP{Slt:01-Ax:I}Mtr dm2_slits_i inboard MC03 (RGA) 2 +XF:06BMA-OP{Slt:01-Ax:T}Mtr dm2_slits_t top MC03 (RGA) 3 +XF:06BMA-OP{Slt:01-Ax:B}Mtr dm2_slits_b bottom MC03 (RGA) 4 +============================== ============ ==================== ====================== ============== + + +DM2 fluorescence screen +~~~~~~~~~~~~~~~~~~~~~~~ + +============================== ============ ==================== ====================== ============== +PV alias Motor Description controller motor number +============================== ============ ==================== ====================== ============== +XF:06BMA-BI{Diag:02-Ax:Y}Mtr dm2_fs vertical MC04 (RGA) 7 +============================== ============ ==================== ====================== ============== + +Focusing mirror, M2 +~~~~~~~~~~~~~~~~~~~ + + +============================== ============ ==================== ====================== ============== +PV alias Motor Description controller motor number +============================== ============ ==================== ====================== ============== +XF:06BMA-OP{Mir:M2-Ax:YU}Mtr m2_yu us jack MC04 (RGA) 1 +XF:06BMA-OP{Mir:M2-Ax:YDO}Mtr m2_ydo ds, outboard jack MC04 (RGA) 2 +XF:06BMA-OP{Mir:M2-Ax:YDI}Mtr m2_ydi ds, inboard jack MC04 (RGA) 3 +XF:06BMA-OP{Mir:M2-Ax:XU}Mtr m2_xu us lateral MC04 (RGA) 4 +XF:06BMA-OP{Mir:M2-Ax:XD}Mtr m2_xd ds lateral MC04 (RGA) 5 +XF:06BMA-OP{Mir:M2-Ax:Bend}Mtr m2_bender bender MC04 (RGA) 6 +============================== ============ ==================== ====================== ============== + +Harmonic rejection mirror, M3 +~~~~~~~~~~~~~~~~~~~~~~~~~~~~~ + +============================== ============ ==================== ====================== ============== +PV alias Motor Description controller motor number +============================== ============ ==================== ====================== ============== +XF:06BMA-OP{Mir:M3-Ax:YU}Mtr m3_yu us jack MC05 (RGA) 1 +XF:06BMA-OP{Mir:M3-Ax:YDO}Mtr m3_ydo ds, outboard jack MC05 (RGA) 2 +XF:06BMA-OP{Mir:M3-Ax:YDI}Mtr m3_ydi ds, inboard jack MC05 (RGA) 3 +XF:06BMA-OP{Mir:M3-Ax:XU}Mtr m3_xu us lateral MC05 (RGA) 4 +XF:06BMA-OP{Mir:M3-Ax:XD}Mtr m3_xd ds lateral MC05 (RGA) 5 +============================== ============ ==================== ====================== ============== + +Slits 3, DM3 +~~~~~~~~~~~~ + +============================== ============ ==================== ====================== ============== +PV alias Motor Description controller motor number +============================== ============ ==================== ====================== ============== +XF:06BM-BI{Slt:02-Ax:O}Mtr dm3_slits_o outboard MC06 (RGC1) 5 +XF:06BM-BI{Slt:02-Ax:I}Mtr dm3_slits_i inboard MC06 (RGC1) 6 +XF:06BM-BI{Slt:02-Ax:T}Mtr dm3_slits_t top MC06 (RGC1) 7 +XF:06BM-BI{Slt:02-Ax:B}Mtr dm3_slits_b bottom MC06 (RGC1) 8 +============================== ============ ==================== ====================== ============== + +DM3 +~~~ + +============================== ============ ==================== ====================== ============== +PV alias Motor Description controller motor number +============================== ============ ==================== ====================== ============== +XF:06BM-BI{FS:03-Ax:Y}Mtr dm3_fs fluorescent screen MC06 (RGC1) 1 +XF:06BM-BI{Fltr:01-Ax:Y}Mtr dm3_foils foils actuator MC06 (RGC1) 4 +XF:06BM-BI{BCT-Ax:Y}Mtr dm3_bct vertical stage MC06 (RGC1) 3 +XF:06BM-BI{BPM:1-Ax:Y}Mtr dm3_bpm NanoBPM MC06 (RGC1) 2 +============================== ============ ==================== ====================== ============== + +XAFS Table +~~~~~~~~~~ + +=============================== ============ ==================== ====================== ============== +PV alias Motor Description controller motor number +=============================== ============ ==================== ====================== ============== +XF:06BMA-BI{XAFS-Ax:Tbl_YU}Mtr xafs_yu xafs table y us MC07 (RGC1) 1 +XF:06BMA-BI{XAFS-Ax:Tbl_YDO}Mtr xafs_ydo xafs table y ds ob MC07 (RGC1) 2 +XF:06BMA-BI{XAFS-Ax:Tbl_YDI}Mtr xafs_ydi xafs table y ds ib MC07 (RGC1) 3 +XF:06BMA-BI{XAFS-Ax:Tbl_XU}Mtr xafs_xu xafs table x us MC07 (RGC1) 4 +XF:06BMA-BI{XAFS-Ax:Tbl_XD}Mtr xafs_xd xafs table x ds MC07 (RGC1) 5 +=============================== ============ ==================== ====================== ============== + + +XAFS Stages +~~~~~~~~~~~ + +.. todo:: This table needs attention + + +================================ ============ ==================== ====================== ============== +PV alias Motor Description controller motor number +================================ ============ ==================== ====================== ============== +XF:06BMA-BI{XAFS-Ax:LinY}Mtr xafs_liny xafs sample y MC08 (RGC1) 1 +XF:06BMA-BI{XAFS-Ax:LinX}Mtr xafs_linx xafs sample x MC08 (RGC1) 2 +XF:06BMA-BI{XAFS-Ax:LinS}Mtr xafs_lins xafs sample small MC08 (RGC1) 3 +XF:06BMA-BI{XAFS-Ax:LinXS}Mtr xafs_linxs xafs reference MC08 (RGC1) 4 +XF:06BMA-BI{XAFS-Ax:Pitch}Mtr xafs_pitch xafs pitch stage MC08 (RGC1) 5 +XF:06BMA-BI{XAFS-Ax:Roll}Mtr xafs_roll xafs tilt stage MC08 (RGC1) 6 +. . xafs reference wheel MC08 (RGC1) 7 +. . glancing rotation MC08 (RGC1) 8 +XF:06BMA-BI{XAFS-Ax:Tbl_RotH}Mtr xafs_roth xafs Huber MC07 (RGC1) 6 +XF:06BMA-BI{XAFS-Ax:Tbl_RotB}Mtr xafs_rotb xafs black rot stage MC07 (RGC1) 7 +XF:06BMA-BI{XAFS-Ax:Tbl_RotS}Mtr xafs_rots xafs small rot stage MC07 (RGC1) 8 +================================ ============ ==================== ====================== ============== + +Gonimeter circles +~~~~~~~~~~~~~~~~~ + +============================= ============= ===================== ====================== ============== +PV alias Motor Description controller motor number +============================= ============= ===================== ====================== ============== +XF:06BM-ES{SixC-Ax:VTTH}Mtr 6bm:sixc_vtth Vertical two theta MC11 (RGC2) 1 +XF:06BM-ES{SixC-Ax:VTH}Mtr 6bm:sixc_vth Vertical theta MC11 (RGC2) 2 +XF:06BM-ES{SixC-Ax:CHI}Mtr 6bm:sixc_chi Chi MC11 (RGC2) 3 +XF:06BM-ES{SixC-Ax:PHI}Mtr 6bm:sixc_phi Phi MC11 (RGC2) 4 +XF:06BM-ES{SixC-Ax:HTH}Mtr 6bm:sixc_hth Horizontal theta MC11 (RGC2) 5 +XF:06BM-ES{SixC-Ax:HTTH}Mtr 6bm:sixc_htth Horizontal two theta MC11 (RGC2) 6 +XF:06BM-ES{SixC-Ax:ANAL}Mtr 6bm:sixc_anal Analyzer MC11 (RGC2) 7 +XF:06BM-ES{SixC-Ax:DET}Mtr 6bm:sixc_det Detector MC11 (RGC2) 8 +============================= ============= ===================== ====================== ============== + +Goniometer motors +~~~~~~~~~~~~~~~~~ + +============================== ============== ==================== ====================== ============== +PV alias Motor Description controller motor number +============================== ============== ==================== ====================== ============== +XF:06BM-ES{SixC-Ax:DETHOR}Mtr 6bm:sixc_det_h det horiz MC12 (RGC2) 1 +XF:06BM-ES{SixC-Ax:WHEEL1}Mtr 6bm:sixc_wh1 wheel 1 MC12 (RGC2) 2 +XF:06BM-ES{SixC-Ax:WHEEL2}Mtr 6bm:sixc_wh2 wheel 2 MC12 (RGC2) 3 +XF:06BM-ES{SixC-Ax:SAMX}Mtr 6bm:sixc_samx sample X MC12 (RGC2) 4 +XF:06BM-ES{SixC-Ax:SAMY}Mtr 6bm:sixc_samy sample Y MC12 (RGC2) 5 +XF:06BM-ES{SixC-Ax:SAMZ}Mtr 6bm:sixc_samz sample Z MC12 (RGC2) 6 +XF:06BM-ES{SixC-Ax:Tbl_YD}Mtr 6bm:sixc_tyd table Y ds MC12 (RGC2) 7 +XF:06BM-ES{SixC-Ax:Tbl_YUI}Mtr 6bm:sixc_tyui table Y us ib MC12 (RGC2) 8 +============================== ============== ==================== ====================== ============== + + + +Goniometer table +~~~~~~~~~~~~~~~~ + +============================== =============== ==================== ====================== ============== +PV alias Motor Description controller motor number +============================== =============== ==================== ====================== ============== +XF:06BM-ES{SixC-Ax:Tbl_YUO}Mtr 6bm:sixc_tyuo table Y us ob MC13 (RGC2) 1 +XF:06BM-ES{SixC-Ax:Tbl_XU}Mtr 6bm:sixc_txu table X us MC13 (RGC2) 2 +XF:06BM-ES{SixC-Ax:Tbl_XD}Mtr 6bm:sixc_txd table X ds MC13 (RGC2) 3 +XF:06BM-ES{SixC-Ax:Tbl_Z}Mtr 6bm:sixc_tz table Z MC13 (RGC2) 4 +XF:06BM-ES{SixC-Ax:Slt1_T}Mtr 6bm:sixc_slt1_t top slit MC13 (RGC2) 5 +XF:06BM-ES{SixC-Ax:Slt1_B}Mtr 6bm:sixc_slt1_b bottom slit MC13 (RGC2) 6 +XF:06BM-ES{SixC-Ax:Slt1_I}Mtr 6bm:sixc_slt1_i inboard slit MC13 (RGC2) 7 +XF:06BM-ES{SixC-Ax:Slt1_O}Mtr 6bm:sixc_slt1_o outboard slit MC13 (RGC2) 8 +============================== =============== ==================== ====================== ============== + + + +Shutters and screen +~~~~~~~~~~~~~~~~~~~ + +================================ ========= ====================== ====================== ============== +PV alias Motor Description controller motor number +================================ ========= ====================== ====================== ============== +XF:06BM-PPS{Sh:FE}Pos-Sts front end shutter PPS +XF:06BM-PPS{Sh:A}Pos-Sts A hutch shutter PPS +XF:06BMA-OP{FS:1}Pos-Sts fluorescent screen EPS +================================ ========= ====================== ====================== ============== + + + +Front-end slits +~~~~~~~~~~~~~~~ + +================================ ========= ====================== ====================== ============== +PV alias Motor Description controller motor number +================================ ========= ====================== ====================== ============== +FE:C06B-OP{Slt:12-Ax:X}size horizontal size geobrick (mezzanine) virtual +FE:C06B-OP{Slt:12-Ax:X}center horizontal center geobrick (mezzanine) virtual +FE:C06B-OP{Slt:12-Ax:Y}size vertical size geobrick (mezzanine) virtual +FE:C06B-OP{Slt:12-Ax:Y}center vertical center geobrick (mezzanine) virtual +FE:C06B-OP{Slt:1-Ax:Hrz}Mtr Slit 1 horizontal geobrick (mezzanine) +FE:C06B-OP{Slt:1-Ax:Inc}Mtr Slit 1 incline geobrick (mezzanine) +FE:C06B-OP{Slt:1-Ax:O}Mtr Slit 1 X outboard geobrick (mezzanine) +FE:C06B-OP{Slt:1-Ax:T}Mtr Slit 1 Y top geobrick (mezzanine) +FE:C06B-OP{Slt:2-Ax:Hrz}Mtr Slit 2 horizontal geobrick (mezzanine) +FE:C06B-OP{Slt:2-Ax:Inc}Mtr Slit 2 incline geobrick (mezzanine) +FE:C06B-OP{Slt:2-Ax:I}Mtr Slit 2 X inboard geobrick (mezzanine) +FE:C06B-OP{Slt:2-Ax:B}Mtr Slit 2 Y bottom geobrick (mezzanine) +================================ ========= ====================== ====================== ============== + + +Encoder loss second crystal roll +-------------------------------- + +On 9 January, 2018, when attempting to home the mono motors following +the schduled power outage in December, the 2\ :sup:`nd` crystal roll +motor moved to its negative limit, then reported an encoder loss. +With Graeme Elliner's (an FMB-O controls engineer) help, I came to a +resolution of the problem. It has left that axis in an unusual state +that needs to be documented. + +Executive summary: that axis does not use its encoder. It homes by +running to its negative limit, then running back to it's home +position. It does this by counting controller pulses rather than +encoder + +Here are a couple of useful emails from Graeme to me from January 11 +and 12, 2018. + +.. code-block:: text + + Hi Bruce + + We now need to work out where the problem is. + NOTE you will not be able to drive anything while you're doing this or you potentially can break more. + 1: Unplug the Disable Plug from the back of the DCM (this will + force all motors to be disabled) - it’s the small black connector + (bottom right as you look at the back) + 2: Disconnect PL102 & SK102 from IF2 + 3: Disconnect PL103 & SK103 from IF3 + 4: Connect PL103 & SK103 to IF2 + 5: Connect PL102 & SK102 to IF3 + + IF the Red light on the Interpolator stayed with IF3 then there is + a problem with Interpolator - Need to put motor into Open Loop + + IF the Red light on the Interpolator has moved to IF2 then the + Interpolator is fine and it is cabling somewhere - GOTO STEP 6 + + 6: SWAP PL102 and PL103 + + IF the red light has moved back to IF3 then the problem is between + PL103 to the read head on the Xtal2 Roll stage - GOTO STEP 7 + + IF the red light has stayed with IF2 then the problem is between SK103 to the MCS8 + + This cabling is Pin to Pin so a simple continuity test on each pin should identify what has broken + + 7: SWAP SK102 and SK103. The cabling should now be back to the original layout + 8: SWAP SK203-2 and SK203-3 at the feedthroughs on the DCM (FD3-2 & FD3-3 respectively) + + IF the red light has moved to IF2 then the problem is INSIDE the + DCM vessel - Need to put motor into Open Loop (and ultimately open + the vessel to find it) + + If the red light has stayed on IF3 then there is a problem with the + cable to the DCM. This cable is should be Pin to Pin so a simple + continuity test on each pin should identify what has broken + + + To Put the Roll Axis into Open Loop + Have you got PeWin working now?? + Using Pewin backup the config for the DCM and send it to me please. + +A lot of the cable swapping Graeme called for was to try to isolate a +bad connection. The connection between read-head and motor controller +is rather lengthy, with a vacuum feedthrough, a feedtrough on the side +of the service box, and two connections to the interpolators inside +the service box. + +Following the steps laid out by Graeme, I isolated the problem to +being inside the vacuum vessel. Drat! Using the `old MC02 +configuration +`__ +I saved to a file, Graeme made some edits as described below and sent +me `a new configuration file +`__. + +.. code-block:: text + + Hi Bruce + I have modified the config file to now not use the encoder for + position feedback. I have tested that it downloads with no errors + + Details of the mods are listed at the top of the file and below, I + have marked all modifications with either GRE+ (for added code) or + GRE- (for commented code) + + In PLC1 + P446=0 this disables encoder loss detection for axis4 + + + In the Ivars + I430=700 changed to default stepper gain for no encoder + I432=0 changed to default value for no encoder + I7040=8 this forces the system to use steps for feedback + + Use restore config from the backup menu in PEWin to install this + CHECK that the box at the bottom reports NO ERRORS, + in the terminal window you will need to "SAVE" and "$$$". + + You will now find that the position scaling will be completely + different now that you are not using the encoder. This means that + your jog speeds will also be different + + I strongly suggest NOT trying to use EPICS imediately. + + Use the PeWin terminal (or the Jog Ribbon) to move axis 4 to the -ve + limit ("#4j-") at the -ve limit type "#4HMZ" to zero the postion + display and then to the +ve limit ("#4j+"). + + This will tell you how many steps there are between the limits. + + Using this info and the data for the encoded version you should be + able to move th axis to approximately the correct location. + + I have noticed that in PLC14 (the homing PLC for axis 4) that even + when the axis was using the encoder the home routine was not using + the encoder home refernce. + + It is moving to the -ve limit then moving off 51926 encoder counts, + then setting this to be HOME - search for GRE*** in the file. + + This will not be correct now the system is using steps and might + actually be more than you have measured as the range in steps. + + You will need to change this value before you can use EPICS to home the axis. + + Once all this is working in PeWin you can test the homing routine + by entering M1416=1 in the Pewin terminal. + +Following this set of instructions, I found that there are 1,218,299 +steps between the two limits on the 2\ :sup:`nd` crystal roll motor. +It would seem that there are about 10 or 12 steps per encoder count. +The homing procedure works in the sense of finding the negative limit, +then moving to a home position. But that home position seems to be +about 1/10 of the way between the negative limit and the +home-using-encoder-counts. diff --git a/_sources/index.rst b/_sources/index.rst new file mode 100644 index 0000000..75e1d2b --- /dev/null +++ b/_sources/index.rst @@ -0,0 +1,60 @@ +.. + This document was developed primarily by a NIST employee. Pursuant + to title 17 United States Code Section 105, works of NIST employees + are not subject to copyright protection in the United States. Thus + this repository may not be licensed under the same terms as Bluesky + itself. + + See the LICENSE file for details. + + +BMM Users Manual +================ + +BMM is NIST's :red:`B`\ eamline for :red:`M`\ aterials :red:`M`\ easurement. + +.. _fig-bananagrams: +.. figure:: _images/Beamline-bananagrams.jpg + :target: _images/Beamline-bananagrams.jpg + :width: 50% + :align: left + + BMM at a glance! + + +**Contents:** + +.. toctree:: + :maxdepth: 2 + :numbered: + :caption: Running the beamline + + intro.rst + proposal.rst + before.rst + data.rst + instruments.rst + motors.rst + pds.rst + linescans.rst + xafs.rst + other.rst + automation.rst + log.rst + manage.rst + commonchores.rst + cheatsheet.rst + trouble.rst + +.. appendix:: + :maxdepth: 2 + :numbered: + :caption: Developer documentation + + desktop.rst + profile.rst + kafka.rst + ionchambers.rst + details.rst + restore.rst + todo.rst diff --git a/_sources/instruments.rst b/_sources/instruments.rst new file mode 100644 index 0000000..352cd4d --- /dev/null +++ b/_sources/instruments.rst @@ -0,0 +1,366 @@ +.. + This document was developed primarily by a NIST employee. Pursuant + to title 17 United States Code Section 105, works of NIST employees + are not subject to copyright protection in the United States. Thus + this repository may not be licensed under the same terms as Bluesky + itself. + + See the LICENSE file for details. + +.. _instruments: + + + +Beamline instrumentation +======================== + +This section provides a brief overview of the instrumentation +available for various kinds of routine and *in situ* experiments. + +If you have questions about any of these tools or wish to pursue other +experimental options, contact the beamline staff. + +Fluorescence detectors +---------------------- + +The standard fluorescence detector at BMM is a `four element silicon +drift detector +`__ +with an `Xspress3 `_ +readout. This normally sits on a linear stage so that distance to the +sample can be user-controlled and incorporated into :numref:`beamline +automation (Section %s) `. + +We also have a `single element silicon drift detector +`__ +which is useful in certain situations. If your experimental setup +requires placing the detector in an unusual orientation, the single +element detector can be used. Unlike the 4-element detector, the +single element is not required to remain in an upright orientation +during operation. While the single element detector sees fewer +photons, this versatility of setup is occasionally very helpful. + + +.. subfigure:: AB + :layout-sm: AB + :gap: 8px + :subcaptions: above + :name: fig-XRFINST + :class-grid: outline + + .. image:: _images/4element.jpg + + .. image:: _images/1element.jpg + + (Left) Four element silicon drift detector. (Right) One element + silicon drift detector. + + +.. note:: + Thanks to funding from the CHIPS Act, BMM now has a new + 7-element SDD! + + It has been calibrated and fully integrated into the Bluesky data + acquisition system. + + The new XYZ stage has also arrived and will be integrated during + the 2024-2025 winter shutdown. + + .. _fig-7element: + .. figure:: _images/7element.jpg + :target: _images/7element.jpg + :width: 50% + :align: center + + +Electron yield detector +----------------------- + +.. todo:: + + Document this! + +.. todo:: + + Better scheme in Bluesky for specifying that the yield detector is + in use and that its signal should be plotted in real time. + + +Area detector +------------- + +An older model of the `Pilatus 100K +`_ +is available. + +.. _fig-pilatusINST: +.. figure:: _images/pilatus.jpg + :target: _images/pilatus.jpg + :width: 50% + :align: center + + Dectris Pilatus 100K + + +Please note: + ++ BMM offers only limited integration of data output into the XAS end + station workflow. ++ BMM has limited options for mounting and integrating the Pilatus + into your experiments. ++ This Pilatus has a rather small detection area and a rather large + pixel size (about 170 microns). + +BMM does not currently have access to a larger/better/faster detector. + +Using CHIPS Act funding, we are in the process of procuring a new +large area detector for use with the XRD end station. This detector +will eventually become available for experiments at the XAS end +station, perhaps by 2026. + + +.. _sample-wheel: + +Sample wheel +------------ + +At BMM, the standard *ex situ* sample stage is a laser-cut plastic +disk. The disk has 24 or 48 slots cut from the disk. These are the +sample positions. + +This disk is mounted on a rotation stage. The slots are 15 degrees +apart, so moving from sample to sample only involves moving through a +known rotation angle. + +The rotation stage is mounted on the XY stage, allowing alignment of +the sample holder to the incident beam. + + +.. _fig-wheel_stageINST: +.. figure:: _images/wheel_stage.jpg + :target: _images/wheel_stage.jpg + :width: 70% + :align: center + + The standard *ex situ* sample holder is a plastic disk with slots + for the sample positions. + + +Here are photos of some of the sample holder options. There are +designs which use slots or circles for the sample position. The +circular holes are 13 mm, which is a common size for a pellet press. +13 mm pellets can usually slip snugly into those holes. + +Samples can be packed into the slots or holes. More commonly, samples +are prepared in some manner and affixed to the front of the sample +holder with tape. + +There is also a design which is, essentially, a normal disk cut in +half. That one holds fewer samples, but is easier to load and unload +from a glove box during sample preparation. + + +.. subfigure:: ABC + :layout-sm: ABC + :subcaptions: above + :gap: 8px + :name: fig-wheelINST + :class-grid: outline + + .. image:: _images/Samplewheel.jpg + + .. image:: _images/double_wheel_sm.jpg + + .. image:: _images/halfwheel.jpg + + (Left) A single-ring sample wheel with 24 sample positions. + (Center) Double-ring sample wheels with 48 sample positions. For + both styles, there are options with 13mm x 3 mm slots or 13mm + diameter holes. (Right) A half wheel suitable for loading in a + glove box. + + + +We consider these *ex situ* holders to be consumable items. We will +happily mail several of them to you prior to your experiment so that +you can arrive with samples pre-mounted and ready to be measured. You +may keep the sample holders to use again the next time you visit BMM. + + + +Electrochemistry +---------------- + +At BMM, we have a `BioLogic VSP-300 Potentiostat +`_ which is available for +all manner of electrochemistry experimentation. This is a 6 channel +model, allowing you to prep samples during measurements or to run +multiple electrochemistry experiments in parallel, moving those cells +into and out of the beam. + + +.. _fig-biologiclINST: +.. figure:: _images/biologic.png + :target: _images/biologic.png + :width: 50% + :align: center + + The BioLogic VSP-300 Potentiostat + +We run the vendor-supplied control software on a Windows 10 instance +running in a virtual container. Your electrochemical data will be +saved to the ``assets/vsp300-1/`` folder in your :numref:`proposal +folder (Section %s) `. + +.. note:: + + We do not, at this time, have EPICS-level control of the + potentiostat. This limits the level of automation and integration + with the rest of the beamline. + + DSSI has implemented external triggering and collection of current + and voltage using the `Quantum Detectors PandABox + `__. This may + become available to BMM users sometime in 2025. + +BMM does not provide electrochemical cells. The user is expected to +bring their own cells. Ideally, the electrochemistry has been tested +in the cells intended for use at the beamline and are known to produce +reliable electrochemical results. + + +Temperature control +------------------- + +BMM has two options for experiments at elevated or reduced temperature. + + +Linkam stage +~~~~~~~~~~~~ + +The `Linkam stage `_ has LN2 flow +for cooling a sample down to 77K and a resistive heater to go up to +600C. The volume inside can be pumped or exposed to flowing gas. The +sample stage at the center is modified to have a 3mm diameter hole for +transmission XAFS. + + +.. subfigure:: AB + :layout-sm: AB + :subcaptions: above + :gap: 8px + :name: fig-linkamstageINST + :class-grid: outline + + .. image:: _images/linkam.jpg + + .. image:: _images/dewar.jpg + + (Left) The Linkham stage mounted for transmission on the sample + stage. (Right) The 25 L dewar used for cooling the Linkam stage. + +BMM has two dewars for use with the Linkam. The 2 L dewar has enough +capacity for about 2 hours of measurement. The 25 L dewar runs for +about 14 hours and is the standard choice. The advantage of the +smaller dewar is that it is smaller and might be needed for +complicated setups were space is at a premium. + +Extensive :numref:`automation (Section %s) ` is available +for the Linkam stage. + +Displex Cryostat +~~~~~~~~~~~~~~~~ + +BMM also has a helium compression cryostat capable of reducing +temperature at the sample to around 10K and with a resistive heater +allowing a sample temperature range of 10K to about 400K. + +This Displex model is designed for low-vibration applications. as a +result, it is a bit slow to cool down, requiring about 2 hours to get +to 10K from room temperature. Sample changes are a bit laborious +due to the construction of the vacuum shroud. + + +.. subfigure:: AB + :layout-sm: AB + :subcaptions: above + :gap: 8px + :name: fig-displexINST + :class-grid: outline + + .. image:: _images/cryostat.jpg + + .. image:: _images/lakeshore331.png + + (Left) The Displex cryostat and it's compressor. (Right) The + `LakeShore 331 controller + `__, + used to control temperature for the cryostat shown to the left. + + +Extensive :numref:`automation (Section %s) ` is available +for the Displex using the LakeShore 331 temperature controller.. + +.. _glancing-angle-stage: + +Glancing angle and thin film stage +---------------------------------- + +We use this glancing angle stage for high-throughput studies of thin +film and other flat samples. The apparatus shown below rests on a +rotation stage for moving up to 8 samples into and out of the beam. +The rotation stage sits on a tilt stage, allowing fine control of the +incident angle. Each sample position is a spinner, which is used to +suppress diffraction from the substrate. + +In most cases, sample translation and sample alignment is +:numref:`fully automated (Section %s) ` + +.. _fig-glancinganglestageINST: +.. figure:: _images/glancing_angle_stage.jpg + :target: _images/glancing_angle_stage.jpg + :width: 50% + :align: center + + The glancing angle stage with 8 sample positions. + + +While a standing wave experiment might be feasible at BMM, the much +more typical application is a simple glancing angle measurement in +which the point of the shallow angle is to spread the beam out over +the full length of the sample. This significantly increases the +number of atoms involved in the measurement. + + +.. _tilt-stage: + +Tilt and rotation stages +------------------------ + +Also available are a `Huber pitch and roll stage +`__ +in the form of an Eulerian cradle and a `compact rotary stage +`__. + +.. subfigure:: AB + :layout-sm: AB + :subcaptions: above + :gap: 8px + :name: fig-pitchroll_rotation + :class-grid: outline + + .. image:: _images/tilt_stage.jpg + + .. image:: _images/small_rotation_stage.jpg + + (Left) The pitch and roll stage. (Right) The small rotation stage + + +Here are mechanical drawings, including bolt hole patterns. Consult +these if designing an instrument intended to be mounted for rotation, +pitch, or roll. + ++ Tilt stage mechanical drawing: :download:`PDF <_static/5203_10--32263_131839.pdf>` + ++ Rotation stage mechanical drawing: :download:`JPG <_static/RT-2-11-DIMENSIONS.JPG>` + diff --git a/_sources/intro.rst b/_sources/intro.rst new file mode 100644 index 0000000..d74fbb8 --- /dev/null +++ b/_sources/intro.rst @@ -0,0 +1,440 @@ +.. + This document was developed primarily by a NIST employee. Pursuant + to title 17 United States Code Section 105, works of NIST employees + are not subject to copyright protection in the United States. Thus + this repository may not be licensed under the same terms as Bluesky + itself. + + See the LICENSE file for details. + +.. _intro: + +Introduction to BMM +=================== + +BMM is NIST's :red:`B`\ eamline for :red:`M`\ aterials :red:`M`\ easurement. + +At the unix command line, type ``bsui`` to start the BlueSky user +interface. |bsui| is simply an `Ipython shell `_ +with some customizations specific to BlueSky. On top of that, there +are a number of customizations specific to BMM. + + +In this user manual, there are chapters covering most of the chores +one will need to do at the beamline, including: + +#. moving motors +#. changing the state of the photon delivery system +#. making motor scans +#. making energy scans +#. interacting with the beamline's electronic log book +#. troubleshooting common problems + +TL;DR +----- + +**Open/close the shutter** + ``shb.open()`` and ``shb.close()``, see :numref:`{name}, Section {number} ` + +**Change energy** + Use the ``RE(change_edge())`` command, see :numref:`{name}, Section {number} ` + +**Sample alignment scans** + Use the ``RE(linescan())`` command, see :numref:`{name}, Section {number} ` + +**XAFS scan** + Use the ``RE(xafs())`` command, see :numref:`{name}, Section {number} ` + +**Import an automation spreadsheet** + Prepare a spreadsheet, then ``xlsx()``, see :numref:`{name}, Section {number} ` + +**Details, details, details** + :numref:`Command cheatsheet, Section {number} ` + + +Essential links +--------------- + +:NSLS-II: https://www.bnl.gov/nsls2/ +:BMM's page: https://www.bnl.gov/nsls2/beamlines/beamline.php?r=6-BM +:Operations schedule: https://www.bnl.gov/nsls2/schedule/ +:Beamtime schedule: https://scheduler.nsls2.bnl.gov/ +:NSLS Status: https://www.bnl.gov/nsls2/operating-status.php +:BMM at Github: https://github.com/NSLS-II-BMM + + +The user experience +------------------- + +The Ipython/|bsui| prompt at BMM is modified to provide at-a-glance +information about the state of the beamline. + +.. _fig-prompt: +.. figure:: _images/prompt.png + :target: _images/prompt.png + :width: 40% + :align: center + + The BlueSky user prompt at BMM + +* The green ``BMM`` indicates that the beamline is :numref:`set up and + ready for the user (see Section %s) `. When the beamline + is not ready for users, the ``BMM`` string is red. + +* The string ``D.111`` indicates that the photon delivery system is in + :numref:`mode D (see Table %s) ` and that :numref:`the + Si(111) monochromator (Section %s) ` is in use. + +* The green number in square brackets is an incremented count of how + many commands have been issued since |bsui| was started. + +* If the prompt starts with three red exclamation points |nd| something like + + :red:`!!!` :green:`BMM` D.111 :green:`[1]` + + that means that some motors were not connected when Bluesky started. + Contact beamline staff immediately! + +.. _cadashboard: + +CA Dashboard +~~~~~~~~~~~~ + +At the top of the some screens, you see a crude-but-handy beamline +monitor. It looks something like this: + +.. _fig-cadashboard: +.. figure:: _images/cadashboard.png + :target: _images/cadashboard.png + :width: 100% + :align: center + + The CA dashboard beamline monitor + +This provides a concise (but rather color-blind-unfriendly) overview of +the state of the beamline. + +**Top line** + In short, if the top line has no red text, the beamline is all ready to go. + + + BMM is enabled (:green:`green`) or disabled (:gray:`gray`) + + The BM, FE, & user photon shutters are open (:green:`green`) or closed (:gray:`gray`) + + Indicators that the temperature sensors in the racks are in range + (:green:`green`) or running hot (:red:`red`) + + The state of vacuum sections 1 through 7 |nd| :green:`green` means vacuum + level is OK, :red:`red` means vacuum level is high + + The state of the in-vacuum motors, 4 on the DCM, 2 on the + focusing mirror, 2 on the harmonic rejection mirror |nd| :green:`green` + means temperature is OK, :red:`red` means temperature is high + + The open (:green:`green`) or closed (:red:`red`) state of the 3 front end gate + valves and the 6 beamline gate valves + +.. role:: hlyellow +.. role:: hlcyan +.. role:: hlmagenta +.. role:: hlblue + +**Middle line** + + The energy position of the monochromator + + The current element and edge configuration of the beamline + + The ring current + + The signals on the I\ :sub:`0`, I\ :sub:`t`, and I\ :sub:`r` ion + chambers, measured in nanoamps + + The current operation at the beamline, options are: idle (white), + XAFS scan (:hlmagenta:`pink`), line scan (:hlcyan:`cyan`), area scan (:hlyellow:`yellow`), or time + scan (:hlblue:`blue`) + +.. role:: brown + +**Bottom line** + + Positions of common sample motors + + size of sample slits + + the element currently in the reference position + + the state of the instrument being used |nd| in this example, *the + ex-situ* sample wheel + + the :brown:`brown` block at the end is a heartbeat monitor to visually + verify that the dashboard is functioning correctly + +.. todo:: Document cadashboard in appendix of this manual. + +.. _slack: + +Slack and Data Access +~~~~~~~~~~~~~~~~~~~~~ + +At the beginning of your experiment, you will be invited to the BMM +Slack workspace. There you can follow along with the progress of the +experiment in the #beamtime channel. + +Throughout the course of the experiment, messages and figures will be +automatically posted to that channel. This allows someone to keep +track of progress and to keep an eye on data quality without being +physically at the beamline. + +.. _fig-slack: +.. figure:: _images/slack.png + :target: _images/slack.png + :width: 50% + :align: center + + An example of messages and a picture of measured data posted to the + beamline Slack channel. + +.. attention:: + + As of the 2024-1 cycle, BMM is no longer syncing data automatically + to Google Drive. Data are now available to users via sftp with + proper authentication. See :numref:`Data Access (Section %s) + `. + + + + +.. + Electrochemistry experiments + ~~~~~~~~~~~~~~~~~~~~~~~~~~~~ + + .. note:: January 2022 + + Electrochemistry is not yet properly supported in the experimental + work flow. This section is remains aspirational. + + The data acquisition system at BMM has rudimentary support for + electrochemistry experiments using the BioLogic potentiostat. When + starting a new experiment, do this:: + + BMMuser.begin_experiment(name='Betty Cooper', date='2019-02-29', gup=123456, saf=654321, echem=True) + + The ``echem`` argument, when set to ``True`` tells the system to look + for data from the potentiostat in the appropriate place on the Windows + computer running the EC-Lab software. It will make a folder called + ``electrochemistry`` in the data folder and make a folder on the + Windows machine at ``C:Users\xf06nm\My Documents\EC-Lab\Data``. + There will be a folder with the PI's name and a subfolder with the + start date of the experiment. + + At the end of the experiment, the electrochemistry files are copied + from the Windows machine to the data folder. This puts all of the + data in one place and makes sure that the electrochemistry data are + backed up correctly. + + +Getting help at the command line +~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~ + +To see a summary of common commands, use ``%h``: + +.. code-block:: text + + Open the shutter: shb.open() + Close the shutter: shb.close() + + Change energy: RE(mv(dcm.energy, )) + Move a motor, absolute: RE(mv(, )) + Move a motor, relative: RE(mvr(, )) + Where is a motor? %w + + Where is the DCM? %w dcm + Where is M2? %w m2 + Where is M3? %w m3 + Where are the slits? %w slits3 + Where is the XAFS table? %w xafs_table + + Summarize all motor positions: %m + Summarize utilities: %ut + + How long will a scan seq. be? howlong('scan.ini') + Run a scan sequence: RE(xafs('scan.ini')) + Scan a motor, plot a detector: RE(linescan(, , , , )) + Scan 2 motors, plot a detector: RE(areascan(, , , , , , , , )) + Make a log entry: BMM_log_info("blah blah blah") + + DATA = /home/bravel/BMM_Data/bucket + + All the details: https://nsls-ii-bmm.github.io/BeamlineManual/index.html + +and to see a summary of some useful command line hotkeys, use ``%k``: + +.. code-block:: text + + Abort scan: Ctrl-c twice! + Search backwards: Ctrl-r + Quit search: Ctrl-g + Beginning of line: Ctrl-a + End of line: Ctrl-e + Delete character Ctrl-d + Cut text to eol Ctrl-k + Cut text from bol Ctrl-u + Paste text Ctrl-y + + More details: http://readline.kablamo.org/emacs.html + + +The day will come that we have a GUI for running XAFS experiments at +BMM. For now, we have the command line. Read on |nd| it's not too +difficult! + + +BMM and Building 743 +-------------------- + + +BMM is on the south side of the NSLS-II building: `what3words: +///width.corrugated.support +`__ + +You should park at building 743 and enter through the main entrance +of 743. + +.. _fig-sitemap: +.. figure:: _images/map.png + :target: _images/map.png + :width: 60% + :align: center + + Route from the Main Gate to Building 743 + + +Walk though the lobby to the doors that lead out onto the experimental +floor. BMM is just across the walk way from the doors to the 743 lobby. + +.. subfigure:: ABC + :layout-sm: ABC + :gap: 8px + :subcaptions: above + :name: fig-BMMlocation + :class-grid: outline + + .. image:: _images/743lobby.jpg + + .. image:: _images/corridor.jpg + + .. image:: _images/BMMcontrolstation.jpg + + (Left) Approaching the floor through the lobby of + Building 743. (Center) BMM is just across the corridor from the + door to the 743 lobby. (Right) Walk past the diagonal support beam + and head into the BMM control station + + + + +BMM's staff have offices on the outer hallway of Building 743. + + +.. _fig-lob3: +.. figure:: _images/LOB-3.png + :target: _images/LOB-3.png + :width: 90% + :align: center + + Bruce's, Jean's and Vesna's offices in Building 743 + + + + +A Bit about BMM +--------------- + +BMM is an XAS beamline. As such it is on the simpler end of things at +NSLS-II. We use an NSLS-II `three-pole wiggler (3PW) +`_ as our +photon source. This provides broadband radiation throughout the hard +X-ray range, up to about 30 keV. It is a small device |nd| only about +40 cm long and with a magnetic path length of about 12 cm |nd| which +is inserted in a short section between the two bend magnets in the +dual-bend achromat lattice at NSLS-II. The flux is certainly not the +equal of any of the many-pole insertion devices in the straight, but +it is highly performant for many XAS experiment. + +About 13 meters from the source, we have a paraboloid collimating +mirror. This position is well within the storage ring tunnel and +about 12 meters from the entrance to the BMM first optical enclosure. +We placed a mirror at that location to capture the largest possible +swath of the divergent light coming from the 3PW source. A paraboloid +is the correct shape for focusing light in both the horizontal and +vertical directions. However, a paraboloid must be a fixed figure, +fixed angle device in order to optimally collimate the light. Because +the mirror is in the front end, thus inaccessible during operations, +we found the paraboloid to be an attractive solution. Once aligned in +the beam, it should never need adjustment. + +The collimated light is delivered to a double crystal monochromator +(DCM). The DCM has pairs of Si(111) and Si(311) crystals which are +accessed by :numref:`translating the DCM vacuum vessel laterally +(Section %s) ` . A transition between the two +crystal sets takes about 2 minutes. + +After the DCM, we have a toroidal focusing mirror followed by a flat +harmonic rejection mirror. One or both of these mirrors is in the +beam depending on :numref:`the configuration of the XAS experiment +(Section %s) ` in the end station. Because the beam is +deflected upward after the collimating mirror, at least one of the +mirrors after the DCM must be used in order to deflect the beam +through the lengthy transport pipe and into the end station. + +Because the collimating mirror is at a fixed angle, it only serves as +a harmonic rejection mirror above an energy determined by its +operating angle. That turns out to be about 23.5 keV. For XAS +experiments conducted above 8 keV, then, the harmonic rejection +provided by the collimating mirror is adequate. At lower energies, +the flat harmonic rejection mirror is used to provide clean beam. + +With just the harmonic rejection mirror in place, a beam of size 8 mm +by 1 mm is delivered to the end station. For many XAS experiments, +this rather large beam is desirable. Indeed, many of the visitors to +BMM specifically request the large beam for their experiments. With +the focusing mirror in place, that large swath is reduced to a spot of +about 300 |mu| m by 250 |mu| m. + +Acknowledgements +---------------- + +This documentation project uses `Sphinx +`__ and the lovely +`{book}theme +`__ +from the `The Executable Book Project +`__. Appendices are numbered properly +using the ``appendix.py`` extension from +https://github.com/heig-tin-info/handout. + +BMM's `Bluesky `__ profile was mostly +written by Bruce. But this would not have happened without the help +of several members of NSLS-II's DSSI program. In particular, I want +to thank Dan Allan, Tom Caswell, Josh Lynch, Jakub Wlodek, Max +Rakitin, Dmitri Gavrilov, Stuart Campbell, Abby Giles, Garrett Bishof, +Nate Maytan, Matt Snyder, Oksana Ivashkevych, and Ryan Jaskiel. And I +need to thank every BMM user |nd| being a BMM user means being a beta +tester for the beamline software! + +BMM makes use of `lots of great python tools +`__. +Matt Newville's `Larch `__ is used +to process every XAS scan that gets measured and Matt's `lmfit +`__ is used for many alignment +chores. + +This manual uses a GitHub action to build and deploy `(see details +here) +`__ +this document whenever a ``git push`` happens. We are grateful to the +`UIBCDF `__ +developers for this continuous deployment capability. + + + +A note about copyright +---------------------- + +This document and `the BlueSky data collection profile +`__ it covers was +developed primarily by a NIST employee. Pursuant to title 17 United +States Code Section 105, works of NIST employees are not subject to +copyright protection in the United States. Thus this repository may +not be licensed under the same terms as Bluesky itself or its +documentation. + +See the `LICENSE file +`__ +for details. diff --git a/_sources/ionchambers.rst b/_sources/ionchambers.rst new file mode 100644 index 0000000..8a4529e --- /dev/null +++ b/_sources/ionchambers.rst @@ -0,0 +1,119 @@ + +.. _ionchambers: + +Managing ion chamber signal chains +================================== + +In the 2023-3 and 2024-1 cycles, BMM went through a transition in how +direct beam signals (I\ :sub:`0`, I\ :sub:`t`, I\ :sub:`r`) are +measured. + +The old system +-------------- + +BMM has been using the same ion chambers that were used at X23A2 at +the old NSLS. They are excellent ion chambers |nd| robust, linear, +easy to use. At BMM, we used a `QuadEM +`__ +electrometer to read the current signal from each ion chamber. The +QuadEm is a four-channel device, so I\ :sub:`0`, I\ :sub:`t`, and I\ +:sub:`r` were fed into channels 1, 2, and 3. The fourth channel was +occasionally used for an electron yield detector. + +Voltage was supplied from the same Ortec high voltage supply used at +X23A2. The Ortec device sits outside the hutch. The high voltage +was fed via cabling through a roof labyrinth and patch panels. + +An IOC for reading the QuadEM lives on ``xf06bm-ioc2`` and +communicates with the QuadEM device via a cat6 cable. + +.. subfigure:: AB + :layout-sm: AB + :gap: 8px + :subcaptions: above + :name: old_IC + :class-grid: outline + + .. image:: _images/old_IC.jpg + + .. image:: _images/quadem.jpg + + (Left) The old I\ :sub:`r` detector. (Right) The quadem + electrometer used to measure the three old-style ion chambers. + + +The new system +-------------- + +The new ion chambers are integrated devices. The low half of the +enclosure is a gas-filled volume with capacitor plates made from +printed circuit board. The plates are guarded at both ends and +upright print circuit board pieces are used to guard the fringes of +the fields. These upright pieces are strips of copper bridged by +surface-mounted resistors acting as a sequence of voltage dividers. +This keeps the field lines art the periphery of the capacitor plates +nicely parallel. + +The upper part of the enclosure has the full signal chain and a +voltage supply. This includes an amplifier, an analog-to-digital +converter, and a microprocessor running an embedded Debian system and +a two-channel version of the QuadEM IOC. The voltage supply is fully +adjustable up to 2000 V, but are run at 200 V. + +The electronics package and the voltage supply are powered a 6 V DC +power supply. A cat6 cable for each ion chamber allows the IOCs to +communicate on the INST network. + +The ion chambers also have an optical fiber input port so that data +can be time stamped with the global timing signal from the +accelerator. This will be used in the near future to implement +continuous scanning on the monochromator. + + + +.. _fig-new_IC: +.. figure:: _images/new_IC.jpg + :target: _images/new_IC.jpg + :width: 70% + :align: center + + The new I\ :sub:`0` detector with integrated electronics and + voltage supply. + +Configuration in bsui profile +----------------------------- + +I am trying to make it easy to configure ``bsui`` to switch easily +between ion chambers and electrometers. This is a work in progress. +Here I document the current, slightly awkward, state of affairs. + +In `BMM/user_ns/dwelltime +`__ +three boolean parameters are set: ``with_ic0``, ``with_ic1``, and +``with_ic2``. + +These are used to synchronize setting integration times across the +various signal chains via the `LockedDwellTimes +`__ +object. + +Each new ion chamber in use needs to have its flag set to ``True``. + +The detectors themselves need to be configured correctly in `this file +`__ +using the flag values. + +The QuadEM and each individual ion chamber will be configured if +available on the network. If the network connection cannot be +established, a ``noisy_det`` from ``ophyd.sim`` is created with the +same name. + +Notes: + +#. If the ion chamber device can be made, it will be made even if its + flag is set to ``False``. This allows interaction with an ion + chamber even if it is not expected to be used in a scan. +#. Once all 3 new ion chambers are in place and in use, the QuadEM + device will still be made. This will allow use of electron yield + and other detectors as well as measurement of any other current + signals. diff --git a/_sources/kafka.rst b/_sources/kafka.rst new file mode 100644 index 0000000..026a1e0 --- /dev/null +++ b/_sources/kafka.rst @@ -0,0 +1,596 @@ +.. + This document was developed primarily by a NIST employee. Pursuant + to title 17 United States Code Section 105, works of NIST employees + are not subject to copyright protection in the United States. Thus + this repository may not be licensed under the same terms as Bluesky + itself. + + See the LICENSE file for details. + +.. _plotting: + +Kafka workers at BMM +==================== + +In 2023, BMM began using a Kafka client to manage real-time data +visualization. Prior to that, BMM was using a modified version of +Bluesky's `LivePlot +`__. +With the advent of BMM's use of `BlueSky queueserver +`__ alongside |bsui|, we +had to rethink how plots were made. + +In short, a Kafka client was developed to manage real-time +visualization. This client is subscribed to two topics, a private +BMM-only topic and the same topic used by Bluesky to distribute +event documents to databroker. + +Messages on the private topic tell the worker what kind of plot is +requested. The worker then parses event documents and adds data +points to the real-time plots. + +In May 2024, data security upgrades were implemented at BMM. The +upshot of this new data security regime is that the |bsui| or qs process +is not able to write data to the proposal directory on central +storage. Moving all data file output from the |bsui| profile to a new +Kafka worker gave us a path forward. The workers can be run with +adequate privilege to write to central storage. + +At this time (June 2024), there are three Kafka workers running to +manage file IO and data visualization: + +:File manager: running via `systemd `__ with + adequate privilege to write to project directories. + +:Headless plot manager: running via `systemd `__ + with adequate privilege to write to project + directories. This is in charge of exporting + data visualization to png files and other + formats. + +:Visualization plot manager: running as the beamline operator. This + does not have permission to write to the + proposal directory, but it does have + permission to make real-time plots on + screen. + +For now (June 2024), all three are running on xf06bm-ws3. However, +these can run on any machine at the beamline. The visualization plot +manager, though, should be running on a machine with a screen in plain +sight of the experimenter. xf06bm-ws3 is a fine choice. + +.. admonition:: Future Tech! + + Some day the workers will be stable and battle tested. + The systemd workers could then run on a machine out of + sight, like an IOC server. + +.. _start_consumer: + +Starting the Kafka consumer +--------------------------- + +In the ``consumer/bin/`` folder of the `beamline profile +`__ +there are several files used to manage the workers. + +:run-filemanager: a shell script for launching the file manager worker + in the correct conda environemnt + +:run-plotmanager: a shell script for launching the plot manager worker + in the correct conda environemnt + +:set-konsole-tab-title.sh: interact nicely with the Konsole terminal + +:tab-layout: a Konsole configuration file for opening a window with + three tabs for the three workers + +:open-tabs.sh: s shell script for using the tab-layout file + +:xafs_service: a shell script for restarting the workers as system + services, then run journalctl in tail mode to display + worker screen messges + +It is convenient to make symlinks to each of the shell scripts in a +folder in the execution path. ``~/bin/`` makes sense. + +To start, run the ``open-tabs.sh`` script to make a Konsole window +with three tabs. Each tab is named something evocative of what will +be running in it. + +#. In the tab labeled "Visual plot manager", do ``run-filemanager`` as + the beamline operator. + +#. In the table labeled "File manager", do ``su - `` to + become youself rather than the beamline operator, then do + ``xafs_services file``. You will be prompted for your password and + then sent a DUO push. + +#. In the table labeled "Headless plot manager", do ``su - + `` to become youself rather than the beamline operator, + then do ``xafs_services file``. You will be prompted for your + password and then sent a DUO push. + +Note that you must be beamline staff to run the system services. A +beamline user will not be able to execute the second and third steps +above. + +Once all three workers are running, you are ready to start |bsui| or +|qs| and begin collecting data. + + +.. _fig-consumer: +.. figure:: _images/consumer_startup.png + :target: _images/consumer_startup.png + :width: 70% + :align: center + + A terminal window in which the Kafka consumers have been started. + +.. todo:: Update this figure + + + +A few words about Kafka +----------------------- + +Kafka is a message bus. A message bus is a service that sits between +applications that generate messages and applications that want to +consume those messages and act upon them. + +This works via subscription topics. An application can subscribe to +a topic as a producer. That means it can say "I just did something +and here is information about what I did." It will post that message +to Kafka, then move on. + +A separate application can subscribe to the topic as a consumer. It +will be waiting on messages that get posted to the topic. The +consumer will only see messages on a topic to which it is subscribed. +It can then interpret the message to decide if it should act upon it. + +Many applications can subscribe as producers and many applications can +subscribe as consumers. + +In the case of BMM, there are two possible producers of messages |nd| +|bsui| and |qs|. At BMM, there are three consumers |nd| the +three listed above. The consumers are in separate processes, thus can +act upon messages in parallel. + +Because the Kakfa message bus is involved, actions can be taken by +consumers on messages asynchronously with the producer of the +messages. This means, for instance, that |bsui| can carry on with data +collection and let the file worker take care of the details of writing +files. + +This is, admittedly, a lot more complicated than just having |bsui| +handle all those chores by itself. But this complication pays off in +two very significant ways: + +#. The plot worker makes plots regardless of whether |bsui| or |qs| is + running the experiment. Since |qs| is probably not running on the + beamline workstation, that is very handy. +#. The workers run as systemd processes are able to write files to the + secure proposal directory. Neither |bsui| nor |qs| are run + with adequate privilege for that. + + +Plot types +---------- + +The plot worker makes a tightly curated set of plots. The beamline +user has little freedom to adjust the plots. This is by design |nd| +the data visualization is a tool used during data collection. The +entire data collection workflow is intended for streamlined, automated +measurement. Highly specialized data visualization can be made with +the recorded data. + +Communicating over the ``bmm-test`` topic, two sorts of plotting +chores are managed |nd| real-time visualization and visualization +after the end of the mesurement. + +In each case, the documents sent to Kafka are simple dictionaries +which the consumer parses to perform a plotting chore using +`matplotlib `__. + +These dictionaries are not structured like a BlueSky document. There +is no schema. The dictionary simply contains keywords which the +consumer is programmed to recognize. + +.. admonition:: Future Tech! + + Consider a browser-y solution like `Bokeh + `__. + + + +.. _liveline: + +Live linescan plots +~~~~~~~~~~~~~~~~~~~ + +At BMM, :numref:`a linescan (Sec %s) ` is a scan where a +motor is moved and a signal is plotted. A linescan begins by issuing +a message telling the consumer to start a new plot and to begin +looking for BlueSky event documents: + +.. code-block:: python + + {'linescan' : 'start', + 'motor' : 'xafs_x', + 'detector' : 'I0',} + +Those event documents will be parsed to obtain the result of the most +recently measured data point. The new data point is added to the plot +and the plot is redrawn. + +When the linescan finishes, a *stop* message is issued: + +.. code-block:: python + + {'linescan': 'end',} + +This replicates very closely how the BlueSky `LivePlot +`__ +displays data of this sort. + +.. _livetime: + +Live timescan plots +~~~~~~~~~~~~~~~~~~~ + +With the BMM plotter, a timescan and a linescan are made with the +same code. The only difference is that no motor is given for a +timescan and the X-axis is plotted as the time stamp of the current +point minus the time stamp of the first point. Thus the X-axis is in +units of seconds. The signal plotted on the Y-axis is determined the +same as for a linescan and all the internal mechanics of the time plot +are the same as for a motor plot. + +A timescan begins by issuing a message telling the consumer to start a +new plot and to begin looking for BlueSky event documents: + +.. code-block:: python + + kafka_message({'timescan': 'start', + 'detector' : 'if',}) + +When the linescan finishes, a *stop* message is issued: + +.. code-block:: python + + kafka_message({'timescan': 'stop', + 'fname' : outfile, + 'uid' : uid, }) + +The ``fname`` and ``uid`` arguments are optional and are used for +single energy absorption detection (SEAD) scans. The ``uid`` is the +UID of the timescan and the ``fname`` is the filname (without path) of +the output data file. + +.. todo:: Document SEAD scans. + + +.. _livearea: + +Live areascan plots +~~~~~~~~~~~~~~~~~~~ + +.. todo:: + + Explain this in words. Explain how the contour plot is made at the + end of the scan. + +.. code-block:: python + + kafka_message({'areascan' : 'start', + 'slow_motor' : xafs_y.name, + 'slow_start' : -9, + 'slow_stop' : 9, + 'slow_steps' : 91, + 'slow_initial' : xafs_y.position, + 'fast_motor' : xafs_x.name, + 'fast_start' : -9, + 'fast_stop' : 9, + 'fast_steps' : 91, + 'fast_initial' : xafs_x.position, + 'detector' : 'if', + 'element' : BMMuser.element, + 'energy' : dcm.energy.position}) + + + +.. code-block:: python + + kafka_message({'areascan': 'stop', + 'uid' : uid, + 'filename': stub}) + + +.. _livealignment: + +Alignment plots +~~~~~~~~~~~~~~~ + +Various alignment chores at the beamline |nd| for example, aligning a +slot on a :numref:`sample wheel (Sec %s) ` or aligning +the :numref:`glancing angle stage (Sec %s) ` +|nd| involve a series of :numref:`linescans (Sec %s) `, each +of which is plotted in real time |nd| as shown :numref:`above (Sec %s) +` |nd| followed by a plot summarizing the result of the +alignment. + +Using the sample wheel alignment as an example, the sequence is +initiated by this document: + +.. code-block:: python + + {'align_wheel' : 'start'} + +As each linescan in the alignment procedure is completed, some +automated analysis is performed to determine the optimal position of +the motor axis being scanned. The results of this analysis are issued +in a document like this. + +.. code-block:: python + + {'align_wheel' : 'find_slot', + 'motor' : 'xafs_x', + 'detector' : 'it', + 'xaxis' : list_of_axis_positions, + 'data' : list_of_signal_values, + 'best_fit' : list_of_fitted_values, + 'center' : midpoint_value, + 'amplitude' : amplitude_value, + 'uid' : uid} + +From this a plot showing the measured data and the results of the +analysis is made. + +Once all parts of the alignment procedure are finished, this document +is issued: + +.. code-block:: python + + {'align_wheel' : 'end'} + +This tells the consumer to create a plot summarizing the results of +the alignment. + +The alignment of the glancing angle stage works in much the same +manner. + + +.. _fig-find_slot: +.. figure:: _images/find_slot.png + :target: _images/find_slot.png + :width: 50% + :align: center + + An example of the final plot for an alignment of the *ex situ* + sample wheel. The green X marks show the aligned positions in + ``xafs_x`` and ``xafs_y``. + + +.. _livexafs: + +Live XAFS plots +~~~~~~~~~~~~~~~ + +The problem of making live XAFS plots is quite similar to live +linescan plots, but with some additional considerations: + +#. It is common to make multiple repetitions of XAFS scans, thus + successive scans should be overplotted. +#. There are various interesting views of the XAFS data, including + both transmission and fluorescence of the data, transmission of the + energy calibration standard, and a view of the raw I0 spectrum (to + keep an eye on monochromator glitches and other issues). + +.. admonition:: Future Tech! + + Panel for live |chi|\ (k) plots, begin plotting this panel, say, 60 + eV above the edge. + + +.. admonition:: Future Tech! + + Plot electron yield data in a consistent, maintainable manner. + +Like with the linescan, the plot begins with a message issued to tell +the consumer to begin preparing for an XAFS plot and providing enough +information to make that plot. This ``start`` message is issued at +the beginning of the entire scan sequence. + +.. code-block:: python + + {'xafsscan' : 'start', + 'element' : 'Fe', + 'edge' : 'K', + 'mode' : 'fluorescence', + 'filename' : 'example' + 'repetitions': 3, + 'sample' : 'Fe sample', + 'reference_material': 'Fe foil', } + +At the beginning of each individual repetition, a ``next`` message is +sent, telling the consumer to prepare to add a new set of traces to +the plot for the repetition about to begin. + + +.. code-block:: python + + {'xafsscan': 'next', + 'count': 2, } + +Finally, a message is sent telling the consumer that the sequence of +scans has finished, putting the consumer back into a state where it is +ready to receive the next sequence of messages for the next plot. + +.. code-block:: python + + {'xafsscan': 'end',} + +The plot that is made for an XAFS scan depends on whether fluorescence +measurement is available. If so, a 2x2 grid is shown with the +transmission and fluorescence |mu| (E) on the top, a plot of I0 on the +bottom left, and plot of the transmission |mu| (E) of the reference +material on the bottom right. + +For a scan not using the fluorescence detector, the plot is a 3x1 grid +of transmission |mu| (E), I\ :sub:`0`, and the reference spectrum. + + +.. _fig-xafs_live_view: +.. figure:: _images/XAFS_live_view.png + :target: _images/XAFS_live_view.png + :width: 80% + :align: center + + An example of the XAFS live plot made for a fluorescence XAFS scan. + This is a somewhat old example. I\ :sub:`0` is now normalized by + the dwell time, thus is plotting in units of nanoamperes rather than + nanoampere*seconds, as shown (but labeled incorrectly). + +The live plot at the end of the scan sequence is posted to Slack and +included in the :numref:`dossier (Section %s) `. + + +.. _xafssequence: + +Scan sequence data reduction +~~~~~~~~~~~~~~~~~~~~~~~~~~~~ + +At the end of a scan sequence, we show the user a 3-panel plot showing +|mu| (E), |chi| (k), and |chi| (R). (This is the same 3-panel plot +that is written to the :numref:`dossier (Section %s) `. This +plot is of the merge of the scans measured in the scan sequence. +Behind the scenes, Larch is used to make the merge, remove the +background function, and perform the Fourier transform. Additionally, +every time an individual repetition in the scan sequence is finished, +this 3-panel plot is made from the merge of the scans measured thus far. + +At the beginning of a scan sequence, a Kafka document with a payload +like this is issued: + +.. code-block:: python + + {'xafs_sequence' : 'start', + 'element' : 'Fe', + 'edge' : 'K', + 'folder' : BMMuser.folder, + 'repetitions' : 3, + 'mode' : 'fluorescence'} + +The presence of the ``xafs_sequence`` key tells the Kafka consumer to +interpret this document as relevant to the creation of the 3-panel +plot. The value of ``start`` tells the consumer to prepare for making +this plot from data under the conditions specified by the remainder of +the keywords. + +As each scan finishes, the following document is issued. This tells +the consumer that a repetition finished and supplies the UID of the +just-completed scan. `Tiled `__ is +used to grab the data from the just-completed scan. This triggers a +recalculation of the merge and the recreation of the 3-panel plot. + +.. code-block:: python + + {'xafs_sequence' :'add', + 'uid' : uid} + +Finally, at the end of the scan sequence, this document is issued: + +.. code-block:: python + + {'xafs_sequence' : 'stop', + 'filename' : '/path/to/dossier/image'} + +This tells the consumer to make the final version of the 3-panel plot +using all the data and to save a png image of the plot for use in the +dossier. + +.. _fig-triplot: +.. figure:: _images/triplot.png + :target: _images/triplot.png + :width: 50% + :align: center + + An example of a 3-panel plot created by the Kafka consumer. + + + +This motif of issuing a ``start`` message to begin crafting a plot, +messages to ``add`` to the plot, and a message to ``stop`` the plot is +the common thread to how BMM uses Kafka to make plots, both static and +real-time plots. + + +.. todo:: there are more plot actions that need to be documented. + +Headless and visualization workers +---------------------------------- + +There are two plotting workers that share code and behave almost +identically. This seems redundant, so merits a few words of +explanation. + +The visualization worker is run as the beamline operator |nd| +``xf06bm``. The beamline operator owns the screen and is able to make +plots of data to the screen. However, the beamline operator does not +have permission to write data and png images to the proposal +directory. The visualization worker can be run on any machine on the +local network at BMM |nd| even on multiple machines! + +The headless worker does not make visible plot visualization. +Instead, it writes plots to a virtual device which can then be saved +as png images to the proposal directory. It is also able to write +data files to the proposal directory. For example, the XRD data file +measured before each scan sequence is written by the visualization +worker. + +In short, the visualization worker is for the benefit of the humans at +the beamline while the headless worker is responsible for writing +files for the data record of the experiment. + +Credit goes to Dan Allan for suggesting running two instances of the +plot worker using the QtAgg and Agg `matplotlib backends +`__. + + +File management +--------------- + +.. todo:: + + Explain all the file management actions with example dicts. + + +Cleaning up the screen +---------------------- + +Most of the plotting options from the Kafka consumer are good about +closing the last plot before starting a new one. However, linescans, +in general, do not clean up prior plots. + +You can close some or all of the plots made by the Kafka consumer by +issuing a suitable message, either at the command line or in a plan. + +This will close all plots on screen made by the consumer: + +.. code-block:: python + + kafka_message({'close': 'all'}) + +This will close all plots associated with linescans, but not close +plots associated with XAFS scans: + +.. code-block:: python + + kafka_message({'close': 'line'}) + +And this will close the most recent plot: + +.. code-block:: python + + kafka_message({'close': 'last'}) + diff --git a/_sources/linescans.rst b/_sources/linescans.rst new file mode 100644 index 0000000..d36211d --- /dev/null +++ b/_sources/linescans.rst @@ -0,0 +1,310 @@ +.. + This document was developed primarily by a NIST employee. Pursuant + to title 17 United States Code Section 105, works of NIST employees + are not subject to copyright protection in the United States. Thus + this repository may not be licensed under the same terms as Bluesky + itself. + + See the LICENSE file for details. + +.. role:: key + :class: key + + +.. _linescan: + +Sample position scans +===================== + +General line scans +------------------ + +In the nomenclature of BMM, a line scan is a simple scan of a single +motor axis with an on-screen plot. At BMM this sort of scan is +typically done using the ``linescan()`` plan. This is a wrapper +around BlueSky's `rel_scan() +`_. + +In BMM's ``linescan()`` plan, the scan is always a relative scan +around the current position of the motor being scanned. It works like +this:: + + RE(linescan(xafs_x, 'it', -4, 4, 81)) + +The arguments are: + +#. The motor axis to be scanned. This can be either the motor's + name in Bluesky. + +#. A string indicating the detector for the plotted signal. The + choices are: + + * ``it``: display the ratio of ``It/I0`` + * ``if``: display the sum of four silicon drift channels normalized + by ``I0`` + * ``i0``: display the signal on ``I0`` + * ``ir``: display the ratio of ``Ir/It`` + +#. The starting position of the motor scan, relative to the current + position. + +#. The ending position of the motor scan, relative to the current + position. + +#. The number of steps in the scan. + +Note that the first two arguments can be in either order. The plan +will sort it out for you. These are equivalent:: + + RE(linescan('it', xafs_x, -4, 4, 81)) + RE(linescan(xafs_x, 'it', -4, 4, 81)) + + +At the end of the scan, you are prompted with the following question:: + + Pluck motor position from the plot? yes: y the Enter (or just Enter), no: n then Enter + +If you answer :key:`y` then :key:`Enter`, or simply hit +:key:`Enter`, you will be prompted to single click the left mouse +button :mark:`leftclick,.` on the plot. + +You will then be prompted a second time with the value you selected. + +The motor that was scanned will then move to the motor position you +clicked on. + +You can skip the "click for motor position" step by typing +:key:`n` and hitting :key:`Enter`. + + +Plucking a point from a line scan +~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~ + +If you want to repeat the action of clicking on a point and moving the +motor to the click-upon point, do:: + + RE(pluck()) + +This will enable the left mouse :mark:`leftclick,.` click and +subsequent motion on the most recent plot. + +.. warning:: + The ``pluck()`` command works on any plot made through the Kafka + plotter. Be sure you are clicking on the plot you intend to pluck + from! + + +Revisit a line scan +~~~~~~~~~~~~~~~~~~~ + +.. admonition:: Needs to be verified + + Does this still work post-data-security? + +Retrieve data from the database using the database key or scan ID:: + + ls2dat('/path/to/output.file', '') + +This writes from the database to an output file. The output file is a +simple column data file without header or metadata. + + +The first argument is the name of the output data file. The second +argument is either the scan's unique ID |nd| something like +``42447313-46a5-42ef-bf8a-46fedc2c2bd1`` |nd| or the scan's transient +id number. + + + +.. _special-linescans: + +Specific line scans +------------------- + +Some scan types are performed often enough and always with the same +arguments that they have special names. + +**Rocking curve scan** + This command:: + + RE(rocking_curve()) + + does a scan of the pitch of the second mono crystal and plots the + signal on I0. At the end of the scan, it moves to the position of + the center of mass of the rocking curve. This scan is useful after + a large change of energy. It also opens the slits to 3 mm in + vertical size before starting the scan, then returns the slits to + their original height after the scan. + + You can put this scan in a macro using:: + + yield from rocking_curve() + +**Slit height scan** + This command:: + + RE(slit_height()) + + Runs a scan of the DM3 BCT motor around its current position. At + the end of the scan, you are prompted to left click + :mark:`leftclick,.` on the plot to choose a position to move the + slit height to. This scan is useful for verifying that the slits + are in the correct orientation for the delivery of beam from the + mirrors. + + Optionally, the scan will move to the center of mass of the + measurement, skipping the prompt and plot interaction:: + + RE(slit_height(move=True)) + + You can put this scan in a macro using:: + + yield from slit_height() + +**Align ex situ sample holder** + If the *ex situ* sample wheel is in approximately the right + position such that X-rays are passing through a slot on the outer + ring, you can center the slot around the beam with:: + + RE(find_slot()) + + This will run linescans in the X and Y directions, fitting a + lineshape to each scan in order to find the center position. The + same can be done by hand by doing:: + + RE(linescan(xafs_y, 'It', -3, 3, 31)) + RE(linescan(xafs_x, 'It', -10, 10, 31)) + + each time selecting the center when prompted. + + You can put this scan in a macro using:: + + yield from find_slot() + +Area scans +---------- + + +.. todo:: A raster scan type |nd| i.e. ``RE(raster())`` |nd| that does + an areascan as described here, makes a nice figure using + `matplotlib's contourf + `_, + exports data in formats used by popular plotting programs, + and writes a dossier was tested in 2022-2. Current status: + dossier is written, scan works, documentation needs to be written. + + +An area scan is a simple scan of a two motor axes with an on-screen +heat map. At BMM this sort of scan is typically done using the +``areascan()`` plan. This is a wrapper around BlueSky's +`rel_grid_scan() +`_. +Because the sample stages at BMM do not have encoders, the area scan +is made by `retreading the direction +`_ +of the fast motor rather than snaking back and forth. + +In BMM's ``areascan()`` plan, the scan is always a relative scan +around the current positions of both motors being scanned. It works +like this:: + + RE(areascan('it', '', -4, 4, 81, '', -2, 2, 41)) + +The arguments are: + +#. The slow motor axis. This can be either the motor's + BlueSky name or the nickname in :numref:`Table %s `. So, + these are equivalent:: + + RE(areascan('it', 'x', -4, 4, 81, 'y', -2, 2, 41)) + RE(areascan('it', xafs_x, -4, 4, 81, xafs_y, -2, 2, 41)) + RE(areascan('it', xafs_x, -4, 4, 81, 'y', -2, 2, 41)) + + For a motor that does not have a nickname, you must use the BlueSky + name, as in this very silly example:: + + RE(areascan('it', slits3_outboard, -1, 1, 21, dcm_pitch, -2, 2, 41)) + +#. The starting position of the slow motor, relative to the current + position. + +#. The ending position of the slow motor, relative to the current + position. + +#. The number of steps to take on the slow motor. + +#. The fast motor axis. This can be either the motor's + BlueSky name or the nickname in :numref:`Table %s `. + +#. The starting position of the fast motor, relative to the current + position. + +#. The ending position of the fast motor, relative to the current + position. + +#. The number of steps to take on the fast motor. + +#. The detector for the plotted signal. The choices are ``it``, + ``if``, and ``i0``. For the ``it`` choice, the plot will display + the ratio of It/I0. Similarly for the ``if`` choice, the plot will + display the sum of four silicon drift channels normalized by I0. + For the ``i0`` choice, the signal on the I0 chamber will be plotted. + +At the end of the scan, you are prompted with the following question:: + + Pluck motor position from the plot? [Yn] + +If you answer :key:`Y`, or simply hit :key:`Return`, you will be +prompted to single click the left mouse button :mark:`leftclick,.` on +the plot. Both motors will then move to the position you clicked on. + +You can skip the "click for motor position" step by typing +:key:`n` and hitting :key:`Return`. + + +Plucking a point from an area scan +~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~ + +If you want to repeat the action of clicking on a point and moving the +motor to the click-upon point, do:: + + RE(pluck()) + +This will enable the left mouse click :mark:`leftclick,.` and +subsequent motion on the most recent plot. The ``pluck()`` command +*only* works on the most recent plot. You may not pluck from an older +plot that is still displayed on the screen. + +Of course, an older plot remains active in the sense that you can pass +the cursor over the plot and read the mouse coordinates in the bottom, +left corner of the plot window. You can find a point in this way, +then do a movement command:: + + RE(mv(xafs_x, 28.31, xafs_y, 113.97)) + + + +Revisit an area scan +~~~~~~~~~~~~~~~~~~~~ + +.. admonition:: Needs to be verified + + Does this still work post-data-security? + +Retrieve data from the database using the database key or scan ID:: + + as2dat('/path/to/output.file', '') + +This writes from the database to an output file. The output file is a +simple column data file. The format of this data file is columns with +datablocks (i.e. rows of constant value of the slow motor) separated by +blank lines. This is a format that `works with Gnuplot +`_ and other +plotting programs. + +The first argument is the name of the output data file. The second +argument is either the scan's unique ID |nd| something like +``42447313-46a5-42ef-bf8a-46fedc2c2bd1`` |nd| or the scan's transient +id number. + + diff --git a/_sources/log.rst b/_sources/log.rst new file mode 100644 index 0000000..85b4060 --- /dev/null +++ b/_sources/log.rst @@ -0,0 +1,219 @@ +.. + This document was developed primarily by a NIST employee. Pursuant + to title 17 United States Code Section 105, works of NIST employees + are not subject to copyright protection in the United States. Thus + this repository may not be licensed under the same terms as Bluesky + itself. + + See the LICENSE file for details. + +.. _log: + +Experimental Log +================ + +At BMM, we are trying to run the system in a way that automatically +generates a useful electronic log book. This is meant to complement +|nd| not replace |nd| a paper log book or an online system like `Olog +`_. + +The point of the BMM experimental log is to capture with time stamps +all the significant actions at the beamline. In part, this provides a +record of the actions in an experimental campaign. It also is an +attempt to provide enough information to recover from mistakes and +confusions about sample positions and other experimental issues. + +.. _logfile: + +Log file +-------- + +At the beginning of a user experiment, run something like this command:: + + BMMuser.begin_experiment('Betty Cooper', date='2019-02-28', gup=123456, saf=654321) + +Among other things, this instruments the logger to maintain a log file +specifically for the current experiment. The logger also maintains a +master log file which is, effectively, a concatenation of all the +individual experimental logs. + +The log is an attempt to capture a record of all significant actions +taken during an experimental campaign. It errs of the conservative +side in that it likely captures way too much information. + +Experimental events are captures with a time stamp and a brief +explanation of what happened. For example, when a motor is moved, the +motor name and target position are written to the log with a time +stamp. + +Similarly, a line scan or an XAFS scan is captured with enough +information to understand what the scan accomplished. For a line +scan, the motor name, scan range, and starting position are captured. +For an XAFS scan, the contents of the :numref:`INI file (See Section +%s) ` are written to the log along with the motor positions as +reported by the ``ms()`` command :numref:`(see Section %s) `. +The names of the output data files are recorded with timestamps +indicating when they were written. + +Several other activities specific to BMM also recorded to the log file. +The snapshot tool described above is an example. + +Writing to the log file is accomplished in two ways. This function:: + + BMM_log_info('text of message') + +is used to insert most messages into the log. ``BMM_log_info()`` can +be called at any time from the command line to insert any message into +the log. + +BMM also uses `Bluesky's msg_hook +`_. +This is how ``mv()`` and ``mvr()`` commands are captured in the log. +This bespoke message hook parses the document returned by BlueSky for +specific kinds of events and captures a log message when appropriate. + +.. _snap: + +Snapshots +--------- + +.. todo:: Document the USB cameras + +The XAFS scan plan described in :numref:`Section %s ` includes a +step where snapshots are taken from the XAS webcam and the small +analog camera. These photos are written to JPG files with names +related to the XAS data file written as part of the call to the +``xafs()`` plan. This provides automated photographic documentation +of the state of the measurement as it is being made. + +Snapshots can be taken at any time using this command:: + + snap('', filename='/path/to/image.jpg') + +The choices for ```` are + +``XAS`` + Take a snapshot from the XAS webcam + +``XRD`` + Take a snapshot from the XRD webcam + +``analog`` + Take a snapshot of the image currently displayed on the monitor to + the left of the control station. + +The image will then be written to the file specified by the +``filename=`` argument. + +The ``xafs()`` plan does something equivalent to the ``snap()`` +function for each available camera. These are recorded to the scan's +:numref:`dossier (Section %s) `. + + + + +.. subfigure:: AB|CD + :layout-sm: AB|CD + :gap: 8px + :subcaptions: above + :name: fig-snapshots + :class-grid: outline + + .. image:: _images/analog.jpg + + .. image:: _images/XASwebcam.jpg + + .. image:: _images/usbcam1.jpg + + .. image:: _images/usbcam2.jpg + + Snapshots from an experiment using the Linkam stage. Note that + each snapshot is annotated along the bottom, visually displaying + the time and identifying the sample being measured. + (Top left) Snapshot taken with the analog camera. (Top right) + Snapshot taken with the XAS web camera. (Bottom left) Snapshot + taken with USB camera #1. (Bottom left) Snapshot taken with USB + camera #2. + + +.. _video: + +Recording videos +---------------- + +.. caution:: Does this actually work? + +The USB cameras can be used to record short videos of whatever they +are pointing at. The resulting video will be saved to a `.avi +`__ file in +the ``video`` folder under the experiment folder. + +The simplest way to record a video is by this command: + +.. code-block:: python + + usbvideo1.record_video(name, time) + +Here the arguments are: + +``name`` + The stub of the file to be written to disk. The ``.avi`` extension + will be added +``time`` + The length in seconds of the recording + +This is, perhaps, a bit clunky since you have to specify the time of +the recording. You can do the recording in a more hands on manner by +explicitly starting the recording. + +.. code-block:: python + + usbvideo1.start() + +Do whatever you want to capture, then explicitly stop the recording. +Finally, you save the video to a name which means the same thing as is +explained above. + +.. code-block:: python + + usbvideo1.stop() + usbvideo1.save_video(name) + + +.. _dossier: + +Scan dossier +------------ + +The BMM data collection system now captures a dossier for each scan +sequence that is run. The definition of a scan sequence is a call to +:numref:`the xafs plan (Section %s) `, which may +involve multiple repetitions of the scan. + +The dossier is a static html file which captures most of the +information discussed on this page. It includes, links to each +individual data file, the transient ID and UID for each scan, links to +the snapshots, tables of information from :numref:`the INI file +(Section %s) `, a verbatim copy of the INI file, and a table of +motor positions at the time of the beginning of the scan sequence. + +These dossiers aggregate other assets described on this page and +complement the user's paper logbook by providing comprehensive +summaries of all the information relevant to the scan sequence +provided by the user and gleaned from beamline instrumentation. + +.. _fig-dossier: +.. figure:: _images/dossier.png + :target: _images/dossier.png + :width: 70% + :align: center + + An example of the scan sequence dossier, displayed in a web browser. + + +.. admonition:: New in April 2023 + + Along with the dossier, the beamline now records a flat HTML file + which records all the messages sent to :numref:`Slack (Section %s) + `. This can be accessed by the "Timeline" link at the top + of every dossier page. diff --git a/_sources/manage.rst b/_sources/manage.rst new file mode 100644 index 0000000..a62c53b --- /dev/null +++ b/_sources/manage.rst @@ -0,0 +1,849 @@ +.. + This document was developed primarily by a NIST employee. Pursuant + to title 17 United States Code Section 105, works of NIST employees + are not subject to copyright protection in the United States. Thus + this repository may not be licensed under the same terms as Bluesky + itself. + + See the LICENSE file for details. + +.. _manage: + +Managing the beamline +===================== + + +In this section, some recipes are provided for managing the beamline +and meeting the needs and expectations of different experiments. + + + +.. _start_end: + +Starting and ending an experiment +--------------------------------- + +When a new experiment begins, run the command:: + + BMMuser.begin_experiment(name='Betty Cooper', date='2019-02-29', gup=123456, saf=654321) + +This will create that data folder and populate it with an +:numref:`experimental log (Section %s) `, write a template for a +:numref:`macro file (Section %s) `, configure the logger to +write a :numref:`user log file (Section %s) ` for this +experiment, set the GUP and SAF numbers as metadata for output files, +set up :numref:`snapshot (Section %s) ` and :numref:`dossier +(Section %s) ` folders, and perform other experiment start-up +chores. + +The ``name`` should be the PI's full name, preferably transliterated +into normal ASCII. The ``date`` should be the starting day of the +experiment in the ``YYYY-MM-DD`` format. The ``GUP`` and ``SAF`` +numbers can be found on the posted safety approval form. + +Once the experiment is finished, run this command:: + + BMMuser.end_experiment() + +This will reset the logger and the ``BMMuser.folder`` variable and +unset the GUP and SAF numbers. + + + +Change energy +------------- + +Changing energy is simple. Usually, it is as simple as doing + +.. code-block:: python + + RE(change_edge('Fe')) + +replacing the two-letter element symbol with the element you actually +want to measure. This command will move the monochromator, put the +photon delivery system in the correct mode, move the M2 bender to +approximately the correct setting, run a rocking curve scan, +optimize the slit height, move the reference foil holder to the +correct position (if configured), and select the correct ROI channel +(if configured). + + +If you want to reproduce this by hand, here is the command sequence: + + +#. First move the DCM to the new energy position. It is usually a + good idea to move a bit above the target edge energy. Here's an + example for moving 50 eV above the iron K edge energy: + + .. code-block:: python + + RE(mv(dcm.energy, 7112+50)) + +#. Put the beamline in the correct photon delivery system mode. (See + the table just above.) Continuing with the example of the iron K + edge, for unfocused beam: + + .. code-block:: python + + RE(change_mode('E')) + + If the new edge energy is in the same energy range according to the + table above, you can skip this step. For example, Mn and Fe are + both in mode E (or mode C). The ``change_mode()`` command does not + need to be run to move between those edges. + +#. Measure a :numref:`rocking curve scan (Sec %s) ` + to verify that the second crystal of the rocking curve is parallel + to the first crystal. This is more important for large energy + changes. You may find that you can skip this step if you are + changing between nearby edges. + + .. code-block:: python + + RE(rocking_curve()) + + At the end of the scan, the mono pitch will be moved to the top of + the rocking curve. + +#. If using focused beam, make sure that the mirror bender is in the + correct position. For focusing at the XAS table, ``m2_bender`` + should be at about 212000 counts. For focusing at the position of + the goniometer, ``m2_bender`` should be about 112000 counts. + + .. code-block:: python + + RE(mv(m2_bender, 212000)) + +#. Next, verify that the :numref:`height of the hutch slits (Sec %s) + ` is optimized for the beam height. In + principle, this should be correct after changing photon delivery + system mode. But it doesn't hurt to verify. + + .. code-block:: python + + RE(slit_height()) + + At the end of the scan, you will need to pluck the correct position + from the plot. + +#. Next, if you are using a reference foil, you should move the + reference foil holder to the slot containing the correct foil. The + command is something like: + + .. code-block:: python + + RE(reference('Fe')) + + choosing the correct element for your measurement. + +#. Finally, select the correct ROI channel: + + .. code-block:: python + + BMMuser.verify_roi(xs, 'Fe', 'K') + + +As a reminder, here is the table of operating modes. + +.. _pds_modes: +.. table:: Photon delivery modes + :name: pds-modes2 + :align: left + + ====== ============ ========================= + Mode focused energy range + ====== ============ ========================= + A |checkmark| above 8 keV + B |checkmark| below 6 keV + C |checkmark| 6 keV |nd| 8 keV + D |xmark| above 8 keV + E |xmark| 6 keV |nd| 8 keV + F |xmark| below 6 keV + XRD |checkmark| above 8 keV + ====== ============ ========================= + + +.. + Change mode + ----------- + + Suppose that you want to change from high-energy, unfocused operations + to low energy, focused. That is, you are changing from mode D to mode + B, for example moving from a large sample at the yttrium K edge to a + small sample at the vanadium K edge. + + .. code-block:: python + + RE(change_mode('B')) + RE(mv(dcm.energy, 5465+50)) + RE(rocking_curve()) + RE(slit_height()) + + + #. If the beam has recently been focused at the XRD station, you will + also need to adjust the bender on M2 to optimize vertical focus at + the XAS station (or vice versa). This is best done with the small + CCD camera sitting in the XAS sample stage. + + #. Again, iterating the optimization of the rocking curve and slit + height might be necessary. + +Change crystals +--------------- + +Suppose you wanted to change from the Pt L3 edge (11564 eV) on the +Si(111) crystal to the same energy on the Si(311) crystal. + +.. code-block:: python + + RE(change_xtal('311')) + +This will move the lateral motor of the DCM and optimize the roll and +pitch of the second crystal. It will then move the DCM to the energy +that you started at with the other crystal set and run a rocking curve +scan. + +Note that some of these motions can be a bit surprising in the sense +that the monochromator will briefly report itself as being outside the +normal operating range of the beamline. They will, however, +eventually return to sensible places. + + +Change XAS |larr| XRD +--------------------- + +To move the photon delivery system to delivery of focused beam to the +goniometer: + +.. code-block:: python + + RE(change_edge('Ni', xrd=True, energy=8600)) + +The element symbol in the first argument is not actually used in any +way when ``xrd=True`` is used, however the funtion requires +`something` as its first argument. Setting ``xrd=True`` forces the +``focus=True`` and ``target=0`` arguments to the ``change_edge()`` +command to be set. This will move to the specified energy, place the +photon delivery mode in `XRD` mode, optimize the second +crystal and the slit height, and move to an approximately M2 bender +position. + +To do all of that by hand, you would do the follow commands: + +.. code-block:: python + + RE(change_mode('XRD')) + RE(mv(dcm.energy, 8600)) + RE(rocking_curve()) + RE(slit_height()) + +This change of mode should have the beam in good focus at the position +of the goniometer. 8600 eV is the nominal operating energy for the +goniometer. If a higher energy is required, substitute the correct +energy for ``8600`` in the second line. + +.. note:: The I\ :sub:`0` chamber should be left in place. This will + facilitate changing energy while doing scattering + experiments. The flight path can be put in place at any time. + +.. todo:: Determine look-up table for lower energy operations using + both M2 and M3. This will require a new XAFS table and + adjustments to the limit switches on ``m3_ydo`` and + ``m3_ydi``. + + +.. _use333: + +XAFS with Si(333) +----------------- + +Using the Si(111) monochromator, it is possible to use the third +harmonic |nd| the Si(333) reflection |nd| to measure XAS with slightly +higher energy resolution. In this section, we explain how to set up +the beamline to measure the Ge K edge at 11103 eV using the Si(333). + +You cannot use the ``change_edge()`` command to do this. Use of the +Si(111) (or Si(311)) is hard-wired into that plan. You have to set up +the beamline by hand. + +First, put the photon delivery system in mode D (or mode A if using +the focusing mirror): + +.. code-block:: python + + RE(change_mode('D')) + +Next, move the monochromator to a few 10s of eV above the absorption +edge, as measured with the third harmonic. The Ge K edge is at 11103 +eV, so we need to move the monochromator to 11103/3 = 3701 eV. + +.. code-block:: python + + RE(mv(dcm.energy, (11103+27)/3)) + +or simply + +.. code-block:: python + + RE(mv(dcm.energy, 3701+9)) + +This will put the third harmonic energy 27 eV above the Ge K edge. + +Now, run a rocking curve scan: + +.. code-block:: python + + RE(rocking_curve()) + +This will produce a plot that looks something like this: + +.. _fig-rocking333: +.. figure:: _images/rocking_curve_333_E=3716.png + :target: _images/rocking_curve_333_E=3716.png + :width: 70% + :align: center + + A rocking curve scan with the photon delivery system in mode D and + the mono at 3716 eV. + +The broad base of this curve is the Si(111) rocking curve with photons +at 3710 eV. The sharp spike in the middle is the Si(333) rocking curve +with photons at 11130 eV. + +Optimize the slit_height: + +.. code-block:: python + + RE(slit_height()) + +You are ready to measure XAS with the Si(333) reflection! + +Here's an example ``scan.ini`` file for XANES of elemental Ge: + +.. code-block:: ini + + [scan] + experimenters = Bruce Ravel + + filename = Ge + sample = elemental Ge, crystalline + prep = standard sample + comment = measured with Si(333) reflection, 25um Al foil in beam path before I0 + + ththth = True + e0 = 11103 + element = Ge + edge = K + + nscans = 1 + start = next + + ## mode is one of transmission, fluorescence, both, or reference + mode = transmission + + ## Ge Si(333) + bounds = -45 -18 -9 36 150 + steps = 9 0.9 0.3 0.9 + times = 0.5 0.5 0.5 0.5 + + +Several things to note: + +#. Note that the actual value for E0 is specified, not the divided-by-3 value. +#. Actual energy bounds and steps are specified, the xafs scan plan + will convert them to appropriately sized steps for the Si(111). +#. By setting the 333 flag to True, the correct thing will happen, + including writing the correct energy axis to the output data file. +#. The on-screen plot will show the fundamental |nd| Si(111) |nd| energy, however. +#. Also, you still need to set up the photon delivery system up by hand. + + + +Motor controller kill switches +------------------------------ + +The MCS8 motor controllers supplied by FMBO have a kill switch for +power cycling the Phytron amplifier cards. This is implemented by the +vendor as connector plugged into the back of the chassis which shorts +the two leads of the receptacle. To kill the amplifiers, this plug is +removed and reinserted. + +That's fine, but the motor controllers are on top of the FOE |nd| not +a convenient location. + +The new kill switch system uses DIODE to close the kill switch +circuit. Two-conductor cable is run from each motor controller to a +remote DIODE box mounted on the inboard wall of the end station. + +The Bluesky interface is defined `here +`__ + + + +From the docstring of the class: + +.. code-block:: none + + A simple interface to the DIODE kill switches for the Phytron + amplifiers on the FMBO Delta Tau motor controllers. + + In the BMM DIODE box, these are implemented on channels 0 to 4 of + slot 4. + + attributes + ---------- + dcm + kill switch for MC02, monochromator + slits2 + kill switch for MC03, DM2 slits + m2 + kill switch for MC04, focusing mirror + m3 + kill switch for MC05, harmonic rejection mirror + dm3 + kill switch for MC06, hutch slits and diagnostics + + methods + ------- + kill(mc) + disable Phytron + enable(mc) + activate Phytron + cycle(mc) + disable, wait 5 seconds, reactivate, then re-enable all motors + + Specify the motor controller as a string, i.e. 'dcm', 'slits2', 'm2', 'm3', 'dm3' + + Here is a common problem which is resolved using a kill switch. + + BMM E.111 [36] ▶ RE(mvr(m2.pitch, 0.05)) + INFO:BMM_logger: Moving m2_pitch to 2.550 + + Moving m2_pitch to 2.550 + ERROR:ophyd.objects:Motion failed: m2_yu is in an alarm state status=AlarmStatus.STATE severity=AlarmSeverity.MAJOR + ERROR:ophyd.objects:Motion failed: m2_yu is in an alarm state status=AlarmStatus.STATE severity=AlarmSeverity.MAJOR + ERROR:ophyd.objects:Motion failed: m2_ydi is in an alarm state status=AlarmStatus.STATE severity=AlarmSeverity.MAJOR + ERROR:ophyd.objects:Motion failed: m2_ydi is in an alarm state status=AlarmStatus.STATE severity=AlarmSeverity.MAJOR + Out[36]: () + + This is telling you that the amplifiers for two of the M2 jacks + went into an alarm state. In the vast majority of cases, this + simply requires killing and reactivating those amplifiers. + + The solution to this one is: + + BMM E.111 [1] ▶ ks.cycle('m2') + Cycling amplifiers on m2 motor controller + killing amplifiers + reactivating amplifiers + enabling motors + + +Old kill switch system +~~~~~~~~~~~~~~~~~~~~~~ + +There is a row of switches on rack D, the rack next to the control +station, that are used to disable the amplifiers for the MCS8 motor +controllers. The cabling for this system still exists, but is not +plugged into the controllers. Should the DIODE system somehow fail, +this can be redeployed easily. + + +.. _fig-killswitches: +.. figure:: _images/Kill_switches.jpg + :target: _images/Kill_switches.jpg + :width: 70% + :align: center + + The manual kill switch system + + +When you suspect that a motor has an amplifier fault, toggle the +appropriate switch to the off position. Wait 10 seconds (to be very +safe...). Then toggle the switch back to the on position. The motor +should be ready to go. These switches replace the shorted plugs that +came attached to the "disable" port on the back side of the MCS8s. + +======= ========================== =============================== ================================= + MCS8 RGA label RGD label motors +======= ========================== =============================== ================================= + MC02 6BM-100149-RG:A1-PT1B3-A 6BM-100149-RG:A1-PT1B3-B DCM + MC03 6BM-100150-RG:A1-PT1B3-A 6BM-100150-RG:A1-PT1B3-B slits2 + MC04 6BM-100151-RG:A1-PT1B3-A 6BM-100151-RG:A1-PT1B3-B M2 + DM2 FS + MC05 6BM-100152-RG:A1-PT1B3-A 6BM-100152-RG:A1-PT1B3-B M3 + Filters + MC06 DM3 (bct,bpm,fs,foils)+ Slits3 +======= ========================== =============================== ================================= + +In the situation where toggling the switch does not clear the +amplifier fault, the next troubleshooting step is to power cycle the +MCS8. This is done by toggling the red, illuminated switch on the +front of the MCS8. Wait for the red amplifier lights to stop +flickering after turning off the MCS8, then turn the MCS8 back on. + +After power cycling the MCS8, it is necessary to re-home all the +motors controlled by the MCS8. + + +MCS8 Connector +~~~~~~~~~~~~~~ + +The disable plug on the back of the MCS8 controllers is a Binder RS +connector, part number 468-885. `Here's an +example. `__ + +And here is the wiring diagram. Short the prongs on the side opposite +to the alignment groove. + +.. _fig-killswitcheconnector: +.. figure:: _images/Kill_switch_connector.png + :target: _images/Kill_switch_connector.png + :width: 30% + :align: center + +Tutorial for how to put together the Binder connectors: :download:`PDF <_static/Binder-instructions.pdf>` + + + +Windows VM and BioLogic +----------------------- + +We use the `BioLogic EC_lab software +`__ to run +the `VSP-300 potentiostat +`__. Since there is not a +dedicated Windows machine at BMM, EC-Lab is run on a virtual machine +that is spun up when needed. + +Similarly, Hiden's control software is a Window's only produce. It is +run on the same firtual machine. + +Here are the instructions for starting and interacting with the +the VM. + +Starting the virtual machine +~~~~~~~~~~~~~~~~~~~~~~~~~~~~ + ++ At a command line, do ``rdesktop xf06bm-srv2 &`` ++ This will open a new window and display a Windows login + screen. Normal BNL credentials do not work. Log in as user + ``xf06bm`` using the password known by beamline staff. ++ The Windows desktop might start with a full-screen management + application that looks like the figure below. You can close + or minimize that window. ++ Double-click on the EC-lab or Hiden icon. ++ Do some electrochemistry or mass spectrometry. ++ Save your electrochemistry or mass spectrometry data to the assets + folder as explained below. + +.. _fig-winvm: +.. figure:: _images/Winvm_startup.png + :target: _images/Winvm_startup.png + :width: 70% + :align: center + + VM management window. You can minimize or close this. + +Storing electrochemistry or mass spectrometry data +~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~ + +The Windows VM has permission to connect to central storage with +permissions to write files from EC-lab or the Hiden software to the +correct location. + +To start a new experiment, you first have to disconnect the old drive +(if connected). This is likely mounted as the ``Z:`` drive. + +.. _fig-winstaledrive: +.. figure:: _images/Windows_stale_folder.png + :target: _images/Windows_stale_folder.png + :width: 70% + :align: center + + An example of a stale folder from a previous experiment mounted as + the ``Z:`` drive. + +Disconnect the stale ``Z:`` drive by right clicking on its entry in +the side bar and selecting "Disconnect". + +.. _fig-windisconnect: +.. figure:: _images/Windows_disconnect.png + :target: _images/Windows_disconnect.png + :width: 70% + :align: center + + Disconnect the stale ``Z:`` drive. + +Next, click on "This PC" in the sidebar, then click on the button that +says "Map network drive". + +.. _fig-winmapnewdrive: +.. figure:: _images/Windows_map_new_drive.png + :target: _images/Windows_map_new_drive.png + :width: 70% + :align: center + + Map a new network drive to the VM. + +On the "Map Network Drive" page, you need to fill in the path to the +current experiment's proposal folder. Suppose the current cycle is +2024-3 and the current proposal number is 316832. Using ``Z:`` as the +drive letter, enter the following as the "Folder" + +.. code-block:: none + + \\data3.nsls2.bnl.gov\bmm\proposals\2024-3\pass-316832 + +Note that the backslashes are important. Also substitute the correct +cycle and proposal numbers. + +Be sure to leave "Reconnect at sign-in" checked. Note that "Connect +using different credentials" should be unchecked. + + +.. _fig-winspecifynewdrive: +.. figure:: _images/Windows_specify_new_drive.png + :target: _images/Windows_specify_new_drive.png + :width: 70% + :align: center + + Map a new network drive to the VM. + +Click the finish button. The connection will take several seconds, +but then the new entry will show up in the side bar. + +The new network drive can now be clicked into. + +Configure EC-lab to write its data files into the +``assets\vsp300-1`` folder. + +Configure the Hiden software to write its data files into the +``assets\hpr20-1`` folder. + +.. _fig-winsassetsfolder: +.. figure:: _images/Windows_assets_folder.png + :target: _images/Windows_assets_folder.png + :width: 70% + :align: center + + The ``assets\vsp300-1`` folder is the correct place for data from + EC-lab to be written. The ``assets\hpr20-1`` folder is the correct + place for data from the Hiden to be written. + +By following this procedure, the electrochemistry data from EC-lab and +mass spectrometry data from the Hiden will be available to the user in +the :numref:`same manner as their XAS data (Section %s) `. + + + + +Calibrate the mono +------------------ + +The typical calibration procedure involves measuring the angular +position of the Bragg axis for the edge energies of 10 metals: Fe, Co, +Ni, Cu, Zn, Pt, Au, Pb, Nb, and Mo. + +The tabulated values of edge energies from Table 1 in `Kraft, et +al. `__ are used in the +calibration. + + +#. Be sure that all 10 of these elements are actually mounted on the + reference wheel and configured in the ``xafs_ref.mapping`` dict. + (They should be. It would be very unusual for any of these foils + to have been removed from the reference wheel.) + +#. Run the command + + .. code-block:: python + + RE(calibrate(mono='111')) + + Use the ``mono='311'`` argument for the Si(311) monochromator. + This will, in sequence, move to each edge and measure a XANES scan + over a wide enough range that it should cover the edge (unless the + mono is currently calibrated VERY wrongly). This will write a file + called :file:`edges111.ini` (or :file:`edges3111.ini`). Each XANES + scan uses the file + :file:`/home/xf06bm/Data/Staff/mono_calibration/cal.ini` as the INI + file. Edge appropriate command line parameters will be added by + the ``calibrate()`` plan. + +#. Examine the data in |athena|. Make sure E\ :sub:`0` is selected + correctly for all 10 edges. Copy those values into the first column + of :file:`edges111.ini` (or :file:`edges311.ini`). + + .. attention:: + + It is no longer necessary to compute the angular positions of + the monochromator. Those will be computed from the edge energy + values you edited into the INI file by hand. + + .. todo:: + + Implement on-the-fly determination of E\ :sub:`0` to obviate the + step of editing the INI file. Pb is tricky. Nb and Mo are kind + of tricky. + + + .. + Compute the + angular positions using + .. code-block:: python + dcm.e2a() + and copy those numbers into the :file:`edgeH11.ini` file. + +#. Run the command + + .. code-block:: python + + calibrate_mono(mono='111') + + (or use the ``'311'`` argument). This will show the fitting + results and plot the best fit. It will also print in a text box + instructions for modifying the :file:`BMM/dcm-parameters.py` file + to use the new calibration values. + + .. _fig-calibrate: + .. figure:: _images/Calibration_111.png + :target: _images/Calibration_111.png + :width: 70% + :align: center + + Example calibration curve + + +#. Edit :file:`BMM/dcm-parameters.py` as indicated. + +#. Do + + .. code-block:: python + + %run -i 'home/xf06bm/.ipython/profile_collection/startup/BMM/dcm-parameters.py' + + then do + + .. code-block:: python + + dcm.set_crystal() + + Or simply restart |bsui|, which is usually the easier thing. + +#. Finally, do + + .. code-block:: python + + calibrate_pitch(mono='111') + + This performs a simple linear fit to the rocking curve peak + positions for ``dcm_pitch`` found at each edge. Use the fitted + slope and offset to modify ``approximate_pitch`` in + :file:`BMM/functions.py`. + +The mono should now be correctly calibrated using the new calibration +parameters. + + +Provision a new beamline computer +--------------------------------- + +This is a list of notes on how to finish the provisioning of a new +beamline computer. + +Firstly, make sure that ``/nsls2/data`` is a symlink to +``/nsls2/data3``. If it is not, ask for help from DSSI. + + +install additional packages +~~~~~~~~~~~~~~~~~~~~~~~~~~~ + ++ plasma-desktop (just ... better) ++ redis (essential for operation of |bsui|) ++ most (used as the pager in BMM's |bsui| profile) ++ ag (powerful ack-like grep alternative) ++ fswebcam (used to capture analog pinhole camera) ++ demeter and perl-Graphics-GnuplotIF (something silly, no doubt) ++ slack (communications) ++ ark (compression, useful in file manager) + + +To install these, do: + +.. code-block:: sh + + dzdo dnf install redis most ag fswebcam demeter perl-Graphics-GnuplotIF slack ark + dzdo dnf install --skip-broken --nobest @kde-desktop + +The second command installs the KDE metapackage, skipping missing +packages. + +To finish installing ``sddm``, do + +.. code-block:: sh + + dzdo systemctl stop gdm + dzdo systemctl start sddm + +Desktop wallpaper +~~~~~~~~~~~~~~~~~ + +This may not be provisioned correctly out of the box. Find the +beamline wallpapers in ``/usr/share/nsls2/wallpapers/beamlines``. + +Right click on the desktop and select "Configure Desktop and +Wallpaper". Click on "Add Image" and navigate to the folder above. + +Things to install from git +~~~~~~~~~~~~~~~~~~~~~~~~~~ + ++ BMM stuff: ``git clone git@github.com:NSLS-II-BMM/BMM-beamline-configuration.git`` + + then, ``cd ~/bin`` and ``ln -s ~/git/BMM-beamline-configuration/tools/run-cadashboard`` ++ BMM user manual: ``git clone git@github.com:NSLS-II-BMM/BeamlineManual.git`` ++ BMM standards: ``git clone git@github.com:NSLS-II-BMM/bmm-standards.git`` ++ Switch visualization: ``git clone git@github.com:NSLS-II-BMM/switch-pretty-printer.git`` + +Also do ``cd ~/bin`` and ``ln -s ~/.ipython/profile_collection/startup/consumer/run-consumer`` + +Workspace folders +~~~~~~~~~~~~~~~~~ + +Make the local data collection folders. The +:numref:`BMMuser.begin_experiment() command (Section %s) ` +will make symlinks under those folders to the correct place on central +storage. + +.. code-block:: text + + mkdir ~/Workspace + mkdir ~/Workspace/Visitors + mkdir ~/Workspace/Staff + + + +Manage Silicon Drift Detectors +------------------------------ + +The assumption in the data acquisition system is that one of the three +silicon drift detectors will be the primary detector in an experiment. +At the |bsui| command line (or in |qs|) the ``xs`` symbol should +point at the correct detector. Also, a parameter is set in Redis +allowing other processes (such as the Kafka plotting agent) to know +which detector to be paying attention to. + + +.. code-block:: python + + xs = xspress3_set_detector(7) + +where the argument to ``xspress3_set_detector`` is 1, 4, or 7. Since +October 2024, use of the seven element detector is the default. + +This sets ``xs`` to the selected detector object |nd| ``xs1``, +``xs4``, or ``xs7``. + +Also set is the Redis parameter ``BMM:xspress3``, which is set to 1, +4, or 7 (and represented as a b-string). + +.. code-block:: python + + n_elements = int(rkvs.get('BMM:xspress3')) diff --git a/_sources/modules.rst b/_sources/modules.rst new file mode 100644 index 0000000..824cc6c --- /dev/null +++ b/_sources/modules.rst @@ -0,0 +1,6 @@ +BeamlineManual +============== + +.. toctree:: + :maxdepth: 4 + diff --git a/_sources/motors.rst b/_sources/motors.rst new file mode 100644 index 0000000..4ad0e1e --- /dev/null +++ b/_sources/motors.rst @@ -0,0 +1,427 @@ +.. + This document was developed primarily by a NIST employee. Pursuant + to title 17 United States Code Section 105, works of NIST employees + are not subject to copyright protection in the United States. Thus + this repository may not be licensed under the same terms as Bluesky + itself. + + See the LICENSE file for details. + +.. |nbsp| unicode:: 0xA0 + :trim: + +.. _motors: + +Moving and querying motors +========================== + +To get an overview of the state of the beamline motors, do ``%m`` at +the |bsui| command line. Here is an example: + +.. code-block:: text + + In [1897]: %m + ============================================================================== + Energy = 19300.1 reflection = Si(111) mode = fixed + Bragg = 5.87946 2nd Xtal Perp = 15.0792 2nd Xtal Para = 146.4328 + + M2 + vertical = 6.000 mm YU = 6.000 + lateral = 0.000 mm YDO = 6.000 + pitch = 0.000 mrad YDI = 6.000 + roll = -0.001 mrad XU = -0.129 + yaw = 0.200 mrad XD = 0.129 + + M3 + vertical = 0.000 mm YU = -1.167 + lateral = 15.001 mm YDO = 1.167 + pitch = 3.500 mrad YDI = 1.167 + roll = 0.000 mrad XU = 15.001 + yaw = 0.001 mrad XD = 15.001 + + Slits3: vsize vcenter hsize hcenter top bottom outboard inboard + 1.350 0.000 8.000 -0.000 0.675 -0.675 4.000 -4.000 + + DM3_BCT: 45.004 mm + + XAFS table: + vertical pitch roll YU YDO YDI + 132.000 0.000 0.000 132.000 132.000 132.000 + + XAFS stages: + x y roll pitch linxs roth wheel rots + 9.224 115.000 0.840 0.000 -45.000 0.000 -59.000 0.000 + ============================================================================== + + +.. _sample_stages: + +Sample stages +------------- + +.. |widdershins| replace:: `widdershins `__ + + +These stages sit on top of the XAFS optical table. + +.. table:: XAFS sample stages + :name: xafs-stages + :align: left + + ============== =========== ========= ======================= ===================================== + motor type units notes directions + ============== =========== ========= ======================= ===================================== + ``xafs_x`` linear mm main sample stage |plus| outboard, - inboard + ``xafs_y`` linear mm main sample stage |plus| up, - down + ``xafs_det`` linear mm detector mount |plus| away from sample, - closer + ``xafs_wheel`` rotary degrees *ex situ* sample wheel |plus| clockwise, - |widdershins| + ``xafs_linxs`` linear mm ref wheel vertical |plus| up, - down + ``xafs_ref`` rotary degrees reference stage |plus| clockwise, - |widdershins| + ``xafs_refx`` linear mm reference stage |plus| outboard, - inboard + ``xafs_refy`` linear mm reference stage |plus| up, - down + ``xafs_pitch`` tilt degrees Huber tilt stage |plus| more positive + ``xafs_roll`` tilt degrees Huber tilt stage |plus| more positive + ``xafs_rots`` rotary degrees small rotary stage |plus| clockwise, - |widdershins| + ============== =========== ========= ======================= ===================================== + +Configuration and position of the motors can be queried easily. In +the following examples, the ``xafs_y`` motor is used. The commands +are the same for all sample stage motors. + +**Querying position** + The position of a motor can be queried with a command line like :: + + %w xafs_y + + or :: + + xafs_y.position + +**Moving to a new position** + Always move motors through the run engine, for example: :: + + RE(mvr(xafs_y, 10)) + + ``mvr`` is the relative move command |nd| the numerical argument is + the amount by which the motor will move from the current position. + + ``mv``, as in:: + + RE(mv(xafs_y, 37.63)) + + is the absolute move command. The numerical argument is the + position to which the motor will move. + +.. All movements are logged in the :numref:`experimental log (Section %s) ` + +**Moving to a new position in a plan** + To move a sample stage as part of a :numref:`macro (Section %s) + ` , do:: + + yield from mv(xafs_y, 37.36) + + You can combine motions of two or more motors in a single + synchronous movement:: + + yield from mv(xafs_y, 37.36, xafs_x, 15.79) + + Similarly:: + + yield from mvr(xafs_y, 5) + +**Querying soft limits** + To know the soft limits on a sample stage, do ``xafs_y.limits`` or + ``xafs_y.llm.get()`` or ``xafs_y.hlm.get()`` to query the low or + high limits individually. + +**Setting soft limits** + To set the soft limits on a sample stage, do something like + ``xafs_y.llm.put(5)`` or ``xafs_y.hlm.put(85)`` + +**Reference wheel** + :numref:`The reference stage (Section %s) ` is a + rotation stage with a sample wheel holding up to 48 reference + foils. It is calibrated such that the beam passes through the + center of a slot every 15 degrees. The slots are indexed such that + they can be accessed by the symbol of the element being measured. + To move to a new reference foil:: + + RE(reference('Fe')) + + To see the available foils, do ``%se`` or look at the value of + ``xafs_ref.mapping``. + + :numref:`See Section %s for a full explanation of the the reference wheel contents. ` + + `Here is a complete list of standards + `__ + in BMM's collection. These standards are mounted on sample wheels + and stored in the hutch for ready access by users. + + +Sample wheel +------------ + +The ``xafs_wheel`` motor is a rotary stage that is typically mounted +on the XY stage. It can be mounted face-on to the beam or at 45 +degrees for use with the fluorescence detector. + +Sample plates laser cut from plastic sheet (initially we used `Delrin +`_, since COVID made +supply difficult, we use whatever we can get) are attached to the +rotation stage. The single-ring version of these plates have 24 slots +arranged around the periphery, evenly spaced 15 degree apart. The +double-ring version has concentric rings of 24 slots each. These are +still 15 degrees apart. The radius of the outer ring is 26 mm larger +than the radius of the inner ring. + +While you can move from slot to slot in increments of 15 degrees, i.e. + +.. code-block:: python + + RE(mvr(xafs_wheel, 15*3)) + +it is somewhat easier to move by slot number. The sample plates are +cut with sample numbers for slots 1, 7, 13, and 19, making it clear +which slot is which. The wheel is mounted such that the numbers can +be read normally on the side facing the beam. + +To move, for instance, to slot 5, do: + +.. code-block:: python + + RE(slot(5)) + +In a macro, do + +.. code-block:: python + + yield from slot(5) + +To move to the inner or outer ring, do + +.. code-block:: python + + RE(xafs_wheel.inner()) + RE(xafs_wheel.outer()) + +This translates ``xafs_x`` by |nbsp| |pm| 26 mm. + +In a macro, do + +.. code-block:: python + + yield from xafs_wheel.inner() + yield from xafs_wheel.outer() + + +.. + Sample spinner + -------------- + + The sample spinner is a 12 volt CPU cooling fan mounted on a plate + which is mounted on the tilt stage. It is used to spin crystalline + samples in an effort to suppress Bragg peaks which might enter the + fluorescence detector. + + To turn the spinner on and off:: + + fan.on() + fan.off() + + To turn the spinner on or off in a :numref:`macro (Section %s) `:: + + yield from fan.on_plan() + yield from fan.off_plan() + + The spinner should **always** be turned off before entering the end + station. It is a good idea to always have a camera pointed at the + spinner while it is use. + + +Glancing angle stage +-------------------- + +The glancing angle stage, shown in :numref:`Figure %s `, +can hold up to eight samples and allows each sample to spin +independently. The spinning allows spurious diffraction from a +crystalline substrate into the fluorescence detector to be suppressed. + +.. _fig-gastage: +.. figure:: _images/glancing_angle_stage.jpg + :target: _images/glancing_angle_stage.jpg + :width: 50% + :align: center + + The glancing angle stage with 8 sample positions. + +To move to a sample position:: + + RE(ga.to(3)) + +where the argument is a number from 1 to 8, as shown by the labels in +:numref:`Figure %s `. This command will turn off all +other spinners, rotate that sample into the beam path, and start the +sample spinning. + +To turn a spinner on or off, where the argument is a number from 1 to 8:: + + RE(ga.on(3)) + RE(ga.off(3)) + +To turn off all spinners:: + + RE(ga.alloff()) + +In a plan:: + + yield from ga.on_plan() + yield from ga.off_plan() + yield from ga.alloff_plan() + + + +Sample alignment +~~~~~~~~~~~~~~~~ + +A sample is aligned into the beam by moving the tilt stage to an +approximately flat position:: + + RE(mv(xafs_pitch(0)) + +Then performing the following sequence:: + + RE(linescan(xafs_y, 'It', -1, 1, 41)) + RE(linescan(xafs_pitch, 'It', -2, 2, 41)) + +At the and of the ``xafs_y`` scan, pick the position halfway down the +edge in the It signal. At the end of the ``xafs_pitch`` scan, select +the peak position. This will place the sample such that it is flat +relative to the incident beam direction and halfway blocking the beam. + +You may choose to iterate those two scans. + +Next move the sample to the measurement angle. Suppose the +measurement angle is 2.5 degrees:: + + RE(mv(xafs_pitch, 2.5)) + +Finally, position the sample so that the beam is hitting the center of +the sample:: + + RE(linescan(xafs_y, 'If', -1, 1, 41)) + +Since the sample is not at the eucentric of the tilt stage, this final +vertical scan is always necessary. When first aligning the sample, +you may need to center the sample in ``xafs_x`` as well:: + + RE(linescan(xafs_x, 'If', -6, 6, 41)) + +You will almost certainly need to scan over a longer range. Make sure +the detector is retracted far enough to allow for this motion. + + +Automated alignment +~~~~~~~~~~~~~~~~~~~ + +The sequence described above can be automated in many cases:: + + RE(ga.auto_align(2.5)) + +This will run the sequence of alignment scans described above, +pitching the sample to the user-specified angle before the vertical +scan measuring the fluorescence signal. This works by fitting an +error function to the ``xafs_y`` scan versus It, selecting the peak of +the pitch scan, then selecting the peak of the ``xafs_y`` scan versus +fluorescence. + +.. _fig-ga_alignment: +.. figure:: _images/spinner-alignment.png + :target: _images/spinner-alignment.png + :width: 50% + :align: center + + If all goes well, the result of the sample alignment looks like + this. A picture like this is posted to :numref:`Slack (Section %s) + `. + + +For very flat samples which are square or circular and about 5mm +across or larger, this alignment algorithm is very robust. For oddly +shaped samples, verify that the automation works before relying upon +it. Otherwise, simply do the alignment by hand. + +Table motors +------------ + +Typically, table motors are not moved individually. When changing +:numref:`photon delivery system modes (Section %s) `, the +table should be put into the correct orientation such that the beam +passes through the center of the ion chambers. It is very easy to put +the beamline in a confusing state by changing the table motors outside +of the ``change_mode()`` command. + +The lateral table motors |nd| and its yaw |nd| are normally disabled. + + +.. table:: XAFS table motors + :name: xafs-table + :align: left + + ============== ======== ================================= + motor units notes + ============== ======== ================================= + xafs_yu mm upstream table jack + xafs_ydi mm downstream, inboard table jack + xafs_ydo mm downstream, outboard table jack + xafs_vertical mm coordinated linear motion + xafs_pitch degrees coordinated table pitch + xafs_roll degrees coordinated table roll + ============== ======== ================================= + + +**Querying table position** + The position of any motor can be queried with a command line like + ``%w xafs_table``. + +**Moving table motors** + The normal movement commands work on the real and virtual motors, + e.g.:: + + RE(mvr(xafs_ydi, 3)) + RE(mv(xafs_vertical, 107)) + + Again, this is rarely necessary. The mode changing plan should + leave the table in the correct location for your experiment. + + All table movements are recorded in the :numref:`experimental log + (Section %s) `. + +Examine Motor Axes +------------------ + +Some BlueSky functionality related to the axes controlled by the FMBO +MCS8 motor controllers. These include: + ++ Collimating mirror (``m1_*``) ++ Filter assemblies (``dm1_*``) ++ Monochromator (``dcm_*``) ++ Second diagnostic module (``dm2_*``) ++ Focusing mirror (``m2_*``) ++ Harmonic rejection mirror (``m3_*``) ++ Third diagnostic module (``dm3_*``) + +(38 axes motors in total) but not any of the end station motors +(``xafs_*``), which are run using NSLS-II standard GeoBricks. + +**Homing** + Any of these axes can be homed with, for example, ``dm3_bct.home()`` + +**Summarize the status of a motor** + To show the values of all the status flags, for example, ``dm3_bct.status()`` + +**Which motors have been homed?** + Do this command: ``homed()`` + +**Which motors have their amplifiers enabled?** + Do this command: ``ampen()`` diff --git a/_sources/other.rst b/_sources/other.rst new file mode 100644 index 0000000..8211196 --- /dev/null +++ b/_sources/other.rst @@ -0,0 +1,35 @@ + +.. + This document was developed primarily by a NIST employee. Pursuant + to title 17 United States Code Section 105, works of NIST employees + are not subject to copyright protection in the United States. Thus + this repository may not be licensed under the same terms as Bluesky + itself. + + See the LICENSE file for details. + + +.. _other_measurements: + +Other measurements +================== + +.. attention:: Unfinished section + +SEAD +---- + +Single energy absorption detection + +Term coined in `a paper by Filipponi et al. `__ + + +explain INI file + + +Raster +------ + +Single ROI area scans. A map of sorts. + +explain INI file diff --git a/_sources/pds.rst b/_sources/pds.rst new file mode 100644 index 0000000..0a71359 --- /dev/null +++ b/_sources/pds.rst @@ -0,0 +1,613 @@ +.. + This document was developed primarily by a NIST employee. Pursuant + to title 17 United States Code Section 105, works of NIST employees + are not subject to copyright protection in the United States. Thus + this repository may not be licensed under the same terms as Bluesky + itself. + + See the LICENSE file for details. + +.. _pds: + +Photon Delivery System +====================== + +Configure the Photon Delivery System +------------------------------------ + +Configuring the photon delivery system for a specific measurement is +usually quite simple. When moving to a new absorption edge, do the +following: + +.. code-block:: python + + RE(change_edge('Fe')) + +substituting the two-letter symbol for the element you want to +measure. This will: + +* move the monochromator :numref:`(Section %s) ` +* put the photon delivery system in the correct mode :numref:`(Section + %s) ` +* measure the rocking curve of the monochromator :numref:`(Section %s) + ` +* optimize the height of the hutch slits :numref:`(Section %s) + ` +* move the reference foil holder to the correct position + :numref:`(Section %s) ` +* set the active Xspress3 ROI to the correct emission line + +This whole process takes at most 7 minutes, sometimes under 3 +minutes. After that, the beamline is ready to collect data. + +If using the focusing mirror, do this: + +.. code-block:: python + + RE(change_edge('Fe', focus=True)) + +Excluding the ``focus`` argument |nd| or setting it to False |nd| +indicates setup for collimated beam. + +This edge change can be put into a :numref:`macro (see Section %s) +` like so: + +.. code-block:: python + + yield from change_edge('Fe') + +or + +.. code-block:: python + + yield from change_edge('Fe', focus=True) + +In this way, a macro can manage energy changes while you sleep! + +.. _foilholder: + +Automating reference foil changes +~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~ + +A wheel is used to hold and switch between reference foils and stable +oxides. The standard reference wheel has most of the elements +accessible at BMM, including all the lanthanides (except Pm!). A +double wheel (:numref:`see Figure %s `) is used to +hold the standards. The wheel is mounted on a rotation stage which +is, in turn, mounted on an XY stage for alignment. See +:numref:`Table %s ` for the contents of reference +wheel. + +.. _fig-ref_wheel: +.. figure:: _images/ref_wheel.jpg + :target: _images/ref_wheel.jpg + :width: 50% + :align: center + + The reference wheel. + + +To select, for example, the iron reference foil: + +.. code-block:: python + + RE(reference('Fe')) + +In a plan: + +.. code-block:: python + + yield from reference('Fe') + +The argument is simply the one- or two-element symbol for the target +element. + +This selects the correct reference by rotating to the correct slot and +translating to the correct ring on the wheel. + +The ``change_edge()`` command does this automatically, so long as the +target edge is available on the reference holder. + +The reference wheel content is configured as a python dictionary. See +``xafs_ref.mapping``, `defined here +`__. + +This dictionary identifies the positions in ``xafs_ref`` and +``xafs_refx`` for each reference sample. It also identifies the form +of the reference samples and its chemical composition. + +To see the available reference materials and their positions on the +reference wheel, do ``%se``. + +`Here is a complete list of standards +`__ in +BMM's collection. + + +.. + Automating fluorescence ROI changes + ~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~ + + To make sure that the correct ROI channel is selected, you need to + configure the ROI readout. Suppose that you have configured the + analog detector readout system to measure three of those transition + metals. Then you would execute a command like this to configure the + detector readout: + + .. code-block:: python + + rois.set('Fe Co Ni') + + Unfortunately, the ROI channels and reference holder have the hot + dog/hot dog bun problem. There are only three output channels for the + analog detector readout system, thus only three elements can be + configured at a time. + + When you change edge to an element that is configured as an ROI + channel, the data acquisition system will take its fluorescence data + from the corresponding channels of the Struck multichannel scalar. It + will also perform the dead-time correction using the correct signal + chains for the selected element. + +Parameters for the change_edge() command +~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~ + +Typically the ``change_edge()`` command is called with one or two +arguments, the mandatory element symbol and the the ``focus`` +argument, which can be ``True`` or ``False``. + +The full set of parameters for the ``change_edge()`` plan are: + +.. code-block:: python + + RE(change_edge(element, focus=False, edge='K', energy=None, tune=True, slits=True, calibrating=False, target=300.)) + +where, + +``element`` + The one- or two-letter element symbol or Z number. + +``focus`` + ``True``: set up for using the focusing mirror, modes A, B, C; + ``False``: collimated beam, modes D, E, F. Default is ``False``. + +``edge`` + If not specified, use K or L3, as appropriate for the energy range + of the beamline. Use this argument to specify an L1, L2, or M edge. + +``energy`` + Use an E0 value that is not obtained from the look-up table. + Default is unspecified, i.e. use ``element`` and look-up table. + This is rarely necessary, except when setting up for XRD. + +``tune`` + ``True``: optimize DCM second crystal pitch; ``False``: skip ``rocking_curve()`` + scan. Default is ``True``. Skipping this is rarely a good idea. + +``slits`` + ``True``: optimize slit height; ``False``: skip ``slit_hight()`` + scan. Default is ``True``. Skipping this is rarely a good idea. + +``no_ref`` + ``True``: skip moving to the correct reference foil. Default is + ``False``. Used when the reference stages have been repurposed + for other use in an experiment. + +``calibrating`` + ``True``: used when performing beamline maintenance. Default is + ``False``. Rarely used. + +``target`` + The energy above e0 at which to perform the rocking curve scan. + Default is 300. Care is taken not to exceed an L\ :sub:`2` edge + energy (or L\ :sub:`1` when measuring L\ :sub:`2`). + + +Except for ``edge`` and ``focus``, most of those parameters are rarely +used. If you need to set up for measuring an L\ :sub:`2` or L\ +:sub:`1` edge, you must specify ``edge``. For example: + +.. code-block:: python + + RE(change_edge('Pt', edge='L1')) + + + +For all the details about the individual parts of the photon delivery +system, read on! + + +.. _shutters: + +Shutters +-------- + +**Open and close the photon shutter** + In the nomenclature of BMM, the photon shutter is ``shb``. Open + and close this shutter with:: + + shb.open() + shb.close() + + These plans are somewhat more elaborate than simply toggling the + state of the shutters. It happens from time to time that the + shutter does not trigger when told to open or close. So, these + plans try up to three times to open or close the photon shutter, + with a 1.5 second pause between attempts. + + If you wish to open or close the photon shutter (using the same + multiple attempt algorithm) in a :numref:`macro (Section %s) + `, do:: + + yield from shb.open_plan() + yield from shb.close_plan() + +**Open and close the safety shutter** + This is the front-end shutter. Closing it takes light off the + monochromator, which is not something you typically want to do + during an experiment. That said, the safety shutter is ``sha`` in + the BMM nomenclature:: + + sha.open() + sha.close() + + and:: + + yield from sha.open_plan() + yield from sha.close_plan() + + +.. _dcm: + +Monochromator +------------- + +The monochromator consists of 8 motors. It should never be necessary +to interact directly with any of the physical motors. Plans exist for +facilitating any actions a user should ever need. + +**Query the current energy** + To know the position and energy of the monochromator: ``%w dcm`` + + This returns a short report like this: + + .. code-block:: text + + Energy = 19300.1 reflection = Si(111) + current: Bragg = 5.87946 2nd Xtal Perp = 15.0792 2nd Xtal Para = 146.4328 + + + This report shows the current energy, the crystal set currently in + use, and the position of the parallel and perpendicular motors of + the second crystal carriage. + +**Move to a new energy** + The ``dcm.energy`` virtual motor coordinates the Bragg, parallel, + and perpendicular motors to maintain a fixed exit height and set + the energy of the mono. To move to the copper K edge energy:: + + RE(mv(dcm.energy, 8979)) + + To move 50 eV above the copper K edge energy:: + + RE(mv(dcm.energy, 8979+50)) + + Note that the BlueSky command line is able to do simple arithmetic + (and a whole lot more!). It is a good idea to leave the arithmetic + to the computer. + +**Move to a new energy in a macro** + An energy change can be a part of a :numref:`macro (Section %s) + `. Simply do:: + + yield from mv(dcm.energy, 8979+50)) + +**Tune the second crystal of the mono** + After a long move, you might need to retune the second crystal. To + find the peak of the rocking curve and move to that peak:: + + RE(rocking_curve()) + + This will run a scan of the pitch of the second crystal. At the + end of the scan, it moves to the center of mass of the measured + intensity profile. + + You can do the rocking curve scan by looking at the signal on the + Bicron which is used as the incident beam monitor for the XRD end + station. Do:: + + RE(rocking_curve(detector='Bicron')) + + You can tune the second crystal by hand with these commands:: + + tu() + td() + + Those stand for "tune up" and "tune down". Do not + think that "up" and "down" refer to measured + intensity. Rather, they refer to the direction of motion of the + motor which adjusts the second crystal pitch. When you move to + higher energy, you usually need to tune in ``td()`` direction. + When you move to a lower energy, you usually need to tune in the + ``tu()`` direction. Obviously..... + +**Fixed-exit and pseudo-channelcut modes** + The mono can be run in either fixed-exit or pseudo-channelcut + modes. + + Fixed exit means that the second monochromator crystal will be + moved in directions parallel and perpendicular to its diffracting + surface in order to maintain a fixed exit height of the beam coming + from the second crystal. Without fixed-exit mode, it would not be + possible to change the energy over the entire energy range of the + beamline. The aperture after the monochromator is only a few + millimeters tall. The vertical displacement of the beam over a + lerge energy change would be sufficient to move the beam out of the + aperture. + + However, the stability of the monochromator suffers with respect to + EXAFS data quality when measuring an energy scan in fixed-exit + mode. We find it is better to disable the parallel and + perpendicular motions when measuring XAFS, suffering a small + vertical displacement of the beam. + + The mono mode is controlled by a parameter: + + .. code-block:: python + + dcm.mode = 'fixed' + + or + + .. code-block:: python + + dcm.mode = 'channelcut' + + In practice, the monochromator is normally left in fixed-exit mode. + That way, the monochromator can be moved without having to worry + about the beam height and the monochromator exit aperture. In the + :numref:`XAFS scan plan (Section %s) `, the monochromator + first moves |nd| in fixed-exit mode |nd| to the center of the + angular range of the scan, then sets ``dcm.mode`` to + ``channelcut``. Once the sequence of scan repititions is finished, + the monochromator is moved back to the center of the angular range + and the monochromator is returned to fixed-exit mode. + + +Post-mono slits +--------------- + +After the mono, before the focusing mirror, in Diagnostic Module 2, +there is a four-blade slit system. These are used to define the beam +size on the mirrors and to refine energy resolution for the focused +beam.. + + +.. table:: Post mono slit motors + :name: slits2-motors + :align: left + + =============== ======== ======================= =================== + motor units notes motion type + =============== ======== ======================= =================== + slits2_top mm top blade position single axis + slits2_bottom mm bottom blade position single axis + slits2_inboard mm inboard blade position single axis + slits2_outboard mm outboard blade position single axis + slits2_hsize mm horizontal size coordinated motion + slits2_hcenter mm horizontal center coordinated motion + slits2_vsize mm vertical size coordinated motion + slits2_vcenter mm vertical center coordinated motion + =============== ======== ======================= =================== + + +The individual blades are moved like any other motor:: + + RE(mv(slits2.outboard, -0.5)) + RE(mvr(slits2.top, -0.1)) + + +Coordinated motions are moved the same way:: + + RE(mv(slits2.hsize, 6)) + RE(mvr(slits2.vcenter, -0.1)) + +To know the current positions of the slit blades and their coordinated +motions, use ``%w slits2`` + +.. code-block:: text + + In [1966]: %w slits2 + SLITS2: + vertical size = 1.350 mm Top = 0.675 + vertical center = 0.000 mm Bottom = -0.675 + horizontal size = 8.000 mm Outboard = 4.000 + horizontal center = 0.000 mm Inboard = -4.000 + +Mirrors +------- + +Mirrors are set as part of the mode changing plan. Unless you know +exactly what you are doing, you probably don't want to move the +mirrors outside of the ``change_mode()`` command. (Adjusting M1 by +hand is a horrible idea -- unless you know exactly what you are doing +and why.) Changing the mirror positions in any way changes the height +and inclination of the beam as it enters the end station. This +requires changes in positions of the slits, the XAFS table, and other +parts of the photon delivery system. + +Outside of the use of the ``change_mode()`` command, it should not be +necessary for users to move the mirror motors. It is **very easy** to +lose the beam entirely when moving mirror motors. Without a clear +understanding of how the optics work, re-finding the beam can be quite +challenging. If you loose the beam by moving motors, the best +solution is probably to rerun the ``change_mode()`` command. + +That said, if you want to know the current positions of the motors on +the focusing mirror, use ``%w m2`` + + +.. code-block:: text + + In [1903]: %w m2 + M2: + vertical = 6.000 mm YU = 6.000 + lateral = 0.000 mm YDO = 6.000 + pitch = 0.000 mrad YDI = 6.000 + roll = -0.001 mrad XU = -0.129 + yaw = 0.200 mrad XD = 0.129 + bender = 163789.0 steps + +For the harmonic rejection mirror, use ``%w m3`` + +.. code-block:: text + + In [1904]: %w m3 + M3: (Rh/Pt stripe) + vertical = 0.000 mm YU = -1.167 + lateral = 15.001 mm YDO = 1.167 + pitch = 3.500 mrad YDI = 1.167 + roll = 0.000 mrad XU = 15.001 + yaw = 0.001 mrad XD = 15.001 + +The front-end collimating mirror, the focusing mirror, and one stripe +of the harmonic rejection mirror are coated with 5 nm of Rh deposited +on 30 nm of Pt on silicon. See `MA Marcus, et +al., J. Synch. Radiat. (2004) 11, 239-247 DOI: +10.1107/S0909049504005837 +`__ for an explanation of +the advantages of this coating scheme. + + +.. _slits3: + +End station slits +----------------- + +Near the end of the photon delivery system, in Diagnostic Module 3 in +the end station, there is a four-blade slit system. These are used +to define the beam size on the sample. + + +.. table:: End station slit motors + :name: slits3-motors + :align: left + + =============== ======== ======================= =================== + motor units notes motion type + =============== ======== ======================= =================== + slits3_top mm top blade position single axis + slits3_bottom mm bottom blade position single axis + slits3_inboard mm inboard blade position single axis + slits3_outboard mm outboard blade position single axis + slits3_hsize mm horizontal size coordinated motion + slits3_hcenter mm horizontal center coordinated motion + slits3_vsize mm vertical size coordinated motion + slits3_vcenter mm vertical center coordinated motion + =============== ======== ======================= =================== + + +The individual blades are moved like any other motor, for example:: + + RE(mv(slits3.outboard, -0.5)) + RE(mvr(slits3.top, -0.1)) + + +Coordinated motions are moved the same way, for example:: + + RE(mv(slits3.hsize, 6)) + RE(mvr(slits3.vcenter, -0.1)) + +To know the current positions of the slit blades and their coordinated +motions, use ``%w slits3`` + +.. code-block:: text + + In [1966]: %w slits3 + SLITS3: + vertical size = 1.350 mm Top = 0.675 + vertical center = 0.000 mm Bottom = -0.675 + horizontal size = 8.000 mm Outboard = 4.000 + horizontal center = 0.000 mm Inboard = -4.000 + +Configurations +-------------- + +.. _change-mode: + +Photon delivery modes +~~~~~~~~~~~~~~~~~~~~~ + +A look-up table is used to move the elements of the photon delivery +system to their correct locations for the different energy ranges and +focusing conditions. Here is a table of different photon delivery +modes. Modes A-F are for delivery of light to the XAS end station. +Mode XRD delivers high energy, focused beam to the goniometer. + + +.. _photon_delivery_modes: + +.. table:: Photon delivery modes + :name: pds-modes + :align: left + + ====== ============ ========================= + Mode focused energy range + ====== ============ ========================= + A |checkmark| above 8 keV + B |checkmark| below 6 keV + C |checkmark| 6 keV |nd| 8 keV + D |xmark| above 8 keV + E |xmark| 6 keV |nd| 8 keV + F |xmark| below 6 keV + XRD |checkmark| above 8 keV + ====== ============ ========================= + +.. todo:: The lookup table is not complete for mode B. In fact, the + ydo and ydi jacks of M3 cannot move low enough to enter mode B. In + practice, mode B is not available. Elements that should be + measured in mode B are, instead, measured in mode C and we live + with incomplete harmonic rejection. + + +To move between modes, do:: + + RE(change_mode('')) + +where ```` is one of the strings in the first column of +:numref:`Table %s `. For example:: + + RE(change_mode('D')) + +This will move 17 motors all at the same time and should take less +than 5 minutes. + +Focusing at the XAS end station requires that bender be near its upper +limit. Focusing at the XRD station has the bender near the middle of +its range. + +.. _change-crystals: + +Monochromator crystals +~~~~~~~~~~~~~~~~~~~~~~ + +To change between the Si(111) and Si(311) crystals, do:: + + RE(change_xtals('111')) + +or:: + + RE(change_xtals('311')) + +This will move the lateral motor of the monochromator between the two +crystal sets and adjust the pitch of the second crystal to be nearly +in tune and the roll to deliver the beam to nearly the same location +for both crystals. It will also return the monochromator to the +starting energy. + +This takes about 5 minutes. + +The ``change_xtals()`` plan also runs the :numref:`rocking curve +(Section %s) ` macro to fix the tuning of the +second crystal. + diff --git a/_sources/profile.rst b/_sources/profile.rst new file mode 100644 index 0000000..5e7ec0c --- /dev/null +++ b/_sources/profile.rst @@ -0,0 +1,381 @@ +.. + This document was developed primarily by a NIST employee. Pursuant + to title 17 United States Code Section 105, works of NIST employees + are not subject to copyright protection in the United States. Thus + this repository may not be licensed under the same terms as Bluesky + itself. + + See the LICENSE file for details. + +.. _profile: + +An overview of BMM's profile +============================ + +Once upon a time, the advice from Bluesky's developers was to populate +a ``profile_collection`` folder with python files containing the +beamline customizations. These files would start with a two digit +number from ``00-`` to ``99-`` |nd| that is, files like +``30-detectors.py`` or ``17-motors.py``. These files would be +imported into the main namespace of the IPython session in +numeric order. + +As BMM's profile grew in size and complexity, this system of ordered, +numbered files all imported into the main namespace became +increasingly unweilding, hard to maintain, and hard to extend. + +At Dan Allan's suggestion, we took a different approach. + +The startup folder +------------------ + +The startup folder is a symlink to a folder on Lustre where BMM's +profile is kept. If |bsui| does not start with BMM's profile, make that +symlink by doing the following: + +.. code-block:: sh + + cd ~/.ipython + ln -s /nsls2/data3/bmm/shared/config/bluesky/profile_collection + +If a folder called ``profile_collection`` already exists at that +location |nd| which would be the case if you have already used |bsui| +or ipython |nd| go ahead and rename or delete that folder. + +The top folder of the profile contains a single python file. It +follows the old convention in that it is named +``00-populate-namespace.py`` and is read as soon as IPython starts. + +There are three files in the ``startup`` folder: + +``00-populate-namespace.py`` + This is the first file ipython loads as it starts |bsui|. + :numref:`See Section %s `. + +``rois.json`` + This is a small database of ROI definitions used by the XSpress3 + configuration. + +``user_group_permissions.yaml`` + This is a configuration file used by |qs| to grant access to plans + according to authentication + + + +Beneath the ``startup`` folder, there are several sub-folders: + +``BMM/`` + the bulk of the code used for data acquisition. + +``consumer/`` + the code used by the process that captures Kafka messages and + generates plots for real-time and other data visualization + :numref:`(Section %s) ` + +``BMM_common/`` + code shared by data acquisition and the Kafka consumer + +``lookup_table/`` + the spreadsheet with the look up table of motor positions used by + the ``change_edge()`` command :numref:`(Section %s) ` + +``standards/`` + the spreadsheet for working with `BMM's collection of measurement + standards `__ + +``ML/`` + the data used for the machine learning data evaluation + model :numref:`(Section %s) ` + +``telemetry/`` + the data used for the time estimates of XAFS and other + scans :numref:`(Section %s) ` + +``dossier/`` + files used to construct dossiers :numref:`(Section %s) ` + +``tmpl/`` + template files used for dossiers, INI files, and other + templated materials + +``xlsx/`` + empty spreadsheet examples for the various forms of + automation :numref:`(Section %s) ` + + + + +.. _namespace: + +The IPython namespace +--------------------- + +The ``00-populate-namespace.py`` file contains a single line: + +.. sourcecode:: python + + from BMM.user_ns import * + +This tells python to read the ``BMM/user_ns/__init__.py`` file and +follow it's instructions. That file, in turn, imports each of the +files in the ``BMM/user_ns/`` and imports those symbols into the main +IPython namespace. + +This is different from the old-fashioned approach in that the files in +``BMM/user_ns/`` contain a more carefully curated group of symbols to +be imported into the main namespace. + +Most of the code for creating ophyd objects, defining plans, +establishing automation, and so on is contained in the files found in +the ``BMM`` folder. And most of that is not imported into the main +namespace. + +A motif used in almost every file in the profile is this one: + +.. sourcecode:: python + + from BMM import user_ns as user_ns_module + user_ns = vars(user_ns_module) + +This allows functions and plans defined in the files in the ``BMM`` +folder to have access to symbols from the user namespace without +either importing the entire main namespace or exporting additional +symbols to the main namespace. + +This, perhaps, makes the code in ``BMM`` a bit clunkier. For example, +a motor name like ``xafs_x`` which is defined in the main namespace +cannot be directly accessed by a module in ``BMM``. Instead, it is +accessed as ``user_ns['xafs_x']``. A bit of extra typing, but it +makes for code that is more robust and more readily maintainable and +extensible. + + +Managing |bsui| and |qs| +------------------------ + +A common motif found in many files, including +``BMM/user_ns/__init__.py``, looks like this: + +.. sourcecode:: python + + try: + from bluesky_queueserver import is_re_worker_active + except ImportError: + def is_re_worker_active(): + return False + +This is used to allow a plan or some other bit of code to know whether +it is being run under |bsui| or |qs|. + +|bsui| is, by design, run on the same workstation with which the +experimenter is interacting. |qs| is, by design, run on a +remote server. Having a way to distinguish the two is essential. For +example, there are many plans which, when run with |bsui|, stop to +prompt for an interaction from the user. Such prompts need to be +disabled when running under |qs|. + + +Profile start-up as a narrative +------------------------------- + +Early in the loading of the profile, a function called +``run_report()`` is defined. This function is defined in +``BMM/functions.py`` and called near the top of +``BMM/user_ns/bmm.py``, which is the second file loaded by +``BMM/user_ns/__init__.py``. So, it is defined very early in the +process of loading the profile. + +This is used to write a message to the screen explaining what chore is +being done during start-up or what file is being loaded. As such it +is very similar to the common python idiom of + +.. sourcecode:: python + + print(__file__) + +to identify the module or source code file being loaded. While +similar in concept, it is a bit more suited to our purpose. + +.. _fig-bsui_startup: +.. figure:: _images/bsui_startup.png + :target: _images/bsui_startup.png + :width: 80% + :align: center + + Screen messages during |bsui| start-up. + + +For one thing, it applies consistent coloring to the text. In that +way the user knows that that color is a progress report explaining +what is happening at that moment. That is helpful for debugging +problems in that it gives a hint where to look when the problem +presents itself. + +For another, it is a consistent way to write any progress message to +the screen. For example, + +.. sourcecode:: python + + run_report(__file__) + +would behave very similarly to the idiom. However, the way it is used +throughout the profile is in lines like this: + +.. sourcecode:: python + + run_report('\tglancing angle stage') + +This is the message that appears on screen as the ophyd objects and +automation procedures related to the glancing angle stage +:numref:`(Section %s) ` are imported. As you +can see in the screenshot, one such message is issued for every major +component of the profile. + +Taken together, this sequence of messages provides a start-up +narrative that tells the user something about what capabilities are +available and provides the code maintainer/developer some hints about +where to look in the code base for various features. + + +Profile start-up as acceptance testing +-------------------------------------- + +At NSLS-II, beamline staff are asked to develop ways to do acceptance +testing to verify things like recovery from power failures, or upgrades +of computer operating systems, upgrades of conda and python. + +At BMM, we have chosen *not* to develop one-off or on-delivery +acceptance testing practices. Instead, acceptance testing is built +right into BMM's profile. + +Very early in profile start-up, several basic functions are checked +for, including: + +#. Verify that Channel Access Security is configured for read/write + access +#. Verify that the LAN is up and that IOC servers can be pinged +#. Verify that various necessary folders on the local machine can be + found +#. Verify that Lustre mounts can be found +#. Verify that authentication keys (e.g. for Slack) can be found +#. Verify that a redis server can be found + +If any of these tests fail, the profile stops loading and issues a +(hopefully) useful error message. + +As the profile continues loading, it runs a variety of tests, such as: + +#. Verify that beamline state and user configuration can be obtained + from redis +#. Establish all necessary user configuration +#. Check each axis to verify that it is connected +#. Check each axis to verify that it is homed, or identify those that + are used without homing +#. Verify that detectors are started correctly (e.g. the XSpress3 + needs to save an HDF5 file to initialize file saving) +#. Initialize the hinted ROI from the XSpress3 using data from redis + +In short, the concept is that the profile is instrumented to do +acceptance testing *every time it starts*. If anything is found to be +missing, it can be noticed and addressed immediately. + + +Core Bluesky functionality +-------------------------- + +The file ``BMM/user_ns/base.py`` is used to define core features of +the bluesky ecosystem: + ++ The plan stubs ``mv``, ``mvr``, and ``sleep`` are imported into the + main namespace + ++ `nslsii.configure_base() + `__ + is called appropriately for |bsui| or |qs| + ++ Some configuration of best effort callbacks is done + ++ A `Tiled `__ catalog is created + ++ A ZeroMQ publisher is defined. + +All of this happens before any BMM-specific code is run. + + +Everything in the BMM folder +---------------------------- + +Here's a brief summary of every module in the BMM profile. + + ============================ =================================================== + file purpose + ============================ =================================================== + ``actuators.py`` define shutters and valves + ``agent_plans.py`` BMM-specific ML agents + ``areascan.py`` define an area scan plan + ``busy.py`` define a "wall clock" motor + ``camera_device.py`` interact with AD and non-AD cameras + ``db.py`` utilities for working with *DataBroker* + ``dcm.py`` define monochromator ophyd objects + ``dcm_parameters.py`` mono calibration parameters + ``demeter.py`` |athena| and |hephaestus| integration + ``derivedplot.py`` deprecated plotting capabilities + ``desc_string.py`` fix epics motor ``DESC`` fields for CSS + ``detector_mount.py`` deprecated tools for ``xafs_det`` + ``dossier.py`` manage writing of dossier files + ``dwelltime.py`` coordinate dwell time across detectors + ``edge.py`` change edge + ``electrometer.py`` ophyd objects for QuadEM and related + ``energystep.py`` simple plan for I0 vs. energy + ``fmbo.py`` FMBO motor controller tools + ``frontend.py`` ophyd objects for front-end devices + ``functions.py`` miscellaneous utilities + ``gdrive.py`` :silver:`interact with Google drive (deprecated)` + ``glancing_angle.py`` define glancing angle stage + automation + ``grid.py`` generic grid automation + ``handler.py`` DataBroker handler for images + ``kafka.py`` Kafka producer for profile + ``killswitch.py`` interact with FMBO controller kill switches + ``lakeshore.py`` LakeShore temp. controller + ``larch_interface.py`` connect to Larch XAFS functionality + ``linescans.py`` generic and specific motor scans + ``linkam.py`` Linkam stage temp. controller + ``logging.py`` various logging tools + ``macrobuilder.py`` base class for automation + ``metadata.py`` manage metadata + ``mirror_trigonometry.py`` incomplete tools for mirrors and distances + ``ml.py`` data evaluation tools + ``modes.py`` manage lookup table and PDS modes + ``mono_calibration.py`` energy calibration tools + ``motor_status.py`` motor status reporting tools + ``motors.py`` ophyd objects for motors + ``periodictable.py`` periodic table tools + ``pilatus.py`` Pilatus tools + ``plans.py`` simple plans, power cycle recovery plans + ``prompt.py`` customize IPython prompts + ``purpose.py`` deprecated + ``raster.py`` areascan measurement + dossier + ``resting_state.py`` put beamline in a defined resting state + ``rois.py`` deprecated + ``slits.py`` ophyd objects and tools for BL slits + ``struck.py`` deprecated |nd| interact with Struck + ``suspenders.py`` define suspenders + ``telemetry.py`` tools for scan telemetry and time estimates + ``timescan.py`` time sequence plan + ``usb_camera.py`` interact with USB cameras + ``user.py`` define and manage the BMMuser object + ``utilities.py`` inetract with BL EPS and utilities + ``video.py`` record videos from USB cameras + ``wafer.py`` tools for wafer samples + ``wdywtd.py`` deprecated user help tools + ``webcam_device.py`` interact with Axis webcams + ``wheel.py`` tools for *ex situ* sample wheel + ``workspace.py`` acceptance tests for working environment + ``xafs.py`` XAFS plan definition + dossier + ``xafs_functions.py`` XAFS-related tools + ``xdi.py`` XDI formatting tools + ``xspress3.py`` XSpress3 tools + ``xspress3_1element.py`` customizations for 1 element detector + ``xspress3_4element.py`` customizations for 4 element detector + ``xspress3_7element.py`` customizations for 7 element detector + ============================ =================================================== diff --git a/_sources/proposal.rst b/_sources/proposal.rst new file mode 100644 index 0000000..21bda5c --- /dev/null +++ b/_sources/proposal.rst @@ -0,0 +1,209 @@ +.. + This document was developed primarily by a NIST employee. Pursuant + to title 17 United States Code Section 105, works of NIST employees + are not subject to copyright protection in the United States. Thus + this repository may not be licensed under the same terms as Bluesky + itself. + + See the LICENSE file for details. + +.. _proposal: + +Proposal Writing for BMM +======================== + +XAS Experiments +--------------- + +`Bruce Ravel `__ supports XAS experiments at +BMM. If you have any questions about the capabilities of the beamline +for XAS experiments or questions about an XAS proposal, contact him. + +When writing a proposal for beamtime, it is important to remember that +you are justifying your *need for beamtime*. This, then, is of a +different sort than a funding proposal. While justifying the +importance of your science is certainly a part of the beamtime +proposal, the main purpose is to tell the review panel and the +beamline staff how you intend to use your beamtime effectively. + ++ Explicitly state how many samples you intend to bring. ++ Explicitly state what the differences are between the samples. ++ If you will be measuring under *in situ* conditions |nd| + temperature, electrochemical potential, gas flow, or anything else + |nd| state what those conditions are. ++ If you are measuring a dopant, state the concentration of the dopant. ++ If you are measuring a low concentration sample |nd| for example the + amount of arsenic in a natural sample |nd| provide an estimate of + the concentration. Even better, do elemental analysis of an aliquot + in the electron microscope or with ICP. ++ If you are measuring a thin film sample, state the substrate + material, the thickness of the sample, and the areal dimensions of + the sample. ++ If you have a matrix of sample conditions |nd| for example a range + of dopants, a range of absorption edges, and a range of annealing + conditions |nd| do the math. Tell the reader how many XAS + measurements are required. ++ Make sensible estimates about your time requirements. Be prepared + to prioritize your samples if you are unable to get the full time + you request. + +.. admonition:: Ask questions early! + :class: note + + If you have any questions about beamline capabilities, measurement + times, experimental feasibility, or any other issue pertinent to + your beamtime proposal, contact the beamline staff well before the + proposal deadline. + +XRD Experiments +--------------- + +`Vesna Stanic `__ and `Jean Jordan-Sweet +`__ support scattering experiments at BMM. If you have +any questions about the capabilities of the beamline for scattering +experiments or questions about a scattering proposal, contact them. + +BMM is equipped with a 6-circle psi goniometer, a strip detector, a +small area detector, and point detectors. This installation supports +diffraction in the Bragg geometry and is optimized for reflectivity, +pole figures, truncation rods, and other surface scattering +measurements. + +If you are looking for any +scattering experiment in the Laue geometry, consider applying for time +at `QAS `__, +`PDF `__, +or `XPD +`__. + +To be very clear, `pair distribution function analysis +`__ is not available at BMM. +Any proposal requesting PDF analysis will be denied at feasibility +review. + +.. admonition:: Ask questions early! + :class: note + + If you have any questions about beamline capabilities, measurement + times, experimental feasibility, or any other issue pertinent to + your beamtime proposal, contact the beamline staff well before the + proposal deadline. + + +Multi-modal Experiments +----------------------- + +From time to time, users want to do both spectroscopy and scattering +at BMM. NSLS-II allows a user to request multiple measurement types +in a single proposal. This is called a "multi-modal" experiment. + +However, asking for XAS and XRD in a single proposal at BMM is an +edge-case for the NSLS-II multi-modal program. + +It turns out that there is no way to request a multi-modal proposal +where :maroon:`different` measurement techniques are performed at the +:maroon:`same` beamline. Thus, there is no way to send both parts of the +multi-modal proposal to the relevant proposal review panels. + +As a result, if you wish to request both spectroscopy and scattering +at BMM, you need to write **two proposals**. That is the only way to +route your work through the review system correctly. + +.. caution:: + + A proposal requesting both XAS and XRD at BMM will be denied at + feasibility and a link to this web page with be given as the + reason. + +NIST Partner User Proposals +--------------------------- + +.. note:: + + This section only applies to NIST employees visiting BMM. + +.. caution:: + + If you are not a NIST employee, **do not** submit a PU-P proposal. It + will be deleted without further communication. + +As of the 2024-2 cycle, NIST staff must use the new **Partner User - +Project** proposal type. This new proposal type is necessary to +accommodate :numref:`the new data access and security policies +(Section %s) `. Writing this proposal is a :red:`requirement`. It +is no longer possible to access the beamline via the main partner user +proposal. + +Happily, this proposal is mostly an administrative exercise. It will +not be evaluated by the normal proposal review panel nor by anyone +else. While it is necessary to document what you plan to do with your +beamtime, the text fields do not have to be as complete as they would +for a normally evaluated proposal. + +When filling out your PU-P proposal, be sure to + +#. List all experiments on the "Experimenters" tab. *This is required + so that all the people that will need access to the data will get + access.* +#. Answer all the research screening questions on the "Research" tab. + *This is required to comply with DOE export control regulations.* +#. Make a time request for the appropriate cycle on the "Time Request" + tab. *This is required so that the beamtime can be allocated.* + +As for the text fields on the "Research" tab, you cannot leave them +blank. Briefly describing your experiment |nd| a sentence or two |nd| +is adequate. + +To start, go to https:/pass.bnl.gov and authenticate yourself using +your BNL password and DUO two-factor authentication. Once on the +landing page, start a new proposal as shown in :numref:`Figure %s +`. + +.. _fig-pup-start: +.. figure:: _images/pup-start.png + :target: _images/pup-start.png + :width: 50% + :align: center + + Getting started on a new proposal. + + +When asked for the proposal type, select "Partner User - Project", as +shown in :numref:`Figure %s `. + + +.. _fig-pup-select: +.. figure:: _images/pup-select.png + :target: _images/pup-select.png + :width: 50% + :align: center + + Getting started on a new proposal. + + +To make a time request |nd| either against your new proposal or +against an existing PU-P proposal |nd| go to the "Time Request" tab +and click the "Create New Time Request" button, as +shown in :numref:`Figure %s `. + +.. _fig-pup-btr: +.. figure:: _images/pup-btr.png + :target: _images/pup-btr.png + :width: 50% + :align: center + + Making a beam time request. + +Next, select the cycle in which you will do the experiment, as shown +in :numref:`Figure %s `. Then fill in the text fields +with a sentence or two. + +.. _fig-pup-cycle: +.. figure:: _images/pup-cycle.png + :target: _images/pup-cycle.png + :width: 50% + :align: center + + Selecting the cycle for the beam time request. + +Any questions? Contact `Bruce `__. diff --git a/_sources/restore.rst b/_sources/restore.rst new file mode 100644 index 0000000..7c2f2e6 --- /dev/null +++ b/_sources/restore.rst @@ -0,0 +1,321 @@ +.. + This document was developed primarily by a NIST employee. Pursuant + to title 17 United States Code Section 105, works of NIST employees + are not subject to copyright protection in the United States. Thus + this repository may not be licensed under the same terms as Bluesky + itself. + + See the LICENSE file for details. + +.. _restore: + +Restore beamline after a power outage +===================================== + +.. note:: + + This section was started after the scheduled power shutdown of + April 20, 2024. I am trying to capture the things I did to bring + the beamline back to life that seemed non-obvious or were a source + of friction. YMMV and this section should be expanded in the future. + +Channel Access +-------------- + +By default, channel access security is set such that access to +beamline PVs is disabled. This means that motors cannot be moved, +detectors cannot be triggered, and so on. + +To enable channel access, do the following: + +.. code-block:: bash + + caget XF:06BM-CT{}Prmt:RemoteExp-Sel 1 + +Thus **must** be done as yourself, not as the beamline operator +account. + +If that PV is being reported as disconnected |nd| which is indicated +by the "Workstation Access" buttons on the BMM Main CSS screen or by +the command above returning ``Channel connect timed out`` |nd| then +you need to restart the CAS Switch IOC. + +To do that, ssh to xf06bm-ioc2 as youself (not as the operator +account) and do + +.. code-block:: bash + + dzdo manage-iocs restart cas-switch + +Once that IOC restarts, try again to set +``XF:06BM-CT{}Prmt:RemoteExp-Sel``. + +Redis +----- + +Operations at BMM require that a redis server is running on +xf06bm-ioc2. For whatever reason, the redis server never starts +correctly after a reboot. + +This will become apparent when |bsui| and cadashboard fail to start, +complaining about ``Connection refused`` with ``xf06bm-ioc2:6379``. +6379 is the port that redis uses for communication. + +To fix this, ssh to xf06bm-ioc2 as yourself (not the +user account) and do this command: + +.. code-block:: bash + + dzdo systemctl restart redis + + +Xspress3 +-------- + +To re-power the Xspress3 and its associated server: + +#. Verify that the power button on the back of the Xspress3 unit is + switched on. It should be glowing red. +#. Press the front power button. +#. Once running, restart the relevant IOC on ``xf06bm-ioc2``. For the + seven element detector, use ``xs3-7-1``. + + +Other IOCs +---------- + +The startup acceptance tests in the |bsui| profile may eventually fail +when trying to connect to instruments. For example, this: + +.. code-block:: text + + TimeoutError: XF:06BM-ES:{LINKAM}:MODEL could not connect within 10.0-second timeout. + +indicates that the Linkam controller is powered off and/or the +linkam3 IOC is not running. After verifying power to the instrument, +ssh to xf06bm-ioc2 and do: + +.. code-block:: bash + + dzdo manage-iocs restart linkam3 + +To get a list of all IOCs and their status, do: + +.. code-block:: bash + + manage-iocs status + +Find the name of the relevant IOC and restart it using the +``manage-iocs rastart`` command. + +It sometimes helps to know what port number each IOC is communicating +on: + +.. code-block:: bash + + dzdo manage-iocs report + + +IOCs on xf06bm-ioc2 +~~~~~~~~~~~~~~~~~~~ + +``xf06bm-ioc2`` is the main IOC server at BMM. It is a much beefier +machine than ``xf06bm-ioc1``. + +Here is a list of all the IOCs on ``xf06bm-ioc2`` and what they do: + +================ ================================================= +IOC name purpose +================ ================================================= + axis-caproto5 XRD Axis web camera + axis-caproto6 XAS Axis web camera + cam01 Prosilica camera #1 (DM1) + cam02 Prosilica camera #1 (DM2) + cam03 Prosilica camera #3 (DM3) + cam04 :silver:`??` + cam07 :silver:`??` + cas-switch enables channel access security management + dante Dante controller for Ge detector + diode DIODE controller (filters, spinner stage) + EigerTest1 :silver:`placeholder` + F460 FMBO current monitor (not in use) + flag1 Front end flag (not in use) + I400 FMBO electrometer (not in use) + lakeshore331 LakeShore temperature controller (Displex) + linkam3 Linkam controller + logitechF710 Game controllers + MC01 Collimating mirror + MC02 Monochromator + MC03 Slits2 + MC04 Focusing mirror + MC05 Harmonic rejection mirror and DM1 filters + MC06 DM3 diagnostics and slits3 + MC07 xafs_* motors + MC08 xafs_* motors + MC11 goniometer motors + MC12 goniometer motors + MC13 goniometer motors + mythen1k Mythen (in use??) + omega_i_series ?? + onewire 1Wire temerature sensors near mono + piE625-M2 M2 piezo controller + piE625-M3 M3 piezo controller + piE625-mono mono piezo controller + plc1 PLC IOC + pscdrv ?? + quadEM-1 QuadEM box 1 + quadEM-2 QuadEm box 2 + recsyncIOC ?? + simDetector ?? + va-1 Vacuum controllers and gauges + xf06bmAlarmIOC Alarm server + xs3-8ch :silver:`deprecated XSpress3 server, do not run` + xs3-7-1 XSpress3 server for use with 7-element detector + xs3-4-1 XSpress3 server for use with 4-element detector +================ ================================================= + +IOCs on xf06bm-ioc1 +~~~~~~~~~~~~~~~~~~~ + +Additionally, there is one IOC that regularly runs on ``xf06bm-ioc1``. + +================ ================================================= +IOC name purpose +================ ================================================= + Pilatus100K Pilatus 100k +================ ================================================= + +This IOC does a lot of file I/O, so it seemed like a good idea to +isolate it from the other IOCs. + +All other IOCs on ``xf06bm-ioc1`` must be in the ``stopped`` state. + + +Motor controllers +----------------- + +FMBO MCS8 +~~~~~~~~~ + +Save/restore will not correctly remember motor positions on any opf +the FMBO-supplied axes (i.e. everything except the XAFS and XRD end +stations). + +Restore power to the motor controllers. It should not necessary to +restart the IOCs (MC02 through MC06), but do so if motors are not +moving after powering up the controllers. + +The steps below are the commands in |bsui| for homing sets of axes. The +``ks.cycle()`` steps are not, strictly speaking, necessary. But it is +a good idea to be sure the amplifiers are in a good state. If any +amplifier faults trigger upon starting the homing process, the motors +will be left in a confused state. + + +.. code-block:: python + + ks.cycle('slits2') + RE(recover_slits2()) + + ks.cycle('dm3') + RE(recover_slits3()) + RE(recover_diagnostics()) + + ks.cycle('m2') + RE(recover_m2()) + + ks.cycle('m3') + RE(recover_m3()) + + ks.cycle('dcm') + RE(dcm.recover()) + + +After homing, the monochromator should be at 7134.3 eV, which is an +energy within photon delivery mode E. The mirrors and ``dm3_bct`` +should be at positions consistent with mode E. + +Some of these take quite a while to go through their homing procedure. +The diagnostics recovery takes almost an hour because a couple of the +motors are **very** slow and have a long way to go to hit their limit +switches. + +The M2 bender does not have a homing routine. To verify its +position, move it by hand to its negative limit: + +.. code-block:: python + + RE(mvrbender(-10000)) + +That command is a wrapper around killing the amplifier, then moving by +the specified amount. Feel free to take larger steps. + +Once it hits the negative limit, reset its offset + +.. code-block:: python + + reset_offset(m2_bender, 0) + +then move it back to position and kill the amplifier: + +.. code-block:: python + + RE(mvbender(BMMuser.bender_xas)) + m2_bender.kill() + +For reference, the XAS position for the bender is around 212,000. The +XRD position is around 107,000. + + +.. note:: + + Never home M1, the collimating mirror. It is close enough to the + right position and should never be moved. In fact, there is no + reason to power up the motor controller. + + The fear is that an axis might fail far from the correct position. + + The M1 motor controller is in rack MC7-RG-E4 on the mezzanine. It + is near the bottom and is the only one with FMBO branding. + +Homing MSC8s via PEWIN +~~~~~~~~~~~~~~~~~~~~~~ + +If homing from the |bsui| command line fails, your best bet is to find +the laptop with PEWIN and connect to the motor controller with a USB +cable. + +First, go to xf06bm-ioc2 and stop the reelvant IOC. + +Fire up PEWIN. In the PEWIN command console, issue this command: +``M1x16=1``, where ``x`` is a number from 1 to 8 and indicates the +axis that you want to home. + +You can home multiple axes simultaneously by issuing ``M1x16=1`` +instruction while other axes are in the process of homing. PEWIN is +happy to multitask. + +Note that any axes that involve coordinated motion |nd| mirror +vertical, mirror horizontal, slit vertical or horizontal |nd| work +such that all coordinated axes are triggered for homing when any +individual axis is triggered. For example, to home the M3 vertical +axes, you do not need to do ``M1116=1``, ``M1216=1``, and +``M1316=1``. When you issue any one of those three instructions, all +three axes will begin moving. + + + +geobrick +~~~~~~~~ + +Few or none of the motors on the NSLS-II standard geobricks are +equipped with home or limit switches. This includes the motor +controllers in racks RGC-1 and RGC-2. + +Save/restore should remember positions. + +The IOCs for the ``xafs_*`` controllers (MC07 and MC08) did not need +to be restarted, however all the goniometer controllers (MC11, MC12, +MC13) did. + + diff --git a/_sources/todo.rst b/_sources/todo.rst new file mode 100644 index 0000000..6a909b8 --- /dev/null +++ b/_sources/todo.rst @@ -0,0 +1,64 @@ +.. + This document was developed primarily by a NIST employee. Pursuant + to title 17 United States Code Section 105, works of NIST employees + are not subject to copyright protection in the United States. Thus + this repository may not be licensed under the same terms as Bluesky + itself. + + See the LICENSE file for details. + + +.. role:: strike + :class: strike + +.. _todo_list: + +To Do List +========== + +Chapters needed: + +#. Dealing with XRF spectra, including ``%xrf``, ``xrfat()``, saving + XRF spectra +#. Valence maps +#. Document why M1 is the way it is, explain why moving it is a BAD + idea, explain front end slits + + +This is an aggregate list of things missing from the BMM beamline +documentation manual. + +#. Replace ``db.v2`` with explicit use of Tiled catalog +#. Document use of ``XDI_record`` to control what motors get recorded + in the XDI header +#. Flesh out documentation on using automation + spreadsheets +#. Explain contents of dossier more fully +#. Current batch of cameras +#. Document using the Xspress3 +#. Better abstraction of spreadsheet-to-macro components, less cargo-cult code +#. Document how redis is used at the BL, explain state+redis+BMMuser object +#. Section about instrumentation development plans at BL +#. Document timescan and SEAD, discuss in plotting section + Citation: A Filipponi et al 1998 J. Phys.: Condens. Matter 10 235 DOI: + 10.1088/0953-8984/10/1/026 + + +This is an aggregate list of things mentioned in this document that +the BMM data collection system needs to do. + +#. Mode C lookup table +#. Lookup table for E < 8 keV at goniometer + + +How to add a scan.ini parameter +------------------------------- + +#. Add its entry into ``scan_metadata()`` in :file:`BMM/energyscan.py` +#. Add its default value to the ``BMM_configuration`` class in :file:`BMM/modes.py` +#. Add it to ``html_dict`` in :file:`BMM/energyscan.py` around line 912 +#. Correspondingly, add to the call arguments of + ``scan_sequence_static_html()`` in :file:`BMM/energyscan.py` +#. make corresponding entries in the various automation spreadsheets + +and, of course, use the new parameter in whatever way it needs to be used. + diff --git a/_sources/trouble.rst b/_sources/trouble.rst new file mode 100644 index 0000000..a788310 --- /dev/null +++ b/_sources/trouble.rst @@ -0,0 +1,145 @@ +.. + This document was developed primarily by a NIST employee. Pursuant + to title 17 United States Code Section 105, works of NIST employees + are not subject to copyright protection in the United States. Thus + this repository may not be licensed under the same terms as Bluesky + itself. + + See the LICENSE file for details. + + +Troubleshooting +=============== + +In this section, solutions are given for problems that BMM's visitors +occasionally encounter. + +Pausing BlueSky +--------------- + +There are a small number of ways that you can unintentionally find +yourself outside of BlueSky. One of them is to accidentally hit +``Ctrl-z``, which is unfortunately located close +to ``Ctrl-c``. + +``Ctrl-z`` serves to suspend BlueSky, temporarily +returning you to the Unix command line. It looks like this: + +.. _fig-ctrlz: +.. figure:: _images/ctrl-z.jpg + :target: _images/ctrl-z.jpg + :width: 70% + :align: center + + Accidentally exiting BlueSky and returning to the Unix command line. + +Note that BlueSky said ``Stopped``, then the yellow and magenta +prompt is presented. This indicates that BlueSky is paused rather +than exited. + +To resume BlueSky, type the command ``fg`` and hit ``Enter``. +You will find yourself back at the BlueSky prompt and can carry on +normally. + +Exiting or crashing |bsui| +-------------------------- + +Another possibility is that BlueSky has exited entirely |nd| possibly +because something has happened to put the program into an unworkable +state. This will often be accompanied by a lengthy `stack trace`, +i.e. a bunch of weird code and error messages printed to the terminal +window, followed by the yellow and magenta prompt seen in the picture +above. Other times, there may simply be a message about a +"Segmentation fault" and a "core dump". An example of a segmentation +fault is shown here: + +.. _fig-segfault: +.. figure:: _images/segfault.png + :target: _images/segfault.png + :width: 100% + :align: center + + An example of |bsui| crashing with a segfault + +The previous |bsui| session cannot be recovered at this point. You will +have to restart |bsui| and restart your data collection. + +In this case, at the command line, type ``bsui`` and hit ``Enter``. + +This will start a new BlueSky session and should restore the state of +your experiment from before the crash. + + +Amplifier fault +--------------- + +From time to time, a fault is triggered on one of the motor +amplifiers. The most common examples involve the jacks controlling +the height and pitch of the focusing and harmonic rejection mirrors, +M2 and M3. This is usually observed when trying to use the +``change_edge()`` command (which, among other things, moves the +mirrors to the correct positions). + +The error message on screen will look something like this + +.. todo:: Capture an example of this + +This problem is usually resolved by cycling the kill switch to the amplifiers. + +.. code-block:: python + + ks.cycle('m2') + +where the possible arguments (one for each FMB Oxford motor +controller) are ``m2``, ``m3``, ``slits2``, ``dcm``, or ``dm3``. + +Once that finishes, try doing the thing that triggered the fault +message. Should work.... + + + + +.. + The first solution is to try killing the power to the amplifiers on + the correct MCS8. Switch the corresponding switch to the off + |circle| position, wait at least 10 seconds, then flip the + switch back to the on |verbar| position. Try moving + the motors again. + + .. _fig-killswitch: + .. figure:: _images/Kill_switches.jpg + :target: _images/Kill_switches.jpg + :width: 70% + :align: center + + The MCS8 kill switches on rack D. + + If toggling the switch does not clear the problem, the next solution + to try is to power cycle the appropriate MCS8. You should stop the + corresponding IOC before cycling the power, then restart the IOC + afterwards. Contact Bruce or other beamline staff before doing this. + +Failed hutch search +------------------- + +Sometimes the hutch search fails for mysterious reasons. A likely +cause is that the door `bounced` a bit as it closed. This +confuses the circuit that checks to see that the magnetic latch +holding the door closed is engaged. + +When that (or some other thing out of your control) happens to confuse +the personnel protection system, the search fails and reports the +failure by printing a message in yellow text on the HDMI screen. Here +is what that looks like: + +.. _fig-hdmi: +.. figure:: _images/hdmi.jpg + :target: _images/hdmi.jpg + :width: 70% + :align: center + + The hutch HDMI display showing the yellow text of a failed search. + +When this happens, it is usually sufficient to simply repeat the +search. If the yellow text failure happens again, call the floor +coordinator at extension 5046. diff --git a/_sources/xafs.rst b/_sources/xafs.rst new file mode 100644 index 0000000..d28230b --- /dev/null +++ b/_sources/xafs.rst @@ -0,0 +1,1204 @@ +.. + This document was developed primarily by a NIST employee. Pursuant + to title 17 United States Code Section 105, works of NIST employees + are not subject to copyright protection in the United States. Thus + this repository may not be licensed under the same terms as Bluesky + itself. + + See the LICENSE file for details. + +.. role:: key + :class: key + +.. _xafs: + +XAFS scans +========== + +The simplest access to an XAFS scan starts by editing an `INI file +`_. This is a simple +keyword/value system where the keyword is separated from the value by +an equals sign and each line contains a single keyword/value pair. + +This file captures the set of metadata that the user supplies to +specify the details of the step scan, the basic configuration of the +beamline, and some details about the sample preparation. Most of this +metadata is captured in the beamline database and written to the +header of the output data files. + +This file is typically stored in the same folder where the data files +are written and is called by name when running an XAFS scan. + +Note that the concepts and language used in the INI file is repeated +throughout the spreadsheets used for :numref:`beamline automation +(Section %s) `. + + +.. _ini: + +The INI file +------------ + +.. sourcecode:: ini + :linenos: + + [scan] + filename = cufoil + experimenters = Betty Cooper, Veronica Lodge, Archibald Andrews + + element = Cu + edge = K + sample = Cu metal + prep = standard foil + comment = Welcome to BMM + + nscans = 3 + start = next + + # mode is transmission, fluorescence, both, or reference + mode = transmission + + ## regions relative + ## to e0: 1 2 3 4 + bounds = -200 -30 -10 15.5 15k + steps = 10 2.0 0.3 0.05k + times = 0.5 0.5 0.5 0.25k + + +Here is a complete explanation of the contents of the INI file. + +``filename`` (line 2) + The stub for the names of the output data files and the + snapshots. In the example above, the files written disk will be + called ``cufoil.001``, ``cufoil.002``, and so on. + +``experimenters`` (line 3) + The name of the participants in the measurement. Because all data + and metadata reside in a database, specifying experimenter names + makes it easy to search the database for data associated with a + specific person. + +``element`` (line 5) + The one- or two-letter symbol for the element. The edge energy, + ``e0``, will be looked up using ``element`` and ``edge``. + +``edge`` (line 6) + The symbol (``K``, ``L3``, ``L2``, ``L1``) of the edge being + measured. + +``sample`` (line 7) + This is intended to capture the stoichiometry or composition of the + sample being measured. + +``prep`` (line 8) + This is intended to capture the details of how the sample was + prepared for the XAFS measurement. + +``comment`` (line 9) + This is for anything else you might want to say about your sample. + +``nscans`` and ``start`` (lines 11 - 12) + These are used to form the file extension of the output data file. + They indicate the starting value of the number used as the file + extension and how many repetitions of the scan to make. The file + extension is always a zero-padded, three-digit number, + e.g. :file:`cufoil.001`, :file:`cufoil.002`, and so on. + + The most common value for this parameters is the word ``next``, in + which case the ``folder`` will be searched for files starting with + ``filename`` and ending in a number. The subsequent number will be + used. E.g. if ``cuedge.007`` is the highest numbered file in the + ``cuedge`` sequence, running XAFS again using the same INI file + will start with ``cuedge.008``. + + ``start`` can also be a positive integer, in which case it will be + the first number used in the sequence of scans. So if the value is + 137, the first file will be called ``cufoil.137``. + +``mode`` (line 15) + Indicate how data should be displayed on screen during a scan. The + options are ``transmission``, ``fluorescence``, ``both``, or + ``reference``. ``both`` means to display *both* the transmission + and fluorescence during the scan. + + This parameter also controls what gets written to the output data + files. In all cases, the signals from I0, It, and Ir are written to + the data file. If ``mode`` is ``fluorescence`` or ``both``, then 4 + columns related to the fluorescence detector are also written. The + columns are labeled something like ``Cu1``, ``Cu2``, ``Cu3``, and + ``Cu4`` (where ``Cu`` would be replaced by the symbol of the element + +Comments begin with the hash (``#``) character and are ignored. + + + +Scan regions +~~~~~~~~~~~~ + +In a typical step scan, we measure data on a coarse grid in the +pre-edge, a fine grid through the edge region, and on a constant grid +in photoelectron wavenumber in the extended region. The ``bounds``, +``steps``, and ``times`` keywords (lines 19-21) are used to set this +grid. + + +``bounds`` indicates the energies |nd| relative to the ``e0`` value +|nd| where the step sizes and dwell times will change. There **must** +always be one more value in the ``bounds`` list than in the ``steps`` +and ``times`` lists. + +For the ``bounds`` and ``steps`` lists, values **must** be either a +number or a string consisting of a number followed by the letter +``k``. Numbers followed by ``k`` are interpreted as being values in +photoelectron wavenumber and are only sensible above the edge. + +You may switch back and forth between energy and wavenumber values. +The ``bounds`` and ``steps`` lists are converted to energy values +before being used. + +In the ``bounds`` lists, an energy value indicates an energy below or +above the ``e0`` value. A wavenumber value inidcates a wavenumber +value above the edge. + +In the ``steps`` list, an energy value indicates a step size in eV. A +wavenumber value indicates a step size in |AA|:sup:`-1`. + +In the ``times`` list, a number indicates a dwell time in seconds. A +number followed by ``k`` indicates that the dwell time will grow as a +function of wavenumber above the edge. I.e., a value of ``0.25k`` +means that the dwell time will be 1 second at 4 |AA|:sup:`-1`, 2 +seconds at 8 |AA|:sup:`-1`, and so on. + + +More options +~~~~~~~~~~~~ + +There are several aspects of the XAFS scan plan that can be enabled or +disabled from the INI file. The sample INI file written by the +:numref:`BMMuser.begin_experiment() command (Section %s) ` +does not include these options, but they can be added to the INI file +if needed. + +``e0`` + The edge energy for the element and edge of this measurement. This + is the energy reference for the ``bounds``. Normally, the + tabulated value determined from ``element`` and ``edge`` will be + used. This can be specified to override the tabulated value. + +``usbstick`` + ``True`` will examine the user-supplied filename for characters + that cannot be part of a `filename + `_ + on a standard USB memory stick. If any are found, the filename + will be modified in a way that retains the meaning of the replaced + characters, but which can be successfully written to a memory + stick. Since this is mostly an issue with Windows file systems, + users who want to do their data analysis on a Windows computer + should use this option. :numref:`See Section %s `. + Default: ``True`` + +``snapshots`` + ``True`` to take :numref:`snapshots (Section %s) ` from the + XAS webcam and analog camera before beginning the scan sequence. + ``False`` to skip the snapshots. Default: ``True`` + +``channelcut`` + ``True`` to measure XAFS with the monochromator in pseudo-channelcut + mode. ``False`` to measure in fixed exit mode. Default: ``True`` + +``rockingcurve`` + ``True`` to measure a :numref:`rocking curve scan (Section %s) + ` after moving to the pseudo-channelcut mode + energy. Default: ``False`` + +``bothways`` + ``True`` to measure XAFS in both directions of the monochromator. + ``False`` to always measure in the positive energy (negative angle) + direction. Default: ``False`` + +``htmlpage`` + ``False`` to disable writing of the :numref:`static HTML dossier + (Section %s) `. Default: ``True`` + +``ththth`` + ``True`` to measure with :numref:`Si(333) reflection (Section %s) + ` of the Si(111) monochromator . Default: ``False`` + + +You can explicitly specify a destination folder for the data and other +output files. This is not a great idea, but might be useful in +special situations. The output folder is usually specified +:numref:`when starting an experiment (Section %s) ` and +rarely needs to be changed during the course of an experiment. + +``folder`` + The fully resolved path to the data folder + + +k-weighted integration times +~~~~~~~~~~~~~~~~~~~~~~~~~~~~ + +As discussed above, you can specify k-weighted integration times in +the EXAFS region. While not strictly necessary, it is nice to choose +scan boundaries and integration times that do not result in a +discontinuity in integration time at the transition into the EXAFS +region. + +Here are some suggestions for scan parameters that transition smoothly +between the XANES and EXAFS regions: + +Here are parameters for a 1/2 second base integration time +transitioning into k-weighted integration multiplied by 1/4 (e.g. 2.5 +second integration at k=10). + +.. code-block:: ini + + # 1/2 sec base, k/4 time + bounds = -200 -30 -2 15.5 25 14k + steps = 10 2 0.3 0.3 0.05k + times = 0.5 0.5 0.5 0.25k 0.25k + +Here are parameters for a 1 second base integration time transitioning +into k-weighted integration multiplied by 1/4 (e.g. 2.5 second +integration at k=10). This is Bruce's favorite suggestion for an +experiment needing k-weighted integration time. It's a good balance +between good statistics and reasonable scan time (about 17 minutes). + +.. code-block:: ini + + # 1 sec base, k/2 time + bounds = -200 -30 -2 25 61 14k + steps = 10 2 0.3 0.05k 0.05k + times = 1 1 1 1 0.25k + +Here are parameters for a 1/2 second base integration time +transitioning into k-weighted integration multiplied by 1/2 (e.g. 5 +second integration at k=10). This results in quite long integration +times by the end of the scan, which may be useful for +low-concentration or otherwise noisy EXAFS data.. + +.. code-block:: ini + + # 1/2 sec base, k/2 time + bounds = -200 -30 -2 3.81 25 14k + steps = 10 2 0.3 0.3 0.05k + times = 0.5 0.5 0.5 0.5k 0.5k + +Here are parameters for a 1 second base integration time transitioning +into k-weighted integration multiplied by 1/2 (e.g. 5 second +integration at k=10). This results in quite long integration times by +the end of the scan, which may be useful for low-concentration or +otherwise noisy EXAFS data.. + +.. code-block:: ini + + # 1 sec base, k/2 time + bounds = -200 -30 -2 15.3 25 14k + steps = 10 2 0.3 0.3 0.05k + times = 1 1 1 0.5k 0.5k + + + +.. _howlong: + +Scan run time +------------- + +To get an approximation of the time a scan will take, do:: + + howlong('scan') + +The argument is the path to the INI file described above. Like for +the ``xafs()`` command, the INI file is presumed to be in the user's +data folder and the ``.ini`` need not be specified. It is assumed +that the INI file ends in ``.ini``. + +If you leave off the argument, you will be shown a numbered list of +all :file:`.ini` files in your data folder, something like this: + +.. sourcecode:: text + + Select your INI file: + + 1: Fe.ini + 2: Mn.ini + 3: Zr.ini + 4: scan.ini + + r: return + + Select a file > + +Select number of the :file:`.ini` file you want to read. + +This will make a guess of scan time for an individual scan using a +rather crude heuristic for scan overhead. It will also multiply by +the number of scans to give a total time in hours for the scan +sequence. + +.. code-block:: text + + reading ini file: /home/bravel/BMM_Data/303169/scan.ini + + Each scan will take about 17.9 minutes + The sequence of 6 scans will take about 1.8 hours + + + +.. _usbsafe: + +Safe filenames for USB sticks +----------------------------- + +`These characters are problematic for filenames +`_: + +.. code-block:: text + + ? * / \ % : | " < > + +While there is no issue using these characters in filenames on the +beamline computer, you will find that files containing these names +cannot be written to a normal USB memory stick. The file system used +on many memory sticks (`FAT32 +`__) does +not allow those characters in filenames. This is true even if the +system the memory stick is connected to will allow those characters +(i.e. the beamline linux computer). + + +.. table:: Character translations in filenames + :name: usb-characters + + ================ ================== ======================= + character name character substitution string + ================ ================== ======================= + question mark |mquad| ? ``_QM_`` + asterisk |mquad| |ast| ``_STAR_`` + forward slash |mquad| / ``_SLASH_`` + backslash |mquad| \\ ``_BACKSLASH_`` + percent |mquad| % ``_PERCENT_`` + colon |mquad| : ``_COLON_`` + vertical bar |mquad| |verbar| ``_VERBAR_`` + greater than |mquad| > ``_GT_`` + less than |mquad| < ``_LT_`` + ================ ================== ======================= + + +As an example, a filename like + +.. code-block:: text + + Fe precipitate <60 mM + +will be converted to + +.. code-block:: text + + Fe precipitate _LT_60 mM + +such that the output files will be called + +.. code-block:: text + + Fe precipitate _LT_60 mM.001 + Fe precipitate _LT_60 mM.002 + ... + +Note that spaces are fine in filenames as are all the other keyboard +characters. + + +.. _xafsscan: + +Run an XAFS scan +---------------- + +To run a scan, do this:: + + RE(xafs('scan')) + +The argument is the path to the INI file, as described above. +Specifically, the INI file is assumed to be in the user's data folder +and is assumed to have the ``.ini`` extension. The location of the +user's data folder is set when :numref:`beginning an experiment +(Section %s) `. + +This plan is a wrapper around `BlueSky's scan_nd() plan +`_. +It does the following chores: + +#. Verifies the content of the INI file with a user prompt + +#. Makes an entry in the :numref:`experimental log (Section %s) + ` indicating the INI contents and the current motor + positions of all the important motors + +#. Takes :numref:`snapshots (Section %s) ` of the XAS webcam and + the analog camera near the sample + +#. Moves the monochromator to the center of the angular range of + motion of the scan and enters pseudo-channel-cut mode + +#. If using the Xspress3 to measure fluorescence with the Si-drift + detector, an XRF spectrum will be recorded at that energy. + +#. Generates a plotting subscription appropriate to the value of + ``mode`` in the INI file + +#. Enables a :numref:`set of suspenders (Section %s) ` + which will suspend the current XAFS scan in the event of a beam + dump or a shutter closing (the suspenders are disabled at the end + of the scan sequence) + +#. Moves to the beginning of the scan range and begins taking scans + using the ``scan_nd()`` plan and `cyclers + `_ for energy values and dwell + times constructed from the values of ``bounds``, ``steps``, and + ``times`` read from the INI file + +#. For each scan, notes the start and end times of the scan in the + :numref:`experimental log (Section %s) ` along with the + unique and transient IDs of the scan in the beamline database + +#. After each scan, extracts the data table from the database and writes + an ASCII file in the `XDI format + `_ + +#. After the full sequence of scans, write :numref:`a dossier (Section + %s) ` containing a fairly complete record of the + measurement |nd| including a crude first pass at the data reduction + and processing |nd| made by the XAFS plan. + +The plan also provides some tools to cleanup correctly (i.e. kill +certain motors, reset certain parameters) after a scan sequence ends +or is terminated. + + +Location of scan.ini file +~~~~~~~~~~~~~~~~~~~~~~~~~ + +You may start the XAFS scan by doing:: + + RE(xafs()) + +without specifying an argument. In that case, your data folder will +be searched for INI files and you will presented with an option menu +of the INI files found, as explained in :numref:`Section %s `. + +You may also specify which INI file to use. When you launch an XAFS +scan doing:: + + RE(xafs('myscan')) + +This assumes that there is a file called ``myscan.ini`` in the user's +data directory. Note that you can drop the ``.ini`` |nd| the program +is smart enough to know that you want the ``.ini`` file by that name. +So that is completely equivalent to:: + + RE(xafs('myscan.ini')) + +For instance, if the user's directory (``DATA``) is +``/home/bravel/BMM_Data/303303/``, then the scan plan will look for +the file ``/home/bravel/BMM_Data/303303/scan.ini``. This is +equivalent to:: + + RE(xafs(DATA + 'scan.ini')) + +where ``+`` is the python string concatenation operator. + +You can also explicitly state where your INI file is located, as in:: + + RE(xafs('/home/bravel/BMM_Data/303303/scan.ini')) + +In that case, the explicit location of the INI file will be used. + +The ``DATA`` variable is set when the ``new_experiment()`` command is +run at the beginning of the experiment (:numref:`see Section %s +`). To know the value of the ``DATA`` variable, simply +type ``DATA`` at the command line and hit :key:`Enter`. + + +.. _interrupt: + +Interrupt an XAFS scan +~~~~~~~~~~~~~~~~~~~~~~ + +There are several scenarios where you may need to interrupt or halt an +XAFS scan. + +Pause a scan and *resume* + You can pause a scan at any time by + hitting :key:`Ctrl`-:key:`C` twice. This will return you to + the command line, leaving the scan in a paused state. To *resume* + the scan, do:: + + RE.resume() + + The scan will then continue from where it left off. + +*Stop* a scan + You can pause a scan at any time by hitting + :key:`Ctrl`-:key:`C` twice. This will return you to the + command line, leaving the scan in a paused state. To *end* the + scan, do:: + + RE.stop() + + The scan will then terminate, returning all motors and detectors to + their resting state. + + This will also terminate a paused scan:: + + RE.abort() + + The difference is that ``RE.stop()`` will tag the database entry of + the current scan as ``success`` while ``RE.abort()`` will tag it as + ``failed``. In every other way, the two are equivalent |nd| each + one will shut the scan down gracefully. + +Pause a scan due to external events + When the XAFS scan starts, it initiates a set of `suspenders + `_ + which respond to various external events, such as a shutter closing + or the ring current dumping. When one of these suspenders triggers, + the scan will enter a paused state. It will resume once the + condition causing the suspension is resolved. For example, when the + closed shutter is re-opened or current is restored to the ring. In + general, a short bit of time is required to pass once the suspension + condition is resolved before the scan resumes. For instance, + 5 seconds are allowed to pass after a shutter is re-opened. + +`Here is a summary of pausing, resuming, and stopping scans using +BlueSky +`_. + +Revisit an XAFS scan +-------------------- + +.. admonition:: Needs to be verified + + Does this still work post-data-security? + +Grab a database entry and write it to an XDI file:: + + db2xdi('/path/to/data/file', '') + +The first argument is the name of the output data file. The second +argument is either the scan's unique ID |nd| something like +``f6619ed7-a8e5-41c2-a499-f793b0fcacec`` |nd| or the scan's transient +id number. Both the unique and transient ids can be found in +:numref:`the dossier (Section %s) `. + +.. _macro: + +Scan sequence macro +------------------- + +.. note:: + + Many types of experiments can be automated using the established, + spreadsheet-based systems described in :numref:`Section %s + `. This section is helpful for those situation where + you need to roll your own bespoke automation plans. + +A macro at BMM is a short bit of python code which sequentially moves +motors and initiates scans. A common way of doing this is to make an +INI file for each sample that intend to measure. The macro then moves +to each sample and runs the ``xafs()`` for each sample using the same +INI file. + +.. sourcecode:: python + :linenos: + + def sample_sequence(): + '''User-defined macro for running a sequence of motor motions and + XAFS measurements''' + (ok, text) = BMM_clear_to_start() + if ok is False: + print(error_msg('\n'+text) + bold_msg('Quitting macro....\n')) + return(yield from null()) + + BMMuser.macro_dryrun = False + BMMuser.prompt = False + BMM_log_info('Beginning sample macro') + def main_plan(): + ### --------------------------------------------------------------------------------------- + ### BOILERPLATE ABOVE THIS LINE ----------------------------------------------------------- + ## EDIT BELOW THIS LINE + #<--indentation matters! + + ## sample 1 + yield from slot(1) + yield from xafs('sample1.ini') + close_last_plot() # this command closes the plot on screen + + ## sample 2 + yield from slot(2) + yield from xafs('sample2.ini') + close_last_plot() + + ## EDIT ABOVE THIS LINE + ### BOILERPLATE BELOW THIS LINE ----------------------------------------------------------- + ### --------------------------------------------------------------------------------------- + def cleanup_plan(): + yield from end_of_macro() + + yield from bluesky.preprocessors.finalize_wrapper(main_plan(), cleanup_plan()) + yield from end_of_macro() + BMM_log_info('Sample macro finished!') + +The commented (by ``#``) lines at lines 13-16 and 28-30 are comments +indicating that parts of the macro are intended for editing by the +user while other parts are boilerplate that make the macro work +correctly. In general, you only want to edit the lines between those +two comment blocks, leaving the lines above and below untouched. + +The calls to ``BMM_info()`` at lines 11 and 35 insert lines in the +:numref:`experiment log (Section %s) ` indicating the times that +the scan sequence begins and ends. + +Setting the ``BMMuser.prompt`` parameter to ``False`` at line 9 skips +the step in the ``xafs()`` macro where the user is prompted to verify +that the scan is set up correctly. + +This macro is for samples mounted on the sample wheel. At lines 19 +and 24, the wheel is rotated to the correct slot before launching the +``xafs()`` command. + +Alternately, you can use a single, master :file:`scan.ini` file that +covers all the metadata common to all the samples in a sequence. +Then, as part of the argument to the ``xafs()`` plan, specify those +metadata items specific to the sample. (This has proven to be the more +popular option among BMM users.) + +.. sourcecode:: python + :linenos: + + def sample_sequence(): + '''User-defined macro for running a sequence of motor motions and + XAFS measurements''' + (ok, text) = BMM_clear_to_start() + if ok is False: + print(error_msg('\n'+text) + bold_msg('Quitting macro....\n')) + return(yield from null()) + + BMMuser.macro_dryrun = False + BMMuser.prompt = False + BMM_log_info('Beginning sample macro') + def main_plan(): + ### --------------------------------------------------------------------------------------- + ### BOILERPLATE ABOVE THIS LINE ----------------------------------------------------------- + ## EDIT BELOW THIS LINE + #<--indentation matters! + + ## sample 1 + yield from slot(1) + yield from xafs('scan.ini', filename='samp1', sample='first sample') + close_last_plot() # this command closes the plot on screen + + ## sample 2 + yield from slot(2) + yield from xafs('scan.ini', filename='samp2', sample='another sample', comment='my comment') + close_last_plot() + + ## EDIT ABOVE THIS LINE + ### BOILERPLATE BELOW THIS LINE ----------------------------------------------------------- + ### --------------------------------------------------------------------------------------- + def cleanup_plan(): + yield from end_of_macro() + + yield from bluesky.preprocessors.finalize_wrapper(main_plan(), cleanup_plan()) + yield from end_of_macro() + BMM_log_info('Sample macro finished!') + +:numref:`Any keyword (Section %s) ` from the INI file can be used +as a command argument in the call to ``xafs()``. Arguments to +``xafs()`` will take priority over values in the INI file. + + +Assuming your macro file is stored in your data folder under the name +``macro.py``, you can load or reload the macro into the running +BlueSky session:: + + %run -i BMMuser.data+'macro.py' + +This creates (or overwrites) a new kind of plan called +``sample_sequence()`` (at line 1, you ``def``\ -ine a function of that +name). + +You can then run the macro by invoking the ``sample_sequence()`` +function through the run engine:: + + RE(scan_sequence()) + +Every time you edit the macro file, you **must** reload it into the +running BlueSky session. + +The name of the macro file is not proscribed. If it would be +convenient to have, say, ``macroFe.py`` and ``macroPt.py``, that's +fine. Just be sure to explicitly ``%run -i`` the file using the +correct name. Neither is the name of the command defined in the macro +proscribed. It can be called almost anything (you should avoid +reserved words in Python and names already used for other things in +BlueSky) and run through the run engine (i.e. ``RE()``) like any other +BlueSky plan. + + + + +.. _xdiexample: + + +XAFS data file +-------------- + +XAFS data files are written to the `XDI format +`_. Here is +an example. You can see how the metadata from the INI file and +elsewhere is captured in the output XDI file. + +.. todo:: Document use of ``XDI_record`` dictionary to control which + xafs motors and/or temperatures get recorded in the XDI header + +.. admonition:: New as of Fall 2024 + + There is a new XDI header in use in BMM's datafiles: + ``Scan.xspress3_hdf5_file``. This captures the name of the + associated HDF5 file for fluorescence XAS measurements. + + The value is the path to the asset location beneath the current + proposal folder. + +.. code-block:: text + + # XDI/1.0 BlueSky/1.3.0 + # Beamline.name: BMM (06BM) -- Beamline for Materials Measurement + # Beamline.xray_source: NSLS-II three-pole wiggler + # Beamline.collimation: paraboloid mirror, 5 nm Rh on 30 nm Pt + # Beamline.focusing: torroidal mirror with bender, 5 nm Rh on 30 nm Pt + # Beamline.harmonic_rejection: none + # Detector.I0: 10 cm N2 + # Detector.I1: 25 cm N2 + # Detector.I2: 25 cm N2 + # Detector.fluorescence: SII Vortex ME4 (4-element silicon drift) + # Element.symbol: Mo + # Element.edge: K + # Facility.name: NSLS-II + # Facility.current: 374.3 mA + # Facility.energy: 3.0 GeV + # Facility.mode: top-off + # Facility.GUP: 333333 + # Facility.SAF: 344344 + # Mono.name: Si(311) + # Mono.d_spacing: 1.6376385 Å + # Mono.encoder_resolution: 0.0000050 deg/ct + # Mono.angle_offset: 15.9943932 deg + # Mono.scan_mode: pseudo channel cut + # Mono.scan_type: step + # Mono.direction: forward in energy + # Sample.name: Sedovite + # Sample.prep: speck of mineral in a holder in a gel cap + # Sample.x_position: 2.750 + # Sample.y_position: 147.670 + # Scan.edge_energy: 20000.0 + # Scan.start_time: 2018-07-08T16:26:49 + # Scan.end_time: 2018-07-08T16:44:22 + # Scan.transient_id: 1447 + # Scan.uid: 442bb882-1e46-4607-a12d-1bca2efa74af + # Scan.plot_hint: (DTC1 + DTC2 + DTC3 + DTC4) / I0 -- ($7+$8+$9+$10) / $4 + # Column.1: energy eV + # Column.2: requested_energy eV + # Column.3: measurement_time seconds + # Column.4: I0 nA + # Column.5: It nA + # Column.6: Ir nA + # Column.7: DTC1 + # Column.8: DTC2 + # Column.9: DTC3 + # Column.10: DTC4 + # Column.11: ROI1 counts + # Column.12: ICR1 counts + # Column.13: OCR1 counts + # Column.14: ROI2 counts + # Column.15: ICR2 counts + # Column.16: OCR2 counts + # Column.17: ROI3 counts + # Column.18: ICR3 counts + # Column.19: OCR3 counts + # Column.20: ROI4 counts + # Column.21: ICR4 counts + # Column.22: OCR4 counts + # /////////// + # focused beam, Kyzylsai Dep., Chu-lli Mts., Zhambyl Dist., Kazakhstan 3852 + # ----------- + # energy requested_energy measurement_time I0 It Ir DTC1 DTC2 DTC3 DTC4 ROI1 ICR1 OCR1 ROI2 ICR2 OCR2 ROI3 ICR3 OCR3 ROI4 ICR4 OCR4 + 19809.967 19810.000 0.500 22.780277 28.026418 5.844915 3393.671531 3512.331211 2189.485830 2294.254018 2984.0 86162.0 79706.0 3085.0 86771.0 80213.0 2018.0 57884.0 55169.0 2085.0 64398.0 60757.0 + 19820.016 19820.000 0.500 23.017712 28.316410 5.912596 3607.981130 3515.807498 2272.542220 2255.901234 3160.0 87991.0 81171.0 3088.0 87790.0 81205.0 2093.0 58242.0 55481.0 2036.0 66029.0 61927.0 + 19830.022 19830.000 0.500 23.191409 28.546075 5.971688 3398.408050 3343.071835 2237.827496 2348.453171 2983.0 88018.0 81376.0 2930.0 88064.0 81298.0 2061.0 59218.0 56443.0 2120.0 66896.0 62787.0 + 19840.073 19840.000 0.500 23.022700 28.346179 5.941913 3424.112880 3464.005608 2199.187023 2294.868496 3007.0 87171.0 80589.0 3042.0 87734.0 81137.0 2023.0 58516.0 55684.0 2075.0 66318.0 62324.0 + . + . + . + + +.. _telemetry: + +Telemetry +--------- + +Whenever you run the ``xafs()`` plan or :numref:`import a spreadsheet +(Section %s) ` with the ``xlsx()``, you are given an +estimate of how long it will take. This is estimate is ... pretty +good. Not great, but decent. Here's where it comes from. + +``xafs()`` plan time + For each element, the database is searched for XAFS scans on that + element that ran to completion. The data base record has start and + stop times for the scan as well as a record of point-by-point + integration times. + + For each scan at an element, the sum of integration times is + computed, as is the difference between the end and start times. The + difference between those is the overhead (monochromator movement and + anything else the plan does between the issuing of the start and + stop documents). The average difference is computed and recorded. + + So, the estimated time for an ``xafs()`` plan is the sum of its + integration times plus this historical average of overhead. + +Additional ``xafs()`` plan overhead + The ``xafs()`` plan does a bunch of measurements related to metadata + prior to the start document for the scan being issued. This is a + little harder to compute from the database (certainly not + impossible, but it hasn't yet been worked on), so Bruce has made an + observation through experience to approximate the amount of time + needed to capture photos, measure an XRF spectrum, move to the + :numref:`pseudo-channelcut energy (Section %s) `, etc. When + computing time for a spreadsheet, this is added to the time for each + ``xafs()`` plan run. Until this is measured properly, the value of + ``BMMuser.tweak_xas_time`` is used. + +Changing temperature + For spreadsheets using the Linkam stage or Lakeshore temperature + controller, times for temperature changes are made considering the + ramp rate and the settling time. + +Moving motors, rotating sample wheels + Motor movement for a grid spreadsheet or wheel rotation for wheel + spreadsheet are not considered in the time estimate. The assumption + is that the motors are fast compared to almost everything else. + +Changing edges + For the time estimate in a spreadsheet file, a flat 5 minutes is + used. The range of time for the :numref:`change_edge() (Sample %s) + ` command is about 2.5 minutes when moving between nearby edges + in the same :numref:`photon delivery mode (Table %s) + ` to about 7 minutes for a change between + modes. So this is a source of error in a spreadsheet time estimate. + +Aligning the glancing angle stage + A flat 3 minutes is used to account for the time it takes to do the + automated alignment. + +For a single XAFS scan, the time estimate is the sum of the first two +items in the list above. For a spreadsheet, all applicable items from +the list are added together for each row of the spreadsheet. The +times for each row are added up. + +.. warning:: The time estimate is a good faith estimate. It should be + used as a decent suggestion, but high accuracy should not + be expected! + + +.. _dataevaluation: + +Data evaluation +--------------- + +The thing about :numref:`automation of measurements (Section %s) +` is that the beamline is left unattended for extended +periods. Sometimes things happen at the unattended beamline, +detectors can malfunction, software can get into a weird state, +samples can fall off of sample holders. In short, things can happen +that need human attention and intervention. + +At BMM, we have a sort of a warning system for such things. A machine +agent has been trained to recognize what XAFS data looks like. When a +spectrum is measured that looks like data, i.e. it has an obvious edge +step towards the beginning of the spectrum which is followed by +oscillations, the data evaluator returns a positive result. If the +measurement does not look like that, it returns a negative result. +Examples are shown in :numref:`Figure %s `. + +The result of the data evaluation is printed to the screen. More +importantly, it is posted to :numref:`Slack (Section %s) ` +where it might be seen by the user or the beamline staff. + + +.. subfigure:: AB + :layout-sm: AB + :subcaptions: above + :name: dataeval + :class-grid: outline + + .. image:: _images/good_evaluation.png + + .. image:: _images/bad_evaluation.png + + Examples of data being evaluated as good (left) and bad (right) XAS + data. The data on the left has an obvious edge step followed by + oscillations. It, therefore, looks like XAFS data. The data on + the right is an example of a marginal measurement. There `is` a + step, but it's not very big. Thus it looks like it might be a + problematic measurement. It certainly is something that needs the + attention of a human. + +This machine agent is a trained learning model. It uses a corpus of +data measured at BMM and tagged by Bruce. The corpus includes +hundreds of examples of good spectra and hundreds of examples of +problematic measurements of all sorts. These are human-tagged as such +and trained using a `random forest classifier +`__. +Subsequent spectra are evaluated using this trained classifier. This +evaluation happens upon completion of each XAFS scan repetition. + +Experience so far with this model has been quite good. The training +set is over 98% successful when tested against a subset of the +training corpus. False positives (i.e. bad data identified as being +good data) are exceedingly rare. False negatives (i.e. good data +falsely identified as bad data) are much more common, happening most +days. + +That's a fundamentally useful result. A false negative draws human +attention to the beamline for a situation that might not require it. +Frequent false positives would be much more problematic. + +All negative results are logged so that the training model can be +further refined by having a human tag each those negatives +appropriately and adding them to the training corpus. + +The random forest (RF) classifier was chosen because it is fairly simple +and because it works well. Also tested were K-neighbors (KN) and a +Multi-layer Perceptron (MLP). KN is certainly the simplest of the +models tried |nd| it is the model usually associated with the `classic +iris classification problem +`__. It +actually works quite well, although RF and MLP are both improvements. +MLP was the suggestion of a local machine learning expert and performs +similarly to RF on this trained data corpus. + + +Extract XRF spectra from fluorescence XAS +----------------------------------------- + +BMM offers a handy tool for examining the XRF spectra of a +fluorescence XAS scan on a point-by-point basis. Given the UID of a +scan |nd| which can be found in :numref:`the dossier (Section %s) +` or in :numref:`the header of the data file (Section %s) +` |nd| you can plot the XRF spectrum at a given point in +the scan. + +.. sourcecode:: python + + xrfat(uid, energy) + +Here, ``uid`` is a string containing the scan UID and ``energy`` is +one of the following: + ++ an energy point in the scan range, the nearest energy point will be + used ++ a negative integer, the energy point that many steps from the *end* + of the scan will be used ++ a positive integer (smaller than the first energy value in the + scan), the energy point that many steps from the *beginning* of the + scan will be used ++ a list of any of the above, resulting in XRF spectra from each + energy in the list being over-plotted. + +For example, + +.. sourcecode:: python + + xrfat(uid, -1) + +will plot the XRF spectrum measured at the last point in the scan. + +Here's a good example of why this is useful. Some visitors to BMM +were measuring a sample with a rather low concentration of neodymium +(L\ :sub:`3`\ edge energy of 6208). The |chi|\ (k) data were +noticeably distorted about 330 eV (or about 9.3 inverse Angstrom) +above the edge. This corresponds to the K edge energy (about 6539) of +Mn. We eventually determined that the BN used as a diluant was +slightly contaminated with Mn. + +Here are the plots from below and above the Mn K edge: + +.. sourcecode:: python + + xrfat(uid, 6510) + xrfat(uid, 6560) + +.. subfigure:: AB + :layout-sm: AB + :gap: 8px + :subcaptions: above + :name: Ndsample + :class-grid: outline + + .. image:: _images/Nd-6510.png + + .. image:: _images/Nd-6560.png + + (Left) The XRF spectra from the Nd-bearing sample measured at 6510 eV. + (Right) The XRF spectra from the Nd-bearing sample measured at 6560 eV. + +Those don't look very different. However, overplotting the two +spectra and displaying on a log scale on the y-axis: + +.. _fig-Ndcompare: +.. figure:: _images/Nd-compare.png + :target: _images/Nd-compare.png + :width: 70% + :align: center + + The XRF spectra from the Nd-bearing sample measured at 6510 eV and + at 6560 eV. There is a very small peak at the Mn K\ |alpha| + energy, marked by the green circle. + +While tiny, this Mn contamination had a noticeable impact on the +measured EXAFS data. This sort of forensic work is enabled by the +``xrfat`` command. + +The plots shown in :numref:`Ndsample` can be overplotted using the +list argument of ``energy``, like so: + +.. sourcecode:: python + + xrfat(uid, [6510, 6560]) + + +The full signature of this function is + +.. sourcecode:: python + + def xrfat(uid, energy=-1, xrffile=None, add=True, only=None, xmax=1500): + +where + +``xrffile`` + If not None, is the name of the column data file to be written to + the users ``XRF`` folder. + +``add`` + If True, add the signals from the four channels + +``only`` + If specified as an integer (1, 2, 3, or 4), plot only that detector + channel + +``xmax`` + Specify the maximum energy plotted on the x-axis in units of energy + above the measured fluorescence line energy + +.. _reference-wheel: + +Reference spectra +----------------- + +BMM has a wide variety of reference materials mounted in the reference +position. The collection includes metal foils, metal powders, stable +oxides, or other stable compound of 44 of the elements measurable at +BMM. + +The materials shown and listed below are always available for +measurement. As part of the `command for changing edge +`__, +the reference wheel will rotate to the position of the selected +element. Every XAS scan will include the signal from the I\ :sub:`r` +chamber (whether any signal makes it to that detector depends on the +sample being measured, of course). + +.. _fig-refwheel: +.. figure:: _images/Ref_wheel.jpg + :target: _images/Ref_wheel.jpg + :width: 70% + :align: center + + The reference wheel at BMM + +.. |Gaoxide| replace:: Ga\ :sub:`2`\ O\ :sub:`3` +.. |Geoxide| replace:: GeO\ :sub:`2` +.. |Asoxide| replace:: As\ :sub:`2`\ O\ :sub:`3` +.. |Bioxide| replace:: BiO\ :sub:`2` +.. |Yoxide| replace:: Y\ :sub:`2`\ O\ :sub:`3` +.. |Srtitanate| replace:: SrTiO\ :sub:`3` +.. |Csnitrate| replace:: CsNO\ :sub:`3` +.. |Lahydroxide| replace:: La(OH)\ :sub:`3` +.. |Ceoxide| replace:: Ce\ :sub:`2`\ O\ :sub:`3` +.. |Proxide| replace:: Pr\ :sub:`6`\ O\ :sub:`11` +.. |Ndoxide| replace:: Nd\ :sub:`2`\ O\ :sub:`3` +.. |Smoxide| replace:: Sm\ :sub:`2`\ O\ :sub:`3` +.. |Euoxide| replace:: Eu\ :sub:`2`\ O\ :sub:`3` +.. |Gdoxide| replace:: Gd\ :sub:`2`\ O\ :sub:`3` +.. |Tboxide| replace:: Tb\ :sub:`4`\ O\ :sub:`9` +.. |Dyoxide| replace:: Dy\ :sub:`2`\ O\ :sub:`3` +.. |Hooxide| replace:: Ho\ :sub:`2`\ O\ :sub:`3` +.. |Eroxide| replace:: Er\ :sub:`2`\ O\ :sub:`3` +.. |Tmoxide| replace:: Tm\ :sub:`2`\ O\ :sub:`3` +.. |Yboxide| replace:: Yb\ :sub:`2`\ O\ :sub:`3` +.. |Luoxide| replace:: Lu\ :sub:`2`\ O\ :sub:`3` +.. |Rbcarbonate| replace:: RbCO\ :sub:`3` +.. |Hfoxide| replace:: HfO\ :sub:`2` +.. |Taoxide| replace:: Ta\ :sub:`2`\ O\ :sub:`5` +.. |Reoxide| replace:: ReO\ :sub:`2` +.. |Ruoxide| replace:: RuO\ :sub:`2` + + + +.. table:: Reference wheel contents + :name: tab-reference-wheel + :align: left + + ============ ======== =================== ============ ======== =============== + Ring / slot Element Material Ring / slot Element Material + ============ ======== =================== ============ ======== =============== + Outer 1 empty for alignment Inner 1 Cs |Csnitrate| + Outer 2 Ti foil Inner 2 La |Lahydroxide| + Outer 3 V foil Inner 3 Ce |Ceoxide| + Outer 4 Cr foil Inner 4 Pr |Proxide| + Outer 5 Mn metal powder Inner 5 Nd |Ndoxide| + Outer 6 Fe foil Inner 6 Sm |Smoxide| + Outer 7 Co foil Inner 7 Eu |Euoxide| + Outer 8 Ni foil Inner 8 Gd |Gdoxide| + Outer 9 Cu foil Inner 9 Tb |Tboxide| + Outer 10 Zn foil Inner 10 Dy |Dyoxide| + Outer 11 Ga |Gaoxide| Inner 11 Ho |Hooxide| + Outer 12 Ge |Geoxide| Inner 12 Er |Eroxide| + Outer 13 As |Asoxide| Inner 13 Tm |Tmoxide| + Outer 14 Se metal powder Inner 14 Yb |Yboxide| + Outer 15 Br bromophenol blue Inner 15 Lu |Luoxide| + Outer 16 Zr foil Inner 16 Rb |Rbcarbonate| + Outer 17 Nb foil Inner 17 Ba ** + Outer 18 Mo foil Inner 18 Hf |Hfoxide| + Outer 19 Pt foil Inner 19 Ta |Taoxide| + Outer 20 Au foil Inner 20 W ** + Outer 21 Pb foil Inner 21 Re |Reoxide| + Outer 22 Bi |Bioxide| Inner 22 Os ** + Outer 23 Sr |Srtitanate| Inner 23 Sc metal powder + Outer 24 Y foil Inner 24 Ru |Ruoxide| + ============ ======== =================== ============ ======== =============== + ++ For Th L\ :sub:`3`: Bi\ :sub:`1` will be used (outer 22) ++ For U L\ :sub:`3`: Y K will be used (outer 24) ++ For Pu L\ :sub:`3`: Zr K will be used (outer 16) + +Four elements are missing: Ba, W, & Os, and Ir. + +See also `BMM's complete list of standard materials +`__. + +Here is a `spreadsheet +`__ +for :numref:`automation (see Section %s) ` +with the content of standards wheel. diff --git a/_static/5203_10--32263_131839.pdf b/_static/5203_10--32263_131839.pdf new file mode 100644 index 0000000..d1a0e5b Binary files /dev/null and b/_static/5203_10--32263_131839.pdf differ diff --git a/_static/Binder-instructions.pdf b/_static/Binder-instructions.pdf new file mode 100644 index 0000000..6475354 Binary files /dev/null and b/_static/Binder-instructions.pdf differ diff --git a/_static/HutchSearch.docx b/_static/HutchSearch.docx new file mode 100644 index 0000000..a5e2f8c Binary files /dev/null and b/_static/HutchSearch.docx differ diff --git a/_static/RT-2-11-DIMENSIONS.JPG b/_static/RT-2-11-DIMENSIONS.JPG new file mode 100644 index 0000000..1a0f5e0 Binary files /dev/null and b/_static/RT-2-11-DIMENSIONS.JPG differ diff --git a/_static/basic.css b/_static/basic.css new file mode 100644 index 0000000..6157296 --- /dev/null +++ b/_static/basic.css @@ -0,0 +1,903 @@ +/* + * basic.css + * ~~~~~~~~~ + * + * Sphinx stylesheet -- basic theme. + * + * :copyright: Copyright 2007-2023 by the Sphinx team, see AUTHORS. + * :license: BSD, see LICENSE for details. + * + */ + +/* -- main layout ----------------------------------------------------------- */ + +div.clearer { + clear: both; +} + +div.section::after { + display: block; + content: ''; + clear: left; +} + +/* -- relbar ---------------------------------------------------------------- */ + +div.related { + width: 100%; + font-size: 90%; +} + +div.related h3 { + display: none; +} + +div.related ul { + margin: 0; + padding: 0 0 0 10px; + list-style: none; +} + +div.related li { + display: inline; +} + +div.related li.right { + float: right; + margin-right: 5px; +} + +/* -- sidebar --------------------------------------------------------------- */ + +div.sphinxsidebarwrapper { + padding: 10px 5px 0 10px; +} + +div.sphinxsidebar { + float: left; + width: 270px; + margin-left: -100%; + font-size: 90%; + word-wrap: break-word; + overflow-wrap : break-word; +} + +div.sphinxsidebar ul { + list-style: none; +} + +div.sphinxsidebar ul ul, +div.sphinxsidebar ul.want-points { + margin-left: 20px; + list-style: square; +} + +div.sphinxsidebar ul ul { + margin-top: 0; + margin-bottom: 0; +} + +div.sphinxsidebar form { + margin-top: 10px; +} + +div.sphinxsidebar input { + border: 1px solid #98dbcc; + font-family: sans-serif; + font-size: 1em; +} + +div.sphinxsidebar #searchbox form.search { + overflow: hidden; +} + +div.sphinxsidebar #searchbox input[type="text"] { + float: left; + width: 80%; + padding: 0.25em; + box-sizing: border-box; +} + +div.sphinxsidebar #searchbox input[type="submit"] { + float: left; + width: 20%; + border-left: none; + padding: 0.25em; + box-sizing: border-box; +} + + +img { + border: 0; + max-width: 100%; +} + +/* -- search page ----------------------------------------------------------- */ + +ul.search { + margin: 10px 0 0 20px; + padding: 0; +} + +ul.search li { + padding: 5px 0 5px 20px; + background-image: url(file.png); + background-repeat: no-repeat; + background-position: 0 7px; +} + +ul.search li a { + font-weight: bold; +} + +ul.search li p.context { + color: #888; + margin: 2px 0 0 30px; + text-align: left; +} + +ul.keywordmatches li.goodmatch a { + font-weight: bold; +} + +/* -- index page ------------------------------------------------------------ */ + +table.contentstable { + width: 90%; + margin-left: auto; + margin-right: auto; +} + +table.contentstable p.biglink { + line-height: 150%; +} + +a.biglink { + font-size: 1.3em; +} + +span.linkdescr { + font-style: italic; + padding-top: 5px; + font-size: 90%; +} + +/* -- general index --------------------------------------------------------- */ + +table.indextable { + width: 100%; +} + +table.indextable td { + text-align: left; + vertical-align: top; +} + +table.indextable ul { + margin-top: 0; + margin-bottom: 0; + list-style-type: none; +} + +table.indextable > tbody > tr > td > ul { + padding-left: 0em; +} + +table.indextable tr.pcap { + height: 10px; +} + +table.indextable tr.cap { + margin-top: 10px; + background-color: #f2f2f2; +} + +img.toggler { + margin-right: 3px; + margin-top: 3px; + cursor: pointer; +} + +div.modindex-jumpbox { + border-top: 1px solid #ddd; + border-bottom: 1px solid #ddd; + margin: 1em 0 1em 0; + padding: 0.4em; +} + +div.genindex-jumpbox { + border-top: 1px solid #ddd; + border-bottom: 1px solid #ddd; + margin: 1em 0 1em 0; + padding: 0.4em; +} + +/* -- domain module index --------------------------------------------------- */ + +table.modindextable td { + padding: 2px; + border-collapse: collapse; +} + +/* -- general body styles --------------------------------------------------- */ + +div.body { + min-width: 360px; + max-width: 800px; +} + +div.body p, div.body dd, div.body li, div.body blockquote { + -moz-hyphens: auto; + -ms-hyphens: auto; + -webkit-hyphens: auto; + hyphens: auto; +} + +a.headerlink { + visibility: hidden; +} + +h1:hover > a.headerlink, +h2:hover > a.headerlink, +h3:hover > a.headerlink, +h4:hover > a.headerlink, +h5:hover > a.headerlink, +h6:hover > a.headerlink, +dt:hover > a.headerlink, +caption:hover > a.headerlink, +p.caption:hover > a.headerlink, +div.code-block-caption:hover > a.headerlink { + visibility: visible; +} + +div.body p.caption { + text-align: inherit; +} + +div.body td { + text-align: left; +} + +.first { + margin-top: 0 !important; +} + +p.rubric { + margin-top: 30px; + font-weight: bold; +} + +img.align-left, figure.align-left, .figure.align-left, object.align-left { + clear: left; + float: left; + margin-right: 1em; +} + +img.align-right, figure.align-right, .figure.align-right, object.align-right { + clear: right; + float: right; + margin-left: 1em; +} + +img.align-center, figure.align-center, .figure.align-center, object.align-center { + display: block; + margin-left: auto; + margin-right: auto; +} + +img.align-default, figure.align-default, .figure.align-default { + display: block; + margin-left: auto; + margin-right: auto; +} + +.align-left { + text-align: left; +} + +.align-center { + text-align: center; +} + +.align-default { + text-align: center; +} + +.align-right { + text-align: right; +} + +/* -- sidebars -------------------------------------------------------------- */ + +div.sidebar, +aside.sidebar { + margin: 0 0 0.5em 1em; + border: 1px solid #ddb; + padding: 7px; + background-color: #ffe; + width: 40%; + float: right; + clear: right; + overflow-x: auto; +} + +p.sidebar-title { + font-weight: bold; +} + +nav.contents, +aside.topic, +div.admonition, div.topic, blockquote { + clear: left; +} + +/* -- topics ---------------------------------------------------------------- */ + +nav.contents, +aside.topic, +div.topic { + border: 1px solid #ccc; + padding: 7px; + margin: 10px 0 10px 0; +} + +p.topic-title { + font-size: 1.1em; + font-weight: bold; + margin-top: 10px; +} + +/* -- admonitions ----------------------------------------------------------- */ + +div.admonition { + margin-top: 10px; + margin-bottom: 10px; + padding: 7px; +} + +div.admonition dt { + font-weight: bold; +} + +p.admonition-title { + margin: 0px 10px 5px 0px; + font-weight: bold; +} + +div.body p.centered { + text-align: center; + margin-top: 25px; +} + +/* -- content of sidebars/topics/admonitions -------------------------------- */ + +div.sidebar > :last-child, +aside.sidebar > :last-child, +nav.contents > :last-child, +aside.topic > :last-child, +div.topic > :last-child, +div.admonition > :last-child { + margin-bottom: 0; +} + +div.sidebar::after, +aside.sidebar::after, +nav.contents::after, +aside.topic::after, +div.topic::after, +div.admonition::after, +blockquote::after { + display: block; + content: ''; + clear: both; +} + +/* -- tables ---------------------------------------------------------------- */ + +table.docutils { + margin-top: 10px; + margin-bottom: 10px; + border: 0; + border-collapse: collapse; +} + +table.align-center { + margin-left: auto; + margin-right: auto; +} + +table.align-default { + margin-left: auto; + margin-right: auto; +} + +table caption span.caption-number { + font-style: italic; +} + +table caption span.caption-text { +} + +table.docutils td, table.docutils th { + padding: 1px 8px 1px 5px; + border-top: 0; + border-left: 0; + border-right: 0; + border-bottom: 1px solid #aaa; +} + +th { + text-align: left; + padding-right: 5px; +} + +table.citation { + border-left: solid 1px gray; + margin-left: 1px; +} + +table.citation td { + border-bottom: none; +} + +th > :first-child, +td > :first-child { + margin-top: 0px; +} + +th > :last-child, +td > :last-child { + margin-bottom: 0px; +} + +/* -- figures --------------------------------------------------------------- */ + +div.figure, figure { + margin: 0.5em; + padding: 0.5em; +} + +div.figure p.caption, figcaption { + padding: 0.3em; +} + +div.figure p.caption span.caption-number, +figcaption span.caption-number { + font-style: italic; +} + +div.figure p.caption span.caption-text, +figcaption span.caption-text { +} + +/* -- field list styles ----------------------------------------------------- */ + +table.field-list td, table.field-list th { + border: 0 !important; +} + +.field-list ul { + margin: 0; + padding-left: 1em; +} + +.field-list p { + margin: 0; +} + +.field-name { + -moz-hyphens: manual; + -ms-hyphens: manual; + -webkit-hyphens: manual; + hyphens: manual; +} + +/* -- hlist styles ---------------------------------------------------------- */ + +table.hlist { + margin: 1em 0; +} + +table.hlist td { + vertical-align: top; +} + +/* -- object description styles --------------------------------------------- */ + +.sig { + font-family: 'Consolas', 'Menlo', 'DejaVu Sans Mono', 'Bitstream Vera Sans Mono', monospace; +} + +.sig-name, code.descname { + background-color: transparent; + font-weight: bold; +} + +.sig-name { + font-size: 1.1em; +} + +code.descname { + font-size: 1.2em; +} + +.sig-prename, code.descclassname { + background-color: transparent; +} + +.optional { + font-size: 1.3em; +} + +.sig-paren { + font-size: larger; +} + +.sig-param.n { + font-style: italic; +} + +/* C++ specific styling */ + +.sig-inline.c-texpr, +.sig-inline.cpp-texpr { + font-family: unset; +} + +.sig.c .k, .sig.c .kt, +.sig.cpp .k, .sig.cpp .kt { + color: #0033B3; +} + +.sig.c .m, +.sig.cpp .m { + color: #1750EB; +} + +.sig.c .s, .sig.c .sc, +.sig.cpp .s, .sig.cpp .sc { + color: #067D17; +} + + +/* -- other body styles ----------------------------------------------------- */ + +ol.arabic { + list-style: decimal; +} + +ol.loweralpha { + list-style: lower-alpha; +} + +ol.upperalpha { + list-style: upper-alpha; +} + +ol.lowerroman { + list-style: lower-roman; +} + +ol.upperroman { + list-style: upper-roman; +} + +:not(li) > ol > li:first-child > :first-child, +:not(li) > ul > li:first-child > :first-child { + margin-top: 0px; +} + +:not(li) > ol > li:last-child > :last-child, +:not(li) > ul > li:last-child > :last-child { + margin-bottom: 0px; +} + +ol.simple ol p, +ol.simple ul p, +ul.simple ol p, +ul.simple ul p { + margin-top: 0; +} + +ol.simple > li:not(:first-child) > p, +ul.simple > li:not(:first-child) > p { + margin-top: 0; +} + +ol.simple p, +ul.simple p { + margin-bottom: 0; +} + +aside.footnote > span, +div.citation > span { + float: left; +} +aside.footnote > span:last-of-type, +div.citation > span:last-of-type { + padding-right: 0.5em; +} +aside.footnote > p { + margin-left: 2em; +} +div.citation > p { + margin-left: 4em; +} +aside.footnote > p:last-of-type, +div.citation > p:last-of-type { + margin-bottom: 0em; +} +aside.footnote > p:last-of-type:after, +div.citation > p:last-of-type:after { + content: ""; + clear: both; +} + +dl.field-list { + display: grid; + grid-template-columns: fit-content(30%) auto; +} + +dl.field-list > dt { + font-weight: bold; + word-break: break-word; + padding-left: 0.5em; + padding-right: 5px; +} + +dl.field-list > dd { + padding-left: 0.5em; + margin-top: 0em; + margin-left: 0em; + margin-bottom: 0em; +} + +dl { + margin-bottom: 15px; +} + +dd > :first-child { + margin-top: 0px; +} + +dd ul, dd table { + margin-bottom: 10px; +} + +dd { + margin-top: 3px; + margin-bottom: 10px; + margin-left: 30px; +} + +dl > dd:last-child, +dl > dd:last-child > :last-child { + margin-bottom: 0; +} + +dt:target, span.highlighted { + background-color: #fbe54e; +} + +rect.highlighted { + fill: #fbe54e; +} + +dl.glossary dt { + font-weight: bold; + font-size: 1.1em; +} + +.versionmodified { + font-style: italic; +} + +.system-message { + background-color: #fda; + padding: 5px; + border: 3px solid red; +} + +.footnote:target { + background-color: #ffa; +} + +.line-block { + display: block; + margin-top: 1em; + margin-bottom: 1em; +} + +.line-block .line-block { + margin-top: 0; + margin-bottom: 0; + margin-left: 1.5em; +} + +.guilabel, .menuselection { + font-family: sans-serif; +} + +.accelerator { + text-decoration: underline; +} + +.classifier { + font-style: oblique; +} + +.classifier:before { + font-style: normal; + margin: 0 0.5em; + content: ":"; + display: inline-block; +} + +abbr, acronym { + border-bottom: dotted 1px; + cursor: help; +} + +/* -- code displays --------------------------------------------------------- */ + +pre { + overflow: auto; + overflow-y: hidden; /* fixes display issues on Chrome browsers */ +} + +pre, div[class*="highlight-"] { + clear: both; +} + +span.pre { + -moz-hyphens: none; + -ms-hyphens: none; + -webkit-hyphens: none; + hyphens: none; + white-space: nowrap; +} + +div[class*="highlight-"] { + margin: 1em 0; +} + +td.linenos pre { + border: 0; + background-color: transparent; + color: #aaa; +} + +table.highlighttable { + display: block; +} + +table.highlighttable tbody { + display: block; +} + +table.highlighttable tr { + display: flex; +} + +table.highlighttable td { + margin: 0; + padding: 0; +} + +table.highlighttable td.linenos { + padding-right: 0.5em; +} + +table.highlighttable td.code { + flex: 1; + overflow: hidden; +} + +.highlight .hll { + display: block; +} + +div.highlight pre, +table.highlighttable pre { + margin: 0; +} + +div.code-block-caption + div { + margin-top: 0; +} + +div.code-block-caption { + margin-top: 1em; + padding: 2px 5px; + font-size: small; +} + +div.code-block-caption code { + background-color: transparent; +} + +table.highlighttable td.linenos, +span.linenos, +div.highlight span.gp { /* gp: Generic.Prompt */ + user-select: none; + -webkit-user-select: text; /* Safari fallback only */ + -webkit-user-select: none; /* Chrome/Safari */ + -moz-user-select: none; /* Firefox */ + -ms-user-select: none; /* IE10+ */ +} + +div.code-block-caption span.caption-number { + padding: 0.1em 0.3em; + font-style: italic; +} + +div.code-block-caption span.caption-text { +} + +div.literal-block-wrapper { + margin: 1em 0; +} + +code.xref, a code { + background-color: transparent; + font-weight: bold; +} + +h1 code, h2 code, h3 code, h4 code, h5 code, h6 code { + background-color: transparent; +} + +.viewcode-link { + float: right; +} + +.viewcode-back { + float: right; + font-family: sans-serif; +} + +div.viewcode-block:target { + margin: -1px -10px; + padding: 0 10px; +} + +/* -- math display ---------------------------------------------------------- */ + +img.math { + vertical-align: middle; +} + +div.body div.math p { + text-align: center; +} + +span.eqno { + float: right; +} + +span.eqno a.headerlink { + position: absolute; + z-index: 1; +} + +div.math:hover a.headerlink { + visibility: visible; +} + +/* -- printout stylesheet --------------------------------------------------- */ + +@media print { + div.document, + div.documentwrapper, + div.bodywrapper { + margin: 0 !important; + width: 100%; + } + + div.sphinxsidebar, + div.related, + div.footer, + #top-link { + display: none; + } +} \ No newline at end of file diff --git a/_static/color.css b/_static/color.css new file mode 100644 index 0000000..84b44e2 --- /dev/null +++ b/_static/color.css @@ -0,0 +1,157 @@ +/* + * CSS hacks and small modification for my Sphinx website + * :copyright: Copyright 2013-2016 Lilian Besson + * :license: GPLv3, see LICENSE for details. + */ + + +/* Colors and text decoration. + For example, :black:`text in black` or :blink:`text blinking` in rST. */ + +.brown { + color: brown; +} + +.black { + color: black; +} + +.gray { + color: gray; +} + +.grey { + color: gray; +} + +.silver { + color: silver; +} + +.white { + color: white; +} + +.maroon { + color: maroon; +} + +.red { + color: red; +} + +.magenta { + color: magenta; +} +.hlmagenta { + background-color: magenta; +} + +.fuchsia { + color: fuchsia; +} + +.pink { + color: pink; +} + +.orange { + color: orange; +} + +.yellow { + color: yellow; +} +.hlyellow { + background-color: yellow; +} + +.lime { + color: lime; +} + +.green { + color: green; +} + +.olive { + color: olive; +} + +.teal { + color: teal; +} + +.cyan { + color: cyan; +} +.hlcyan { + background-color: cyan; +} + +.aqua { + color: aqua; +} + +.blue { + color: blue; +} +.hlblue { + color: white; + background-color: blue; +} + +.navy { + color: navy; +} + +.purple { + color: purple; +} + +.under { + text-decoration: underline; +} + +.over { + text-decoration: overline; +} + +.blink { + text-decoration: blink; +} + +.line { + text-decoration: line-through; +} + +.strike { + text-decoration: line-through; +} + +.bolditalic { + font-weight: bold; + font-style: italic; +} + +.smallcaps { + font-variant: small-caps; + color: #8f4800; +} + +.it { + font-style: italic; +} + +.ob { + font-style: oblique; +} + +.small { + font-size: small; +} + +.large { + font-size: large; +} + diff --git a/_static/demeter.css b/_static/demeter.css new file mode 100644 index 0000000..ed91228 --- /dev/null +++ b/_static/demeter.css @@ -0,0 +1,68 @@ +/***********************************************************************/ +/* some css swiped from sphinxtr to get subfigures to display sensibly */ +/***********************************************************************/ + +/* change figure captions to get highlighted when targeted with a link */ +div.figure:target > p.caption, +div.subfigure:target > p.caption, +div.figure:target > center > p.caption, +div.math:target span.eqno { + background-color: #FFA; +} +div.figure:target, +div.subfigure:target { + border: 1px dashed #333; +} +div.figure.compound { + padding: 0; + padding-top: 10px; +} +div.figure.compound > p.caption { + margin-left: 7px; + margin-right: 7px; +} +div.subfigure { + display: inline-block; + vertical-align: top; + padding-left: 7px; + padding-right: 7px; +} + + +cite { + background-color: #2e3434; + color: #eff0f1; + outline-style: double; + outline-color: #2e3434; + margin-left: 5px; + margin-right: 5px; +} + + +/*****************************************/ +/* some color for the caution admonition */ +/*****************************************/ + +div.caution { + background-color: #ffeaea; + border: 1px solid #c00; +} + +div.admonition-note { + background-color: #ffeaea; + border: 1px solid #c00; +} + +.strike { + text-decoration: line-through; +} + +/* code, pre { */ +/* padding: 0; */ +/* } */ + +/* The disappears the damn "Back to top" button from Pydata */ +/* which is not otherwise configurable using html_theme_options */ +button.rounded-pill { + display: none !important; +} diff --git a/_static/doctools.js b/_static/doctools.js new file mode 100644 index 0000000..d06a71d --- /dev/null +++ b/_static/doctools.js @@ -0,0 +1,156 @@ +/* + * doctools.js + * ~~~~~~~~~~~ + * + * Base JavaScript utilities for all Sphinx HTML documentation. + * + * :copyright: Copyright 2007-2023 by the Sphinx team, see AUTHORS. + * :license: BSD, see LICENSE for details. + * + */ +"use strict"; + +const BLACKLISTED_KEY_CONTROL_ELEMENTS = new Set([ + "TEXTAREA", + "INPUT", + "SELECT", + "BUTTON", +]); + +const _ready = (callback) => { + if (document.readyState !== "loading") { + callback(); + } else { + document.addEventListener("DOMContentLoaded", callback); + } +}; + +/** + * Small JavaScript module for the documentation. + */ +const Documentation = { + init: () => { + Documentation.initDomainIndexTable(); + Documentation.initOnKeyListeners(); + }, + + /** + * i18n support + */ + TRANSLATIONS: {}, + PLURAL_EXPR: (n) => (n === 1 ? 0 : 1), + LOCALE: "unknown", + + // gettext and ngettext don't access this so that the functions + // can safely bound to a different name (_ = Documentation.gettext) + gettext: (string) => { + const translated = Documentation.TRANSLATIONS[string]; + switch (typeof translated) { + case "undefined": + return string; // no translation + case "string": + return translated; // translation exists + default: + return translated[0]; // (singular, plural) translation tuple exists + } + }, + + ngettext: (singular, plural, n) => { + const translated = Documentation.TRANSLATIONS[singular]; + if (typeof translated !== "undefined") + return translated[Documentation.PLURAL_EXPR(n)]; + return n === 1 ? singular : plural; + }, + + addTranslations: (catalog) => { + Object.assign(Documentation.TRANSLATIONS, catalog.messages); + Documentation.PLURAL_EXPR = new Function( + "n", + `return (${catalog.plural_expr})` + ); + Documentation.LOCALE = catalog.locale; + }, + + /** + * helper function to focus on search bar + */ + focusSearchBar: () => { + document.querySelectorAll("input[name=q]")[0]?.focus(); + }, + + /** + * Initialise the domain index toggle buttons + */ + initDomainIndexTable: () => { + const toggler = (el) => { + const idNumber = el.id.substr(7); + const toggledRows = document.querySelectorAll(`tr.cg-${idNumber}`); + if (el.src.substr(-9) === "minus.png") { + el.src = `${el.src.substr(0, el.src.length - 9)}plus.png`; + toggledRows.forEach((el) => (el.style.display = "none")); + } else { + el.src = `${el.src.substr(0, el.src.length - 8)}minus.png`; + toggledRows.forEach((el) => (el.style.display = "")); + } + }; + + const togglerElements = document.querySelectorAll("img.toggler"); + togglerElements.forEach((el) => + el.addEventListener("click", (event) => toggler(event.currentTarget)) + ); + togglerElements.forEach((el) => (el.style.display = "")); + if (DOCUMENTATION_OPTIONS.COLLAPSE_INDEX) togglerElements.forEach(toggler); + }, + + initOnKeyListeners: () => { + // only install a listener if it is really needed + if ( + !DOCUMENTATION_OPTIONS.NAVIGATION_WITH_KEYS && + !DOCUMENTATION_OPTIONS.ENABLE_SEARCH_SHORTCUTS + ) + return; + + document.addEventListener("keydown", (event) => { + // bail for input elements + if (BLACKLISTED_KEY_CONTROL_ELEMENTS.has(document.activeElement.tagName)) return; + // bail with special keys + if (event.altKey || event.ctrlKey || event.metaKey) return; + + if (!event.shiftKey) { + switch (event.key) { + case "ArrowLeft": + if (!DOCUMENTATION_OPTIONS.NAVIGATION_WITH_KEYS) break; + + const prevLink = document.querySelector('link[rel="prev"]'); + if (prevLink && prevLink.href) { + window.location.href = prevLink.href; + event.preventDefault(); + } + break; + case "ArrowRight": + if (!DOCUMENTATION_OPTIONS.NAVIGATION_WITH_KEYS) break; + + const nextLink = document.querySelector('link[rel="next"]'); + if (nextLink && nextLink.href) { + window.location.href = nextLink.href; + event.preventDefault(); + } + break; + } + } + + // some keyboard layouts may need Shift to get / + switch (event.key) { + case "/": + if (!DOCUMENTATION_OPTIONS.ENABLE_SEARCH_SHORTCUTS) break; + Documentation.focusSearchBar(); + event.preventDefault(); + } + }); + }, +}; + +// quick alias for translations +const _ = Documentation.gettext; + +_ready(Documentation.init); diff --git a/_static/documentation_options.js b/_static/documentation_options.js new file mode 100644 index 0000000..8fba697 --- /dev/null +++ b/_static/documentation_options.js @@ -0,0 +1,14 @@ +var DOCUMENTATION_OPTIONS = { + URL_ROOT: document.getElementById("documentation_options").getAttribute('data-url_root'), + VERSION: '0.1', + LANGUAGE: 'en', + COLLAPSE_INDEX: false, + BUILDER: 'html', + FILE_SUFFIX: '.html', + LINK_SUFFIX: '.html', + HAS_SOURCE: true, + SOURCELINK_SUFFIX: '', + NAVIGATION_WITH_KEYS: false, + SHOW_SEARCH_SUMMARY: true, + ENABLE_SEARCH_SHORTCUTS: true, +}; \ No newline at end of file diff --git a/_static/file.png b/_static/file.png new file mode 100644 index 0000000..a858a41 Binary files /dev/null and b/_static/file.png differ diff --git a/_static/floor_mat.png b/_static/floor_mat.png new file mode 100644 index 0000000..e661ff8 Binary files /dev/null and b/_static/floor_mat.png differ diff --git a/_static/images/logo_binder.svg b/_static/images/logo_binder.svg new file mode 100644 index 0000000..45fecf7 --- /dev/null +++ b/_static/images/logo_binder.svg @@ -0,0 +1,19 @@ + + + + +logo + + + + + + + + diff --git a/_static/images/logo_colab.png b/_static/images/logo_colab.png new file mode 100644 index 0000000..b7560ec Binary files /dev/null and b/_static/images/logo_colab.png differ diff --git a/_static/images/logo_deepnote.svg b/_static/images/logo_deepnote.svg new file mode 100644 index 0000000..fa77ebf --- /dev/null +++ b/_static/images/logo_deepnote.svg @@ -0,0 +1 @@ + diff --git a/_static/images/logo_jupyterhub.svg b/_static/images/logo_jupyterhub.svg new file mode 100644 index 0000000..60cfe9f --- /dev/null +++ b/_static/images/logo_jupyterhub.svg @@ -0,0 +1 @@ +logo_jupyterhubHub diff --git a/_static/keys.css b/_static/keys.css new file mode 100644 index 0000000..a3a8c5d --- /dev/null +++ b/_static/keys.css @@ -0,0 +1,81 @@ +/** + * KEYS.css + * + * A simple stylesheet for rendering beautiful keyboard-style elements. + * + * Author: Michael Hüneburg + * Website: http://michaelhue.com/keyscss + * License: MIT License (see LICENSE.txt) + */ + +/* Base style, essential for every key. */ +kbd, .key { + display: inline; + display: inline-block; + min-width: 1em; + padding: .2em .3em; + font: normal .85em/1 "Lucida Grande", Lucida, Arial, sans-serif; + text-align: center; + text-decoration: none; + font-weight: bold; + -moz-border-radius: .3em; + -webkit-border-radius: .3em; + border-radius: .3em; + border: none; + cursor: default; + -moz-user-select: none; + -webkit-user-select: none; + user-select: none; +} +kbd[title], .key[title] { + cursor: help; +} + +/* I added 20 to all the dark numbers to make it a little less dark (BR) */ +/* Dark style for display on light background. This is the default style. */ +kbd, kbd.dark, .dark-keys kbd, .key, .key.dark, .dark-keys .key { + background: rgb(100, 100, 100); + background: -moz-linear-gradient(top, rgb(80, 80, 80), rgb(100, 100, 100)); + background: -webkit-gradient(linear, left top, left bottom, from(rgb(80, 80, 80)), to(rgb(100, 100, 100))); + color: rgb(250, 250, 250); + text-shadow: -1px -1px 0 rgb(90, 90, 90); + -moz-box-shadow: inset 0 0 1px rgb(170, 170, 170), inset 0 -.05em .4em rgb(100, 100, 100), 0 .1em 0 rgb(50, 50, 50), 0 .1em .1em rgba(0, 0, 0, .3); + -webkit-box-shadow: inset 0 0 1px rgb(170, 170, 170), inset 0 -.05em .4em rgb(100, 100, 100), 0 .1em 0 rgb(50, 50, 50), 0 .1em .1em rgba(0, 0, 0, .3); + box-shadow: inset 0 0 1px rgb(170, 170, 170), inset 0 -.05em .4em rgb(100, 100, 100), 0 .1em 0 rgb(50, 50, 50), 0 .1em .1em rgba(0, 0, 0, .3); +} + +/* Light style for display on dark background. */ +/* kbd.light, .light-keys kbd, .key.light, .light-keys .key { */ +/* background: rgb(250, 250, 250); */ +/* background: -moz-linear-gradient(top, rgb(210, 210, 210), rgb(255, 255, 255)); */ +/* background: -webkit-gradient(linear, left top, left bottom, from(rgb(210, 210, 210)), to(rgb(255, 255, 255))); */ +/* color: rgb(50, 50, 50); */ +/* text-shadow: 0 0 2px rgb(255, 255, 255); */ +/* -moz-box-shadow: inset 0 0 1px rgb(255, 255, 255), inset 0 0 .4em rgb(200, 200, 200), 0 .1em 0 rgb(130, 130, 130), 0 .11em 0 rgba(0, 0, 0, .4), 0 .1em .11em rgba(0, 0, 0, .9); */ +/* -webkit-box-shadow: inset 0 0 1px rgb(255, 255, 255), inset 0 0 .4em rgb(200, 200, 200), 0 .1em 0 rgb(130, 130, 130), 0 .11em 0 rgba(0, 0, 0, .4), 0 .1em .11em rgba(0, 0, 0, .9); */ +/* box-shadow: inset 0 0 1px rgb(255, 255, 255), inset 0 0 .4em rgb(200, 200, 200), 0 .1em 0 rgb(130, 130, 130), 0 .11em 0 rgba(0, 0, 0, .4), 0 .1em .11em rgba(0, 0, 0, .9); */ +/* } */ + +/* /\* purple and orange styles added for demeter document *\/ */ + +/* kbd.purple, .purple-keys kbd, .key, .key.purple, .purple-keys { */ +/* background: rgb(163, 113, 250); */ +/* background: -moz-linear-gradient(top, rgb(137, 123, 250), rgb(163, 113, 250)); */ +/* background: -webkit-gradient(linear, left top, left bottom, from(rgb(137, 123, 250)), to(rgb(163, 113, 250))); */ +/* color: rgb(250, 250, 250); */ +/* text-shadow: -1px -1px 0 rgb(139, 133, 250); */ +/* -moz-box-shadow: inset 0 0 1px rgb(87, 50, 250), inset 0 -.05em .4em rgb(163, 113, 250), 0 .1em 0 rgb(186, 156, 250), 0 .1em .1em rgba(0, 0, 0, .3); */ +/* -webkit-box-shadow: inset 0 0 1px rgb(87, 50, 250), inset 0 -.05em .4em rgb(163, 113, 250), 0 .1em 0 rgb(186, 156, 250), 0 .1em .1em rgba(0, 0, 0, .3); */ +/* box-shadow: inset 0 0 1px rgb(87, 50, 250), inset 0 -.05em .4em rgb(163, 113, 250), 0 .1em 0 rgb(186, 156, 250), 0 .1em .1em rgba(0, 0, 0, .3); */ +/* } */ + +/* kbd.orange, .orange-keys kbd, .key, .key.orange, .orange-keys { */ +/* background: rgb(252, 221, 159); */ +/* background: -moz-linear-gradient(top, rgb(252, 227, 182), rgb(252, 221, 159)); */ +/* background: -webkit-gradient(linear, left top, left bottom, from(rgb(252, 227, 182)), to(rgb(252, 221, 159))); */ +/* color: rgb(50, 50, 50); */ +/* text-shadow: -1px -1px 0 rgb(252, 185, 129); */ +/* -moz-box-shadow: inset 0 0 1px rgb(252, 153, 91), inset 0 -.05em .4em rgb(252, 221, 159), 0 .1em 0 rgb(252, 173, 53), 0 .1em .1em rgba(0, 0, 0, .3); */ +/* -webkit-box-shadow: inset 0 0 1px rgb(252, 153, 91), inset 0 -.05em .4em rgb(252, 221, 159), 0 .1em 0 rgb(252, 173, 53), 0 .1em .1em rgba(0, 0, 0, .3); */ +/* box-shadow: inset 0 0 1px rgb(252, 153, 91), inset 0 -.05em .4em rgb(252, 221, 159), 0 .1em 0 rgb(252, 173, 53), 0 .1em .1em rgba(0, 0, 0, .3); */ +/* } */ diff --git a/_static/language_data.js b/_static/language_data.js new file mode 100644 index 0000000..250f566 --- /dev/null +++ b/_static/language_data.js @@ -0,0 +1,199 @@ +/* + * language_data.js + * ~~~~~~~~~~~~~~~~ + * + * This script contains the language-specific data used by searchtools.js, + * namely the list of stopwords, stemmer, scorer and splitter. + * + * :copyright: Copyright 2007-2023 by the Sphinx team, see AUTHORS. + * :license: BSD, see LICENSE for details. + * + */ + +var stopwords = ["a", "and", "are", "as", "at", "be", "but", "by", "for", "if", "in", "into", "is", "it", "near", "no", "not", "of", "on", "or", "such", "that", "the", "their", "then", "there", "these", "they", "this", "to", "was", "will", "with"]; + + +/* Non-minified version is copied as a separate JS file, is available */ + +/** + * Porter Stemmer + */ +var Stemmer = function() { + + var step2list = { + ational: 'ate', + tional: 'tion', + enci: 'ence', + anci: 'ance', + izer: 'ize', + bli: 'ble', + alli: 'al', + entli: 'ent', + eli: 'e', + ousli: 'ous', + ization: 'ize', + ation: 'ate', + ator: 'ate', + alism: 'al', + iveness: 'ive', + fulness: 'ful', + ousness: 'ous', + aliti: 'al', + iviti: 'ive', + biliti: 'ble', + logi: 'log' + }; + + var step3list = { + icate: 'ic', + ative: '', + alize: 'al', + iciti: 'ic', + ical: 'ic', + ful: '', + ness: '' + }; + + var c = "[^aeiou]"; // consonant + var v = "[aeiouy]"; // vowel + var C = c + "[^aeiouy]*"; // consonant sequence + var V = v + "[aeiou]*"; // vowel sequence + + var mgr0 = "^(" + C + ")?" + V + C; // [C]VC... is m>0 + var meq1 = "^(" + C + ")?" + V + C + "(" + V + ")?$"; // [C]VC[V] is m=1 + var mgr1 = "^(" + C + ")?" + V + C + V + C; // [C]VCVC... is m>1 + var s_v = "^(" + C + ")?" + v; // vowel in stem + + this.stemWord = function (w) { + var stem; + var suffix; + var firstch; + var origword = w; + + if (w.length < 3) + return w; + + var re; + var re2; + var re3; + var re4; + + firstch = w.substr(0,1); + if (firstch == "y") + w = firstch.toUpperCase() + w.substr(1); + + // Step 1a + re = /^(.+?)(ss|i)es$/; + re2 = /^(.+?)([^s])s$/; + + if (re.test(w)) + w = w.replace(re,"$1$2"); + else if (re2.test(w)) + w = w.replace(re2,"$1$2"); + + // Step 1b + re = /^(.+?)eed$/; + re2 = /^(.+?)(ed|ing)$/; + if (re.test(w)) { + var fp = re.exec(w); + re = new RegExp(mgr0); + if (re.test(fp[1])) { + re = /.$/; + w = w.replace(re,""); + } + } + else if (re2.test(w)) { + var fp = re2.exec(w); + stem = fp[1]; + re2 = new RegExp(s_v); + if (re2.test(stem)) { + w = stem; + re2 = /(at|bl|iz)$/; + re3 = new RegExp("([^aeiouylsz])\\1$"); + re4 = new RegExp("^" + C + v + "[^aeiouwxy]$"); + if (re2.test(w)) + w = w + "e"; + else if (re3.test(w)) { + re = /.$/; + w = w.replace(re,""); + } + else if (re4.test(w)) + w = w + "e"; + } + } + + // Step 1c + re = /^(.+?)y$/; + if (re.test(w)) { + var fp = re.exec(w); + stem = fp[1]; + re = new RegExp(s_v); + if (re.test(stem)) + w = stem + "i"; + } + + // Step 2 + re = /^(.+?)(ational|tional|enci|anci|izer|bli|alli|entli|eli|ousli|ization|ation|ator|alism|iveness|fulness|ousness|aliti|iviti|biliti|logi)$/; + if (re.test(w)) { + var fp = re.exec(w); + stem = fp[1]; + suffix = fp[2]; + re = new RegExp(mgr0); + if (re.test(stem)) + w = stem + step2list[suffix]; + } + + // Step 3 + re = /^(.+?)(icate|ative|alize|iciti|ical|ful|ness)$/; + if (re.test(w)) { + var fp = re.exec(w); + stem = fp[1]; + suffix = fp[2]; + re = new RegExp(mgr0); + if (re.test(stem)) + w = stem + step3list[suffix]; + } + + // Step 4 + re = /^(.+?)(al|ance|ence|er|ic|able|ible|ant|ement|ment|ent|ou|ism|ate|iti|ous|ive|ize)$/; + re2 = /^(.+?)(s|t)(ion)$/; + if (re.test(w)) { + var fp = re.exec(w); + stem = fp[1]; + re = new RegExp(mgr1); + if (re.test(stem)) + w = stem; + } + else if (re2.test(w)) { + var fp = re2.exec(w); + stem = fp[1] + fp[2]; + re2 = new RegExp(mgr1); + if (re2.test(stem)) + w = stem; + } + + // Step 5 + re = /^(.+?)e$/; + if (re.test(w)) { + var fp = re.exec(w); + stem = fp[1]; + re = new RegExp(mgr1); + re2 = new RegExp(meq1); + re3 = new RegExp("^" + C + v + "[^aeiouwxy]$"); + if (re.test(stem) || (re2.test(stem) && !(re3.test(stem)))) + w = stem; + } + re = /ll$/; + re2 = new RegExp(mgr1); + if (re.test(w) && re2.test(w)) { + re = /.$/; + w = w.replace(re,""); + } + + // and turn initial Y back to y + if (firstch == "y") + w = firstch.toLowerCase() + w.substr(1); + return w; + } +} + diff --git a/_static/leftclick.png b/_static/leftclick.png new file mode 100644 index 0000000..a9f559f Binary files /dev/null and b/_static/leftclick.png differ diff --git a/_static/linksbox.html b/_static/linksbox.html new file mode 100644 index 0000000..a24a89f --- /dev/null +++ b/_static/linksbox.html @@ -0,0 +1,13 @@ + diff --git a/_static/locales/ar/LC_MESSAGES/booktheme.mo b/_static/locales/ar/LC_MESSAGES/booktheme.mo new file mode 100644 index 0000000..15541a6 Binary files /dev/null and b/_static/locales/ar/LC_MESSAGES/booktheme.mo differ diff --git a/_static/locales/ar/LC_MESSAGES/booktheme.po b/_static/locales/ar/LC_MESSAGES/booktheme.po new file mode 100644 index 0000000..34d404c --- /dev/null +++ b/_static/locales/ar/LC_MESSAGES/booktheme.po @@ -0,0 +1,75 @@ + +msgid "" +msgstr "" +"Project-Id-Version: Sphinx-Book-Theme\n" +"MIME-Version: 1.0\n" +"Content-Type: text/plain; charset=UTF-8\n" +"Content-Transfer-Encoding: 8bit\n" +"Language: ar\n" +"Plural-Forms: nplurals=2; plural=(n != 1);\n" + +msgid "Print to PDF" +msgstr "طباعة إلى PDF" + +msgid "Theme by the" +msgstr "موضوع بواسطة" + +msgid "Download source file" +msgstr "تنزيل ملف المصدر" + +msgid "open issue" +msgstr "قضية مفتوحة" + +msgid "Contents" +msgstr "محتويات" + +msgid "previous page" +msgstr "الصفحة السابقة" + +msgid "Download notebook file" +msgstr "تنزيل ملف دفتر الملاحظات" + +msgid "Copyright" +msgstr "حقوق النشر" + +msgid "Download this page" +msgstr "قم بتنزيل هذه الصفحة" + +msgid "Source repository" +msgstr "مستودع المصدر" + +msgid "By" +msgstr "بواسطة" + +msgid "repository" +msgstr "مخزن" + +msgid "Last updated on" +msgstr "آخر تحديث في" + +msgid "Toggle navigation" +msgstr "تبديل التنقل" + +msgid "Sphinx Book Theme" +msgstr "موضوع كتاب أبو الهول" + +msgid "suggest edit" +msgstr "أقترح تحرير" + +msgid "Open an issue" +msgstr "افتح قضية" + +msgid "Launch" +msgstr "إطلاق" + +msgid "Fullscreen mode" +msgstr "وضع ملء الشاشة" + +msgid "Edit this page" +msgstr "قم بتحرير هذه الصفحة" + +msgid "By the" +msgstr "بواسطة" + +msgid "next page" +msgstr "الصفحة التالية" diff --git a/_static/locales/bg/LC_MESSAGES/booktheme.mo b/_static/locales/bg/LC_MESSAGES/booktheme.mo new file mode 100644 index 0000000..da95120 Binary files /dev/null and b/_static/locales/bg/LC_MESSAGES/booktheme.mo differ diff --git a/_static/locales/bg/LC_MESSAGES/booktheme.po b/_static/locales/bg/LC_MESSAGES/booktheme.po new file mode 100644 index 0000000..7420c19 --- /dev/null +++ b/_static/locales/bg/LC_MESSAGES/booktheme.po @@ -0,0 +1,75 @@ + +msgid "" +msgstr "" +"Project-Id-Version: Sphinx-Book-Theme\n" +"MIME-Version: 1.0\n" +"Content-Type: text/plain; charset=UTF-8\n" +"Content-Transfer-Encoding: 8bit\n" +"Language: bg\n" +"Plural-Forms: nplurals=2; plural=(n != 1);\n" + +msgid "Print to PDF" +msgstr "Печат в PDF" + +msgid "Theme by the" +msgstr "Тема от" + +msgid "Download source file" +msgstr "Изтеглете изходния файл" + +msgid "open issue" +msgstr "отворен брой" + +msgid "Contents" +msgstr "Съдържание" + +msgid "previous page" +msgstr "предишна страница" + +msgid "Download notebook file" +msgstr "Изтеглете файла на бележника" + +msgid "Copyright" +msgstr "Авторско право" + +msgid "Download this page" +msgstr "Изтеглете тази страница" + +msgid "Source repository" +msgstr "Хранилище на източника" + +msgid "By" +msgstr "От" + +msgid "repository" +msgstr "хранилище" + +msgid "Last updated on" +msgstr "Последна актуализация на" + +msgid "Toggle navigation" +msgstr "Превключване на навигацията" + +msgid "Sphinx Book Theme" +msgstr "Тема на книгата Sphinx" + +msgid "suggest edit" +msgstr "предложи редактиране" + +msgid "Open an issue" +msgstr "Отворете проблем" + +msgid "Launch" +msgstr "Стартиране" + +msgid "Fullscreen mode" +msgstr "Режим на цял екран" + +msgid "Edit this page" +msgstr "Редактирайте тази страница" + +msgid "By the" +msgstr "По" + +msgid "next page" +msgstr "Следваща страница" diff --git a/_static/locales/bn/LC_MESSAGES/booktheme.mo b/_static/locales/bn/LC_MESSAGES/booktheme.mo new file mode 100644 index 0000000..6b96639 Binary files /dev/null and b/_static/locales/bn/LC_MESSAGES/booktheme.mo differ diff --git a/_static/locales/bn/LC_MESSAGES/booktheme.po b/_static/locales/bn/LC_MESSAGES/booktheme.po new file mode 100644 index 0000000..63a07c3 --- /dev/null +++ b/_static/locales/bn/LC_MESSAGES/booktheme.po @@ -0,0 +1,63 @@ + +msgid "" +msgstr "" +"Project-Id-Version: Sphinx-Book-Theme\n" +"MIME-Version: 1.0\n" +"Content-Type: text/plain; charset=UTF-8\n" +"Content-Transfer-Encoding: 8bit\n" +"Language: bn\n" +"Plural-Forms: nplurals=2; plural=(n != 1);\n" + +msgid "Print to PDF" +msgstr "পিডিএফ প্রিন্ট করুন" + +msgid "Theme by the" +msgstr "থিম দ্বারা" + +msgid "Download source file" +msgstr "উত্স ফাইল ডাউনলোড করুন" + +msgid "open issue" +msgstr "খোলা সমস্যা" + +msgid "previous page" +msgstr "আগের পৃষ্ঠা" + +msgid "Download notebook file" +msgstr "নোটবুক ফাইল ডাউনলোড করুন" + +msgid "Copyright" +msgstr "কপিরাইট" + +msgid "Download this page" +msgstr "এই পৃষ্ঠাটি ডাউনলোড করুন" + +msgid "Source repository" +msgstr "উত্স সংগ্রহস্থল" + +msgid "By" +msgstr "দ্বারা" + +msgid "Last updated on" +msgstr "সর্বশেষ আপডেট" + +msgid "Toggle navigation" +msgstr "নেভিগেশন টগল করুন" + +msgid "Sphinx Book Theme" +msgstr "স্পিনিক্স বুক থিম" + +msgid "Open an issue" +msgstr "একটি সমস্যা খুলুন" + +msgid "Launch" +msgstr "শুরু করা" + +msgid "Edit this page" +msgstr "এই পৃষ্ঠাটি সম্পাদনা করুন" + +msgid "By the" +msgstr "দ্বারা" + +msgid "next page" +msgstr "পরবর্তী পৃষ্ঠা" diff --git a/_static/locales/ca/LC_MESSAGES/booktheme.mo b/_static/locales/ca/LC_MESSAGES/booktheme.mo new file mode 100644 index 0000000..a4dd30e Binary files /dev/null and b/_static/locales/ca/LC_MESSAGES/booktheme.mo differ diff --git a/_static/locales/ca/LC_MESSAGES/booktheme.po b/_static/locales/ca/LC_MESSAGES/booktheme.po new file mode 100644 index 0000000..8fb358b --- /dev/null +++ b/_static/locales/ca/LC_MESSAGES/booktheme.po @@ -0,0 +1,66 @@ + +msgid "" +msgstr "" +"Project-Id-Version: Sphinx-Book-Theme\n" +"MIME-Version: 1.0\n" +"Content-Type: text/plain; charset=UTF-8\n" +"Content-Transfer-Encoding: 8bit\n" +"Language: ca\n" +"Plural-Forms: nplurals=2; plural=(n != 1);\n" + +msgid "Print to PDF" +msgstr "Imprimeix a PDF" + +msgid "Theme by the" +msgstr "Tema del" + +msgid "Download source file" +msgstr "Baixeu el fitxer font" + +msgid "open issue" +msgstr "número obert" + +msgid "previous page" +msgstr "Pàgina anterior" + +msgid "Download notebook file" +msgstr "Descarregar fitxer de quadern" + +msgid "Copyright" +msgstr "Copyright" + +msgid "Download this page" +msgstr "Descarregueu aquesta pàgina" + +msgid "Source repository" +msgstr "Dipòsit de fonts" + +msgid "By" +msgstr "Per" + +msgid "Last updated on" +msgstr "Darrera actualització el" + +msgid "Toggle navigation" +msgstr "Commuta la navegació" + +msgid "Sphinx Book Theme" +msgstr "Tema del llibre Esfinx" + +msgid "suggest edit" +msgstr "suggerir edició" + +msgid "Open an issue" +msgstr "Obriu un número" + +msgid "Launch" +msgstr "Llançament" + +msgid "Edit this page" +msgstr "Editeu aquesta pàgina" + +msgid "By the" +msgstr "Per la" + +msgid "next page" +msgstr "pàgina següent" diff --git a/_static/locales/cs/LC_MESSAGES/booktheme.mo b/_static/locales/cs/LC_MESSAGES/booktheme.mo new file mode 100644 index 0000000..c39e01a Binary files /dev/null and b/_static/locales/cs/LC_MESSAGES/booktheme.mo differ diff --git a/_static/locales/cs/LC_MESSAGES/booktheme.po b/_static/locales/cs/LC_MESSAGES/booktheme.po new file mode 100644 index 0000000..c6ef469 --- /dev/null +++ b/_static/locales/cs/LC_MESSAGES/booktheme.po @@ -0,0 +1,75 @@ + +msgid "" +msgstr "" +"Project-Id-Version: Sphinx-Book-Theme\n" +"MIME-Version: 1.0\n" +"Content-Type: text/plain; charset=UTF-8\n" +"Content-Transfer-Encoding: 8bit\n" +"Language: cs\n" +"Plural-Forms: nplurals=2; plural=(n != 1);\n" + +msgid "Print to PDF" +msgstr "Tisk do PDF" + +msgid "Theme by the" +msgstr "Téma od" + +msgid "Download source file" +msgstr "Stáhněte si zdrojový soubor" + +msgid "open issue" +msgstr "otevřené číslo" + +msgid "Contents" +msgstr "Obsah" + +msgid "previous page" +msgstr "předchozí stránka" + +msgid "Download notebook file" +msgstr "Stáhnout soubor poznámkového bloku" + +msgid "Copyright" +msgstr "autorská práva" + +msgid "Download this page" +msgstr "Stáhněte si tuto stránku" + +msgid "Source repository" +msgstr "Zdrojové úložiště" + +msgid "By" +msgstr "Podle" + +msgid "repository" +msgstr "úložiště" + +msgid "Last updated on" +msgstr "Naposledy aktualizováno" + +msgid "Toggle navigation" +msgstr "Přepnout navigaci" + +msgid "Sphinx Book Theme" +msgstr "Téma knihy Sfinga" + +msgid "suggest edit" +msgstr "navrhnout úpravy" + +msgid "Open an issue" +msgstr "Otevřete problém" + +msgid "Launch" +msgstr "Zahájení" + +msgid "Fullscreen mode" +msgstr "Režim celé obrazovky" + +msgid "Edit this page" +msgstr "Upravit tuto stránku" + +msgid "By the" +msgstr "Podle" + +msgid "next page" +msgstr "další strana" diff --git a/_static/locales/da/LC_MESSAGES/booktheme.mo b/_static/locales/da/LC_MESSAGES/booktheme.mo new file mode 100644 index 0000000..f43157d Binary files /dev/null and b/_static/locales/da/LC_MESSAGES/booktheme.mo differ diff --git a/_static/locales/da/LC_MESSAGES/booktheme.po b/_static/locales/da/LC_MESSAGES/booktheme.po new file mode 100644 index 0000000..306a38e --- /dev/null +++ b/_static/locales/da/LC_MESSAGES/booktheme.po @@ -0,0 +1,75 @@ + +msgid "" +msgstr "" +"Project-Id-Version: Sphinx-Book-Theme\n" +"MIME-Version: 1.0\n" +"Content-Type: text/plain; charset=UTF-8\n" +"Content-Transfer-Encoding: 8bit\n" +"Language: da\n" +"Plural-Forms: nplurals=2; plural=(n != 1);\n" + +msgid "Print to PDF" +msgstr "Udskriv til PDF" + +msgid "Theme by the" +msgstr "Tema af" + +msgid "Download source file" +msgstr "Download kildefil" + +msgid "open issue" +msgstr "åbent nummer" + +msgid "Contents" +msgstr "Indhold" + +msgid "previous page" +msgstr "forrige side" + +msgid "Download notebook file" +msgstr "Download notesbog-fil" + +msgid "Copyright" +msgstr "ophavsret" + +msgid "Download this page" +msgstr "Download denne side" + +msgid "Source repository" +msgstr "Kildelager" + +msgid "By" +msgstr "Ved" + +msgid "repository" +msgstr "lager" + +msgid "Last updated on" +msgstr "Sidst opdateret den" + +msgid "Toggle navigation" +msgstr "Skift navigation" + +msgid "Sphinx Book Theme" +msgstr "Sphinx bogtema" + +msgid "suggest edit" +msgstr "foreslå redigering" + +msgid "Open an issue" +msgstr "Åbn et problem" + +msgid "Launch" +msgstr "Start" + +msgid "Fullscreen mode" +msgstr "Fuldskærmstilstand" + +msgid "Edit this page" +msgstr "Rediger denne side" + +msgid "By the" +msgstr "Ved" + +msgid "next page" +msgstr "Næste side" diff --git a/_static/locales/de/LC_MESSAGES/booktheme.mo b/_static/locales/de/LC_MESSAGES/booktheme.mo new file mode 100644 index 0000000..648b565 Binary files /dev/null and b/_static/locales/de/LC_MESSAGES/booktheme.mo differ diff --git a/_static/locales/de/LC_MESSAGES/booktheme.po b/_static/locales/de/LC_MESSAGES/booktheme.po new file mode 100644 index 0000000..4925360 --- /dev/null +++ b/_static/locales/de/LC_MESSAGES/booktheme.po @@ -0,0 +1,75 @@ + +msgid "" +msgstr "" +"Project-Id-Version: Sphinx-Book-Theme\n" +"MIME-Version: 1.0\n" +"Content-Type: text/plain; charset=UTF-8\n" +"Content-Transfer-Encoding: 8bit\n" +"Language: de\n" +"Plural-Forms: nplurals=2; plural=(n != 1);\n" + +msgid "Print to PDF" +msgstr "In PDF drucken" + +msgid "Theme by the" +msgstr "Thema von der" + +msgid "Download source file" +msgstr "Quelldatei herunterladen" + +msgid "open issue" +msgstr "offenes Thema" + +msgid "Contents" +msgstr "Inhalt" + +msgid "previous page" +msgstr "vorherige Seite" + +msgid "Download notebook file" +msgstr "Notebook-Datei herunterladen" + +msgid "Copyright" +msgstr "Urheberrechte ©" + +msgid "Download this page" +msgstr "Laden Sie diese Seite herunter" + +msgid "Source repository" +msgstr "Quell-Repository" + +msgid "By" +msgstr "Durch" + +msgid "repository" +msgstr "Repository" + +msgid "Last updated on" +msgstr "Zuletzt aktualisiert am" + +msgid "Toggle navigation" +msgstr "Navigation umschalten" + +msgid "Sphinx Book Theme" +msgstr "Sphinx-Buch-Thema" + +msgid "suggest edit" +msgstr "vorschlagen zu bearbeiten" + +msgid "Open an issue" +msgstr "Öffnen Sie ein Problem" + +msgid "Launch" +msgstr "Starten" + +msgid "Fullscreen mode" +msgstr "Vollbildmodus" + +msgid "Edit this page" +msgstr "Bearbeite diese Seite" + +msgid "By the" +msgstr "Bis zum" + +msgid "next page" +msgstr "Nächste Seite" diff --git a/_static/locales/el/LC_MESSAGES/booktheme.mo b/_static/locales/el/LC_MESSAGES/booktheme.mo new file mode 100644 index 0000000..fca6e93 Binary files /dev/null and b/_static/locales/el/LC_MESSAGES/booktheme.mo differ diff --git a/_static/locales/el/LC_MESSAGES/booktheme.po b/_static/locales/el/LC_MESSAGES/booktheme.po new file mode 100644 index 0000000..3e01acb --- /dev/null +++ b/_static/locales/el/LC_MESSAGES/booktheme.po @@ -0,0 +1,75 @@ + +msgid "" +msgstr "" +"Project-Id-Version: Sphinx-Book-Theme\n" +"MIME-Version: 1.0\n" +"Content-Type: text/plain; charset=UTF-8\n" +"Content-Transfer-Encoding: 8bit\n" +"Language: el\n" +"Plural-Forms: nplurals=2; plural=(n != 1);\n" + +msgid "Print to PDF" +msgstr "Εκτύπωση σε PDF" + +msgid "Theme by the" +msgstr "Θέμα από το" + +msgid "Download source file" +msgstr "Λήψη αρχείου προέλευσης" + +msgid "open issue" +msgstr "ανοιχτό ζήτημα" + +msgid "Contents" +msgstr "Περιεχόμενα" + +msgid "previous page" +msgstr "προηγούμενη σελίδα" + +msgid "Download notebook file" +msgstr "Λήψη αρχείου σημειωματάριου" + +msgid "Copyright" +msgstr "Πνευματική ιδιοκτησία" + +msgid "Download this page" +msgstr "Λήψη αυτής της σελίδας" + +msgid "Source repository" +msgstr "Αποθήκη πηγής" + +msgid "By" +msgstr "Με" + +msgid "repository" +msgstr "αποθήκη" + +msgid "Last updated on" +msgstr "Τελευταία ενημέρωση στις" + +msgid "Toggle navigation" +msgstr "Εναλλαγή πλοήγησης" + +msgid "Sphinx Book Theme" +msgstr "Θέμα βιβλίου Sphinx" + +msgid "suggest edit" +msgstr "προτείνω επεξεργασία" + +msgid "Open an issue" +msgstr "Ανοίξτε ένα ζήτημα" + +msgid "Launch" +msgstr "Εκτόξευση" + +msgid "Fullscreen mode" +msgstr "ΛΕΙΤΟΥΡΓΙΑ ΠΛΗΡΟΥΣ ΟΘΟΝΗΣ" + +msgid "Edit this page" +msgstr "Επεξεργαστείτε αυτήν τη σελίδα" + +msgid "By the" +msgstr "Από το" + +msgid "next page" +msgstr "επόμενη σελίδα" diff --git a/_static/locales/eo/LC_MESSAGES/booktheme.mo b/_static/locales/eo/LC_MESSAGES/booktheme.mo new file mode 100644 index 0000000..d1072bb Binary files /dev/null and b/_static/locales/eo/LC_MESSAGES/booktheme.mo differ diff --git a/_static/locales/eo/LC_MESSAGES/booktheme.po b/_static/locales/eo/LC_MESSAGES/booktheme.po new file mode 100644 index 0000000..f7ed226 --- /dev/null +++ b/_static/locales/eo/LC_MESSAGES/booktheme.po @@ -0,0 +1,75 @@ + +msgid "" +msgstr "" +"Project-Id-Version: Sphinx-Book-Theme\n" +"MIME-Version: 1.0\n" +"Content-Type: text/plain; charset=UTF-8\n" +"Content-Transfer-Encoding: 8bit\n" +"Language: eo\n" +"Plural-Forms: nplurals=2; plural=(n != 1);\n" + +msgid "Print to PDF" +msgstr "Presi al PDF" + +msgid "Theme by the" +msgstr "Temo de la" + +msgid "Download source file" +msgstr "Elŝutu fontodosieron" + +msgid "open issue" +msgstr "malferma numero" + +msgid "Contents" +msgstr "Enhavo" + +msgid "previous page" +msgstr "antaŭa paĝo" + +msgid "Download notebook file" +msgstr "Elŝutu kajeran dosieron" + +msgid "Copyright" +msgstr "Kopirajto" + +msgid "Download this page" +msgstr "Elŝutu ĉi tiun paĝon" + +msgid "Source repository" +msgstr "Fonto-deponejo" + +msgid "By" +msgstr "De" + +msgid "repository" +msgstr "deponejo" + +msgid "Last updated on" +msgstr "Laste ĝisdatigita la" + +msgid "Toggle navigation" +msgstr "Ŝalti navigadon" + +msgid "Sphinx Book Theme" +msgstr "Sfinksa Libro-Temo" + +msgid "suggest edit" +msgstr "sugesti redaktadon" + +msgid "Open an issue" +msgstr "Malfermu numeron" + +msgid "Launch" +msgstr "Lanĉo" + +msgid "Fullscreen mode" +msgstr "Plenekrana reĝimo" + +msgid "Edit this page" +msgstr "Redaktu ĉi tiun paĝon" + +msgid "By the" +msgstr "Per la" + +msgid "next page" +msgstr "sekva paĝo" diff --git a/_static/locales/es/LC_MESSAGES/booktheme.mo b/_static/locales/es/LC_MESSAGES/booktheme.mo new file mode 100644 index 0000000..ba2ee4d Binary files /dev/null and b/_static/locales/es/LC_MESSAGES/booktheme.mo differ diff --git a/_static/locales/es/LC_MESSAGES/booktheme.po b/_static/locales/es/LC_MESSAGES/booktheme.po new file mode 100644 index 0000000..5e0029e --- /dev/null +++ b/_static/locales/es/LC_MESSAGES/booktheme.po @@ -0,0 +1,75 @@ + +msgid "" +msgstr "" +"Project-Id-Version: Sphinx-Book-Theme\n" +"MIME-Version: 1.0\n" +"Content-Type: text/plain; charset=UTF-8\n" +"Content-Transfer-Encoding: 8bit\n" +"Language: es\n" +"Plural-Forms: nplurals=2; plural=(n != 1);\n" + +msgid "Print to PDF" +msgstr "Imprimir en PDF" + +msgid "Theme by the" +msgstr "Tema por el" + +msgid "Download source file" +msgstr "Descargar archivo fuente" + +msgid "open issue" +msgstr "Tema abierto" + +msgid "Contents" +msgstr "Contenido" + +msgid "previous page" +msgstr "pagina anterior" + +msgid "Download notebook file" +msgstr "Descargar archivo de cuaderno" + +msgid "Copyright" +msgstr "Derechos de autor" + +msgid "Download this page" +msgstr "Descarga esta pagina" + +msgid "Source repository" +msgstr "Repositorio de origen" + +msgid "By" +msgstr "Por" + +msgid "repository" +msgstr "repositorio" + +msgid "Last updated on" +msgstr "Ultima actualización en" + +msgid "Toggle navigation" +msgstr "Navegación de palanca" + +msgid "Sphinx Book Theme" +msgstr "Tema del libro de la esfinge" + +msgid "suggest edit" +msgstr "sugerir editar" + +msgid "Open an issue" +msgstr "Abrir un problema" + +msgid "Launch" +msgstr "Lanzamiento" + +msgid "Fullscreen mode" +msgstr "Modo de pantalla completa" + +msgid "Edit this page" +msgstr "Edita esta página" + +msgid "By the" +msgstr "Por el" + +msgid "next page" +msgstr "siguiente página" diff --git a/_static/locales/et/LC_MESSAGES/booktheme.mo b/_static/locales/et/LC_MESSAGES/booktheme.mo new file mode 100644 index 0000000..983b823 Binary files /dev/null and b/_static/locales/et/LC_MESSAGES/booktheme.mo differ diff --git a/_static/locales/et/LC_MESSAGES/booktheme.po b/_static/locales/et/LC_MESSAGES/booktheme.po new file mode 100644 index 0000000..8680982 --- /dev/null +++ b/_static/locales/et/LC_MESSAGES/booktheme.po @@ -0,0 +1,75 @@ + +msgid "" +msgstr "" +"Project-Id-Version: Sphinx-Book-Theme\n" +"MIME-Version: 1.0\n" +"Content-Type: text/plain; charset=UTF-8\n" +"Content-Transfer-Encoding: 8bit\n" +"Language: et\n" +"Plural-Forms: nplurals=2; plural=(n != 1);\n" + +msgid "Print to PDF" +msgstr "Prindi PDF-i" + +msgid "Theme by the" +msgstr "Teema" + +msgid "Download source file" +msgstr "Laadige alla lähtefail" + +msgid "open issue" +msgstr "avatud küsimus" + +msgid "Contents" +msgstr "Sisu" + +msgid "previous page" +msgstr "eelmine leht" + +msgid "Download notebook file" +msgstr "Laadige sülearvuti fail alla" + +msgid "Copyright" +msgstr "Autoriõigus" + +msgid "Download this page" +msgstr "Laadige see leht alla" + +msgid "Source repository" +msgstr "Allikahoidla" + +msgid "By" +msgstr "Kõrval" + +msgid "repository" +msgstr "hoidla" + +msgid "Last updated on" +msgstr "Viimati uuendatud" + +msgid "Toggle navigation" +msgstr "Lülita navigeerimine sisse" + +msgid "Sphinx Book Theme" +msgstr "Sfinksiraamatu teema" + +msgid "suggest edit" +msgstr "soovita muuta" + +msgid "Open an issue" +msgstr "Avage probleem" + +msgid "Launch" +msgstr "Käivitage" + +msgid "Fullscreen mode" +msgstr "Täisekraanirežiim" + +msgid "Edit this page" +msgstr "Muutke seda lehte" + +msgid "By the" +msgstr "Autor" + +msgid "next page" +msgstr "järgmine leht" diff --git a/_static/locales/fi/LC_MESSAGES/booktheme.mo b/_static/locales/fi/LC_MESSAGES/booktheme.mo new file mode 100644 index 0000000..d8ac054 Binary files /dev/null and b/_static/locales/fi/LC_MESSAGES/booktheme.mo differ diff --git a/_static/locales/fi/LC_MESSAGES/booktheme.po b/_static/locales/fi/LC_MESSAGES/booktheme.po new file mode 100644 index 0000000..34dac21 --- /dev/null +++ b/_static/locales/fi/LC_MESSAGES/booktheme.po @@ -0,0 +1,75 @@ + +msgid "" +msgstr "" +"Project-Id-Version: Sphinx-Book-Theme\n" +"MIME-Version: 1.0\n" +"Content-Type: text/plain; charset=UTF-8\n" +"Content-Transfer-Encoding: 8bit\n" +"Language: fi\n" +"Plural-Forms: nplurals=2; plural=(n != 1);\n" + +msgid "Print to PDF" +msgstr "Tulosta PDF-tiedostoon" + +msgid "Theme by the" +msgstr "Teeman tekijä" + +msgid "Download source file" +msgstr "Lataa lähdetiedosto" + +msgid "open issue" +msgstr "avoin ongelma" + +msgid "Contents" +msgstr "Sisällys" + +msgid "previous page" +msgstr "Edellinen sivu" + +msgid "Download notebook file" +msgstr "Lataa muistikirjatiedosto" + +msgid "Copyright" +msgstr "Tekijänoikeus" + +msgid "Download this page" +msgstr "Lataa tämä sivu" + +msgid "Source repository" +msgstr "Lähteen arkisto" + +msgid "By" +msgstr "Tekijä" + +msgid "repository" +msgstr "arkisto" + +msgid "Last updated on" +msgstr "Viimeksi päivitetty" + +msgid "Toggle navigation" +msgstr "Vaihda navigointia" + +msgid "Sphinx Book Theme" +msgstr "Sphinx-kirjan teema" + +msgid "suggest edit" +msgstr "ehdottaa muokkausta" + +msgid "Open an issue" +msgstr "Avaa ongelma" + +msgid "Launch" +msgstr "Tuoda markkinoille" + +msgid "Fullscreen mode" +msgstr "Koko näytön tila" + +msgid "Edit this page" +msgstr "Muokkaa tätä sivua" + +msgid "By the" +msgstr "Mukaan" + +msgid "next page" +msgstr "seuraava sivu" diff --git a/_static/locales/fr/LC_MESSAGES/booktheme.mo b/_static/locales/fr/LC_MESSAGES/booktheme.mo new file mode 100644 index 0000000..f663d39 Binary files /dev/null and b/_static/locales/fr/LC_MESSAGES/booktheme.mo differ diff --git a/_static/locales/fr/LC_MESSAGES/booktheme.po b/_static/locales/fr/LC_MESSAGES/booktheme.po new file mode 100644 index 0000000..8991a1b --- /dev/null +++ b/_static/locales/fr/LC_MESSAGES/booktheme.po @@ -0,0 +1,75 @@ + +msgid "" +msgstr "" +"Project-Id-Version: Sphinx-Book-Theme\n" +"MIME-Version: 1.0\n" +"Content-Type: text/plain; charset=UTF-8\n" +"Content-Transfer-Encoding: 8bit\n" +"Language: fr\n" +"Plural-Forms: nplurals=2; plural=(n != 1);\n" + +msgid "Print to PDF" +msgstr "Imprimer au format PDF" + +msgid "Theme by the" +msgstr "Thème par le" + +msgid "Download source file" +msgstr "Télécharger le fichier source" + +msgid "open issue" +msgstr "signaler un problème" + +msgid "Contents" +msgstr "Contenu" + +msgid "previous page" +msgstr "page précédente" + +msgid "Download notebook file" +msgstr "Télécharger le fichier notebook" + +msgid "Copyright" +msgstr "droits d'auteur" + +msgid "Download this page" +msgstr "Téléchargez cette page" + +msgid "Source repository" +msgstr "Dépôt source" + +msgid "By" +msgstr "Par" + +msgid "repository" +msgstr "dépôt" + +msgid "Last updated on" +msgstr "Dernière mise à jour le" + +msgid "Toggle navigation" +msgstr "Basculer la navigation" + +msgid "Sphinx Book Theme" +msgstr "Thème du livre Sphinx" + +msgid "suggest edit" +msgstr "suggestion de modification" + +msgid "Open an issue" +msgstr "Ouvrez un problème" + +msgid "Launch" +msgstr "lancement" + +msgid "Fullscreen mode" +msgstr "Mode plein écran" + +msgid "Edit this page" +msgstr "Modifier cette page" + +msgid "By the" +msgstr "Par le" + +msgid "next page" +msgstr "page suivante" diff --git a/_static/locales/hr/LC_MESSAGES/booktheme.mo b/_static/locales/hr/LC_MESSAGES/booktheme.mo new file mode 100644 index 0000000..eca4a1a Binary files /dev/null and b/_static/locales/hr/LC_MESSAGES/booktheme.mo differ diff --git a/_static/locales/hr/LC_MESSAGES/booktheme.po b/_static/locales/hr/LC_MESSAGES/booktheme.po new file mode 100644 index 0000000..42c4233 --- /dev/null +++ b/_static/locales/hr/LC_MESSAGES/booktheme.po @@ -0,0 +1,75 @@ + +msgid "" +msgstr "" +"Project-Id-Version: Sphinx-Book-Theme\n" +"MIME-Version: 1.0\n" +"Content-Type: text/plain; charset=UTF-8\n" +"Content-Transfer-Encoding: 8bit\n" +"Language: hr\n" +"Plural-Forms: nplurals=2; plural=(n != 1);\n" + +msgid "Print to PDF" +msgstr "Ispis u PDF" + +msgid "Theme by the" +msgstr "Tema autora" + +msgid "Download source file" +msgstr "Preuzmi izvornu datoteku" + +msgid "open issue" +msgstr "otvoreno izdanje" + +msgid "Contents" +msgstr "Sadržaj" + +msgid "previous page" +msgstr "Prethodna stranica" + +msgid "Download notebook file" +msgstr "Preuzmi datoteku bilježnice" + +msgid "Copyright" +msgstr "Autorska prava" + +msgid "Download this page" +msgstr "Preuzmite ovu stranicu" + +msgid "Source repository" +msgstr "Izvorno spremište" + +msgid "By" +msgstr "Po" + +msgid "repository" +msgstr "spremište" + +msgid "Last updated on" +msgstr "Posljednje ažuriranje:" + +msgid "Toggle navigation" +msgstr "Uključi / isključi navigaciju" + +msgid "Sphinx Book Theme" +msgstr "Tema knjige Sphinx" + +msgid "suggest edit" +msgstr "predloži uređivanje" + +msgid "Open an issue" +msgstr "Otvorite izdanje" + +msgid "Launch" +msgstr "Pokrenite" + +msgid "Fullscreen mode" +msgstr "Način preko cijelog zaslona" + +msgid "Edit this page" +msgstr "Uredite ovu stranicu" + +msgid "By the" +msgstr "Od strane" + +msgid "next page" +msgstr "sljedeća stranica" diff --git a/_static/locales/id/LC_MESSAGES/booktheme.mo b/_static/locales/id/LC_MESSAGES/booktheme.mo new file mode 100644 index 0000000..d07a06a Binary files /dev/null and b/_static/locales/id/LC_MESSAGES/booktheme.mo differ diff --git a/_static/locales/id/LC_MESSAGES/booktheme.po b/_static/locales/id/LC_MESSAGES/booktheme.po new file mode 100644 index 0000000..b8d8d89 --- /dev/null +++ b/_static/locales/id/LC_MESSAGES/booktheme.po @@ -0,0 +1,75 @@ + +msgid "" +msgstr "" +"Project-Id-Version: Sphinx-Book-Theme\n" +"MIME-Version: 1.0\n" +"Content-Type: text/plain; charset=UTF-8\n" +"Content-Transfer-Encoding: 8bit\n" +"Language: id\n" +"Plural-Forms: nplurals=2; plural=(n != 1);\n" + +msgid "Print to PDF" +msgstr "Cetak ke PDF" + +msgid "Theme by the" +msgstr "Tema oleh" + +msgid "Download source file" +msgstr "Unduh file sumber" + +msgid "open issue" +msgstr "masalah terbuka" + +msgid "Contents" +msgstr "Isi" + +msgid "previous page" +msgstr "halaman sebelumnya" + +msgid "Download notebook file" +msgstr "Unduh file notebook" + +msgid "Copyright" +msgstr "hak cipta" + +msgid "Download this page" +msgstr "Unduh halaman ini" + +msgid "Source repository" +msgstr "Repositori sumber" + +msgid "By" +msgstr "Oleh" + +msgid "repository" +msgstr "gudang" + +msgid "Last updated on" +msgstr "Terakhir diperbarui saat" + +msgid "Toggle navigation" +msgstr "Alihkan navigasi" + +msgid "Sphinx Book Theme" +msgstr "Tema Buku Sphinx" + +msgid "suggest edit" +msgstr "menyarankan edit" + +msgid "Open an issue" +msgstr "Buka masalah" + +msgid "Launch" +msgstr "Meluncurkan" + +msgid "Fullscreen mode" +msgstr "Mode layar penuh" + +msgid "Edit this page" +msgstr "Edit halaman ini" + +msgid "By the" +msgstr "Oleh" + +msgid "next page" +msgstr "halaman selanjutnya" diff --git a/_static/locales/it/LC_MESSAGES/booktheme.mo b/_static/locales/it/LC_MESSAGES/booktheme.mo new file mode 100644 index 0000000..53ba476 Binary files /dev/null and b/_static/locales/it/LC_MESSAGES/booktheme.mo differ diff --git a/_static/locales/it/LC_MESSAGES/booktheme.po b/_static/locales/it/LC_MESSAGES/booktheme.po new file mode 100644 index 0000000..36fca59 --- /dev/null +++ b/_static/locales/it/LC_MESSAGES/booktheme.po @@ -0,0 +1,75 @@ + +msgid "" +msgstr "" +"Project-Id-Version: Sphinx-Book-Theme\n" +"MIME-Version: 1.0\n" +"Content-Type: text/plain; charset=UTF-8\n" +"Content-Transfer-Encoding: 8bit\n" +"Language: it\n" +"Plural-Forms: nplurals=2; plural=(n != 1);\n" + +msgid "Print to PDF" +msgstr "Stampa in PDF" + +msgid "Theme by the" +msgstr "Tema di" + +msgid "Download source file" +msgstr "Scarica il file sorgente" + +msgid "open issue" +msgstr "questione aperta" + +msgid "Contents" +msgstr "Contenuti" + +msgid "previous page" +msgstr "pagina precedente" + +msgid "Download notebook file" +msgstr "Scarica il file del taccuino" + +msgid "Copyright" +msgstr "Diritto d'autore" + +msgid "Download this page" +msgstr "Scarica questa pagina" + +msgid "Source repository" +msgstr "Repository di origine" + +msgid "By" +msgstr "Di" + +msgid "repository" +msgstr "repository" + +msgid "Last updated on" +msgstr "Ultimo aggiornamento il" + +msgid "Toggle navigation" +msgstr "Attiva / disattiva la navigazione" + +msgid "Sphinx Book Theme" +msgstr "Tema del libro della Sfinge" + +msgid "suggest edit" +msgstr "suggerisci modifica" + +msgid "Open an issue" +msgstr "Apri un problema" + +msgid "Launch" +msgstr "Lanciare" + +msgid "Fullscreen mode" +msgstr "Modalità schermo intero" + +msgid "Edit this page" +msgstr "Modifica questa pagina" + +msgid "By the" +msgstr "Dal" + +msgid "next page" +msgstr "pagina successiva" diff --git a/_static/locales/iw/LC_MESSAGES/booktheme.mo b/_static/locales/iw/LC_MESSAGES/booktheme.mo new file mode 100644 index 0000000..a45c657 Binary files /dev/null and b/_static/locales/iw/LC_MESSAGES/booktheme.mo differ diff --git a/_static/locales/iw/LC_MESSAGES/booktheme.po b/_static/locales/iw/LC_MESSAGES/booktheme.po new file mode 100644 index 0000000..dede9cb --- /dev/null +++ b/_static/locales/iw/LC_MESSAGES/booktheme.po @@ -0,0 +1,75 @@ + +msgid "" +msgstr "" +"Project-Id-Version: Sphinx-Book-Theme\n" +"MIME-Version: 1.0\n" +"Content-Type: text/plain; charset=UTF-8\n" +"Content-Transfer-Encoding: 8bit\n" +"Language: iw\n" +"Plural-Forms: nplurals=2; plural=(n != 1);\n" + +msgid "Print to PDF" +msgstr "הדפס לקובץ PDF" + +msgid "Theme by the" +msgstr "נושא מאת" + +msgid "Download source file" +msgstr "הורד את קובץ המקור" + +msgid "open issue" +msgstr "בעיה פתוחה" + +msgid "Contents" +msgstr "תוכן" + +msgid "previous page" +msgstr "עמוד קודם" + +msgid "Download notebook file" +msgstr "הורד קובץ מחברת" + +msgid "Copyright" +msgstr "זכויות יוצרים" + +msgid "Download this page" +msgstr "הורד דף זה" + +msgid "Source repository" +msgstr "מאגר המקורות" + +msgid "By" +msgstr "על ידי" + +msgid "repository" +msgstr "מאגר" + +msgid "Last updated on" +msgstr "עודכן לאחרונה ב" + +msgid "Toggle navigation" +msgstr "החלף ניווט" + +msgid "Sphinx Book Theme" +msgstr "נושא ספר ספינקס" + +msgid "suggest edit" +msgstr "מציע לערוך" + +msgid "Open an issue" +msgstr "פתח גיליון" + +msgid "Launch" +msgstr "לְהַשִׁיק" + +msgid "Fullscreen mode" +msgstr "מצב מסך מלא" + +msgid "Edit this page" +msgstr "ערוך דף זה" + +msgid "By the" +msgstr "דרך" + +msgid "next page" +msgstr "עמוד הבא" diff --git a/_static/locales/ja/LC_MESSAGES/booktheme.mo b/_static/locales/ja/LC_MESSAGES/booktheme.mo new file mode 100644 index 0000000..1cefd29 Binary files /dev/null and b/_static/locales/ja/LC_MESSAGES/booktheme.mo differ diff --git a/_static/locales/ja/LC_MESSAGES/booktheme.po b/_static/locales/ja/LC_MESSAGES/booktheme.po new file mode 100644 index 0000000..2615f0d --- /dev/null +++ b/_static/locales/ja/LC_MESSAGES/booktheme.po @@ -0,0 +1,75 @@ + +msgid "" +msgstr "" +"Project-Id-Version: Sphinx-Book-Theme\n" +"MIME-Version: 1.0\n" +"Content-Type: text/plain; charset=UTF-8\n" +"Content-Transfer-Encoding: 8bit\n" +"Language: ja\n" +"Plural-Forms: nplurals=2; plural=(n != 1);\n" + +msgid "Print to PDF" +msgstr "PDFに印刷" + +msgid "Theme by the" +msgstr "のテーマ" + +msgid "Download source file" +msgstr "ソースファイルをダウンロード" + +msgid "open issue" +msgstr "未解決の問題" + +msgid "Contents" +msgstr "目次" + +msgid "previous page" +msgstr "前のページ" + +msgid "Download notebook file" +msgstr "ノートブックファイルをダウンロード" + +msgid "Copyright" +msgstr "Copyright" + +msgid "Download this page" +msgstr "このページをダウンロード" + +msgid "Source repository" +msgstr "ソースリポジトリ" + +msgid "By" +msgstr "著者" + +msgid "repository" +msgstr "リポジトリ" + +msgid "Last updated on" +msgstr "最終更新日" + +msgid "Toggle navigation" +msgstr "ナビゲーションを切り替え" + +msgid "Sphinx Book Theme" +msgstr "スフィンクスの本のテーマ" + +msgid "suggest edit" +msgstr "編集を提案する" + +msgid "Open an issue" +msgstr "問題を報告" + +msgid "Launch" +msgstr "起動" + +msgid "Fullscreen mode" +msgstr "全画面モード" + +msgid "Edit this page" +msgstr "このページを編集" + +msgid "By the" +msgstr "によって" + +msgid "next page" +msgstr "次のページ" diff --git a/_static/locales/ko/LC_MESSAGES/booktheme.mo b/_static/locales/ko/LC_MESSAGES/booktheme.mo new file mode 100644 index 0000000..06c7ec9 Binary files /dev/null and b/_static/locales/ko/LC_MESSAGES/booktheme.mo differ diff --git a/_static/locales/ko/LC_MESSAGES/booktheme.po b/_static/locales/ko/LC_MESSAGES/booktheme.po new file mode 100644 index 0000000..c9e13a4 --- /dev/null +++ b/_static/locales/ko/LC_MESSAGES/booktheme.po @@ -0,0 +1,75 @@ + +msgid "" +msgstr "" +"Project-Id-Version: Sphinx-Book-Theme\n" +"MIME-Version: 1.0\n" +"Content-Type: text/plain; charset=UTF-8\n" +"Content-Transfer-Encoding: 8bit\n" +"Language: ko\n" +"Plural-Forms: nplurals=2; plural=(n != 1);\n" + +msgid "Print to PDF" +msgstr "PDF로 인쇄" + +msgid "Theme by the" +msgstr "테마별" + +msgid "Download source file" +msgstr "소스 파일 다운로드" + +msgid "open issue" +msgstr "열린 문제" + +msgid "Contents" +msgstr "내용" + +msgid "previous page" +msgstr "이전 페이지" + +msgid "Download notebook file" +msgstr "노트북 파일 다운로드" + +msgid "Copyright" +msgstr "저작권" + +msgid "Download this page" +msgstr "이 페이지 다운로드" + +msgid "Source repository" +msgstr "소스 저장소" + +msgid "By" +msgstr "으로" + +msgid "repository" +msgstr "저장소" + +msgid "Last updated on" +msgstr "마지막 업데이트" + +msgid "Toggle navigation" +msgstr "탐색 전환" + +msgid "Sphinx Book Theme" +msgstr "스핑크스 도서 테마" + +msgid "suggest edit" +msgstr "편집 제안" + +msgid "Open an issue" +msgstr "이슈 열기" + +msgid "Launch" +msgstr "시작하다" + +msgid "Fullscreen mode" +msgstr "전체 화면으로보기" + +msgid "Edit this page" +msgstr "이 페이지 편집" + +msgid "By the" +msgstr "에 의해" + +msgid "next page" +msgstr "다음 페이지" diff --git a/_static/locales/lt/LC_MESSAGES/booktheme.mo b/_static/locales/lt/LC_MESSAGES/booktheme.mo new file mode 100644 index 0000000..4468ba0 Binary files /dev/null and b/_static/locales/lt/LC_MESSAGES/booktheme.mo differ diff --git a/_static/locales/lt/LC_MESSAGES/booktheme.po b/_static/locales/lt/LC_MESSAGES/booktheme.po new file mode 100644 index 0000000..35eabd9 --- /dev/null +++ b/_static/locales/lt/LC_MESSAGES/booktheme.po @@ -0,0 +1,75 @@ + +msgid "" +msgstr "" +"Project-Id-Version: Sphinx-Book-Theme\n" +"MIME-Version: 1.0\n" +"Content-Type: text/plain; charset=UTF-8\n" +"Content-Transfer-Encoding: 8bit\n" +"Language: lt\n" +"Plural-Forms: nplurals=2; plural=(n != 1);\n" + +msgid "Print to PDF" +msgstr "Spausdinti į PDF" + +msgid "Theme by the" +msgstr "Tema" + +msgid "Download source file" +msgstr "Atsisiųsti šaltinio failą" + +msgid "open issue" +msgstr "atviras klausimas" + +msgid "Contents" +msgstr "Turinys" + +msgid "previous page" +msgstr "Ankstesnis puslapis" + +msgid "Download notebook file" +msgstr "Atsisiųsti nešiojamojo kompiuterio failą" + +msgid "Copyright" +msgstr "Autorių teisės" + +msgid "Download this page" +msgstr "Atsisiųskite šį puslapį" + +msgid "Source repository" +msgstr "Šaltinio saugykla" + +msgid "By" +msgstr "Iki" + +msgid "repository" +msgstr "saugykla" + +msgid "Last updated on" +msgstr "Paskutinį kartą atnaujinta" + +msgid "Toggle navigation" +msgstr "Perjungti naršymą" + +msgid "Sphinx Book Theme" +msgstr "Sfinkso knygos tema" + +msgid "suggest edit" +msgstr "pasiūlyti redaguoti" + +msgid "Open an issue" +msgstr "Atidarykite problemą" + +msgid "Launch" +msgstr "Paleiskite" + +msgid "Fullscreen mode" +msgstr "Pilno ekrano režimas" + +msgid "Edit this page" +msgstr "Redaguoti šį puslapį" + +msgid "By the" +msgstr "Prie" + +msgid "next page" +msgstr "Kitas puslapis" diff --git a/_static/locales/lv/LC_MESSAGES/booktheme.mo b/_static/locales/lv/LC_MESSAGES/booktheme.mo new file mode 100644 index 0000000..74aa4d8 Binary files /dev/null and b/_static/locales/lv/LC_MESSAGES/booktheme.mo differ diff --git a/_static/locales/lv/LC_MESSAGES/booktheme.po b/_static/locales/lv/LC_MESSAGES/booktheme.po new file mode 100644 index 0000000..ee1bd08 --- /dev/null +++ b/_static/locales/lv/LC_MESSAGES/booktheme.po @@ -0,0 +1,75 @@ + +msgid "" +msgstr "" +"Project-Id-Version: Sphinx-Book-Theme\n" +"MIME-Version: 1.0\n" +"Content-Type: text/plain; charset=UTF-8\n" +"Content-Transfer-Encoding: 8bit\n" +"Language: lv\n" +"Plural-Forms: nplurals=2; plural=(n != 1);\n" + +msgid "Print to PDF" +msgstr "Drukāt PDF formātā" + +msgid "Theme by the" +msgstr "Autora tēma" + +msgid "Download source file" +msgstr "Lejupielādēt avota failu" + +msgid "open issue" +msgstr "atklāts jautājums" + +msgid "Contents" +msgstr "Saturs" + +msgid "previous page" +msgstr "iepriekšējā lapa" + +msgid "Download notebook file" +msgstr "Lejupielādēt piezīmju grāmatiņu" + +msgid "Copyright" +msgstr "Autortiesības" + +msgid "Download this page" +msgstr "Lejupielādējiet šo lapu" + +msgid "Source repository" +msgstr "Avota krātuve" + +msgid "By" +msgstr "Autors" + +msgid "repository" +msgstr "krātuve" + +msgid "Last updated on" +msgstr "Pēdējoreiz atjaunināts" + +msgid "Toggle navigation" +msgstr "Pārslēgt navigāciju" + +msgid "Sphinx Book Theme" +msgstr "Sfinksa grāmatas tēma" + +msgid "suggest edit" +msgstr "ieteikt rediģēt" + +msgid "Open an issue" +msgstr "Atveriet problēmu" + +msgid "Launch" +msgstr "Uzsākt" + +msgid "Fullscreen mode" +msgstr "Pilnekrāna režīms" + +msgid "Edit this page" +msgstr "Rediģēt šo lapu" + +msgid "By the" +msgstr "Ar" + +msgid "next page" +msgstr "nākamā lapaspuse" diff --git a/_static/locales/ml/LC_MESSAGES/booktheme.mo b/_static/locales/ml/LC_MESSAGES/booktheme.mo new file mode 100644 index 0000000..2736e8f Binary files /dev/null and b/_static/locales/ml/LC_MESSAGES/booktheme.mo differ diff --git a/_static/locales/ml/LC_MESSAGES/booktheme.po b/_static/locales/ml/LC_MESSAGES/booktheme.po new file mode 100644 index 0000000..d471277 --- /dev/null +++ b/_static/locales/ml/LC_MESSAGES/booktheme.po @@ -0,0 +1,66 @@ + +msgid "" +msgstr "" +"Project-Id-Version: Sphinx-Book-Theme\n" +"MIME-Version: 1.0\n" +"Content-Type: text/plain; charset=UTF-8\n" +"Content-Transfer-Encoding: 8bit\n" +"Language: ml\n" +"Plural-Forms: nplurals=2; plural=(n != 1);\n" + +msgid "Print to PDF" +msgstr "PDF- ലേക്ക് പ്രിന്റുചെയ്യുക" + +msgid "Theme by the" +msgstr "പ്രമേയം" + +msgid "Download source file" +msgstr "ഉറവിട ഫയൽ ഡൗൺലോഡുചെയ്യുക" + +msgid "open issue" +msgstr "തുറന്ന പ്രശ്നം" + +msgid "previous page" +msgstr "മുൻപത്തെ താൾ" + +msgid "Download notebook file" +msgstr "നോട്ട്ബുക്ക് ഫയൽ ഡൺലോഡ് ചെയ്യുക" + +msgid "Copyright" +msgstr "പകർപ്പവകാശം" + +msgid "Download this page" +msgstr "ഈ പേജ് ഡൗൺലോഡുചെയ്യുക" + +msgid "Source repository" +msgstr "ഉറവിട ശേഖരം" + +msgid "By" +msgstr "എഴുതിയത്" + +msgid "Last updated on" +msgstr "അവസാനം അപ്‌ഡേറ്റുചെയ്‌തത്" + +msgid "Toggle navigation" +msgstr "നാവിഗേഷൻ ടോഗിൾ ചെയ്യുക" + +msgid "Sphinx Book Theme" +msgstr "സ്ഫിങ്ക്സ് പുസ്തക തീം" + +msgid "suggest edit" +msgstr "എഡിറ്റുചെയ്യാൻ നിർദ്ദേശിക്കുക" + +msgid "Open an issue" +msgstr "ഒരു പ്രശ്നം തുറക്കുക" + +msgid "Launch" +msgstr "സമാരംഭിക്കുക" + +msgid "Edit this page" +msgstr "ഈ പേജ് എഡിറ്റുചെയ്യുക" + +msgid "By the" +msgstr "എഴുതിയത്" + +msgid "next page" +msgstr "അടുത്ത പേജ്" diff --git a/_static/locales/mr/LC_MESSAGES/booktheme.mo b/_static/locales/mr/LC_MESSAGES/booktheme.mo new file mode 100644 index 0000000..fe53010 Binary files /dev/null and b/_static/locales/mr/LC_MESSAGES/booktheme.mo differ diff --git a/_static/locales/mr/LC_MESSAGES/booktheme.po b/_static/locales/mr/LC_MESSAGES/booktheme.po new file mode 100644 index 0000000..f3694ac --- /dev/null +++ b/_static/locales/mr/LC_MESSAGES/booktheme.po @@ -0,0 +1,66 @@ + +msgid "" +msgstr "" +"Project-Id-Version: Sphinx-Book-Theme\n" +"MIME-Version: 1.0\n" +"Content-Type: text/plain; charset=UTF-8\n" +"Content-Transfer-Encoding: 8bit\n" +"Language: mr\n" +"Plural-Forms: nplurals=2; plural=(n != 1);\n" + +msgid "Print to PDF" +msgstr "पीडीएफवर मुद्रित करा" + +msgid "Theme by the" +msgstr "द्वारा थीम" + +msgid "Download source file" +msgstr "स्त्रोत फाइल डाउनलोड करा" + +msgid "open issue" +msgstr "खुला मुद्दा" + +msgid "previous page" +msgstr "मागील पान" + +msgid "Download notebook file" +msgstr "नोटबुक फाईल डाउनलोड करा" + +msgid "Copyright" +msgstr "कॉपीराइट" + +msgid "Download this page" +msgstr "हे पृष्ठ डाउनलोड करा" + +msgid "Source repository" +msgstr "स्त्रोत भांडार" + +msgid "By" +msgstr "द्वारा" + +msgid "Last updated on" +msgstr "अखेरचे अद्यतनित" + +msgid "Toggle navigation" +msgstr "नेव्हिगेशन टॉगल करा" + +msgid "Sphinx Book Theme" +msgstr "स्फिंक्स बुक थीम" + +msgid "suggest edit" +msgstr "संपादन सुचवा" + +msgid "Open an issue" +msgstr "एक मुद्दा उघडा" + +msgid "Launch" +msgstr "लाँच करा" + +msgid "Edit this page" +msgstr "हे पृष्ठ संपादित करा" + +msgid "By the" +msgstr "द्वारा" + +msgid "next page" +msgstr "पुढील पृष्ठ" diff --git a/_static/locales/ms/LC_MESSAGES/booktheme.mo b/_static/locales/ms/LC_MESSAGES/booktheme.mo new file mode 100644 index 0000000..f02603f Binary files /dev/null and b/_static/locales/ms/LC_MESSAGES/booktheme.mo differ diff --git a/_static/locales/ms/LC_MESSAGES/booktheme.po b/_static/locales/ms/LC_MESSAGES/booktheme.po new file mode 100644 index 0000000..65b7c60 --- /dev/null +++ b/_static/locales/ms/LC_MESSAGES/booktheme.po @@ -0,0 +1,66 @@ + +msgid "" +msgstr "" +"Project-Id-Version: Sphinx-Book-Theme\n" +"MIME-Version: 1.0\n" +"Content-Type: text/plain; charset=UTF-8\n" +"Content-Transfer-Encoding: 8bit\n" +"Language: ms\n" +"Plural-Forms: nplurals=2; plural=(n != 1);\n" + +msgid "Print to PDF" +msgstr "Cetak ke PDF" + +msgid "Theme by the" +msgstr "Tema oleh" + +msgid "Download source file" +msgstr "Muat turun fail sumber" + +msgid "open issue" +msgstr "isu terbuka" + +msgid "previous page" +msgstr "halaman sebelumnya" + +msgid "Download notebook file" +msgstr "Muat turun fail buku nota" + +msgid "Copyright" +msgstr "hak cipta" + +msgid "Download this page" +msgstr "Muat turun halaman ini" + +msgid "Source repository" +msgstr "Repositori sumber" + +msgid "By" +msgstr "Oleh" + +msgid "Last updated on" +msgstr "Terakhir dikemas kini pada" + +msgid "Toggle navigation" +msgstr "Togol navigasi" + +msgid "Sphinx Book Theme" +msgstr "Tema Buku Sphinx" + +msgid "suggest edit" +msgstr "cadangkan edit" + +msgid "Open an issue" +msgstr "Buka masalah" + +msgid "Launch" +msgstr "Lancarkan" + +msgid "Edit this page" +msgstr "Edit halaman ini" + +msgid "By the" +msgstr "Oleh" + +msgid "next page" +msgstr "muka surat seterusnya" diff --git a/_static/locales/nl/LC_MESSAGES/booktheme.mo b/_static/locales/nl/LC_MESSAGES/booktheme.mo new file mode 100644 index 0000000..e59e7ec Binary files /dev/null and b/_static/locales/nl/LC_MESSAGES/booktheme.mo differ diff --git a/_static/locales/nl/LC_MESSAGES/booktheme.po b/_static/locales/nl/LC_MESSAGES/booktheme.po new file mode 100644 index 0000000..71bd1cd --- /dev/null +++ b/_static/locales/nl/LC_MESSAGES/booktheme.po @@ -0,0 +1,75 @@ + +msgid "" +msgstr "" +"Project-Id-Version: Sphinx-Book-Theme\n" +"MIME-Version: 1.0\n" +"Content-Type: text/plain; charset=UTF-8\n" +"Content-Transfer-Encoding: 8bit\n" +"Language: nl\n" +"Plural-Forms: nplurals=2; plural=(n != 1);\n" + +msgid "Print to PDF" +msgstr "Afdrukken naar pdf" + +msgid "Theme by the" +msgstr "Thema door de" + +msgid "Download source file" +msgstr "Download het bronbestand" + +msgid "open issue" +msgstr "open probleem" + +msgid "Contents" +msgstr "Inhoud" + +msgid "previous page" +msgstr "vorige pagina" + +msgid "Download notebook file" +msgstr "Download notebookbestand" + +msgid "Copyright" +msgstr "auteursrechten" + +msgid "Download this page" +msgstr "Download deze pagina" + +msgid "Source repository" +msgstr "Bronopslagplaats" + +msgid "By" +msgstr "Door" + +msgid "repository" +msgstr "repository" + +msgid "Last updated on" +msgstr "Laatst geupdate op" + +msgid "Toggle navigation" +msgstr "Schakel navigatie" + +msgid "Sphinx Book Theme" +msgstr "Sphinx-boekthema" + +msgid "suggest edit" +msgstr "suggereren bewerken" + +msgid "Open an issue" +msgstr "Open een probleem" + +msgid "Launch" +msgstr "Lancering" + +msgid "Fullscreen mode" +msgstr "Volledig scherm" + +msgid "Edit this page" +msgstr "bewerk deze pagina" + +msgid "By the" +msgstr "Door de" + +msgid "next page" +msgstr "volgende bladzijde" diff --git a/_static/locales/no/LC_MESSAGES/booktheme.mo b/_static/locales/no/LC_MESSAGES/booktheme.mo new file mode 100644 index 0000000..6cd15c8 Binary files /dev/null and b/_static/locales/no/LC_MESSAGES/booktheme.mo differ diff --git a/_static/locales/no/LC_MESSAGES/booktheme.po b/_static/locales/no/LC_MESSAGES/booktheme.po new file mode 100644 index 0000000..b21346a --- /dev/null +++ b/_static/locales/no/LC_MESSAGES/booktheme.po @@ -0,0 +1,75 @@ + +msgid "" +msgstr "" +"Project-Id-Version: Sphinx-Book-Theme\n" +"MIME-Version: 1.0\n" +"Content-Type: text/plain; charset=UTF-8\n" +"Content-Transfer-Encoding: 8bit\n" +"Language: no\n" +"Plural-Forms: nplurals=2; plural=(n != 1);\n" + +msgid "Print to PDF" +msgstr "Skriv ut til PDF" + +msgid "Theme by the" +msgstr "Tema av" + +msgid "Download source file" +msgstr "Last ned kildefilen" + +msgid "open issue" +msgstr "åpent nummer" + +msgid "Contents" +msgstr "Innhold" + +msgid "previous page" +msgstr "forrige side" + +msgid "Download notebook file" +msgstr "Last ned notatbokfilen" + +msgid "Copyright" +msgstr "opphavsrett" + +msgid "Download this page" +msgstr "Last ned denne siden" + +msgid "Source repository" +msgstr "Kildedepot" + +msgid "By" +msgstr "Av" + +msgid "repository" +msgstr "oppbevaringssted" + +msgid "Last updated on" +msgstr "Sist oppdatert den" + +msgid "Toggle navigation" +msgstr "Bytt navigasjon" + +msgid "Sphinx Book Theme" +msgstr "Sphinx boktema" + +msgid "suggest edit" +msgstr "foreslå redigering" + +msgid "Open an issue" +msgstr "Åpne et problem" + +msgid "Launch" +msgstr "Start" + +msgid "Fullscreen mode" +msgstr "Fullskjerm-modus" + +msgid "Edit this page" +msgstr "Rediger denne siden" + +msgid "By the" +msgstr "Ved" + +msgid "next page" +msgstr "neste side" diff --git a/_static/locales/pl/LC_MESSAGES/booktheme.mo b/_static/locales/pl/LC_MESSAGES/booktheme.mo new file mode 100644 index 0000000..9ebb584 Binary files /dev/null and b/_static/locales/pl/LC_MESSAGES/booktheme.mo differ diff --git a/_static/locales/pl/LC_MESSAGES/booktheme.po b/_static/locales/pl/LC_MESSAGES/booktheme.po new file mode 100644 index 0000000..1b7233f --- /dev/null +++ b/_static/locales/pl/LC_MESSAGES/booktheme.po @@ -0,0 +1,75 @@ + +msgid "" +msgstr "" +"Project-Id-Version: Sphinx-Book-Theme\n" +"MIME-Version: 1.0\n" +"Content-Type: text/plain; charset=UTF-8\n" +"Content-Transfer-Encoding: 8bit\n" +"Language: pl\n" +"Plural-Forms: nplurals=2; plural=(n != 1);\n" + +msgid "Print to PDF" +msgstr "Drukuj do PDF" + +msgid "Theme by the" +msgstr "Motyw autorstwa" + +msgid "Download source file" +msgstr "Pobierz plik źródłowy" + +msgid "open issue" +msgstr "otwarty problem" + +msgid "Contents" +msgstr "Zawartość" + +msgid "previous page" +msgstr "Poprzednia strona" + +msgid "Download notebook file" +msgstr "Pobierz plik notatnika" + +msgid "Copyright" +msgstr "prawa autorskie" + +msgid "Download this page" +msgstr "Pobierz tę stronę" + +msgid "Source repository" +msgstr "Repozytorium źródłowe" + +msgid "By" +msgstr "Przez" + +msgid "repository" +msgstr "magazyn" + +msgid "Last updated on" +msgstr "Ostatnia aktualizacja" + +msgid "Toggle navigation" +msgstr "Przełącz nawigację" + +msgid "Sphinx Book Theme" +msgstr "Motyw książki Sphinx" + +msgid "suggest edit" +msgstr "zaproponuj edycję" + +msgid "Open an issue" +msgstr "Otwórz problem" + +msgid "Launch" +msgstr "Uruchomić" + +msgid "Fullscreen mode" +msgstr "Pełny ekran" + +msgid "Edit this page" +msgstr "Edytuj tę strone" + +msgid "By the" +msgstr "Przez" + +msgid "next page" +msgstr "Następna strona" diff --git a/_static/locales/pt/LC_MESSAGES/booktheme.mo b/_static/locales/pt/LC_MESSAGES/booktheme.mo new file mode 100644 index 0000000..d0ddb87 Binary files /dev/null and b/_static/locales/pt/LC_MESSAGES/booktheme.mo differ diff --git a/_static/locales/pt/LC_MESSAGES/booktheme.po b/_static/locales/pt/LC_MESSAGES/booktheme.po new file mode 100644 index 0000000..1b27314 --- /dev/null +++ b/_static/locales/pt/LC_MESSAGES/booktheme.po @@ -0,0 +1,75 @@ + +msgid "" +msgstr "" +"Project-Id-Version: Sphinx-Book-Theme\n" +"MIME-Version: 1.0\n" +"Content-Type: text/plain; charset=UTF-8\n" +"Content-Transfer-Encoding: 8bit\n" +"Language: pt\n" +"Plural-Forms: nplurals=2; plural=(n != 1);\n" + +msgid "Print to PDF" +msgstr "Imprimir em PDF" + +msgid "Theme by the" +msgstr "Tema por" + +msgid "Download source file" +msgstr "Baixar arquivo fonte" + +msgid "open issue" +msgstr "questão aberta" + +msgid "Contents" +msgstr "Conteúdo" + +msgid "previous page" +msgstr "página anterior" + +msgid "Download notebook file" +msgstr "Baixar arquivo de notebook" + +msgid "Copyright" +msgstr "direito autoral" + +msgid "Download this page" +msgstr "Baixe esta página" + +msgid "Source repository" +msgstr "Repositório fonte" + +msgid "By" +msgstr "De" + +msgid "repository" +msgstr "repositório" + +msgid "Last updated on" +msgstr "Última atualização em" + +msgid "Toggle navigation" +msgstr "Alternar de navegação" + +msgid "Sphinx Book Theme" +msgstr "Tema do livro Sphinx" + +msgid "suggest edit" +msgstr "sugerir edição" + +msgid "Open an issue" +msgstr "Abra um problema" + +msgid "Launch" +msgstr "Lançamento" + +msgid "Fullscreen mode" +msgstr "Modo tela cheia" + +msgid "Edit this page" +msgstr "Edite essa página" + +msgid "By the" +msgstr "Pelo" + +msgid "next page" +msgstr "próxima página" diff --git a/_static/locales/ro/LC_MESSAGES/booktheme.mo b/_static/locales/ro/LC_MESSAGES/booktheme.mo new file mode 100644 index 0000000..3c36ab1 Binary files /dev/null and b/_static/locales/ro/LC_MESSAGES/booktheme.mo differ diff --git a/_static/locales/ro/LC_MESSAGES/booktheme.po b/_static/locales/ro/LC_MESSAGES/booktheme.po new file mode 100644 index 0000000..1783ad2 --- /dev/null +++ b/_static/locales/ro/LC_MESSAGES/booktheme.po @@ -0,0 +1,75 @@ + +msgid "" +msgstr "" +"Project-Id-Version: Sphinx-Book-Theme\n" +"MIME-Version: 1.0\n" +"Content-Type: text/plain; charset=UTF-8\n" +"Content-Transfer-Encoding: 8bit\n" +"Language: ro\n" +"Plural-Forms: nplurals=2; plural=(n != 1);\n" + +msgid "Print to PDF" +msgstr "Imprimați în PDF" + +msgid "Theme by the" +msgstr "Tema de" + +msgid "Download source file" +msgstr "Descărcați fișierul sursă" + +msgid "open issue" +msgstr "problema deschisă" + +msgid "Contents" +msgstr "Cuprins" + +msgid "previous page" +msgstr "pagina anterioară" + +msgid "Download notebook file" +msgstr "Descărcați fișierul notebook" + +msgid "Copyright" +msgstr "Drepturi de autor" + +msgid "Download this page" +msgstr "Descarcă această pagină" + +msgid "Source repository" +msgstr "Depozit sursă" + +msgid "By" +msgstr "De" + +msgid "repository" +msgstr "repertoriu" + +msgid "Last updated on" +msgstr "Ultima actualizare la" + +msgid "Toggle navigation" +msgstr "Comutare navigare" + +msgid "Sphinx Book Theme" +msgstr "Tema Sphinx Book" + +msgid "suggest edit" +msgstr "sugerează editare" + +msgid "Open an issue" +msgstr "Deschideți o problemă" + +msgid "Launch" +msgstr "Lansa" + +msgid "Fullscreen mode" +msgstr "Modul ecran întreg" + +msgid "Edit this page" +msgstr "Editați această pagină" + +msgid "By the" +msgstr "Langa" + +msgid "next page" +msgstr "pagina următoare" diff --git a/_static/locales/ru/LC_MESSAGES/booktheme.mo b/_static/locales/ru/LC_MESSAGES/booktheme.mo new file mode 100644 index 0000000..6b8ca41 Binary files /dev/null and b/_static/locales/ru/LC_MESSAGES/booktheme.mo differ diff --git a/_static/locales/ru/LC_MESSAGES/booktheme.po b/_static/locales/ru/LC_MESSAGES/booktheme.po new file mode 100644 index 0000000..b1176b7 --- /dev/null +++ b/_static/locales/ru/LC_MESSAGES/booktheme.po @@ -0,0 +1,75 @@ + +msgid "" +msgstr "" +"Project-Id-Version: Sphinx-Book-Theme\n" +"MIME-Version: 1.0\n" +"Content-Type: text/plain; charset=UTF-8\n" +"Content-Transfer-Encoding: 8bit\n" +"Language: ru\n" +"Plural-Forms: nplurals=2; plural=(n != 1);\n" + +msgid "Print to PDF" +msgstr "Распечатать в PDF" + +msgid "Theme by the" +msgstr "Тема от" + +msgid "Download source file" +msgstr "Скачать исходный файл" + +msgid "open issue" +msgstr "открытый вопрос" + +msgid "Contents" +msgstr "Содержание" + +msgid "previous page" +msgstr "Предыдущая страница" + +msgid "Download notebook file" +msgstr "Скачать файл записной книжки" + +msgid "Copyright" +msgstr "авторское право" + +msgid "Download this page" +msgstr "Загрузите эту страницу" + +msgid "Source repository" +msgstr "Исходный репозиторий" + +msgid "By" +msgstr "По" + +msgid "repository" +msgstr "хранилище" + +msgid "Last updated on" +msgstr "Последнее обновление" + +msgid "Toggle navigation" +msgstr "Переключить навигацию" + +msgid "Sphinx Book Theme" +msgstr "Тема книги Сфинкс" + +msgid "suggest edit" +msgstr "предложить редактировать" + +msgid "Open an issue" +msgstr "Открыть вопрос" + +msgid "Launch" +msgstr "Запуск" + +msgid "Fullscreen mode" +msgstr "Полноэкранный режим" + +msgid "Edit this page" +msgstr "Редактировать эту страницу" + +msgid "By the" +msgstr "Посредством" + +msgid "next page" +msgstr "Следующая страница" diff --git a/_static/locales/sk/LC_MESSAGES/booktheme.mo b/_static/locales/sk/LC_MESSAGES/booktheme.mo new file mode 100644 index 0000000..59bd0dd Binary files /dev/null and b/_static/locales/sk/LC_MESSAGES/booktheme.mo differ diff --git a/_static/locales/sk/LC_MESSAGES/booktheme.po b/_static/locales/sk/LC_MESSAGES/booktheme.po new file mode 100644 index 0000000..6501288 --- /dev/null +++ b/_static/locales/sk/LC_MESSAGES/booktheme.po @@ -0,0 +1,75 @@ + +msgid "" +msgstr "" +"Project-Id-Version: Sphinx-Book-Theme\n" +"MIME-Version: 1.0\n" +"Content-Type: text/plain; charset=UTF-8\n" +"Content-Transfer-Encoding: 8bit\n" +"Language: sk\n" +"Plural-Forms: nplurals=2; plural=(n != 1);\n" + +msgid "Print to PDF" +msgstr "Tlač do PDF" + +msgid "Theme by the" +msgstr "Téma od" + +msgid "Download source file" +msgstr "Stiahnite si zdrojový súbor" + +msgid "open issue" +msgstr "otvorené vydanie" + +msgid "Contents" +msgstr "Obsah" + +msgid "previous page" +msgstr "predchádzajúca strana" + +msgid "Download notebook file" +msgstr "Stiahnite si zošit" + +msgid "Copyright" +msgstr "Autorské práva" + +msgid "Download this page" +msgstr "Stiahnite si túto stránku" + +msgid "Source repository" +msgstr "Zdrojové úložisko" + +msgid "By" +msgstr "Autor:" + +msgid "repository" +msgstr "Úložisko" + +msgid "Last updated on" +msgstr "Posledná aktualizácia dňa" + +msgid "Toggle navigation" +msgstr "Prepnúť navigáciu" + +msgid "Sphinx Book Theme" +msgstr "Téma knihy Sfinga" + +msgid "suggest edit" +msgstr "navrhnúť úpravu" + +msgid "Open an issue" +msgstr "Otvorte problém" + +msgid "Launch" +msgstr "Spustiť" + +msgid "Fullscreen mode" +msgstr "Režim celej obrazovky" + +msgid "Edit this page" +msgstr "Upraviť túto stránku" + +msgid "By the" +msgstr "Podľa" + +msgid "next page" +msgstr "ďalšia strana" diff --git a/_static/locales/sl/LC_MESSAGES/booktheme.mo b/_static/locales/sl/LC_MESSAGES/booktheme.mo new file mode 100644 index 0000000..87bf26d Binary files /dev/null and b/_static/locales/sl/LC_MESSAGES/booktheme.mo differ diff --git a/_static/locales/sl/LC_MESSAGES/booktheme.po b/_static/locales/sl/LC_MESSAGES/booktheme.po new file mode 100644 index 0000000..3c7e3a8 --- /dev/null +++ b/_static/locales/sl/LC_MESSAGES/booktheme.po @@ -0,0 +1,75 @@ + +msgid "" +msgstr "" +"Project-Id-Version: Sphinx-Book-Theme\n" +"MIME-Version: 1.0\n" +"Content-Type: text/plain; charset=UTF-8\n" +"Content-Transfer-Encoding: 8bit\n" +"Language: sl\n" +"Plural-Forms: nplurals=2; plural=(n != 1);\n" + +msgid "Print to PDF" +msgstr "Natisni v PDF" + +msgid "Theme by the" +msgstr "Tema avtorja" + +msgid "Download source file" +msgstr "Prenesite izvorno datoteko" + +msgid "open issue" +msgstr "odprto vprašanje" + +msgid "Contents" +msgstr "Vsebina" + +msgid "previous page" +msgstr "Prejšnja stran" + +msgid "Download notebook file" +msgstr "Prenesite datoteko zvezka" + +msgid "Copyright" +msgstr "avtorske pravice" + +msgid "Download this page" +msgstr "Prenesite to stran" + +msgid "Source repository" +msgstr "Izvorno skladišče" + +msgid "By" +msgstr "Avtor" + +msgid "repository" +msgstr "odlagališče" + +msgid "Last updated on" +msgstr "Nazadnje posodobljeno dne" + +msgid "Toggle navigation" +msgstr "Preklopi navigacijo" + +msgid "Sphinx Book Theme" +msgstr "Tema knjige Sphinx" + +msgid "suggest edit" +msgstr "predlagajte urejanje" + +msgid "Open an issue" +msgstr "Odprite številko" + +msgid "Launch" +msgstr "Kosilo" + +msgid "Fullscreen mode" +msgstr "Celozaslonski način" + +msgid "Edit this page" +msgstr "Uredite to stran" + +msgid "By the" +msgstr "Avtor" + +msgid "next page" +msgstr "Naslednja stran" diff --git a/_static/locales/sr/LC_MESSAGES/booktheme.mo b/_static/locales/sr/LC_MESSAGES/booktheme.mo new file mode 100644 index 0000000..ec740f4 Binary files /dev/null and b/_static/locales/sr/LC_MESSAGES/booktheme.mo differ diff --git a/_static/locales/sr/LC_MESSAGES/booktheme.po b/_static/locales/sr/LC_MESSAGES/booktheme.po new file mode 100644 index 0000000..773b8ad --- /dev/null +++ b/_static/locales/sr/LC_MESSAGES/booktheme.po @@ -0,0 +1,75 @@ + +msgid "" +msgstr "" +"Project-Id-Version: Sphinx-Book-Theme\n" +"MIME-Version: 1.0\n" +"Content-Type: text/plain; charset=UTF-8\n" +"Content-Transfer-Encoding: 8bit\n" +"Language: sr\n" +"Plural-Forms: nplurals=2; plural=(n != 1);\n" + +msgid "Print to PDF" +msgstr "Испис у ПДФ" + +msgid "Theme by the" +msgstr "Тхеме би" + +msgid "Download source file" +msgstr "Преузми изворну датотеку" + +msgid "open issue" +msgstr "отворено издање" + +msgid "Contents" +msgstr "Садржај" + +msgid "previous page" +msgstr "Претходна страница" + +msgid "Download notebook file" +msgstr "Преузмите датотеку бележнице" + +msgid "Copyright" +msgstr "Ауторско право" + +msgid "Download this page" +msgstr "Преузмите ову страницу" + +msgid "Source repository" +msgstr "Изворно спремиште" + +msgid "By" +msgstr "Од стране" + +msgid "repository" +msgstr "спремиште" + +msgid "Last updated on" +msgstr "Последње ажурирање" + +msgid "Toggle navigation" +msgstr "Укључи / искључи навигацију" + +msgid "Sphinx Book Theme" +msgstr "Тема књиге Спхинк" + +msgid "suggest edit" +msgstr "предложи уређивање" + +msgid "Open an issue" +msgstr "Отворите издање" + +msgid "Launch" +msgstr "Лансирање" + +msgid "Fullscreen mode" +msgstr "Режим целог екрана" + +msgid "Edit this page" +msgstr "Уредите ову страницу" + +msgid "By the" +msgstr "Од" + +msgid "next page" +msgstr "Следећа страна" diff --git a/_static/locales/sv/LC_MESSAGES/booktheme.mo b/_static/locales/sv/LC_MESSAGES/booktheme.mo new file mode 100644 index 0000000..b07dc76 Binary files /dev/null and b/_static/locales/sv/LC_MESSAGES/booktheme.mo differ diff --git a/_static/locales/sv/LC_MESSAGES/booktheme.po b/_static/locales/sv/LC_MESSAGES/booktheme.po new file mode 100644 index 0000000..bcac54c --- /dev/null +++ b/_static/locales/sv/LC_MESSAGES/booktheme.po @@ -0,0 +1,75 @@ + +msgid "" +msgstr "" +"Project-Id-Version: Sphinx-Book-Theme\n" +"MIME-Version: 1.0\n" +"Content-Type: text/plain; charset=UTF-8\n" +"Content-Transfer-Encoding: 8bit\n" +"Language: sv\n" +"Plural-Forms: nplurals=2; plural=(n != 1);\n" + +msgid "Print to PDF" +msgstr "Skriv ut till PDF" + +msgid "Theme by the" +msgstr "Tema av" + +msgid "Download source file" +msgstr "Ladda ner källfil" + +msgid "open issue" +msgstr "öppna problemrapport" + +msgid "Contents" +msgstr "Innehåll" + +msgid "previous page" +msgstr "föregående sida" + +msgid "Download notebook file" +msgstr "Ladda ner notebook-fil" + +msgid "Copyright" +msgstr "Upphovsrätt" + +msgid "Download this page" +msgstr "Ladda ner den här sidan" + +msgid "Source repository" +msgstr "Källkodsrepositorium" + +msgid "By" +msgstr "Av" + +msgid "repository" +msgstr "repositorium" + +msgid "Last updated on" +msgstr "Senast uppdaterad den" + +msgid "Toggle navigation" +msgstr "Växla navigering" + +msgid "Sphinx Book Theme" +msgstr "Sphinx Boktema" + +msgid "suggest edit" +msgstr "föreslå ändring" + +msgid "Open an issue" +msgstr "Öppna en problemrapport" + +msgid "Launch" +msgstr "Öppna" + +msgid "Fullscreen mode" +msgstr "Fullskärmsläge" + +msgid "Edit this page" +msgstr "Redigera den här sidan" + +msgid "By the" +msgstr "Av den" + +msgid "next page" +msgstr "nästa sida" diff --git a/_static/locales/ta/LC_MESSAGES/booktheme.mo b/_static/locales/ta/LC_MESSAGES/booktheme.mo new file mode 100644 index 0000000..29f52e1 Binary files /dev/null and b/_static/locales/ta/LC_MESSAGES/booktheme.mo differ diff --git a/_static/locales/ta/LC_MESSAGES/booktheme.po b/_static/locales/ta/LC_MESSAGES/booktheme.po new file mode 100644 index 0000000..b48bdfa --- /dev/null +++ b/_static/locales/ta/LC_MESSAGES/booktheme.po @@ -0,0 +1,66 @@ + +msgid "" +msgstr "" +"Project-Id-Version: Sphinx-Book-Theme\n" +"MIME-Version: 1.0\n" +"Content-Type: text/plain; charset=UTF-8\n" +"Content-Transfer-Encoding: 8bit\n" +"Language: ta\n" +"Plural-Forms: nplurals=2; plural=(n != 1);\n" + +msgid "Print to PDF" +msgstr "PDF இல் அச்சிடுக" + +msgid "Theme by the" +msgstr "வழங்கிய தீம்" + +msgid "Download source file" +msgstr "மூல கோப்பைப் பதிவிறக்குக" + +msgid "open issue" +msgstr "திறந்த பிரச்சினை" + +msgid "previous page" +msgstr "முந்தைய பக்கம்" + +msgid "Download notebook file" +msgstr "நோட்புக் கோப்பைப் பதிவிறக்கவும்" + +msgid "Copyright" +msgstr "பதிப்புரிமை" + +msgid "Download this page" +msgstr "இந்தப் பக்கத்தைப் பதிவிறக்கவும்" + +msgid "Source repository" +msgstr "மூல களஞ்சியம்" + +msgid "By" +msgstr "வழங்கியவர்" + +msgid "Last updated on" +msgstr "கடைசியாக புதுப்பிக்கப்பட்டது" + +msgid "Toggle navigation" +msgstr "வழிசெலுத்தலை நிலைமாற்று" + +msgid "Sphinx Book Theme" +msgstr "ஸ்பிங்க்ஸ் புத்தக தீம்" + +msgid "suggest edit" +msgstr "திருத்த பரிந்துரைக்கவும்" + +msgid "Open an issue" +msgstr "சிக்கலைத் திறக்கவும்" + +msgid "Launch" +msgstr "தொடங்க" + +msgid "Edit this page" +msgstr "இந்தப் பக்கத்தைத் திருத்தவும்" + +msgid "By the" +msgstr "மூலம்" + +msgid "next page" +msgstr "அடுத்த பக்கம்" diff --git a/_static/locales/te/LC_MESSAGES/booktheme.mo b/_static/locales/te/LC_MESSAGES/booktheme.mo new file mode 100644 index 0000000..0a5f4b4 Binary files /dev/null and b/_static/locales/te/LC_MESSAGES/booktheme.mo differ diff --git a/_static/locales/te/LC_MESSAGES/booktheme.po b/_static/locales/te/LC_MESSAGES/booktheme.po new file mode 100644 index 0000000..952278f --- /dev/null +++ b/_static/locales/te/LC_MESSAGES/booktheme.po @@ -0,0 +1,66 @@ + +msgid "" +msgstr "" +"Project-Id-Version: Sphinx-Book-Theme\n" +"MIME-Version: 1.0\n" +"Content-Type: text/plain; charset=UTF-8\n" +"Content-Transfer-Encoding: 8bit\n" +"Language: te\n" +"Plural-Forms: nplurals=2; plural=(n != 1);\n" + +msgid "Print to PDF" +msgstr "PDF కి ముద్రించండి" + +msgid "Theme by the" +msgstr "ద్వారా థీమ్" + +msgid "Download source file" +msgstr "మూల ఫైల్‌ను డౌన్‌లోడ్ చేయండి" + +msgid "open issue" +msgstr "ఓపెన్ ఇష్యూ" + +msgid "previous page" +msgstr "ముందు పేజి" + +msgid "Download notebook file" +msgstr "నోట్బుక్ ఫైల్ను డౌన్లోడ్ చేయండి" + +msgid "Copyright" +msgstr "కాపీరైట్" + +msgid "Download this page" +msgstr "ఈ పేజీని డౌన్‌లోడ్ చేయండి" + +msgid "Source repository" +msgstr "మూల రిపోజిటరీ" + +msgid "By" +msgstr "ద్వారా" + +msgid "Last updated on" +msgstr "చివరిగా నవీకరించబడింది" + +msgid "Toggle navigation" +msgstr "నావిగేషన్‌ను టోగుల్ చేయండి" + +msgid "Sphinx Book Theme" +msgstr "సింహిక పుస్తక థీమ్" + +msgid "suggest edit" +msgstr "సవరించమని సూచించండి" + +msgid "Open an issue" +msgstr "సమస్యను తెరవండి" + +msgid "Launch" +msgstr "ప్రారంభించండి" + +msgid "Edit this page" +msgstr "ఈ పేజీని సవరించండి" + +msgid "By the" +msgstr "ద్వారా" + +msgid "next page" +msgstr "తరువాతి పేజీ" diff --git a/_static/locales/tg/LC_MESSAGES/booktheme.mo b/_static/locales/tg/LC_MESSAGES/booktheme.mo new file mode 100644 index 0000000..b21c6c6 Binary files /dev/null and b/_static/locales/tg/LC_MESSAGES/booktheme.mo differ diff --git a/_static/locales/tg/LC_MESSAGES/booktheme.po b/_static/locales/tg/LC_MESSAGES/booktheme.po new file mode 100644 index 0000000..c33dc42 --- /dev/null +++ b/_static/locales/tg/LC_MESSAGES/booktheme.po @@ -0,0 +1,75 @@ + +msgid "" +msgstr "" +"Project-Id-Version: Sphinx-Book-Theme\n" +"MIME-Version: 1.0\n" +"Content-Type: text/plain; charset=UTF-8\n" +"Content-Transfer-Encoding: 8bit\n" +"Language: tg\n" +"Plural-Forms: nplurals=2; plural=(n != 1);\n" + +msgid "Print to PDF" +msgstr "Чоп ба PDF" + +msgid "Theme by the" +msgstr "Мавзӯъи аз" + +msgid "Download source file" +msgstr "Файли манбаъро зеркашӣ кунед" + +msgid "open issue" +msgstr "барориши кушод" + +msgid "Contents" +msgstr "Мундариҷа" + +msgid "previous page" +msgstr "саҳифаи қаблӣ" + +msgid "Download notebook file" +msgstr "Файли дафтарро зеркашӣ кунед" + +msgid "Copyright" +msgstr "Ҳуқуқи муаллиф" + +msgid "Download this page" +msgstr "Ин саҳифаро зеркашӣ кунед" + +msgid "Source repository" +msgstr "Анбори манбаъ" + +msgid "By" +msgstr "Бо" + +msgid "repository" +msgstr "анбор" + +msgid "Last updated on" +msgstr "Last навсозӣ дар" + +msgid "Toggle navigation" +msgstr "Гузаришро иваз кунед" + +msgid "Sphinx Book Theme" +msgstr "Сфинкс Мавзӯи китоб" + +msgid "suggest edit" +msgstr "пешниҳод вироиш" + +msgid "Open an issue" +msgstr "Масъаларо кушоед" + +msgid "Launch" +msgstr "Оғоз" + +msgid "Fullscreen mode" +msgstr "Ҳолати экрани пурра" + +msgid "Edit this page" +msgstr "Ин саҳифаро таҳрир кунед" + +msgid "By the" +msgstr "Бо" + +msgid "next page" +msgstr "саҳифаи оянда" diff --git a/_static/locales/th/LC_MESSAGES/booktheme.mo b/_static/locales/th/LC_MESSAGES/booktheme.mo new file mode 100644 index 0000000..abede98 Binary files /dev/null and b/_static/locales/th/LC_MESSAGES/booktheme.mo differ diff --git a/_static/locales/th/LC_MESSAGES/booktheme.po b/_static/locales/th/LC_MESSAGES/booktheme.po new file mode 100644 index 0000000..9d24294 --- /dev/null +++ b/_static/locales/th/LC_MESSAGES/booktheme.po @@ -0,0 +1,75 @@ + +msgid "" +msgstr "" +"Project-Id-Version: Sphinx-Book-Theme\n" +"MIME-Version: 1.0\n" +"Content-Type: text/plain; charset=UTF-8\n" +"Content-Transfer-Encoding: 8bit\n" +"Language: th\n" +"Plural-Forms: nplurals=2; plural=(n != 1);\n" + +msgid "Print to PDF" +msgstr "พิมพ์เป็น PDF" + +msgid "Theme by the" +msgstr "ธีมโดย" + +msgid "Download source file" +msgstr "ดาวน์โหลดไฟล์ต้นฉบับ" + +msgid "open issue" +msgstr "เปิดปัญหา" + +msgid "Contents" +msgstr "สารบัญ" + +msgid "previous page" +msgstr "หน้าที่แล้ว" + +msgid "Download notebook file" +msgstr "ดาวน์โหลดไฟล์สมุดบันทึก" + +msgid "Copyright" +msgstr "ลิขสิทธิ์" + +msgid "Download this page" +msgstr "ดาวน์โหลดหน้านี้" + +msgid "Source repository" +msgstr "ที่เก็บซอร์ส" + +msgid "By" +msgstr "โดย" + +msgid "repository" +msgstr "ที่เก็บ" + +msgid "Last updated on" +msgstr "ปรับปรุงล่าสุดเมื่อ" + +msgid "Toggle navigation" +msgstr "ไม่ต้องสลับช่องทาง" + +msgid "Sphinx Book Theme" +msgstr "ธีมหนังสือสฟิงซ์" + +msgid "suggest edit" +msgstr "แนะนำแก้ไข" + +msgid "Open an issue" +msgstr "เปิดปัญหา" + +msgid "Launch" +msgstr "เปิด" + +msgid "Fullscreen mode" +msgstr "โหมดเต็มหน้าจอ" + +msgid "Edit this page" +msgstr "แก้ไขหน้านี้" + +msgid "By the" +msgstr "โดย" + +msgid "next page" +msgstr "หน้าต่อไป" diff --git a/_static/locales/tl/LC_MESSAGES/booktheme.mo b/_static/locales/tl/LC_MESSAGES/booktheme.mo new file mode 100644 index 0000000..8df1b73 Binary files /dev/null and b/_static/locales/tl/LC_MESSAGES/booktheme.mo differ diff --git a/_static/locales/tl/LC_MESSAGES/booktheme.po b/_static/locales/tl/LC_MESSAGES/booktheme.po new file mode 100644 index 0000000..20e0d07 --- /dev/null +++ b/_static/locales/tl/LC_MESSAGES/booktheme.po @@ -0,0 +1,66 @@ + +msgid "" +msgstr "" +"Project-Id-Version: Sphinx-Book-Theme\n" +"MIME-Version: 1.0\n" +"Content-Type: text/plain; charset=UTF-8\n" +"Content-Transfer-Encoding: 8bit\n" +"Language: tl\n" +"Plural-Forms: nplurals=2; plural=(n != 1);\n" + +msgid "Print to PDF" +msgstr "I-print sa PDF" + +msgid "Theme by the" +msgstr "Tema ng" + +msgid "Download source file" +msgstr "Mag-download ng file ng pinagmulan" + +msgid "open issue" +msgstr "bukas na isyu" + +msgid "previous page" +msgstr "Nakaraang pahina" + +msgid "Download notebook file" +msgstr "Mag-download ng file ng notebook" + +msgid "Copyright" +msgstr "Copyright" + +msgid "Download this page" +msgstr "I-download ang pahinang ito" + +msgid "Source repository" +msgstr "Pinagmulan ng imbakan" + +msgid "By" +msgstr "Ni" + +msgid "Last updated on" +msgstr "Huling na-update noong" + +msgid "Toggle navigation" +msgstr "I-toggle ang pag-navigate" + +msgid "Sphinx Book Theme" +msgstr "Tema ng Sphinx Book" + +msgid "suggest edit" +msgstr "iminumungkahi i-edit" + +msgid "Open an issue" +msgstr "Magbukas ng isyu" + +msgid "Launch" +msgstr "Ilunsad" + +msgid "Edit this page" +msgstr "I-edit ang pahinang ito" + +msgid "By the" +msgstr "Sa pamamagitan ng" + +msgid "next page" +msgstr "Susunod na pahina" diff --git a/_static/locales/tr/LC_MESSAGES/booktheme.mo b/_static/locales/tr/LC_MESSAGES/booktheme.mo new file mode 100644 index 0000000..029ae18 Binary files /dev/null and b/_static/locales/tr/LC_MESSAGES/booktheme.mo differ diff --git a/_static/locales/tr/LC_MESSAGES/booktheme.po b/_static/locales/tr/LC_MESSAGES/booktheme.po new file mode 100644 index 0000000..a77eb02 --- /dev/null +++ b/_static/locales/tr/LC_MESSAGES/booktheme.po @@ -0,0 +1,75 @@ + +msgid "" +msgstr "" +"Project-Id-Version: Sphinx-Book-Theme\n" +"MIME-Version: 1.0\n" +"Content-Type: text/plain; charset=UTF-8\n" +"Content-Transfer-Encoding: 8bit\n" +"Language: tr\n" +"Plural-Forms: nplurals=2; plural=(n != 1);\n" + +msgid "Print to PDF" +msgstr "PDF olarak yazdır" + +msgid "Theme by the" +msgstr "Tarafından tema" + +msgid "Download source file" +msgstr "Kaynak dosyayı indirin" + +msgid "open issue" +msgstr "Açık konu" + +msgid "Contents" +msgstr "İçindekiler" + +msgid "previous page" +msgstr "önceki sayfa" + +msgid "Download notebook file" +msgstr "Defter dosyasını indirin" + +msgid "Copyright" +msgstr "Telif hakkı" + +msgid "Download this page" +msgstr "Bu sayfayı indirin" + +msgid "Source repository" +msgstr "Kaynak kod deposu" + +msgid "By" +msgstr "Tarafından" + +msgid "repository" +msgstr "depo" + +msgid "Last updated on" +msgstr "Son güncelleme tarihi" + +msgid "Toggle navigation" +msgstr "Gezinmeyi değiştir" + +msgid "Sphinx Book Theme" +msgstr "Sfenks Kitap Teması" + +msgid "suggest edit" +msgstr "düzenleme öner" + +msgid "Open an issue" +msgstr "Bir sorunu açın" + +msgid "Launch" +msgstr "Başlatmak" + +msgid "Fullscreen mode" +msgstr "Tam ekran modu" + +msgid "Edit this page" +msgstr "Bu sayfayı düzenle" + +msgid "By the" +msgstr "Tarafından" + +msgid "next page" +msgstr "sonraki Sayfa" diff --git a/_static/locales/uk/LC_MESSAGES/booktheme.mo b/_static/locales/uk/LC_MESSAGES/booktheme.mo new file mode 100644 index 0000000..16ab789 Binary files /dev/null and b/_static/locales/uk/LC_MESSAGES/booktheme.mo differ diff --git a/_static/locales/uk/LC_MESSAGES/booktheme.po b/_static/locales/uk/LC_MESSAGES/booktheme.po new file mode 100644 index 0000000..993dd07 --- /dev/null +++ b/_static/locales/uk/LC_MESSAGES/booktheme.po @@ -0,0 +1,75 @@ + +msgid "" +msgstr "" +"Project-Id-Version: Sphinx-Book-Theme\n" +"MIME-Version: 1.0\n" +"Content-Type: text/plain; charset=UTF-8\n" +"Content-Transfer-Encoding: 8bit\n" +"Language: uk\n" +"Plural-Forms: nplurals=2; plural=(n != 1);\n" + +msgid "Print to PDF" +msgstr "Друк у форматі PDF" + +msgid "Theme by the" +msgstr "Тема від" + +msgid "Download source file" +msgstr "Завантажити вихідний файл" + +msgid "open issue" +msgstr "відкритий випуск" + +msgid "Contents" +msgstr "Зміст" + +msgid "previous page" +msgstr "Попередня сторінка" + +msgid "Download notebook file" +msgstr "Завантажте файл блокнота" + +msgid "Copyright" +msgstr "Авторське право" + +msgid "Download this page" +msgstr "Завантажте цю сторінку" + +msgid "Source repository" +msgstr "Джерело сховища" + +msgid "By" +msgstr "Автор" + +msgid "repository" +msgstr "сховище" + +msgid "Last updated on" +msgstr "Останнє оновлення:" + +msgid "Toggle navigation" +msgstr "Переключити навігацію" + +msgid "Sphinx Book Theme" +msgstr "Тема книги \"Сфінкс\"" + +msgid "suggest edit" +msgstr "запропонувати редагувати" + +msgid "Open an issue" +msgstr "Відкрийте випуск" + +msgid "Launch" +msgstr "Запуск" + +msgid "Fullscreen mode" +msgstr "Повноекранний режим" + +msgid "Edit this page" +msgstr "Редагувати цю сторінку" + +msgid "By the" +msgstr "По" + +msgid "next page" +msgstr "Наступна сторінка" diff --git a/_static/locales/ur/LC_MESSAGES/booktheme.mo b/_static/locales/ur/LC_MESSAGES/booktheme.mo new file mode 100644 index 0000000..de8c84b Binary files /dev/null and b/_static/locales/ur/LC_MESSAGES/booktheme.mo differ diff --git a/_static/locales/ur/LC_MESSAGES/booktheme.po b/_static/locales/ur/LC_MESSAGES/booktheme.po new file mode 100644 index 0000000..2f77426 --- /dev/null +++ b/_static/locales/ur/LC_MESSAGES/booktheme.po @@ -0,0 +1,66 @@ + +msgid "" +msgstr "" +"Project-Id-Version: Sphinx-Book-Theme\n" +"MIME-Version: 1.0\n" +"Content-Type: text/plain; charset=UTF-8\n" +"Content-Transfer-Encoding: 8bit\n" +"Language: ur\n" +"Plural-Forms: nplurals=2; plural=(n != 1);\n" + +msgid "Print to PDF" +msgstr "پی ڈی ایف پرنٹ کریں" + +msgid "Theme by the" +msgstr "کے ذریعہ تھیم" + +msgid "Download source file" +msgstr "سورس فائل ڈاؤن لوڈ کریں" + +msgid "open issue" +msgstr "کھلا مسئلہ" + +msgid "previous page" +msgstr "سابقہ ​​صفحہ" + +msgid "Download notebook file" +msgstr "نوٹ بک فائل ڈاؤن لوڈ کریں" + +msgid "Copyright" +msgstr "کاپی رائٹ" + +msgid "Download this page" +msgstr "اس صفحے کو ڈاؤن لوڈ کریں" + +msgid "Source repository" +msgstr "ماخذ ذخیرہ" + +msgid "By" +msgstr "بذریعہ" + +msgid "Last updated on" +msgstr "آخری بار تازہ کاری ہوئی" + +msgid "Toggle navigation" +msgstr "نیویگیشن ٹوگل کریں" + +msgid "Sphinx Book Theme" +msgstr "سپنکس بک تھیم" + +msgid "suggest edit" +msgstr "ترمیم کی تجویز کریں" + +msgid "Open an issue" +msgstr "ایک مسئلہ کھولیں" + +msgid "Launch" +msgstr "لانچ کریں" + +msgid "Edit this page" +msgstr "اس صفحے میں ترمیم کریں" + +msgid "By the" +msgstr "کی طرف" + +msgid "next page" +msgstr "اگلا صفحہ" diff --git a/_static/locales/vi/LC_MESSAGES/booktheme.mo b/_static/locales/vi/LC_MESSAGES/booktheme.mo new file mode 100644 index 0000000..2bb3255 Binary files /dev/null and b/_static/locales/vi/LC_MESSAGES/booktheme.mo differ diff --git a/_static/locales/vi/LC_MESSAGES/booktheme.po b/_static/locales/vi/LC_MESSAGES/booktheme.po new file mode 100644 index 0000000..33159f3 --- /dev/null +++ b/_static/locales/vi/LC_MESSAGES/booktheme.po @@ -0,0 +1,75 @@ + +msgid "" +msgstr "" +"Project-Id-Version: Sphinx-Book-Theme\n" +"MIME-Version: 1.0\n" +"Content-Type: text/plain; charset=UTF-8\n" +"Content-Transfer-Encoding: 8bit\n" +"Language: vi\n" +"Plural-Forms: nplurals=2; plural=(n != 1);\n" + +msgid "Print to PDF" +msgstr "In sang PDF" + +msgid "Theme by the" +msgstr "Chủ đề của" + +msgid "Download source file" +msgstr "Tải xuống tệp nguồn" + +msgid "open issue" +msgstr "vấn đề mở" + +msgid "Contents" +msgstr "Nội dung" + +msgid "previous page" +msgstr "trang trước" + +msgid "Download notebook file" +msgstr "Tải xuống tệp sổ tay" + +msgid "Copyright" +msgstr "Bản quyền" + +msgid "Download this page" +msgstr "Tải xuống trang này" + +msgid "Source repository" +msgstr "Kho nguồn" + +msgid "By" +msgstr "Bởi" + +msgid "repository" +msgstr "kho" + +msgid "Last updated on" +msgstr "Cập nhật lần cuối vào" + +msgid "Toggle navigation" +msgstr "Chuyển đổi điều hướng thành" + +msgid "Sphinx Book Theme" +msgstr "Chủ đề sách nhân sư" + +msgid "suggest edit" +msgstr "đề nghị chỉnh sửa" + +msgid "Open an issue" +msgstr "Mở một vấn đề" + +msgid "Launch" +msgstr "Phóng" + +msgid "Fullscreen mode" +msgstr "Chế độ toàn màn hình" + +msgid "Edit this page" +msgstr "chỉnh sửa trang này" + +msgid "By the" +msgstr "Bằng" + +msgid "next page" +msgstr "Trang tiếp theo" diff --git a/_static/locales/zh_CN/LC_MESSAGES/booktheme.mo b/_static/locales/zh_CN/LC_MESSAGES/booktheme.mo new file mode 100644 index 0000000..0e3235d Binary files /dev/null and b/_static/locales/zh_CN/LC_MESSAGES/booktheme.mo differ diff --git a/_static/locales/zh_CN/LC_MESSAGES/booktheme.po b/_static/locales/zh_CN/LC_MESSAGES/booktheme.po new file mode 100644 index 0000000..2e519ef --- /dev/null +++ b/_static/locales/zh_CN/LC_MESSAGES/booktheme.po @@ -0,0 +1,75 @@ + +msgid "" +msgstr "" +"Project-Id-Version: Sphinx-Book-Theme\n" +"MIME-Version: 1.0\n" +"Content-Type: text/plain; charset=UTF-8\n" +"Content-Transfer-Encoding: 8bit\n" +"Language: zh_CN\n" +"Plural-Forms: nplurals=2; plural=(n != 1);\n" + +msgid "Print to PDF" +msgstr "列印成 PDF" + +msgid "Theme by the" +msgstr "主题作者:" + +msgid "Download source file" +msgstr "下载源文件" + +msgid "open issue" +msgstr "创建议题" + +msgid "Contents" +msgstr "目录" + +msgid "previous page" +msgstr "上一页" + +msgid "Download notebook file" +msgstr "下载笔记本文件" + +msgid "Copyright" +msgstr "版权" + +msgid "Download this page" +msgstr "下载此页面" + +msgid "Source repository" +msgstr "源码库" + +msgid "By" +msgstr "作者:" + +msgid "repository" +msgstr "仓库" + +msgid "Last updated on" +msgstr "上次更新时间:" + +msgid "Toggle navigation" +msgstr "显示或隐藏导航栏" + +msgid "Sphinx Book Theme" +msgstr "Sphinx Book 主题" + +msgid "suggest edit" +msgstr "提出修改建议" + +msgid "Open an issue" +msgstr "创建议题" + +msgid "Launch" +msgstr "启动" + +msgid "Fullscreen mode" +msgstr "全屏模式" + +msgid "Edit this page" +msgstr "编辑此页面" + +msgid "By the" +msgstr "作者:" + +msgid "next page" +msgstr "下一页" diff --git a/_static/locales/zh_TW/LC_MESSAGES/booktheme.mo b/_static/locales/zh_TW/LC_MESSAGES/booktheme.mo new file mode 100644 index 0000000..9116fa9 Binary files /dev/null and b/_static/locales/zh_TW/LC_MESSAGES/booktheme.mo differ diff --git a/_static/locales/zh_TW/LC_MESSAGES/booktheme.po b/_static/locales/zh_TW/LC_MESSAGES/booktheme.po new file mode 100644 index 0000000..beecb07 --- /dev/null +++ b/_static/locales/zh_TW/LC_MESSAGES/booktheme.po @@ -0,0 +1,75 @@ + +msgid "" +msgstr "" +"Project-Id-Version: Sphinx-Book-Theme\n" +"MIME-Version: 1.0\n" +"Content-Type: text/plain; charset=UTF-8\n" +"Content-Transfer-Encoding: 8bit\n" +"Language: zh_TW\n" +"Plural-Forms: nplurals=2; plural=(n != 1);\n" + +msgid "Print to PDF" +msgstr "列印成 PDF" + +msgid "Theme by the" +msgstr "佈景主題作者:" + +msgid "Download source file" +msgstr "下載原始檔" + +msgid "open issue" +msgstr "公開的問題" + +msgid "Contents" +msgstr "目錄" + +msgid "previous page" +msgstr "上一頁" + +msgid "Download notebook file" +msgstr "下載 Notebook 檔案" + +msgid "Copyright" +msgstr "Copyright" + +msgid "Download this page" +msgstr "下載此頁面" + +msgid "Source repository" +msgstr "來源儲存庫" + +msgid "By" +msgstr "作者:" + +msgid "repository" +msgstr "儲存庫" + +msgid "Last updated on" +msgstr "最後更新時間:" + +msgid "Toggle navigation" +msgstr "顯示或隱藏導覽列" + +msgid "Sphinx Book Theme" +msgstr "Sphinx Book 佈景主題" + +msgid "suggest edit" +msgstr "提出修改建議" + +msgid "Open an issue" +msgstr "開啟議題" + +msgid "Launch" +msgstr "啟動" + +msgid "Fullscreen mode" +msgstr "全螢幕模式" + +msgid "Edit this page" +msgstr "編輯此頁面" + +msgid "By the" +msgstr "作者:" + +msgid "next page" +msgstr "下一頁" diff --git a/_static/minus.png b/_static/minus.png new file mode 100644 index 0000000..d96755f Binary files /dev/null and b/_static/minus.png differ diff --git a/_static/plus.png b/_static/plus.png new file mode 100644 index 0000000..7107cec Binary files /dev/null and b/_static/plus.png differ diff --git a/_static/program.css b/_static/program.css new file mode 100644 index 0000000..e8d3c0f --- /dev/null +++ b/_static/program.css @@ -0,0 +1,40 @@ + +/* color of top/bottom bars in nature theme */ +/* div.related { */ +/* background-color: #bc00bc; */ +/* } */ + + +/* div.body h1 { font-size: 150%; background-color: #ebb3c0; } */ +/* div.body h2 { font-size: 150%; background-color: #e2acb9; } */ +/* div.body h3 { font-size: 120%; background-color: #d3a1ac; } */ +/* div.body h4 { font-size: 110%; background-color: #d3a1ac; } */ +/* div.body h5 { font-size: 100%; background-color: #d3a1ac; } */ +/* div.body h6 { font-size: 100%; background-color: #d3a1ac; } */ + +/* div.body h1 { font-size: 150%; background-color: #BED4EB; } */ +/* div.body h2 { font-size: 150%; background-color: #C8D5E3; } */ +/* div.body h3 { font-size: 120%; background-color: #D8DEE3; } */ +/* div.body h4 { font-size: 110%; background-color: #D8DEE3; } */ +/* div.body h5 { font-size: 100%; background-color: #D8DEE3; } */ +/* div.body h6 { font-size: 100%; background-color: #D8DEE3; } */ + + + + +/* guzzle needs some table padding */ + +th, td { + padding-left: 3px; + padding-right: 3px; +} + + +.strike { + text-decoration: line-through; +} + + +#right-column { + max-width: 1100px; +} diff --git a/_static/pygments.css b/_static/pygments.css new file mode 100644 index 0000000..012e6a0 --- /dev/null +++ b/_static/pygments.css @@ -0,0 +1,152 @@ +html[data-theme="light"] .highlight pre { line-height: 125%; } +html[data-theme="light"] .highlight td.linenos .normal { color: inherit; background-color: transparent; padding-left: 5px; padding-right: 5px; } +html[data-theme="light"] .highlight span.linenos { color: inherit; background-color: transparent; padding-left: 5px; padding-right: 5px; } +html[data-theme="light"] .highlight td.linenos .special { color: #000000; background-color: #ffffc0; padding-left: 5px; padding-right: 5px; } +html[data-theme="light"] .highlight span.linenos.special { color: #000000; background-color: #ffffc0; padding-left: 5px; padding-right: 5px; } +html[data-theme="light"] .highlight .hll { background-color: #fae4c2 } +html[data-theme="light"] .highlight { background: #fefefe; color: #080808 } +html[data-theme="light"] .highlight .c { color: #515151 } /* Comment */ +html[data-theme="light"] .highlight .err { color: #a12236 } /* Error */ +html[data-theme="light"] .highlight .k { color: #6730c5 } /* Keyword */ +html[data-theme="light"] .highlight .l { color: #7f4707 } /* Literal */ +html[data-theme="light"] .highlight .n { color: #080808 } /* Name */ +html[data-theme="light"] .highlight .o { color: #00622f } /* Operator */ +html[data-theme="light"] .highlight .p { color: #080808 } /* Punctuation */ +html[data-theme="light"] .highlight .ch { color: #515151 } /* Comment.Hashbang */ +html[data-theme="light"] .highlight .cm { color: #515151 } /* Comment.Multiline */ +html[data-theme="light"] .highlight .cp { color: #515151 } /* Comment.Preproc */ +html[data-theme="light"] .highlight .cpf { color: #515151 } /* Comment.PreprocFile */ +html[data-theme="light"] .highlight .c1 { color: #515151 } /* Comment.Single */ +html[data-theme="light"] .highlight .cs { color: #515151 } /* Comment.Special */ +html[data-theme="light"] .highlight .gd { color: #005b82 } /* Generic.Deleted */ +html[data-theme="light"] .highlight .ge { font-style: italic } /* Generic.Emph */ +html[data-theme="light"] .highlight .gh { color: #005b82 } /* Generic.Heading */ +html[data-theme="light"] .highlight .gs { font-weight: bold } /* Generic.Strong */ +html[data-theme="light"] .highlight .gu { color: #005b82 } /* Generic.Subheading */ +html[data-theme="light"] .highlight .kc { color: #6730c5 } /* Keyword.Constant */ +html[data-theme="light"] .highlight .kd { color: #6730c5 } /* Keyword.Declaration */ +html[data-theme="light"] .highlight .kn { color: #6730c5 } /* Keyword.Namespace */ +html[data-theme="light"] .highlight .kp { color: #6730c5 } /* Keyword.Pseudo */ +html[data-theme="light"] .highlight .kr { color: #6730c5 } /* Keyword.Reserved */ +html[data-theme="light"] .highlight .kt { color: #7f4707 } /* Keyword.Type */ +html[data-theme="light"] .highlight .ld { color: #7f4707 } /* Literal.Date */ +html[data-theme="light"] .highlight .m { color: #7f4707 } /* Literal.Number */ +html[data-theme="light"] .highlight .s { color: #00622f } /* Literal.String */ +html[data-theme="light"] .highlight .na { color: #912583 } /* Name.Attribute */ +html[data-theme="light"] .highlight .nb { color: #7f4707 } /* Name.Builtin */ +html[data-theme="light"] .highlight .nc { color: #005b82 } /* Name.Class */ +html[data-theme="light"] .highlight .no { color: #005b82 } /* Name.Constant */ +html[data-theme="light"] .highlight .nd { color: #7f4707 } /* Name.Decorator */ +html[data-theme="light"] .highlight .ni { color: #00622f } /* Name.Entity */ +html[data-theme="light"] .highlight .ne { color: #6730c5 } /* Name.Exception */ +html[data-theme="light"] .highlight .nf { color: #005b82 } /* Name.Function */ +html[data-theme="light"] .highlight .nl { color: #7f4707 } /* Name.Label */ +html[data-theme="light"] .highlight .nn { color: #080808 } /* Name.Namespace */ +html[data-theme="light"] .highlight .nx { color: #080808 } /* Name.Other */ +html[data-theme="light"] .highlight .py { color: #005b82 } /* Name.Property */ +html[data-theme="light"] .highlight .nt { color: #005b82 } /* Name.Tag */ +html[data-theme="light"] .highlight .nv { color: #a12236 } /* Name.Variable */ +html[data-theme="light"] .highlight .ow { color: #6730c5 } /* Operator.Word */ +html[data-theme="light"] .highlight .pm { color: #080808 } /* Punctuation.Marker */ +html[data-theme="light"] .highlight .w { color: #080808 } /* Text.Whitespace */ +html[data-theme="light"] .highlight .mb { color: #7f4707 } /* Literal.Number.Bin */ +html[data-theme="light"] .highlight .mf { color: #7f4707 } /* Literal.Number.Float */ +html[data-theme="light"] .highlight .mh { color: #7f4707 } /* Literal.Number.Hex */ +html[data-theme="light"] .highlight .mi { color: #7f4707 } /* Literal.Number.Integer */ +html[data-theme="light"] .highlight .mo { color: #7f4707 } /* Literal.Number.Oct */ +html[data-theme="light"] .highlight .sa { color: #00622f } /* Literal.String.Affix */ +html[data-theme="light"] .highlight .sb { color: #00622f } /* Literal.String.Backtick */ +html[data-theme="light"] .highlight .sc { color: #00622f } /* Literal.String.Char */ +html[data-theme="light"] .highlight .dl { color: #00622f } /* Literal.String.Delimiter */ +html[data-theme="light"] .highlight .sd { color: #00622f } /* Literal.String.Doc */ +html[data-theme="light"] .highlight .s2 { color: #00622f } /* Literal.String.Double */ +html[data-theme="light"] .highlight .se { color: #00622f } /* Literal.String.Escape */ +html[data-theme="light"] .highlight .sh { color: #00622f } /* Literal.String.Heredoc */ +html[data-theme="light"] .highlight .si { color: #00622f } /* Literal.String.Interpol */ +html[data-theme="light"] .highlight .sx { color: #00622f } /* Literal.String.Other */ +html[data-theme="light"] .highlight .sr { color: #a12236 } /* Literal.String.Regex */ +html[data-theme="light"] .highlight .s1 { color: #00622f } /* Literal.String.Single */ +html[data-theme="light"] .highlight .ss { color: #005b82 } /* Literal.String.Symbol */ +html[data-theme="light"] .highlight .bp { color: #7f4707 } /* Name.Builtin.Pseudo */ +html[data-theme="light"] .highlight .fm { color: #005b82 } /* Name.Function.Magic */ +html[data-theme="light"] .highlight .vc { color: #a12236 } /* Name.Variable.Class */ +html[data-theme="light"] .highlight .vg { color: #a12236 } /* Name.Variable.Global */ +html[data-theme="light"] .highlight .vi { color: #a12236 } /* Name.Variable.Instance */ +html[data-theme="light"] .highlight .vm { color: #7f4707 } /* Name.Variable.Magic */ +html[data-theme="light"] .highlight .il { color: #7f4707 } /* Literal.Number.Integer.Long */ +html[data-theme="dark"] .highlight pre { line-height: 125%; } +html[data-theme="dark"] .highlight td.linenos .normal { color: inherit; background-color: transparent; padding-left: 5px; padding-right: 5px; } +html[data-theme="dark"] .highlight span.linenos { color: inherit; background-color: transparent; padding-left: 5px; padding-right: 5px; } +html[data-theme="dark"] .highlight td.linenos .special { color: #000000; background-color: #ffffc0; padding-left: 5px; padding-right: 5px; } +html[data-theme="dark"] .highlight span.linenos.special { color: #000000; background-color: #ffffc0; padding-left: 5px; padding-right: 5px; } +html[data-theme="dark"] .highlight .hll { background-color: #ffd9002e } +html[data-theme="dark"] .highlight { background: #2b2b2b; color: #f8f8f2 } +html[data-theme="dark"] .highlight .c { color: #ffd900 } /* Comment */ +html[data-theme="dark"] .highlight .err { color: #ffa07a } /* Error */ +html[data-theme="dark"] .highlight .k { color: #dcc6e0 } /* Keyword */ +html[data-theme="dark"] .highlight .l { color: #ffd900 } /* Literal */ +html[data-theme="dark"] .highlight .n { color: #f8f8f2 } /* Name */ +html[data-theme="dark"] .highlight .o { color: #abe338 } /* Operator */ +html[data-theme="dark"] .highlight .p { color: #f8f8f2 } /* Punctuation */ +html[data-theme="dark"] .highlight .ch { color: #ffd900 } /* Comment.Hashbang */ +html[data-theme="dark"] .highlight .cm { color: #ffd900 } /* Comment.Multiline */ +html[data-theme="dark"] .highlight .cp { color: #ffd900 } /* Comment.Preproc */ +html[data-theme="dark"] .highlight .cpf { color: #ffd900 } /* Comment.PreprocFile */ +html[data-theme="dark"] .highlight .c1 { color: #ffd900 } /* Comment.Single */ +html[data-theme="dark"] .highlight .cs { color: #ffd900 } /* Comment.Special */ +html[data-theme="dark"] .highlight .gd { color: #00e0e0 } /* Generic.Deleted */ +html[data-theme="dark"] .highlight .ge { font-style: italic } /* Generic.Emph */ +html[data-theme="dark"] .highlight .gh { color: #00e0e0 } /* Generic.Heading */ +html[data-theme="dark"] .highlight .gs { font-weight: bold } /* Generic.Strong */ +html[data-theme="dark"] .highlight .gu { color: #00e0e0 } /* Generic.Subheading */ +html[data-theme="dark"] .highlight .kc { color: #dcc6e0 } /* Keyword.Constant */ +html[data-theme="dark"] .highlight .kd { color: #dcc6e0 } /* Keyword.Declaration */ +html[data-theme="dark"] .highlight .kn { color: #dcc6e0 } /* Keyword.Namespace */ +html[data-theme="dark"] .highlight .kp { color: #dcc6e0 } /* Keyword.Pseudo */ +html[data-theme="dark"] .highlight .kr { color: #dcc6e0 } /* Keyword.Reserved */ +html[data-theme="dark"] .highlight .kt { color: #ffd900 } /* Keyword.Type */ +html[data-theme="dark"] .highlight .ld { color: #ffd900 } /* Literal.Date */ +html[data-theme="dark"] .highlight .m { color: #ffd900 } /* Literal.Number */ +html[data-theme="dark"] .highlight .s { color: #abe338 } /* Literal.String */ +html[data-theme="dark"] .highlight .na { color: #ffd900 } /* Name.Attribute */ +html[data-theme="dark"] .highlight .nb { color: #ffd900 } /* Name.Builtin */ +html[data-theme="dark"] .highlight .nc { color: #00e0e0 } /* Name.Class */ +html[data-theme="dark"] .highlight .no { color: #00e0e0 } /* Name.Constant */ +html[data-theme="dark"] .highlight .nd { color: #ffd900 } /* Name.Decorator */ +html[data-theme="dark"] .highlight .ni { color: #abe338 } /* Name.Entity */ +html[data-theme="dark"] .highlight .ne { color: #dcc6e0 } /* Name.Exception */ +html[data-theme="dark"] .highlight .nf { color: #00e0e0 } /* Name.Function */ +html[data-theme="dark"] .highlight .nl { color: #ffd900 } /* Name.Label */ +html[data-theme="dark"] .highlight .nn { color: #f8f8f2 } /* Name.Namespace */ +html[data-theme="dark"] .highlight .nx { color: #f8f8f2 } /* Name.Other */ +html[data-theme="dark"] .highlight .py { color: #00e0e0 } /* Name.Property */ +html[data-theme="dark"] .highlight .nt { color: #00e0e0 } /* Name.Tag */ +html[data-theme="dark"] .highlight .nv { color: #ffa07a } /* Name.Variable */ +html[data-theme="dark"] .highlight .ow { color: #dcc6e0 } /* Operator.Word */ +html[data-theme="dark"] .highlight .pm { color: #f8f8f2 } /* Punctuation.Marker */ +html[data-theme="dark"] .highlight .w { color: #f8f8f2 } /* Text.Whitespace */ +html[data-theme="dark"] .highlight .mb { color: #ffd900 } /* Literal.Number.Bin */ +html[data-theme="dark"] .highlight .mf { color: #ffd900 } /* Literal.Number.Float */ +html[data-theme="dark"] .highlight .mh { color: #ffd900 } /* Literal.Number.Hex */ +html[data-theme="dark"] .highlight .mi { color: #ffd900 } /* Literal.Number.Integer */ +html[data-theme="dark"] .highlight .mo { color: #ffd900 } /* Literal.Number.Oct */ +html[data-theme="dark"] .highlight .sa { color: #abe338 } /* Literal.String.Affix */ +html[data-theme="dark"] .highlight .sb { color: #abe338 } /* Literal.String.Backtick */ +html[data-theme="dark"] .highlight .sc { color: #abe338 } /* Literal.String.Char */ +html[data-theme="dark"] .highlight .dl { color: #abe338 } /* Literal.String.Delimiter */ +html[data-theme="dark"] .highlight .sd { color: #abe338 } /* Literal.String.Doc */ +html[data-theme="dark"] .highlight .s2 { color: #abe338 } /* Literal.String.Double */ +html[data-theme="dark"] .highlight .se { color: #abe338 } /* Literal.String.Escape */ +html[data-theme="dark"] .highlight .sh { color: #abe338 } /* Literal.String.Heredoc */ +html[data-theme="dark"] .highlight .si { color: #abe338 } /* Literal.String.Interpol */ +html[data-theme="dark"] .highlight .sx { color: #abe338 } /* Literal.String.Other */ +html[data-theme="dark"] .highlight .sr { color: #ffa07a } /* Literal.String.Regex */ +html[data-theme="dark"] .highlight .s1 { color: #abe338 } /* Literal.String.Single */ +html[data-theme="dark"] .highlight .ss { color: #00e0e0 } /* Literal.String.Symbol */ +html[data-theme="dark"] .highlight .bp { color: #ffd900 } /* Name.Builtin.Pseudo */ +html[data-theme="dark"] .highlight .fm { color: #00e0e0 } /* Name.Function.Magic */ +html[data-theme="dark"] .highlight .vc { color: #ffa07a } /* Name.Variable.Class */ +html[data-theme="dark"] .highlight .vg { color: #ffa07a } /* Name.Variable.Global */ +html[data-theme="dark"] .highlight .vi { color: #ffa07a } /* Name.Variable.Instance */ +html[data-theme="dark"] .highlight .vm { color: #ffd900 } /* Name.Variable.Magic */ +html[data-theme="dark"] .highlight .il { color: #ffd900 } /* Literal.Number.Integer.Long */ \ No newline at end of file diff --git a/_static/rightclick.png b/_static/rightclick.png new file mode 100644 index 0000000..9c4de39 Binary files /dev/null and b/_static/rightclick.png differ diff --git a/_static/sbt-webpack-macros.html b/_static/sbt-webpack-macros.html new file mode 100644 index 0000000..6cbf559 --- /dev/null +++ b/_static/sbt-webpack-macros.html @@ -0,0 +1,11 @@ + +{% macro head_pre_bootstrap() %} + +{% endmacro %} + +{% macro body_post() %} + +{% endmacro %} diff --git a/_static/scripts/bootstrap.js b/_static/scripts/bootstrap.js new file mode 100644 index 0000000..c8178de --- /dev/null +++ b/_static/scripts/bootstrap.js @@ -0,0 +1,3 @@ +/*! For license information please see bootstrap.js.LICENSE.txt */ +(()=>{"use strict";var t={d:(e,i)=>{for(var n in i)t.o(i,n)&&!t.o(e,n)&&Object.defineProperty(e,n,{enumerable:!0,get:i[n]})},o:(t,e)=>Object.prototype.hasOwnProperty.call(t,e),r:t=>{"undefined"!=typeof Symbol&&Symbol.toStringTag&&Object.defineProperty(t,Symbol.toStringTag,{value:"Module"}),Object.defineProperty(t,"__esModule",{value:!0})}},e={};t.r(e),t.d(e,{afterMain:()=>E,afterRead:()=>v,afterWrite:()=>C,applyStyles:()=>$,arrow:()=>J,auto:()=>a,basePlacements:()=>l,beforeMain:()=>y,beforeRead:()=>_,beforeWrite:()=>A,bottom:()=>s,clippingParents:()=>d,computeStyles:()=>it,createPopper:()=>Dt,createPopperBase:()=>St,createPopperLite:()=>$t,detectOverflow:()=>_t,end:()=>h,eventListeners:()=>st,flip:()=>bt,hide:()=>wt,left:()=>r,main:()=>w,modifierPhases:()=>O,offset:()=>Et,placements:()=>g,popper:()=>f,popperGenerator:()=>Lt,popperOffsets:()=>At,preventOverflow:()=>Tt,read:()=>b,reference:()=>p,right:()=>o,start:()=>c,top:()=>n,variationPlacements:()=>m,viewport:()=>u,write:()=>T});var i={};t.r(i),t.d(i,{Alert:()=>Oe,Button:()=>ke,Carousel:()=>li,Collapse:()=>Ei,Dropdown:()=>Ki,Modal:()=>Ln,Offcanvas:()=>Kn,Popover:()=>bs,ScrollSpy:()=>Ls,Tab:()=>Js,Toast:()=>po,Tooltip:()=>fs});var n="top",s="bottom",o="right",r="left",a="auto",l=[n,s,o,r],c="start",h="end",d="clippingParents",u="viewport",f="popper",p="reference",m=l.reduce((function(t,e){return t.concat([e+"-"+c,e+"-"+h])}),[]),g=[].concat(l,[a]).reduce((function(t,e){return t.concat([e,e+"-"+c,e+"-"+h])}),[]),_="beforeRead",b="read",v="afterRead",y="beforeMain",w="main",E="afterMain",A="beforeWrite",T="write",C="afterWrite",O=[_,b,v,y,w,E,A,T,C];function x(t){return t?(t.nodeName||"").toLowerCase():null}function k(t){if(null==t)return window;if("[object Window]"!==t.toString()){var e=t.ownerDocument;return e&&e.defaultView||window}return t}function L(t){return t instanceof k(t).Element||t instanceof Element}function S(t){return t instanceof k(t).HTMLElement||t instanceof HTMLElement}function D(t){return"undefined"!=typeof ShadowRoot&&(t instanceof k(t).ShadowRoot||t instanceof ShadowRoot)}const $={name:"applyStyles",enabled:!0,phase:"write",fn:function(t){var e=t.state;Object.keys(e.elements).forEach((function(t){var i=e.styles[t]||{},n=e.attributes[t]||{},s=e.elements[t];S(s)&&x(s)&&(Object.assign(s.style,i),Object.keys(n).forEach((function(t){var e=n[t];!1===e?s.removeAttribute(t):s.setAttribute(t,!0===e?"":e)})))}))},effect:function(t){var e=t.state,i={popper:{position:e.options.strategy,left:"0",top:"0",margin:"0"},arrow:{position:"absolute"},reference:{}};return Object.assign(e.elements.popper.style,i.popper),e.styles=i,e.elements.arrow&&Object.assign(e.elements.arrow.style,i.arrow),function(){Object.keys(e.elements).forEach((function(t){var n=e.elements[t],s=e.attributes[t]||{},o=Object.keys(e.styles.hasOwnProperty(t)?e.styles[t]:i[t]).reduce((function(t,e){return t[e]="",t}),{});S(n)&&x(n)&&(Object.assign(n.style,o),Object.keys(s).forEach((function(t){n.removeAttribute(t)})))}))}},requires:["computeStyles"]};function I(t){return t.split("-")[0]}var N=Math.max,P=Math.min,M=Math.round;function j(){var t=navigator.userAgentData;return null!=t&&t.brands&&Array.isArray(t.brands)?t.brands.map((function(t){return t.brand+"/"+t.version})).join(" "):navigator.userAgent}function F(){return!/^((?!chrome|android).)*safari/i.test(j())}function H(t,e,i){void 0===e&&(e=!1),void 0===i&&(i=!1);var n=t.getBoundingClientRect(),s=1,o=1;e&&S(t)&&(s=t.offsetWidth>0&&M(n.width)/t.offsetWidth||1,o=t.offsetHeight>0&&M(n.height)/t.offsetHeight||1);var r=(L(t)?k(t):window).visualViewport,a=!F()&&i,l=(n.left+(a&&r?r.offsetLeft:0))/s,c=(n.top+(a&&r?r.offsetTop:0))/o,h=n.width/s,d=n.height/o;return{width:h,height:d,top:c,right:l+h,bottom:c+d,left:l,x:l,y:c}}function B(t){var e=H(t),i=t.offsetWidth,n=t.offsetHeight;return Math.abs(e.width-i)<=1&&(i=e.width),Math.abs(e.height-n)<=1&&(n=e.height),{x:t.offsetLeft,y:t.offsetTop,width:i,height:n}}function W(t,e){var i=e.getRootNode&&e.getRootNode();if(t.contains(e))return!0;if(i&&D(i)){var n=e;do{if(n&&t.isSameNode(n))return!0;n=n.parentNode||n.host}while(n)}return!1}function z(t){return k(t).getComputedStyle(t)}function R(t){return["table","td","th"].indexOf(x(t))>=0}function q(t){return((L(t)?t.ownerDocument:t.document)||window.document).documentElement}function V(t){return"html"===x(t)?t:t.assignedSlot||t.parentNode||(D(t)?t.host:null)||q(t)}function Y(t){return S(t)&&"fixed"!==z(t).position?t.offsetParent:null}function K(t){for(var e=k(t),i=Y(t);i&&R(i)&&"static"===z(i).position;)i=Y(i);return i&&("html"===x(i)||"body"===x(i)&&"static"===z(i).position)?e:i||function(t){var e=/firefox/i.test(j());if(/Trident/i.test(j())&&S(t)&&"fixed"===z(t).position)return null;var i=V(t);for(D(i)&&(i=i.host);S(i)&&["html","body"].indexOf(x(i))<0;){var n=z(i);if("none"!==n.transform||"none"!==n.perspective||"paint"===n.contain||-1!==["transform","perspective"].indexOf(n.willChange)||e&&"filter"===n.willChange||e&&n.filter&&"none"!==n.filter)return i;i=i.parentNode}return null}(t)||e}function Q(t){return["top","bottom"].indexOf(t)>=0?"x":"y"}function X(t,e,i){return N(t,P(e,i))}function U(t){return Object.assign({},{top:0,right:0,bottom:0,left:0},t)}function G(t,e){return e.reduce((function(e,i){return e[i]=t,e}),{})}const J={name:"arrow",enabled:!0,phase:"main",fn:function(t){var e,i=t.state,a=t.name,c=t.options,h=i.elements.arrow,d=i.modifiersData.popperOffsets,u=I(i.placement),f=Q(u),p=[r,o].indexOf(u)>=0?"height":"width";if(h&&d){var m=function(t,e){return U("number"!=typeof(t="function"==typeof t?t(Object.assign({},e.rects,{placement:e.placement})):t)?t:G(t,l))}(c.padding,i),g=B(h),_="y"===f?n:r,b="y"===f?s:o,v=i.rects.reference[p]+i.rects.reference[f]-d[f]-i.rects.popper[p],y=d[f]-i.rects.reference[f],w=K(h),E=w?"y"===f?w.clientHeight||0:w.clientWidth||0:0,A=v/2-y/2,T=m[_],C=E-g[p]-m[b],O=E/2-g[p]/2+A,x=X(T,O,C),k=f;i.modifiersData[a]=((e={})[k]=x,e.centerOffset=x-O,e)}},effect:function(t){var e=t.state,i=t.options.element,n=void 0===i?"[data-popper-arrow]":i;null!=n&&("string"!=typeof n||(n=e.elements.popper.querySelector(n)))&&W(e.elements.popper,n)&&(e.elements.arrow=n)},requires:["popperOffsets"],requiresIfExists:["preventOverflow"]};function Z(t){return t.split("-")[1]}var tt={top:"auto",right:"auto",bottom:"auto",left:"auto"};function et(t){var e,i=t.popper,a=t.popperRect,l=t.placement,c=t.variation,d=t.offsets,u=t.position,f=t.gpuAcceleration,p=t.adaptive,m=t.roundOffsets,g=t.isFixed,_=d.x,b=void 0===_?0:_,v=d.y,y=void 0===v?0:v,w="function"==typeof m?m({x:b,y}):{x:b,y};b=w.x,y=w.y;var E=d.hasOwnProperty("x"),A=d.hasOwnProperty("y"),T=r,C=n,O=window;if(p){var x=K(i),L="clientHeight",S="clientWidth";x===k(i)&&"static"!==z(x=q(i)).position&&"absolute"===u&&(L="scrollHeight",S="scrollWidth"),(l===n||(l===r||l===o)&&c===h)&&(C=s,y-=(g&&x===O&&O.visualViewport?O.visualViewport.height:x[L])-a.height,y*=f?1:-1),l!==r&&(l!==n&&l!==s||c!==h)||(T=o,b-=(g&&x===O&&O.visualViewport?O.visualViewport.width:x[S])-a.width,b*=f?1:-1)}var D,$=Object.assign({position:u},p&&tt),I=!0===m?function(t,e){var i=t.x,n=t.y,s=e.devicePixelRatio||1;return{x:M(i*s)/s||0,y:M(n*s)/s||0}}({x:b,y},k(i)):{x:b,y};return b=I.x,y=I.y,f?Object.assign({},$,((D={})[C]=A?"0":"",D[T]=E?"0":"",D.transform=(O.devicePixelRatio||1)<=1?"translate("+b+"px, "+y+"px)":"translate3d("+b+"px, "+y+"px, 0)",D)):Object.assign({},$,((e={})[C]=A?y+"px":"",e[T]=E?b+"px":"",e.transform="",e))}const it={name:"computeStyles",enabled:!0,phase:"beforeWrite",fn:function(t){var e=t.state,i=t.options,n=i.gpuAcceleration,s=void 0===n||n,o=i.adaptive,r=void 0===o||o,a=i.roundOffsets,l=void 0===a||a,c={placement:I(e.placement),variation:Z(e.placement),popper:e.elements.popper,popperRect:e.rects.popper,gpuAcceleration:s,isFixed:"fixed"===e.options.strategy};null!=e.modifiersData.popperOffsets&&(e.styles.popper=Object.assign({},e.styles.popper,et(Object.assign({},c,{offsets:e.modifiersData.popperOffsets,position:e.options.strategy,adaptive:r,roundOffsets:l})))),null!=e.modifiersData.arrow&&(e.styles.arrow=Object.assign({},e.styles.arrow,et(Object.assign({},c,{offsets:e.modifiersData.arrow,position:"absolute",adaptive:!1,roundOffsets:l})))),e.attributes.popper=Object.assign({},e.attributes.popper,{"data-popper-placement":e.placement})},data:{}};var nt={passive:!0};const st={name:"eventListeners",enabled:!0,phase:"write",fn:function(){},effect:function(t){var e=t.state,i=t.instance,n=t.options,s=n.scroll,o=void 0===s||s,r=n.resize,a=void 0===r||r,l=k(e.elements.popper),c=[].concat(e.scrollParents.reference,e.scrollParents.popper);return o&&c.forEach((function(t){t.addEventListener("scroll",i.update,nt)})),a&&l.addEventListener("resize",i.update,nt),function(){o&&c.forEach((function(t){t.removeEventListener("scroll",i.update,nt)})),a&&l.removeEventListener("resize",i.update,nt)}},data:{}};var ot={left:"right",right:"left",bottom:"top",top:"bottom"};function rt(t){return t.replace(/left|right|bottom|top/g,(function(t){return ot[t]}))}var at={start:"end",end:"start"};function lt(t){return t.replace(/start|end/g,(function(t){return at[t]}))}function ct(t){var e=k(t);return{scrollLeft:e.pageXOffset,scrollTop:e.pageYOffset}}function ht(t){return H(q(t)).left+ct(t).scrollLeft}function dt(t){var e=z(t),i=e.overflow,n=e.overflowX,s=e.overflowY;return/auto|scroll|overlay|hidden/.test(i+s+n)}function ut(t){return["html","body","#document"].indexOf(x(t))>=0?t.ownerDocument.body:S(t)&&dt(t)?t:ut(V(t))}function ft(t,e){var i;void 0===e&&(e=[]);var n=ut(t),s=n===(null==(i=t.ownerDocument)?void 0:i.body),o=k(n),r=s?[o].concat(o.visualViewport||[],dt(n)?n:[]):n,a=e.concat(r);return s?a:a.concat(ft(V(r)))}function pt(t){return Object.assign({},t,{left:t.x,top:t.y,right:t.x+t.width,bottom:t.y+t.height})}function mt(t,e,i){return e===u?pt(function(t,e){var i=k(t),n=q(t),s=i.visualViewport,o=n.clientWidth,r=n.clientHeight,a=0,l=0;if(s){o=s.width,r=s.height;var c=F();(c||!c&&"fixed"===e)&&(a=s.offsetLeft,l=s.offsetTop)}return{width:o,height:r,x:a+ht(t),y:l}}(t,i)):L(e)?function(t,e){var i=H(t,!1,"fixed"===e);return i.top=i.top+t.clientTop,i.left=i.left+t.clientLeft,i.bottom=i.top+t.clientHeight,i.right=i.left+t.clientWidth,i.width=t.clientWidth,i.height=t.clientHeight,i.x=i.left,i.y=i.top,i}(e,i):pt(function(t){var e,i=q(t),n=ct(t),s=null==(e=t.ownerDocument)?void 0:e.body,o=N(i.scrollWidth,i.clientWidth,s?s.scrollWidth:0,s?s.clientWidth:0),r=N(i.scrollHeight,i.clientHeight,s?s.scrollHeight:0,s?s.clientHeight:0),a=-n.scrollLeft+ht(t),l=-n.scrollTop;return"rtl"===z(s||i).direction&&(a+=N(i.clientWidth,s?s.clientWidth:0)-o),{width:o,height:r,x:a,y:l}}(q(t)))}function gt(t){var e,i=t.reference,a=t.element,l=t.placement,d=l?I(l):null,u=l?Z(l):null,f=i.x+i.width/2-a.width/2,p=i.y+i.height/2-a.height/2;switch(d){case n:e={x:f,y:i.y-a.height};break;case s:e={x:f,y:i.y+i.height};break;case o:e={x:i.x+i.width,y:p};break;case r:e={x:i.x-a.width,y:p};break;default:e={x:i.x,y:i.y}}var m=d?Q(d):null;if(null!=m){var g="y"===m?"height":"width";switch(u){case c:e[m]=e[m]-(i[g]/2-a[g]/2);break;case h:e[m]=e[m]+(i[g]/2-a[g]/2)}}return e}function _t(t,e){void 0===e&&(e={});var i=e,r=i.placement,a=void 0===r?t.placement:r,c=i.strategy,h=void 0===c?t.strategy:c,m=i.boundary,g=void 0===m?d:m,_=i.rootBoundary,b=void 0===_?u:_,v=i.elementContext,y=void 0===v?f:v,w=i.altBoundary,E=void 0!==w&&w,A=i.padding,T=void 0===A?0:A,C=U("number"!=typeof T?T:G(T,l)),O=y===f?p:f,k=t.rects.popper,D=t.elements[E?O:y],$=function(t,e,i,n){var s="clippingParents"===e?function(t){var e=ft(V(t)),i=["absolute","fixed"].indexOf(z(t).position)>=0&&S(t)?K(t):t;return L(i)?e.filter((function(t){return L(t)&&W(t,i)&&"body"!==x(t)})):[]}(t):[].concat(e),o=[].concat(s,[i]),r=o[0],a=o.reduce((function(e,i){var s=mt(t,i,n);return e.top=N(s.top,e.top),e.right=P(s.right,e.right),e.bottom=P(s.bottom,e.bottom),e.left=N(s.left,e.left),e}),mt(t,r,n));return a.width=a.right-a.left,a.height=a.bottom-a.top,a.x=a.left,a.y=a.top,a}(L(D)?D:D.contextElement||q(t.elements.popper),g,b,h),I=H(t.elements.reference),M=gt({reference:I,element:k,strategy:"absolute",placement:a}),j=pt(Object.assign({},k,M)),F=y===f?j:I,B={top:$.top-F.top+C.top,bottom:F.bottom-$.bottom+C.bottom,left:$.left-F.left+C.left,right:F.right-$.right+C.right},R=t.modifiersData.offset;if(y===f&&R){var Y=R[a];Object.keys(B).forEach((function(t){var e=[o,s].indexOf(t)>=0?1:-1,i=[n,s].indexOf(t)>=0?"y":"x";B[t]+=Y[i]*e}))}return B}const bt={name:"flip",enabled:!0,phase:"main",fn:function(t){var e=t.state,i=t.options,h=t.name;if(!e.modifiersData[h]._skip){for(var d=i.mainAxis,u=void 0===d||d,f=i.altAxis,p=void 0===f||f,_=i.fallbackPlacements,b=i.padding,v=i.boundary,y=i.rootBoundary,w=i.altBoundary,E=i.flipVariations,A=void 0===E||E,T=i.allowedAutoPlacements,C=e.options.placement,O=I(C),x=_||(O!==C&&A?function(t){if(I(t)===a)return[];var e=rt(t);return[lt(t),e,lt(e)]}(C):[rt(C)]),k=[C].concat(x).reduce((function(t,i){return t.concat(I(i)===a?function(t,e){void 0===e&&(e={});var i=e,n=i.placement,s=i.boundary,o=i.rootBoundary,r=i.padding,a=i.flipVariations,c=i.allowedAutoPlacements,h=void 0===c?g:c,d=Z(n),u=d?a?m:m.filter((function(t){return Z(t)===d})):l,f=u.filter((function(t){return h.indexOf(t)>=0}));0===f.length&&(f=u);var p=f.reduce((function(e,i){return e[i]=_t(t,{placement:i,boundary:s,rootBoundary:o,padding:r})[I(i)],e}),{});return Object.keys(p).sort((function(t,e){return p[t]-p[e]}))}(e,{placement:i,boundary:v,rootBoundary:y,padding:b,flipVariations:A,allowedAutoPlacements:T}):i)}),[]),L=e.rects.reference,S=e.rects.popper,D=new Map,$=!0,N=k[0],P=0;P=0,B=H?"width":"height",W=_t(e,{placement:M,boundary:v,rootBoundary:y,altBoundary:w,padding:b}),z=H?F?o:r:F?s:n;L[B]>S[B]&&(z=rt(z));var R=rt(z),q=[];if(u&&q.push(W[j]<=0),p&&q.push(W[z]<=0,W[R]<=0),q.every((function(t){return t}))){N=M,$=!1;break}D.set(M,q)}if($)for(var V=function(t){var e=k.find((function(e){var i=D.get(e);if(i)return i.slice(0,t).every((function(t){return t}))}));if(e)return N=e,"break"},Y=A?3:1;Y>0&&"break"!==V(Y);Y--);e.placement!==N&&(e.modifiersData[h]._skip=!0,e.placement=N,e.reset=!0)}},requiresIfExists:["offset"],data:{_skip:!1}};function vt(t,e,i){return void 0===i&&(i={x:0,y:0}),{top:t.top-e.height-i.y,right:t.right-e.width+i.x,bottom:t.bottom-e.height+i.y,left:t.left-e.width-i.x}}function yt(t){return[n,o,s,r].some((function(e){return t[e]>=0}))}const wt={name:"hide",enabled:!0,phase:"main",requiresIfExists:["preventOverflow"],fn:function(t){var e=t.state,i=t.name,n=e.rects.reference,s=e.rects.popper,o=e.modifiersData.preventOverflow,r=_t(e,{elementContext:"reference"}),a=_t(e,{altBoundary:!0}),l=vt(r,n),c=vt(a,s,o),h=yt(l),d=yt(c);e.modifiersData[i]={referenceClippingOffsets:l,popperEscapeOffsets:c,isReferenceHidden:h,hasPopperEscaped:d},e.attributes.popper=Object.assign({},e.attributes.popper,{"data-popper-reference-hidden":h,"data-popper-escaped":d})}},Et={name:"offset",enabled:!0,phase:"main",requires:["popperOffsets"],fn:function(t){var e=t.state,i=t.options,s=t.name,a=i.offset,l=void 0===a?[0,0]:a,c=g.reduce((function(t,i){return t[i]=function(t,e,i){var s=I(t),a=[r,n].indexOf(s)>=0?-1:1,l="function"==typeof i?i(Object.assign({},e,{placement:t})):i,c=l[0],h=l[1];return c=c||0,h=(h||0)*a,[r,o].indexOf(s)>=0?{x:h,y:c}:{x:c,y:h}}(i,e.rects,l),t}),{}),h=c[e.placement],d=h.x,u=h.y;null!=e.modifiersData.popperOffsets&&(e.modifiersData.popperOffsets.x+=d,e.modifiersData.popperOffsets.y+=u),e.modifiersData[s]=c}},At={name:"popperOffsets",enabled:!0,phase:"read",fn:function(t){var e=t.state,i=t.name;e.modifiersData[i]=gt({reference:e.rects.reference,element:e.rects.popper,strategy:"absolute",placement:e.placement})},data:{}},Tt={name:"preventOverflow",enabled:!0,phase:"main",fn:function(t){var e=t.state,i=t.options,a=t.name,l=i.mainAxis,h=void 0===l||l,d=i.altAxis,u=void 0!==d&&d,f=i.boundary,p=i.rootBoundary,m=i.altBoundary,g=i.padding,_=i.tether,b=void 0===_||_,v=i.tetherOffset,y=void 0===v?0:v,w=_t(e,{boundary:f,rootBoundary:p,padding:g,altBoundary:m}),E=I(e.placement),A=Z(e.placement),T=!A,C=Q(E),O="x"===C?"y":"x",x=e.modifiersData.popperOffsets,k=e.rects.reference,L=e.rects.popper,S="function"==typeof y?y(Object.assign({},e.rects,{placement:e.placement})):y,D="number"==typeof S?{mainAxis:S,altAxis:S}:Object.assign({mainAxis:0,altAxis:0},S),$=e.modifiersData.offset?e.modifiersData.offset[e.placement]:null,M={x:0,y:0};if(x){if(h){var j,F="y"===C?n:r,H="y"===C?s:o,W="y"===C?"height":"width",z=x[C],R=z+w[F],q=z-w[H],V=b?-L[W]/2:0,Y=A===c?k[W]:L[W],U=A===c?-L[W]:-k[W],G=e.elements.arrow,J=b&&G?B(G):{width:0,height:0},tt=e.modifiersData["arrow#persistent"]?e.modifiersData["arrow#persistent"].padding:{top:0,right:0,bottom:0,left:0},et=tt[F],it=tt[H],nt=X(0,k[W],J[W]),st=T?k[W]/2-V-nt-et-D.mainAxis:Y-nt-et-D.mainAxis,ot=T?-k[W]/2+V+nt+it+D.mainAxis:U+nt+it+D.mainAxis,rt=e.elements.arrow&&K(e.elements.arrow),at=rt?"y"===C?rt.clientTop||0:rt.clientLeft||0:0,lt=null!=(j=null==$?void 0:$[C])?j:0,ct=z+ot-lt,ht=X(b?P(R,z+st-lt-at):R,z,b?N(q,ct):q);x[C]=ht,M[C]=ht-z}if(u){var dt,ut="x"===C?n:r,ft="x"===C?s:o,pt=x[O],mt="y"===O?"height":"width",gt=pt+w[ut],bt=pt-w[ft],vt=-1!==[n,r].indexOf(E),yt=null!=(dt=null==$?void 0:$[O])?dt:0,wt=vt?gt:pt-k[mt]-L[mt]-yt+D.altAxis,Et=vt?pt+k[mt]+L[mt]-yt-D.altAxis:bt,At=b&&vt?function(t,e,i){var n=X(t,e,i);return n>i?i:n}(wt,pt,Et):X(b?wt:gt,pt,b?Et:bt);x[O]=At,M[O]=At-pt}e.modifiersData[a]=M}},requiresIfExists:["offset"]};function Ct(t,e,i){void 0===i&&(i=!1);var n,s,o=S(e),r=S(e)&&function(t){var e=t.getBoundingClientRect(),i=M(e.width)/t.offsetWidth||1,n=M(e.height)/t.offsetHeight||1;return 1!==i||1!==n}(e),a=q(e),l=H(t,r,i),c={scrollLeft:0,scrollTop:0},h={x:0,y:0};return(o||!o&&!i)&&(("body"!==x(e)||dt(a))&&(c=(n=e)!==k(n)&&S(n)?{scrollLeft:(s=n).scrollLeft,scrollTop:s.scrollTop}:ct(n)),S(e)?((h=H(e,!0)).x+=e.clientLeft,h.y+=e.clientTop):a&&(h.x=ht(a))),{x:l.left+c.scrollLeft-h.x,y:l.top+c.scrollTop-h.y,width:l.width,height:l.height}}function Ot(t){var e=new Map,i=new Set,n=[];function s(t){i.add(t.name),[].concat(t.requires||[],t.requiresIfExists||[]).forEach((function(t){if(!i.has(t)){var n=e.get(t);n&&s(n)}})),n.push(t)}return t.forEach((function(t){e.set(t.name,t)})),t.forEach((function(t){i.has(t.name)||s(t)})),n}var xt={placement:"bottom",modifiers:[],strategy:"absolute"};function kt(){for(var t=arguments.length,e=new Array(t),i=0;iIt.has(t)&&It.get(t).get(e)||null,remove(t,e){if(!It.has(t))return;const i=It.get(t);i.delete(e),0===i.size&&It.delete(t)}},Pt="transitionend",Mt=t=>(t&&window.CSS&&window.CSS.escape&&(t=t.replace(/#([^\s"#']+)/g,((t,e)=>`#${CSS.escape(e)}`))),t),jt=t=>{t.dispatchEvent(new Event(Pt))},Ft=t=>!(!t||"object"!=typeof t)&&(void 0!==t.jquery&&(t=t[0]),void 0!==t.nodeType),Ht=t=>Ft(t)?t.jquery?t[0]:t:"string"==typeof t&&t.length>0?document.querySelector(Mt(t)):null,Bt=t=>{if(!Ft(t)||0===t.getClientRects().length)return!1;const e="visible"===getComputedStyle(t).getPropertyValue("visibility"),i=t.closest("details:not([open])");if(!i)return e;if(i!==t){const e=t.closest("summary");if(e&&e.parentNode!==i)return!1;if(null===e)return!1}return e},Wt=t=>!t||t.nodeType!==Node.ELEMENT_NODE||!!t.classList.contains("disabled")||(void 0!==t.disabled?t.disabled:t.hasAttribute("disabled")&&"false"!==t.getAttribute("disabled")),zt=t=>{if(!document.documentElement.attachShadow)return null;if("function"==typeof t.getRootNode){const e=t.getRootNode();return e instanceof ShadowRoot?e:null}return t instanceof ShadowRoot?t:t.parentNode?zt(t.parentNode):null},Rt=()=>{},qt=t=>{t.offsetHeight},Vt=()=>window.jQuery&&!document.body.hasAttribute("data-bs-no-jquery")?window.jQuery:null,Yt=[],Kt=()=>"rtl"===document.documentElement.dir,Qt=t=>{var e;e=()=>{const e=Vt();if(e){const i=t.NAME,n=e.fn[i];e.fn[i]=t.jQueryInterface,e.fn[i].Constructor=t,e.fn[i].noConflict=()=>(e.fn[i]=n,t.jQueryInterface)}},"loading"===document.readyState?(Yt.length||document.addEventListener("DOMContentLoaded",(()=>{for(const t of Yt)t()})),Yt.push(e)):e()},Xt=(t,e=[],i=t)=>"function"==typeof t?t(...e):i,Ut=(t,e,i=!0)=>{if(!i)return void Xt(t);const n=(t=>{if(!t)return 0;let{transitionDuration:e,transitionDelay:i}=window.getComputedStyle(t);const n=Number.parseFloat(e),s=Number.parseFloat(i);return n||s?(e=e.split(",")[0],i=i.split(",")[0],1e3*(Number.parseFloat(e)+Number.parseFloat(i))):0})(e)+5;let s=!1;const o=({target:i})=>{i===e&&(s=!0,e.removeEventListener(Pt,o),Xt(t))};e.addEventListener(Pt,o),setTimeout((()=>{s||jt(e)}),n)},Gt=(t,e,i,n)=>{const s=t.length;let o=t.indexOf(e);return-1===o?!i&&n?t[s-1]:t[0]:(o+=i?1:-1,n&&(o=(o+s)%s),t[Math.max(0,Math.min(o,s-1))])},Jt=/[^.]*(?=\..*)\.|.*/,Zt=/\..*/,te=/::\d+$/,ee={};let ie=1;const ne={mouseenter:"mouseover",mouseleave:"mouseout"},se=new Set(["click","dblclick","mouseup","mousedown","contextmenu","mousewheel","DOMMouseScroll","mouseover","mouseout","mousemove","selectstart","selectend","keydown","keypress","keyup","orientationchange","touchstart","touchmove","touchend","touchcancel","pointerdown","pointermove","pointerup","pointerleave","pointercancel","gesturestart","gesturechange","gestureend","focus","blur","change","reset","select","submit","focusin","focusout","load","unload","beforeunload","resize","move","DOMContentLoaded","readystatechange","error","abort","scroll"]);function oe(t,e){return e&&`${e}::${ie++}`||t.uidEvent||ie++}function re(t){const e=oe(t);return t.uidEvent=e,ee[e]=ee[e]||{},ee[e]}function ae(t,e,i=null){return Object.values(t).find((t=>t.callable===e&&t.delegationSelector===i))}function le(t,e,i){const n="string"==typeof e,s=n?i:e||i;let o=ue(t);return se.has(o)||(o=t),[n,s,o]}function ce(t,e,i,n,s){if("string"!=typeof e||!t)return;let[o,r,a]=le(e,i,n);if(e in ne){const t=t=>function(e){if(!e.relatedTarget||e.relatedTarget!==e.delegateTarget&&!e.delegateTarget.contains(e.relatedTarget))return t.call(this,e)};r=t(r)}const l=re(t),c=l[a]||(l[a]={}),h=ae(c,r,o?i:null);if(h)return void(h.oneOff=h.oneOff&&s);const d=oe(r,e.replace(Jt,"")),u=o?function(t,e,i){return function n(s){const o=t.querySelectorAll(e);for(let{target:r}=s;r&&r!==this;r=r.parentNode)for(const a of o)if(a===r)return pe(s,{delegateTarget:r}),n.oneOff&&fe.off(t,s.type,e,i),i.apply(r,[s])}}(t,i,r):function(t,e){return function i(n){return pe(n,{delegateTarget:t}),i.oneOff&&fe.off(t,n.type,e),e.apply(t,[n])}}(t,r);u.delegationSelector=o?i:null,u.callable=r,u.oneOff=s,u.uidEvent=d,c[d]=u,t.addEventListener(a,u,o)}function he(t,e,i,n,s){const o=ae(e[i],n,s);o&&(t.removeEventListener(i,o,Boolean(s)),delete e[i][o.uidEvent])}function de(t,e,i,n){const s=e[i]||{};for(const[o,r]of Object.entries(s))o.includes(n)&&he(t,e,i,r.callable,r.delegationSelector)}function ue(t){return t=t.replace(Zt,""),ne[t]||t}const fe={on(t,e,i,n){ce(t,e,i,n,!1)},one(t,e,i,n){ce(t,e,i,n,!0)},off(t,e,i,n){if("string"!=typeof e||!t)return;const[s,o,r]=le(e,i,n),a=r!==e,l=re(t),c=l[r]||{},h=e.startsWith(".");if(void 0===o){if(h)for(const i of Object.keys(l))de(t,l,i,e.slice(1));for(const[i,n]of Object.entries(c)){const s=i.replace(te,"");a&&!e.includes(s)||he(t,l,r,n.callable,n.delegationSelector)}}else{if(!Object.keys(c).length)return;he(t,l,r,o,s?i:null)}},trigger(t,e,i){if("string"!=typeof e||!t)return null;const n=Vt();let s=null,o=!0,r=!0,a=!1;e!==ue(e)&&n&&(s=n.Event(e,i),n(t).trigger(s),o=!s.isPropagationStopped(),r=!s.isImmediatePropagationStopped(),a=s.isDefaultPrevented());const l=pe(new Event(e,{bubbles:o,cancelable:!0}),i);return a&&l.preventDefault(),r&&t.dispatchEvent(l),l.defaultPrevented&&s&&s.preventDefault(),l}};function pe(t,e={}){for(const[i,n]of Object.entries(e))try{t[i]=n}catch(e){Object.defineProperty(t,i,{configurable:!0,get:()=>n})}return t}function me(t){if("true"===t)return!0;if("false"===t)return!1;if(t===Number(t).toString())return Number(t);if(""===t||"null"===t)return null;if("string"!=typeof t)return t;try{return JSON.parse(decodeURIComponent(t))}catch(e){return t}}function ge(t){return t.replace(/[A-Z]/g,(t=>`-${t.toLowerCase()}`))}const _e={setDataAttribute(t,e,i){t.setAttribute(`data-bs-${ge(e)}`,i)},removeDataAttribute(t,e){t.removeAttribute(`data-bs-${ge(e)}`)},getDataAttributes(t){if(!t)return{};const e={},i=Object.keys(t.dataset).filter((t=>t.startsWith("bs")&&!t.startsWith("bsConfig")));for(const n of i){let i=n.replace(/^bs/,"");i=i.charAt(0).toLowerCase()+i.slice(1,i.length),e[i]=me(t.dataset[n])}return e},getDataAttribute:(t,e)=>me(t.getAttribute(`data-bs-${ge(e)}`))};class be{static get Default(){return{}}static get DefaultType(){return{}}static get NAME(){throw new Error('You have to implement the static method "NAME", for each component!')}_getConfig(t){return t=this._mergeConfigObj(t),t=this._configAfterMerge(t),this._typeCheckConfig(t),t}_configAfterMerge(t){return t}_mergeConfigObj(t,e){const i=Ft(e)?_e.getDataAttribute(e,"config"):{};return{...this.constructor.Default,..."object"==typeof i?i:{},...Ft(e)?_e.getDataAttributes(e):{},..."object"==typeof t?t:{}}}_typeCheckConfig(t,e=this.constructor.DefaultType){for(const[n,s]of Object.entries(e)){const e=t[n],o=Ft(e)?"element":null==(i=e)?`${i}`:Object.prototype.toString.call(i).match(/\s([a-z]+)/i)[1].toLowerCase();if(!new RegExp(s).test(o))throw new TypeError(`${this.constructor.NAME.toUpperCase()}: Option "${n}" provided type "${o}" but expected type "${s}".`)}var i}}class ve extends be{constructor(t,e){super(),(t=Ht(t))&&(this._element=t,this._config=this._getConfig(e),Nt.set(this._element,this.constructor.DATA_KEY,this))}dispose(){Nt.remove(this._element,this.constructor.DATA_KEY),fe.off(this._element,this.constructor.EVENT_KEY);for(const t of Object.getOwnPropertyNames(this))this[t]=null}_queueCallback(t,e,i=!0){Ut(t,e,i)}_getConfig(t){return t=this._mergeConfigObj(t,this._element),t=this._configAfterMerge(t),this._typeCheckConfig(t),t}static getInstance(t){return Nt.get(Ht(t),this.DATA_KEY)}static getOrCreateInstance(t,e={}){return this.getInstance(t)||new this(t,"object"==typeof e?e:null)}static get VERSION(){return"5.3.3"}static get DATA_KEY(){return`bs.${this.NAME}`}static get EVENT_KEY(){return`.${this.DATA_KEY}`}static eventName(t){return`${t}${this.EVENT_KEY}`}}const ye=t=>{let e=t.getAttribute("data-bs-target");if(!e||"#"===e){let i=t.getAttribute("href");if(!i||!i.includes("#")&&!i.startsWith("."))return null;i.includes("#")&&!i.startsWith("#")&&(i=`#${i.split("#")[1]}`),e=i&&"#"!==i?i.trim():null}return e?e.split(",").map((t=>Mt(t))).join(","):null},we={find:(t,e=document.documentElement)=>[].concat(...Element.prototype.querySelectorAll.call(e,t)),findOne:(t,e=document.documentElement)=>Element.prototype.querySelector.call(e,t),children:(t,e)=>[].concat(...t.children).filter((t=>t.matches(e))),parents(t,e){const i=[];let n=t.parentNode.closest(e);for(;n;)i.push(n),n=n.parentNode.closest(e);return i},prev(t,e){let i=t.previousElementSibling;for(;i;){if(i.matches(e))return[i];i=i.previousElementSibling}return[]},next(t,e){let i=t.nextElementSibling;for(;i;){if(i.matches(e))return[i];i=i.nextElementSibling}return[]},focusableChildren(t){const e=["a","button","input","textarea","select","details","[tabindex]",'[contenteditable="true"]'].map((t=>`${t}:not([tabindex^="-"])`)).join(",");return this.find(e,t).filter((t=>!Wt(t)&&Bt(t)))},getSelectorFromElement(t){const e=ye(t);return e&&we.findOne(e)?e:null},getElementFromSelector(t){const e=ye(t);return e?we.findOne(e):null},getMultipleElementsFromSelector(t){const e=ye(t);return e?we.find(e):[]}},Ee=(t,e="hide")=>{const i=`click.dismiss${t.EVENT_KEY}`,n=t.NAME;fe.on(document,i,`[data-bs-dismiss="${n}"]`,(function(i){if(["A","AREA"].includes(this.tagName)&&i.preventDefault(),Wt(this))return;const s=we.getElementFromSelector(this)||this.closest(`.${n}`);t.getOrCreateInstance(s)[e]()}))},Ae=".bs.alert",Te=`close${Ae}`,Ce=`closed${Ae}`;class Oe extends ve{static get NAME(){return"alert"}close(){if(fe.trigger(this._element,Te).defaultPrevented)return;this._element.classList.remove("show");const t=this._element.classList.contains("fade");this._queueCallback((()=>this._destroyElement()),this._element,t)}_destroyElement(){this._element.remove(),fe.trigger(this._element,Ce),this.dispose()}static jQueryInterface(t){return this.each((function(){const e=Oe.getOrCreateInstance(this);if("string"==typeof t){if(void 0===e[t]||t.startsWith("_")||"constructor"===t)throw new TypeError(`No method named "${t}"`);e[t](this)}}))}}Ee(Oe,"close"),Qt(Oe);const xe='[data-bs-toggle="button"]';class ke extends ve{static get NAME(){return"button"}toggle(){this._element.setAttribute("aria-pressed",this._element.classList.toggle("active"))}static jQueryInterface(t){return this.each((function(){const e=ke.getOrCreateInstance(this);"toggle"===t&&e[t]()}))}}fe.on(document,"click.bs.button.data-api",xe,(t=>{t.preventDefault();const e=t.target.closest(xe);ke.getOrCreateInstance(e).toggle()})),Qt(ke);const Le=".bs.swipe",Se=`touchstart${Le}`,De=`touchmove${Le}`,$e=`touchend${Le}`,Ie=`pointerdown${Le}`,Ne=`pointerup${Le}`,Pe={endCallback:null,leftCallback:null,rightCallback:null},Me={endCallback:"(function|null)",leftCallback:"(function|null)",rightCallback:"(function|null)"};class je extends be{constructor(t,e){super(),this._element=t,t&&je.isSupported()&&(this._config=this._getConfig(e),this._deltaX=0,this._supportPointerEvents=Boolean(window.PointerEvent),this._initEvents())}static get Default(){return Pe}static get DefaultType(){return Me}static get NAME(){return"swipe"}dispose(){fe.off(this._element,Le)}_start(t){this._supportPointerEvents?this._eventIsPointerPenTouch(t)&&(this._deltaX=t.clientX):this._deltaX=t.touches[0].clientX}_end(t){this._eventIsPointerPenTouch(t)&&(this._deltaX=t.clientX-this._deltaX),this._handleSwipe(),Xt(this._config.endCallback)}_move(t){this._deltaX=t.touches&&t.touches.length>1?0:t.touches[0].clientX-this._deltaX}_handleSwipe(){const t=Math.abs(this._deltaX);if(t<=40)return;const e=t/this._deltaX;this._deltaX=0,e&&Xt(e>0?this._config.rightCallback:this._config.leftCallback)}_initEvents(){this._supportPointerEvents?(fe.on(this._element,Ie,(t=>this._start(t))),fe.on(this._element,Ne,(t=>this._end(t))),this._element.classList.add("pointer-event")):(fe.on(this._element,Se,(t=>this._start(t))),fe.on(this._element,De,(t=>this._move(t))),fe.on(this._element,$e,(t=>this._end(t))))}_eventIsPointerPenTouch(t){return this._supportPointerEvents&&("pen"===t.pointerType||"touch"===t.pointerType)}static isSupported(){return"ontouchstart"in document.documentElement||navigator.maxTouchPoints>0}}const Fe=".bs.carousel",He=".data-api",Be="ArrowLeft",We="ArrowRight",ze="next",Re="prev",qe="left",Ve="right",Ye=`slide${Fe}`,Ke=`slid${Fe}`,Qe=`keydown${Fe}`,Xe=`mouseenter${Fe}`,Ue=`mouseleave${Fe}`,Ge=`dragstart${Fe}`,Je=`load${Fe}${He}`,Ze=`click${Fe}${He}`,ti="carousel",ei="active",ii=".active",ni=".carousel-item",si=ii+ni,oi={[Be]:Ve,[We]:qe},ri={interval:5e3,keyboard:!0,pause:"hover",ride:!1,touch:!0,wrap:!0},ai={interval:"(number|boolean)",keyboard:"boolean",pause:"(string|boolean)",ride:"(boolean|string)",touch:"boolean",wrap:"boolean"};class li extends ve{constructor(t,e){super(t,e),this._interval=null,this._activeElement=null,this._isSliding=!1,this.touchTimeout=null,this._swipeHelper=null,this._indicatorsElement=we.findOne(".carousel-indicators",this._element),this._addEventListeners(),this._config.ride===ti&&this.cycle()}static get Default(){return ri}static get DefaultType(){return ai}static get NAME(){return"carousel"}next(){this._slide(ze)}nextWhenVisible(){!document.hidden&&Bt(this._element)&&this.next()}prev(){this._slide(Re)}pause(){this._isSliding&&jt(this._element),this._clearInterval()}cycle(){this._clearInterval(),this._updateInterval(),this._interval=setInterval((()=>this.nextWhenVisible()),this._config.interval)}_maybeEnableCycle(){this._config.ride&&(this._isSliding?fe.one(this._element,Ke,(()=>this.cycle())):this.cycle())}to(t){const e=this._getItems();if(t>e.length-1||t<0)return;if(this._isSliding)return void fe.one(this._element,Ke,(()=>this.to(t)));const i=this._getItemIndex(this._getActive());if(i===t)return;const n=t>i?ze:Re;this._slide(n,e[t])}dispose(){this._swipeHelper&&this._swipeHelper.dispose(),super.dispose()}_configAfterMerge(t){return t.defaultInterval=t.interval,t}_addEventListeners(){this._config.keyboard&&fe.on(this._element,Qe,(t=>this._keydown(t))),"hover"===this._config.pause&&(fe.on(this._element,Xe,(()=>this.pause())),fe.on(this._element,Ue,(()=>this._maybeEnableCycle()))),this._config.touch&&je.isSupported()&&this._addTouchEventListeners()}_addTouchEventListeners(){for(const t of we.find(".carousel-item img",this._element))fe.on(t,Ge,(t=>t.preventDefault()));const t={leftCallback:()=>this._slide(this._directionToOrder(qe)),rightCallback:()=>this._slide(this._directionToOrder(Ve)),endCallback:()=>{"hover"===this._config.pause&&(this.pause(),this.touchTimeout&&clearTimeout(this.touchTimeout),this.touchTimeout=setTimeout((()=>this._maybeEnableCycle()),500+this._config.interval))}};this._swipeHelper=new je(this._element,t)}_keydown(t){if(/input|textarea/i.test(t.target.tagName))return;const e=oi[t.key];e&&(t.preventDefault(),this._slide(this._directionToOrder(e)))}_getItemIndex(t){return this._getItems().indexOf(t)}_setActiveIndicatorElement(t){if(!this._indicatorsElement)return;const e=we.findOne(ii,this._indicatorsElement);e.classList.remove(ei),e.removeAttribute("aria-current");const i=we.findOne(`[data-bs-slide-to="${t}"]`,this._indicatorsElement);i&&(i.classList.add(ei),i.setAttribute("aria-current","true"))}_updateInterval(){const t=this._activeElement||this._getActive();if(!t)return;const e=Number.parseInt(t.getAttribute("data-bs-interval"),10);this._config.interval=e||this._config.defaultInterval}_slide(t,e=null){if(this._isSliding)return;const i=this._getActive(),n=t===ze,s=e||Gt(this._getItems(),i,n,this._config.wrap);if(s===i)return;const o=this._getItemIndex(s),r=e=>fe.trigger(this._element,e,{relatedTarget:s,direction:this._orderToDirection(t),from:this._getItemIndex(i),to:o});if(r(Ye).defaultPrevented)return;if(!i||!s)return;const a=Boolean(this._interval);this.pause(),this._isSliding=!0,this._setActiveIndicatorElement(o),this._activeElement=s;const l=n?"carousel-item-start":"carousel-item-end",c=n?"carousel-item-next":"carousel-item-prev";s.classList.add(c),qt(s),i.classList.add(l),s.classList.add(l),this._queueCallback((()=>{s.classList.remove(l,c),s.classList.add(ei),i.classList.remove(ei,c,l),this._isSliding=!1,r(Ke)}),i,this._isAnimated()),a&&this.cycle()}_isAnimated(){return this._element.classList.contains("slide")}_getActive(){return we.findOne(si,this._element)}_getItems(){return we.find(ni,this._element)}_clearInterval(){this._interval&&(clearInterval(this._interval),this._interval=null)}_directionToOrder(t){return Kt()?t===qe?Re:ze:t===qe?ze:Re}_orderToDirection(t){return Kt()?t===Re?qe:Ve:t===Re?Ve:qe}static jQueryInterface(t){return this.each((function(){const e=li.getOrCreateInstance(this,t);if("number"!=typeof t){if("string"==typeof t){if(void 0===e[t]||t.startsWith("_")||"constructor"===t)throw new TypeError(`No method named "${t}"`);e[t]()}}else e.to(t)}))}}fe.on(document,Ze,"[data-bs-slide], [data-bs-slide-to]",(function(t){const e=we.getElementFromSelector(this);if(!e||!e.classList.contains(ti))return;t.preventDefault();const i=li.getOrCreateInstance(e),n=this.getAttribute("data-bs-slide-to");return n?(i.to(n),void i._maybeEnableCycle()):"next"===_e.getDataAttribute(this,"slide")?(i.next(),void i._maybeEnableCycle()):(i.prev(),void i._maybeEnableCycle())})),fe.on(window,Je,(()=>{const t=we.find('[data-bs-ride="carousel"]');for(const e of t)li.getOrCreateInstance(e)})),Qt(li);const ci=".bs.collapse",hi=`show${ci}`,di=`shown${ci}`,ui=`hide${ci}`,fi=`hidden${ci}`,pi=`click${ci}.data-api`,mi="show",gi="collapse",_i="collapsing",bi=`:scope .${gi} .${gi}`,vi='[data-bs-toggle="collapse"]',yi={parent:null,toggle:!0},wi={parent:"(null|element)",toggle:"boolean"};class Ei extends ve{constructor(t,e){super(t,e),this._isTransitioning=!1,this._triggerArray=[];const i=we.find(vi);for(const t of i){const e=we.getSelectorFromElement(t),i=we.find(e).filter((t=>t===this._element));null!==e&&i.length&&this._triggerArray.push(t)}this._initializeChildren(),this._config.parent||this._addAriaAndCollapsedClass(this._triggerArray,this._isShown()),this._config.toggle&&this.toggle()}static get Default(){return yi}static get DefaultType(){return wi}static get NAME(){return"collapse"}toggle(){this._isShown()?this.hide():this.show()}show(){if(this._isTransitioning||this._isShown())return;let t=[];if(this._config.parent&&(t=this._getFirstLevelChildren(".collapse.show, .collapse.collapsing").filter((t=>t!==this._element)).map((t=>Ei.getOrCreateInstance(t,{toggle:!1})))),t.length&&t[0]._isTransitioning)return;if(fe.trigger(this._element,hi).defaultPrevented)return;for(const e of t)e.hide();const e=this._getDimension();this._element.classList.remove(gi),this._element.classList.add(_i),this._element.style[e]=0,this._addAriaAndCollapsedClass(this._triggerArray,!0),this._isTransitioning=!0;const i=`scroll${e[0].toUpperCase()+e.slice(1)}`;this._queueCallback((()=>{this._isTransitioning=!1,this._element.classList.remove(_i),this._element.classList.add(gi,mi),this._element.style[e]="",fe.trigger(this._element,di)}),this._element,!0),this._element.style[e]=`${this._element[i]}px`}hide(){if(this._isTransitioning||!this._isShown())return;if(fe.trigger(this._element,ui).defaultPrevented)return;const t=this._getDimension();this._element.style[t]=`${this._element.getBoundingClientRect()[t]}px`,qt(this._element),this._element.classList.add(_i),this._element.classList.remove(gi,mi);for(const t of this._triggerArray){const e=we.getElementFromSelector(t);e&&!this._isShown(e)&&this._addAriaAndCollapsedClass([t],!1)}this._isTransitioning=!0,this._element.style[t]="",this._queueCallback((()=>{this._isTransitioning=!1,this._element.classList.remove(_i),this._element.classList.add(gi),fe.trigger(this._element,fi)}),this._element,!0)}_isShown(t=this._element){return t.classList.contains(mi)}_configAfterMerge(t){return t.toggle=Boolean(t.toggle),t.parent=Ht(t.parent),t}_getDimension(){return this._element.classList.contains("collapse-horizontal")?"width":"height"}_initializeChildren(){if(!this._config.parent)return;const t=this._getFirstLevelChildren(vi);for(const e of t){const t=we.getElementFromSelector(e);t&&this._addAriaAndCollapsedClass([e],this._isShown(t))}}_getFirstLevelChildren(t){const e=we.find(bi,this._config.parent);return we.find(t,this._config.parent).filter((t=>!e.includes(t)))}_addAriaAndCollapsedClass(t,e){if(t.length)for(const i of t)i.classList.toggle("collapsed",!e),i.setAttribute("aria-expanded",e)}static jQueryInterface(t){const e={};return"string"==typeof t&&/show|hide/.test(t)&&(e.toggle=!1),this.each((function(){const i=Ei.getOrCreateInstance(this,e);if("string"==typeof t){if(void 0===i[t])throw new TypeError(`No method named "${t}"`);i[t]()}}))}}fe.on(document,pi,vi,(function(t){("A"===t.target.tagName||t.delegateTarget&&"A"===t.delegateTarget.tagName)&&t.preventDefault();for(const t of we.getMultipleElementsFromSelector(this))Ei.getOrCreateInstance(t,{toggle:!1}).toggle()})),Qt(Ei);const Ai="dropdown",Ti=".bs.dropdown",Ci=".data-api",Oi="ArrowUp",xi="ArrowDown",ki=`hide${Ti}`,Li=`hidden${Ti}`,Si=`show${Ti}`,Di=`shown${Ti}`,$i=`click${Ti}${Ci}`,Ii=`keydown${Ti}${Ci}`,Ni=`keyup${Ti}${Ci}`,Pi="show",Mi='[data-bs-toggle="dropdown"]:not(.disabled):not(:disabled)',ji=`${Mi}.${Pi}`,Fi=".dropdown-menu",Hi=Kt()?"top-end":"top-start",Bi=Kt()?"top-start":"top-end",Wi=Kt()?"bottom-end":"bottom-start",zi=Kt()?"bottom-start":"bottom-end",Ri=Kt()?"left-start":"right-start",qi=Kt()?"right-start":"left-start",Vi={autoClose:!0,boundary:"clippingParents",display:"dynamic",offset:[0,2],popperConfig:null,reference:"toggle"},Yi={autoClose:"(boolean|string)",boundary:"(string|element)",display:"string",offset:"(array|string|function)",popperConfig:"(null|object|function)",reference:"(string|element|object)"};class Ki extends ve{constructor(t,e){super(t,e),this._popper=null,this._parent=this._element.parentNode,this._menu=we.next(this._element,Fi)[0]||we.prev(this._element,Fi)[0]||we.findOne(Fi,this._parent),this._inNavbar=this._detectNavbar()}static get Default(){return Vi}static get DefaultType(){return Yi}static get NAME(){return Ai}toggle(){return this._isShown()?this.hide():this.show()}show(){if(Wt(this._element)||this._isShown())return;const t={relatedTarget:this._element};if(!fe.trigger(this._element,Si,t).defaultPrevented){if(this._createPopper(),"ontouchstart"in document.documentElement&&!this._parent.closest(".navbar-nav"))for(const t of[].concat(...document.body.children))fe.on(t,"mouseover",Rt);this._element.focus(),this._element.setAttribute("aria-expanded",!0),this._menu.classList.add(Pi),this._element.classList.add(Pi),fe.trigger(this._element,Di,t)}}hide(){if(Wt(this._element)||!this._isShown())return;const t={relatedTarget:this._element};this._completeHide(t)}dispose(){this._popper&&this._popper.destroy(),super.dispose()}update(){this._inNavbar=this._detectNavbar(),this._popper&&this._popper.update()}_completeHide(t){if(!fe.trigger(this._element,ki,t).defaultPrevented){if("ontouchstart"in document.documentElement)for(const t of[].concat(...document.body.children))fe.off(t,"mouseover",Rt);this._popper&&this._popper.destroy(),this._menu.classList.remove(Pi),this._element.classList.remove(Pi),this._element.setAttribute("aria-expanded","false"),_e.removeDataAttribute(this._menu,"popper"),fe.trigger(this._element,Li,t)}}_getConfig(t){if("object"==typeof(t=super._getConfig(t)).reference&&!Ft(t.reference)&&"function"!=typeof t.reference.getBoundingClientRect)throw new TypeError(`${Ai.toUpperCase()}: Option "reference" provided type "object" without a required "getBoundingClientRect" method.`);return t}_createPopper(){if(void 0===e)throw new TypeError("Bootstrap's dropdowns require Popper (https://popper.js.org)");let t=this._element;"parent"===this._config.reference?t=this._parent:Ft(this._config.reference)?t=Ht(this._config.reference):"object"==typeof this._config.reference&&(t=this._config.reference);const i=this._getPopperConfig();this._popper=Dt(t,this._menu,i)}_isShown(){return this._menu.classList.contains(Pi)}_getPlacement(){const t=this._parent;if(t.classList.contains("dropend"))return Ri;if(t.classList.contains("dropstart"))return qi;if(t.classList.contains("dropup-center"))return"top";if(t.classList.contains("dropdown-center"))return"bottom";const e="end"===getComputedStyle(this._menu).getPropertyValue("--bs-position").trim();return t.classList.contains("dropup")?e?Bi:Hi:e?zi:Wi}_detectNavbar(){return null!==this._element.closest(".navbar")}_getOffset(){const{offset:t}=this._config;return"string"==typeof t?t.split(",").map((t=>Number.parseInt(t,10))):"function"==typeof t?e=>t(e,this._element):t}_getPopperConfig(){const t={placement:this._getPlacement(),modifiers:[{name:"preventOverflow",options:{boundary:this._config.boundary}},{name:"offset",options:{offset:this._getOffset()}}]};return(this._inNavbar||"static"===this._config.display)&&(_e.setDataAttribute(this._menu,"popper","static"),t.modifiers=[{name:"applyStyles",enabled:!1}]),{...t,...Xt(this._config.popperConfig,[t])}}_selectMenuItem({key:t,target:e}){const i=we.find(".dropdown-menu .dropdown-item:not(.disabled):not(:disabled)",this._menu).filter((t=>Bt(t)));i.length&&Gt(i,e,t===xi,!i.includes(e)).focus()}static jQueryInterface(t){return this.each((function(){const e=Ki.getOrCreateInstance(this,t);if("string"==typeof t){if(void 0===e[t])throw new TypeError(`No method named "${t}"`);e[t]()}}))}static clearMenus(t){if(2===t.button||"keyup"===t.type&&"Tab"!==t.key)return;const e=we.find(ji);for(const i of e){const e=Ki.getInstance(i);if(!e||!1===e._config.autoClose)continue;const n=t.composedPath(),s=n.includes(e._menu);if(n.includes(e._element)||"inside"===e._config.autoClose&&!s||"outside"===e._config.autoClose&&s)continue;if(e._menu.contains(t.target)&&("keyup"===t.type&&"Tab"===t.key||/input|select|option|textarea|form/i.test(t.target.tagName)))continue;const o={relatedTarget:e._element};"click"===t.type&&(o.clickEvent=t),e._completeHide(o)}}static dataApiKeydownHandler(t){const e=/input|textarea/i.test(t.target.tagName),i="Escape"===t.key,n=[Oi,xi].includes(t.key);if(!n&&!i)return;if(e&&!i)return;t.preventDefault();const s=this.matches(Mi)?this:we.prev(this,Mi)[0]||we.next(this,Mi)[0]||we.findOne(Mi,t.delegateTarget.parentNode),o=Ki.getOrCreateInstance(s);if(n)return t.stopPropagation(),o.show(),void o._selectMenuItem(t);o._isShown()&&(t.stopPropagation(),o.hide(),s.focus())}}fe.on(document,Ii,Mi,Ki.dataApiKeydownHandler),fe.on(document,Ii,Fi,Ki.dataApiKeydownHandler),fe.on(document,$i,Ki.clearMenus),fe.on(document,Ni,Ki.clearMenus),fe.on(document,$i,Mi,(function(t){t.preventDefault(),Ki.getOrCreateInstance(this).toggle()})),Qt(Ki);const Qi="backdrop",Xi="show",Ui=`mousedown.bs.${Qi}`,Gi={className:"modal-backdrop",clickCallback:null,isAnimated:!1,isVisible:!0,rootElement:"body"},Ji={className:"string",clickCallback:"(function|null)",isAnimated:"boolean",isVisible:"boolean",rootElement:"(element|string)"};class Zi extends be{constructor(t){super(),this._config=this._getConfig(t),this._isAppended=!1,this._element=null}static get Default(){return Gi}static get DefaultType(){return Ji}static get NAME(){return Qi}show(t){if(!this._config.isVisible)return void Xt(t);this._append();const e=this._getElement();this._config.isAnimated&&qt(e),e.classList.add(Xi),this._emulateAnimation((()=>{Xt(t)}))}hide(t){this._config.isVisible?(this._getElement().classList.remove(Xi),this._emulateAnimation((()=>{this.dispose(),Xt(t)}))):Xt(t)}dispose(){this._isAppended&&(fe.off(this._element,Ui),this._element.remove(),this._isAppended=!1)}_getElement(){if(!this._element){const t=document.createElement("div");t.className=this._config.className,this._config.isAnimated&&t.classList.add("fade"),this._element=t}return this._element}_configAfterMerge(t){return t.rootElement=Ht(t.rootElement),t}_append(){if(this._isAppended)return;const t=this._getElement();this._config.rootElement.append(t),fe.on(t,Ui,(()=>{Xt(this._config.clickCallback)})),this._isAppended=!0}_emulateAnimation(t){Ut(t,this._getElement(),this._config.isAnimated)}}const tn=".bs.focustrap",en=`focusin${tn}`,nn=`keydown.tab${tn}`,sn="backward",on={autofocus:!0,trapElement:null},rn={autofocus:"boolean",trapElement:"element"};class an extends be{constructor(t){super(),this._config=this._getConfig(t),this._isActive=!1,this._lastTabNavDirection=null}static get Default(){return on}static get DefaultType(){return rn}static get NAME(){return"focustrap"}activate(){this._isActive||(this._config.autofocus&&this._config.trapElement.focus(),fe.off(document,tn),fe.on(document,en,(t=>this._handleFocusin(t))),fe.on(document,nn,(t=>this._handleKeydown(t))),this._isActive=!0)}deactivate(){this._isActive&&(this._isActive=!1,fe.off(document,tn))}_handleFocusin(t){const{trapElement:e}=this._config;if(t.target===document||t.target===e||e.contains(t.target))return;const i=we.focusableChildren(e);0===i.length?e.focus():this._lastTabNavDirection===sn?i[i.length-1].focus():i[0].focus()}_handleKeydown(t){"Tab"===t.key&&(this._lastTabNavDirection=t.shiftKey?sn:"forward")}}const ln=".fixed-top, .fixed-bottom, .is-fixed, .sticky-top",cn=".sticky-top",hn="padding-right",dn="margin-right";class un{constructor(){this._element=document.body}getWidth(){const t=document.documentElement.clientWidth;return Math.abs(window.innerWidth-t)}hide(){const t=this.getWidth();this._disableOverFlow(),this._setElementAttributes(this._element,hn,(e=>e+t)),this._setElementAttributes(ln,hn,(e=>e+t)),this._setElementAttributes(cn,dn,(e=>e-t))}reset(){this._resetElementAttributes(this._element,"overflow"),this._resetElementAttributes(this._element,hn),this._resetElementAttributes(ln,hn),this._resetElementAttributes(cn,dn)}isOverflowing(){return this.getWidth()>0}_disableOverFlow(){this._saveInitialAttribute(this._element,"overflow"),this._element.style.overflow="hidden"}_setElementAttributes(t,e,i){const n=this.getWidth();this._applyManipulationCallback(t,(t=>{if(t!==this._element&&window.innerWidth>t.clientWidth+n)return;this._saveInitialAttribute(t,e);const s=window.getComputedStyle(t).getPropertyValue(e);t.style.setProperty(e,`${i(Number.parseFloat(s))}px`)}))}_saveInitialAttribute(t,e){const i=t.style.getPropertyValue(e);i&&_e.setDataAttribute(t,e,i)}_resetElementAttributes(t,e){this._applyManipulationCallback(t,(t=>{const i=_e.getDataAttribute(t,e);null!==i?(_e.removeDataAttribute(t,e),t.style.setProperty(e,i)):t.style.removeProperty(e)}))}_applyManipulationCallback(t,e){if(Ft(t))e(t);else for(const i of we.find(t,this._element))e(i)}}const fn=".bs.modal",pn=`hide${fn}`,mn=`hidePrevented${fn}`,gn=`hidden${fn}`,_n=`show${fn}`,bn=`shown${fn}`,vn=`resize${fn}`,yn=`click.dismiss${fn}`,wn=`mousedown.dismiss${fn}`,En=`keydown.dismiss${fn}`,An=`click${fn}.data-api`,Tn="modal-open",Cn="show",On="modal-static",xn={backdrop:!0,focus:!0,keyboard:!0},kn={backdrop:"(boolean|string)",focus:"boolean",keyboard:"boolean"};class Ln extends ve{constructor(t,e){super(t,e),this._dialog=we.findOne(".modal-dialog",this._element),this._backdrop=this._initializeBackDrop(),this._focustrap=this._initializeFocusTrap(),this._isShown=!1,this._isTransitioning=!1,this._scrollBar=new un,this._addEventListeners()}static get Default(){return xn}static get DefaultType(){return kn}static get NAME(){return"modal"}toggle(t){return this._isShown?this.hide():this.show(t)}show(t){this._isShown||this._isTransitioning||fe.trigger(this._element,_n,{relatedTarget:t}).defaultPrevented||(this._isShown=!0,this._isTransitioning=!0,this._scrollBar.hide(),document.body.classList.add(Tn),this._adjustDialog(),this._backdrop.show((()=>this._showElement(t))))}hide(){this._isShown&&!this._isTransitioning&&(fe.trigger(this._element,pn).defaultPrevented||(this._isShown=!1,this._isTransitioning=!0,this._focustrap.deactivate(),this._element.classList.remove(Cn),this._queueCallback((()=>this._hideModal()),this._element,this._isAnimated())))}dispose(){fe.off(window,fn),fe.off(this._dialog,fn),this._backdrop.dispose(),this._focustrap.deactivate(),super.dispose()}handleUpdate(){this._adjustDialog()}_initializeBackDrop(){return new Zi({isVisible:Boolean(this._config.backdrop),isAnimated:this._isAnimated()})}_initializeFocusTrap(){return new an({trapElement:this._element})}_showElement(t){document.body.contains(this._element)||document.body.append(this._element),this._element.style.display="block",this._element.removeAttribute("aria-hidden"),this._element.setAttribute("aria-modal",!0),this._element.setAttribute("role","dialog"),this._element.scrollTop=0;const e=we.findOne(".modal-body",this._dialog);e&&(e.scrollTop=0),qt(this._element),this._element.classList.add(Cn),this._queueCallback((()=>{this._config.focus&&this._focustrap.activate(),this._isTransitioning=!1,fe.trigger(this._element,bn,{relatedTarget:t})}),this._dialog,this._isAnimated())}_addEventListeners(){fe.on(this._element,En,(t=>{"Escape"===t.key&&(this._config.keyboard?this.hide():this._triggerBackdropTransition())})),fe.on(window,vn,(()=>{this._isShown&&!this._isTransitioning&&this._adjustDialog()})),fe.on(this._element,wn,(t=>{fe.one(this._element,yn,(e=>{this._element===t.target&&this._element===e.target&&("static"!==this._config.backdrop?this._config.backdrop&&this.hide():this._triggerBackdropTransition())}))}))}_hideModal(){this._element.style.display="none",this._element.setAttribute("aria-hidden",!0),this._element.removeAttribute("aria-modal"),this._element.removeAttribute("role"),this._isTransitioning=!1,this._backdrop.hide((()=>{document.body.classList.remove(Tn),this._resetAdjustments(),this._scrollBar.reset(),fe.trigger(this._element,gn)}))}_isAnimated(){return this._element.classList.contains("fade")}_triggerBackdropTransition(){if(fe.trigger(this._element,mn).defaultPrevented)return;const t=this._element.scrollHeight>document.documentElement.clientHeight,e=this._element.style.overflowY;"hidden"===e||this._element.classList.contains(On)||(t||(this._element.style.overflowY="hidden"),this._element.classList.add(On),this._queueCallback((()=>{this._element.classList.remove(On),this._queueCallback((()=>{this._element.style.overflowY=e}),this._dialog)}),this._dialog),this._element.focus())}_adjustDialog(){const t=this._element.scrollHeight>document.documentElement.clientHeight,e=this._scrollBar.getWidth(),i=e>0;if(i&&!t){const t=Kt()?"paddingLeft":"paddingRight";this._element.style[t]=`${e}px`}if(!i&&t){const t=Kt()?"paddingRight":"paddingLeft";this._element.style[t]=`${e}px`}}_resetAdjustments(){this._element.style.paddingLeft="",this._element.style.paddingRight=""}static jQueryInterface(t,e){return this.each((function(){const i=Ln.getOrCreateInstance(this,t);if("string"==typeof t){if(void 0===i[t])throw new TypeError(`No method named "${t}"`);i[t](e)}}))}}fe.on(document,An,'[data-bs-toggle="modal"]',(function(t){const e=we.getElementFromSelector(this);["A","AREA"].includes(this.tagName)&&t.preventDefault(),fe.one(e,_n,(t=>{t.defaultPrevented||fe.one(e,gn,(()=>{Bt(this)&&this.focus()}))}));const i=we.findOne(".modal.show");i&&Ln.getInstance(i).hide(),Ln.getOrCreateInstance(e).toggle(this)})),Ee(Ln),Qt(Ln);const Sn=".bs.offcanvas",Dn=".data-api",$n=`load${Sn}${Dn}`,In="show",Nn="showing",Pn="hiding",Mn=".offcanvas.show",jn=`show${Sn}`,Fn=`shown${Sn}`,Hn=`hide${Sn}`,Bn=`hidePrevented${Sn}`,Wn=`hidden${Sn}`,zn=`resize${Sn}`,Rn=`click${Sn}${Dn}`,qn=`keydown.dismiss${Sn}`,Vn={backdrop:!0,keyboard:!0,scroll:!1},Yn={backdrop:"(boolean|string)",keyboard:"boolean",scroll:"boolean"};class Kn extends ve{constructor(t,e){super(t,e),this._isShown=!1,this._backdrop=this._initializeBackDrop(),this._focustrap=this._initializeFocusTrap(),this._addEventListeners()}static get Default(){return Vn}static get DefaultType(){return Yn}static get NAME(){return"offcanvas"}toggle(t){return this._isShown?this.hide():this.show(t)}show(t){this._isShown||fe.trigger(this._element,jn,{relatedTarget:t}).defaultPrevented||(this._isShown=!0,this._backdrop.show(),this._config.scroll||(new un).hide(),this._element.setAttribute("aria-modal",!0),this._element.setAttribute("role","dialog"),this._element.classList.add(Nn),this._queueCallback((()=>{this._config.scroll&&!this._config.backdrop||this._focustrap.activate(),this._element.classList.add(In),this._element.classList.remove(Nn),fe.trigger(this._element,Fn,{relatedTarget:t})}),this._element,!0))}hide(){this._isShown&&(fe.trigger(this._element,Hn).defaultPrevented||(this._focustrap.deactivate(),this._element.blur(),this._isShown=!1,this._element.classList.add(Pn),this._backdrop.hide(),this._queueCallback((()=>{this._element.classList.remove(In,Pn),this._element.removeAttribute("aria-modal"),this._element.removeAttribute("role"),this._config.scroll||(new un).reset(),fe.trigger(this._element,Wn)}),this._element,!0)))}dispose(){this._backdrop.dispose(),this._focustrap.deactivate(),super.dispose()}_initializeBackDrop(){const t=Boolean(this._config.backdrop);return new Zi({className:"offcanvas-backdrop",isVisible:t,isAnimated:!0,rootElement:this._element.parentNode,clickCallback:t?()=>{"static"!==this._config.backdrop?this.hide():fe.trigger(this._element,Bn)}:null})}_initializeFocusTrap(){return new an({trapElement:this._element})}_addEventListeners(){fe.on(this._element,qn,(t=>{"Escape"===t.key&&(this._config.keyboard?this.hide():fe.trigger(this._element,Bn))}))}static jQueryInterface(t){return this.each((function(){const e=Kn.getOrCreateInstance(this,t);if("string"==typeof t){if(void 0===e[t]||t.startsWith("_")||"constructor"===t)throw new TypeError(`No method named "${t}"`);e[t](this)}}))}}fe.on(document,Rn,'[data-bs-toggle="offcanvas"]',(function(t){const e=we.getElementFromSelector(this);if(["A","AREA"].includes(this.tagName)&&t.preventDefault(),Wt(this))return;fe.one(e,Wn,(()=>{Bt(this)&&this.focus()}));const i=we.findOne(Mn);i&&i!==e&&Kn.getInstance(i).hide(),Kn.getOrCreateInstance(e).toggle(this)})),fe.on(window,$n,(()=>{for(const t of we.find(Mn))Kn.getOrCreateInstance(t).show()})),fe.on(window,zn,(()=>{for(const t of we.find("[aria-modal][class*=show][class*=offcanvas-]"))"fixed"!==getComputedStyle(t).position&&Kn.getOrCreateInstance(t).hide()})),Ee(Kn),Qt(Kn);const Qn={"*":["class","dir","id","lang","role",/^aria-[\w-]*$/i],a:["target","href","title","rel"],area:[],b:[],br:[],col:[],code:[],dd:[],div:[],dl:[],dt:[],em:[],hr:[],h1:[],h2:[],h3:[],h4:[],h5:[],h6:[],i:[],img:["src","srcset","alt","title","width","height"],li:[],ol:[],p:[],pre:[],s:[],small:[],span:[],sub:[],sup:[],strong:[],u:[],ul:[]},Xn=new Set(["background","cite","href","itemtype","longdesc","poster","src","xlink:href"]),Un=/^(?!javascript:)(?:[a-z0-9+.-]+:|[^&:/?#]*(?:[/?#]|$))/i,Gn=(t,e)=>{const i=t.nodeName.toLowerCase();return e.includes(i)?!Xn.has(i)||Boolean(Un.test(t.nodeValue)):e.filter((t=>t instanceof RegExp)).some((t=>t.test(i)))},Jn={allowList:Qn,content:{},extraClass:"",html:!1,sanitize:!0,sanitizeFn:null,template:"
"},Zn={allowList:"object",content:"object",extraClass:"(string|function)",html:"boolean",sanitize:"boolean",sanitizeFn:"(null|function)",template:"string"},ts={entry:"(string|element|function|null)",selector:"(string|element)"};class es extends be{constructor(t){super(),this._config=this._getConfig(t)}static get Default(){return Jn}static get DefaultType(){return Zn}static get NAME(){return"TemplateFactory"}getContent(){return Object.values(this._config.content).map((t=>this._resolvePossibleFunction(t))).filter(Boolean)}hasContent(){return this.getContent().length>0}changeContent(t){return this._checkContent(t),this._config.content={...this._config.content,...t},this}toHtml(){const t=document.createElement("div");t.innerHTML=this._maybeSanitize(this._config.template);for(const[e,i]of Object.entries(this._config.content))this._setContent(t,i,e);const e=t.children[0],i=this._resolvePossibleFunction(this._config.extraClass);return i&&e.classList.add(...i.split(" ")),e}_typeCheckConfig(t){super._typeCheckConfig(t),this._checkContent(t.content)}_checkContent(t){for(const[e,i]of Object.entries(t))super._typeCheckConfig({selector:e,entry:i},ts)}_setContent(t,e,i){const n=we.findOne(i,t);n&&((e=this._resolvePossibleFunction(e))?Ft(e)?this._putElementInTemplate(Ht(e),n):this._config.html?n.innerHTML=this._maybeSanitize(e):n.textContent=e:n.remove())}_maybeSanitize(t){return this._config.sanitize?function(t,e,i){if(!t.length)return t;if(i&&"function"==typeof i)return i(t);const n=(new window.DOMParser).parseFromString(t,"text/html"),s=[].concat(...n.body.querySelectorAll("*"));for(const t of s){const i=t.nodeName.toLowerCase();if(!Object.keys(e).includes(i)){t.remove();continue}const n=[].concat(...t.attributes),s=[].concat(e["*"]||[],e[i]||[]);for(const e of n)Gn(e,s)||t.removeAttribute(e.nodeName)}return n.body.innerHTML}(t,this._config.allowList,this._config.sanitizeFn):t}_resolvePossibleFunction(t){return Xt(t,[this])}_putElementInTemplate(t,e){if(this._config.html)return e.innerHTML="",void e.append(t);e.textContent=t.textContent}}const is=new Set(["sanitize","allowList","sanitizeFn"]),ns="fade",ss="show",os=".tooltip-inner",rs=".modal",as="hide.bs.modal",ls="hover",cs="focus",hs={AUTO:"auto",TOP:"top",RIGHT:Kt()?"left":"right",BOTTOM:"bottom",LEFT:Kt()?"right":"left"},ds={allowList:Qn,animation:!0,boundary:"clippingParents",container:!1,customClass:"",delay:0,fallbackPlacements:["top","right","bottom","left"],html:!1,offset:[0,6],placement:"top",popperConfig:null,sanitize:!0,sanitizeFn:null,selector:!1,template:'',title:"",trigger:"hover focus"},us={allowList:"object",animation:"boolean",boundary:"(string|element)",container:"(string|element|boolean)",customClass:"(string|function)",delay:"(number|object)",fallbackPlacements:"array",html:"boolean",offset:"(array|string|function)",placement:"(string|function)",popperConfig:"(null|object|function)",sanitize:"boolean",sanitizeFn:"(null|function)",selector:"(string|boolean)",template:"string",title:"(string|element|function)",trigger:"string"};class fs extends ve{constructor(t,i){if(void 0===e)throw new TypeError("Bootstrap's tooltips require Popper (https://popper.js.org)");super(t,i),this._isEnabled=!0,this._timeout=0,this._isHovered=null,this._activeTrigger={},this._popper=null,this._templateFactory=null,this._newContent=null,this.tip=null,this._setListeners(),this._config.selector||this._fixTitle()}static get Default(){return ds}static get DefaultType(){return us}static get NAME(){return"tooltip"}enable(){this._isEnabled=!0}disable(){this._isEnabled=!1}toggleEnabled(){this._isEnabled=!this._isEnabled}toggle(){this._isEnabled&&(this._activeTrigger.click=!this._activeTrigger.click,this._isShown()?this._leave():this._enter())}dispose(){clearTimeout(this._timeout),fe.off(this._element.closest(rs),as,this._hideModalHandler),this._element.getAttribute("data-bs-original-title")&&this._element.setAttribute("title",this._element.getAttribute("data-bs-original-title")),this._disposePopper(),super.dispose()}show(){if("none"===this._element.style.display)throw new Error("Please use show on visible elements");if(!this._isWithContent()||!this._isEnabled)return;const t=fe.trigger(this._element,this.constructor.eventName("show")),e=(zt(this._element)||this._element.ownerDocument.documentElement).contains(this._element);if(t.defaultPrevented||!e)return;this._disposePopper();const i=this._getTipElement();this._element.setAttribute("aria-describedby",i.getAttribute("id"));const{container:n}=this._config;if(this._element.ownerDocument.documentElement.contains(this.tip)||(n.append(i),fe.trigger(this._element,this.constructor.eventName("inserted"))),this._popper=this._createPopper(i),i.classList.add(ss),"ontouchstart"in document.documentElement)for(const t of[].concat(...document.body.children))fe.on(t,"mouseover",Rt);this._queueCallback((()=>{fe.trigger(this._element,this.constructor.eventName("shown")),!1===this._isHovered&&this._leave(),this._isHovered=!1}),this.tip,this._isAnimated())}hide(){if(this._isShown()&&!fe.trigger(this._element,this.constructor.eventName("hide")).defaultPrevented){if(this._getTipElement().classList.remove(ss),"ontouchstart"in document.documentElement)for(const t of[].concat(...document.body.children))fe.off(t,"mouseover",Rt);this._activeTrigger.click=!1,this._activeTrigger[cs]=!1,this._activeTrigger[ls]=!1,this._isHovered=null,this._queueCallback((()=>{this._isWithActiveTrigger()||(this._isHovered||this._disposePopper(),this._element.removeAttribute("aria-describedby"),fe.trigger(this._element,this.constructor.eventName("hidden")))}),this.tip,this._isAnimated())}}update(){this._popper&&this._popper.update()}_isWithContent(){return Boolean(this._getTitle())}_getTipElement(){return this.tip||(this.tip=this._createTipElement(this._newContent||this._getContentForTemplate())),this.tip}_createTipElement(t){const e=this._getTemplateFactory(t).toHtml();if(!e)return null;e.classList.remove(ns,ss),e.classList.add(`bs-${this.constructor.NAME}-auto`);const i=(t=>{do{t+=Math.floor(1e6*Math.random())}while(document.getElementById(t));return t})(this.constructor.NAME).toString();return e.setAttribute("id",i),this._isAnimated()&&e.classList.add(ns),e}setContent(t){this._newContent=t,this._isShown()&&(this._disposePopper(),this.show())}_getTemplateFactory(t){return this._templateFactory?this._templateFactory.changeContent(t):this._templateFactory=new es({...this._config,content:t,extraClass:this._resolvePossibleFunction(this._config.customClass)}),this._templateFactory}_getContentForTemplate(){return{[os]:this._getTitle()}}_getTitle(){return this._resolvePossibleFunction(this._config.title)||this._element.getAttribute("data-bs-original-title")}_initializeOnDelegatedTarget(t){return this.constructor.getOrCreateInstance(t.delegateTarget,this._getDelegateConfig())}_isAnimated(){return this._config.animation||this.tip&&this.tip.classList.contains(ns)}_isShown(){return this.tip&&this.tip.classList.contains(ss)}_createPopper(t){const e=Xt(this._config.placement,[this,t,this._element]),i=hs[e.toUpperCase()];return Dt(this._element,t,this._getPopperConfig(i))}_getOffset(){const{offset:t}=this._config;return"string"==typeof t?t.split(",").map((t=>Number.parseInt(t,10))):"function"==typeof t?e=>t(e,this._element):t}_resolvePossibleFunction(t){return Xt(t,[this._element])}_getPopperConfig(t){const e={placement:t,modifiers:[{name:"flip",options:{fallbackPlacements:this._config.fallbackPlacements}},{name:"offset",options:{offset:this._getOffset()}},{name:"preventOverflow",options:{boundary:this._config.boundary}},{name:"arrow",options:{element:`.${this.constructor.NAME}-arrow`}},{name:"preSetPlacement",enabled:!0,phase:"beforeMain",fn:t=>{this._getTipElement().setAttribute("data-popper-placement",t.state.placement)}}]};return{...e,...Xt(this._config.popperConfig,[e])}}_setListeners(){const t=this._config.trigger.split(" ");for(const e of t)if("click"===e)fe.on(this._element,this.constructor.eventName("click"),this._config.selector,(t=>{this._initializeOnDelegatedTarget(t).toggle()}));else if("manual"!==e){const t=e===ls?this.constructor.eventName("mouseenter"):this.constructor.eventName("focusin"),i=e===ls?this.constructor.eventName("mouseleave"):this.constructor.eventName("focusout");fe.on(this._element,t,this._config.selector,(t=>{const e=this._initializeOnDelegatedTarget(t);e._activeTrigger["focusin"===t.type?cs:ls]=!0,e._enter()})),fe.on(this._element,i,this._config.selector,(t=>{const e=this._initializeOnDelegatedTarget(t);e._activeTrigger["focusout"===t.type?cs:ls]=e._element.contains(t.relatedTarget),e._leave()}))}this._hideModalHandler=()=>{this._element&&this.hide()},fe.on(this._element.closest(rs),as,this._hideModalHandler)}_fixTitle(){const t=this._element.getAttribute("title");t&&(this._element.getAttribute("aria-label")||this._element.textContent.trim()||this._element.setAttribute("aria-label",t),this._element.setAttribute("data-bs-original-title",t),this._element.removeAttribute("title"))}_enter(){this._isShown()||this._isHovered?this._isHovered=!0:(this._isHovered=!0,this._setTimeout((()=>{this._isHovered&&this.show()}),this._config.delay.show))}_leave(){this._isWithActiveTrigger()||(this._isHovered=!1,this._setTimeout((()=>{this._isHovered||this.hide()}),this._config.delay.hide))}_setTimeout(t,e){clearTimeout(this._timeout),this._timeout=setTimeout(t,e)}_isWithActiveTrigger(){return Object.values(this._activeTrigger).includes(!0)}_getConfig(t){const e=_e.getDataAttributes(this._element);for(const t of Object.keys(e))is.has(t)&&delete e[t];return t={...e,..."object"==typeof t&&t?t:{}},t=this._mergeConfigObj(t),t=this._configAfterMerge(t),this._typeCheckConfig(t),t}_configAfterMerge(t){return t.container=!1===t.container?document.body:Ht(t.container),"number"==typeof t.delay&&(t.delay={show:t.delay,hide:t.delay}),"number"==typeof t.title&&(t.title=t.title.toString()),"number"==typeof t.content&&(t.content=t.content.toString()),t}_getDelegateConfig(){const t={};for(const[e,i]of Object.entries(this._config))this.constructor.Default[e]!==i&&(t[e]=i);return t.selector=!1,t.trigger="manual",t}_disposePopper(){this._popper&&(this._popper.destroy(),this._popper=null),this.tip&&(this.tip.remove(),this.tip=null)}static jQueryInterface(t){return this.each((function(){const e=fs.getOrCreateInstance(this,t);if("string"==typeof t){if(void 0===e[t])throw new TypeError(`No method named "${t}"`);e[t]()}}))}}Qt(fs);const ps=".popover-header",ms=".popover-body",gs={...fs.Default,content:"",offset:[0,8],placement:"right",template:'',trigger:"click"},_s={...fs.DefaultType,content:"(null|string|element|function)"};class bs extends fs{static get Default(){return gs}static get DefaultType(){return _s}static get NAME(){return"popover"}_isWithContent(){return this._getTitle()||this._getContent()}_getContentForTemplate(){return{[ps]:this._getTitle(),[ms]:this._getContent()}}_getContent(){return this._resolvePossibleFunction(this._config.content)}static jQueryInterface(t){return this.each((function(){const e=bs.getOrCreateInstance(this,t);if("string"==typeof t){if(void 0===e[t])throw new TypeError(`No method named "${t}"`);e[t]()}}))}}Qt(bs);const vs=".bs.scrollspy",ys=`activate${vs}`,ws=`click${vs}`,Es=`load${vs}.data-api`,As="active",Ts="[href]",Cs=".nav-link",Os=`${Cs}, .nav-item > ${Cs}, .list-group-item`,xs={offset:null,rootMargin:"0px 0px -25%",smoothScroll:!1,target:null,threshold:[.1,.5,1]},ks={offset:"(number|null)",rootMargin:"string",smoothScroll:"boolean",target:"element",threshold:"array"};class Ls extends ve{constructor(t,e){super(t,e),this._targetLinks=new Map,this._observableSections=new Map,this._rootElement="visible"===getComputedStyle(this._element).overflowY?null:this._element,this._activeTarget=null,this._observer=null,this._previousScrollData={visibleEntryTop:0,parentScrollTop:0},this.refresh()}static get Default(){return xs}static get DefaultType(){return ks}static get NAME(){return"scrollspy"}refresh(){this._initializeTargetsAndObservables(),this._maybeEnableSmoothScroll(),this._observer?this._observer.disconnect():this._observer=this._getNewObserver();for(const t of this._observableSections.values())this._observer.observe(t)}dispose(){this._observer.disconnect(),super.dispose()}_configAfterMerge(t){return t.target=Ht(t.target)||document.body,t.rootMargin=t.offset?`${t.offset}px 0px -30%`:t.rootMargin,"string"==typeof t.threshold&&(t.threshold=t.threshold.split(",").map((t=>Number.parseFloat(t)))),t}_maybeEnableSmoothScroll(){this._config.smoothScroll&&(fe.off(this._config.target,ws),fe.on(this._config.target,ws,Ts,(t=>{const e=this._observableSections.get(t.target.hash);if(e){t.preventDefault();const i=this._rootElement||window,n=e.offsetTop-this._element.offsetTop;if(i.scrollTo)return void i.scrollTo({top:n,behavior:"smooth"});i.scrollTop=n}})))}_getNewObserver(){const t={root:this._rootElement,threshold:this._config.threshold,rootMargin:this._config.rootMargin};return new IntersectionObserver((t=>this._observerCallback(t)),t)}_observerCallback(t){const e=t=>this._targetLinks.get(`#${t.target.id}`),i=t=>{this._previousScrollData.visibleEntryTop=t.target.offsetTop,this._process(e(t))},n=(this._rootElement||document.documentElement).scrollTop,s=n>=this._previousScrollData.parentScrollTop;this._previousScrollData.parentScrollTop=n;for(const o of t){if(!o.isIntersecting){this._activeTarget=null,this._clearActiveClass(e(o));continue}const t=o.target.offsetTop>=this._previousScrollData.visibleEntryTop;if(s&&t){if(i(o),!n)return}else s||t||i(o)}}_initializeTargetsAndObservables(){this._targetLinks=new Map,this._observableSections=new Map;const t=we.find(Ts,this._config.target);for(const e of t){if(!e.hash||Wt(e))continue;const t=we.findOne(decodeURI(e.hash),this._element);Bt(t)&&(this._targetLinks.set(decodeURI(e.hash),e),this._observableSections.set(e.hash,t))}}_process(t){this._activeTarget!==t&&(this._clearActiveClass(this._config.target),this._activeTarget=t,t.classList.add(As),this._activateParents(t),fe.trigger(this._element,ys,{relatedTarget:t}))}_activateParents(t){if(t.classList.contains("dropdown-item"))we.findOne(".dropdown-toggle",t.closest(".dropdown")).classList.add(As);else for(const e of we.parents(t,".nav, .list-group"))for(const t of we.prev(e,Os))t.classList.add(As)}_clearActiveClass(t){t.classList.remove(As);const e=we.find(`${Ts}.${As}`,t);for(const t of e)t.classList.remove(As)}static jQueryInterface(t){return this.each((function(){const e=Ls.getOrCreateInstance(this,t);if("string"==typeof t){if(void 0===e[t]||t.startsWith("_")||"constructor"===t)throw new TypeError(`No method named "${t}"`);e[t]()}}))}}fe.on(window,Es,(()=>{for(const t of we.find('[data-bs-spy="scroll"]'))Ls.getOrCreateInstance(t)})),Qt(Ls);const Ss=".bs.tab",Ds=`hide${Ss}`,$s=`hidden${Ss}`,Is=`show${Ss}`,Ns=`shown${Ss}`,Ps=`click${Ss}`,Ms=`keydown${Ss}`,js=`load${Ss}`,Fs="ArrowLeft",Hs="ArrowRight",Bs="ArrowUp",Ws="ArrowDown",zs="Home",Rs="End",qs="active",Vs="fade",Ys="show",Ks=".dropdown-toggle",Qs=`:not(${Ks})`,Xs='[data-bs-toggle="tab"], [data-bs-toggle="pill"], [data-bs-toggle="list"]',Us=`.nav-link${Qs}, .list-group-item${Qs}, [role="tab"]${Qs}, ${Xs}`,Gs=`.${qs}[data-bs-toggle="tab"], .${qs}[data-bs-toggle="pill"], .${qs}[data-bs-toggle="list"]`;class Js extends ve{constructor(t){super(t),this._parent=this._element.closest('.list-group, .nav, [role="tablist"]'),this._parent&&(this._setInitialAttributes(this._parent,this._getChildren()),fe.on(this._element,Ms,(t=>this._keydown(t))))}static get NAME(){return"tab"}show(){const t=this._element;if(this._elemIsActive(t))return;const e=this._getActiveElem(),i=e?fe.trigger(e,Ds,{relatedTarget:t}):null;fe.trigger(t,Is,{relatedTarget:e}).defaultPrevented||i&&i.defaultPrevented||(this._deactivate(e,t),this._activate(t,e))}_activate(t,e){t&&(t.classList.add(qs),this._activate(we.getElementFromSelector(t)),this._queueCallback((()=>{"tab"===t.getAttribute("role")?(t.removeAttribute("tabindex"),t.setAttribute("aria-selected",!0),this._toggleDropDown(t,!0),fe.trigger(t,Ns,{relatedTarget:e})):t.classList.add(Ys)}),t,t.classList.contains(Vs)))}_deactivate(t,e){t&&(t.classList.remove(qs),t.blur(),this._deactivate(we.getElementFromSelector(t)),this._queueCallback((()=>{"tab"===t.getAttribute("role")?(t.setAttribute("aria-selected",!1),t.setAttribute("tabindex","-1"),this._toggleDropDown(t,!1),fe.trigger(t,$s,{relatedTarget:e})):t.classList.remove(Ys)}),t,t.classList.contains(Vs)))}_keydown(t){if(![Fs,Hs,Bs,Ws,zs,Rs].includes(t.key))return;t.stopPropagation(),t.preventDefault();const e=this._getChildren().filter((t=>!Wt(t)));let i;if([zs,Rs].includes(t.key))i=e[t.key===zs?0:e.length-1];else{const n=[Hs,Ws].includes(t.key);i=Gt(e,t.target,n,!0)}i&&(i.focus({preventScroll:!0}),Js.getOrCreateInstance(i).show())}_getChildren(){return we.find(Us,this._parent)}_getActiveElem(){return this._getChildren().find((t=>this._elemIsActive(t)))||null}_setInitialAttributes(t,e){this._setAttributeIfNotExists(t,"role","tablist");for(const t of e)this._setInitialAttributesOnChild(t)}_setInitialAttributesOnChild(t){t=this._getInnerElement(t);const e=this._elemIsActive(t),i=this._getOuterElement(t);t.setAttribute("aria-selected",e),i!==t&&this._setAttributeIfNotExists(i,"role","presentation"),e||t.setAttribute("tabindex","-1"),this._setAttributeIfNotExists(t,"role","tab"),this._setInitialAttributesOnTargetPanel(t)}_setInitialAttributesOnTargetPanel(t){const e=we.getElementFromSelector(t);e&&(this._setAttributeIfNotExists(e,"role","tabpanel"),t.id&&this._setAttributeIfNotExists(e,"aria-labelledby",`${t.id}`))}_toggleDropDown(t,e){const i=this._getOuterElement(t);if(!i.classList.contains("dropdown"))return;const n=(t,n)=>{const s=we.findOne(t,i);s&&s.classList.toggle(n,e)};n(Ks,qs),n(".dropdown-menu",Ys),i.setAttribute("aria-expanded",e)}_setAttributeIfNotExists(t,e,i){t.hasAttribute(e)||t.setAttribute(e,i)}_elemIsActive(t){return t.classList.contains(qs)}_getInnerElement(t){return t.matches(Us)?t:we.findOne(Us,t)}_getOuterElement(t){return t.closest(".nav-item, .list-group-item")||t}static jQueryInterface(t){return this.each((function(){const e=Js.getOrCreateInstance(this);if("string"==typeof t){if(void 0===e[t]||t.startsWith("_")||"constructor"===t)throw new TypeError(`No method named "${t}"`);e[t]()}}))}}fe.on(document,Ps,Xs,(function(t){["A","AREA"].includes(this.tagName)&&t.preventDefault(),Wt(this)||Js.getOrCreateInstance(this).show()})),fe.on(window,js,(()=>{for(const t of we.find(Gs))Js.getOrCreateInstance(t)})),Qt(Js);const Zs=".bs.toast",to=`mouseover${Zs}`,eo=`mouseout${Zs}`,io=`focusin${Zs}`,no=`focusout${Zs}`,so=`hide${Zs}`,oo=`hidden${Zs}`,ro=`show${Zs}`,ao=`shown${Zs}`,lo="hide",co="show",ho="showing",uo={animation:"boolean",autohide:"boolean",delay:"number"},fo={animation:!0,autohide:!0,delay:5e3};class po extends ve{constructor(t,e){super(t,e),this._timeout=null,this._hasMouseInteraction=!1,this._hasKeyboardInteraction=!1,this._setListeners()}static get Default(){return fo}static get DefaultType(){return uo}static get NAME(){return"toast"}show(){fe.trigger(this._element,ro).defaultPrevented||(this._clearTimeout(),this._config.animation&&this._element.classList.add("fade"),this._element.classList.remove(lo),qt(this._element),this._element.classList.add(co,ho),this._queueCallback((()=>{this._element.classList.remove(ho),fe.trigger(this._element,ao),this._maybeScheduleHide()}),this._element,this._config.animation))}hide(){this.isShown()&&(fe.trigger(this._element,so).defaultPrevented||(this._element.classList.add(ho),this._queueCallback((()=>{this._element.classList.add(lo),this._element.classList.remove(ho,co),fe.trigger(this._element,oo)}),this._element,this._config.animation)))}dispose(){this._clearTimeout(),this.isShown()&&this._element.classList.remove(co),super.dispose()}isShown(){return this._element.classList.contains(co)}_maybeScheduleHide(){this._config.autohide&&(this._hasMouseInteraction||this._hasKeyboardInteraction||(this._timeout=setTimeout((()=>{this.hide()}),this._config.delay)))}_onInteraction(t,e){switch(t.type){case"mouseover":case"mouseout":this._hasMouseInteraction=e;break;case"focusin":case"focusout":this._hasKeyboardInteraction=e}if(e)return void this._clearTimeout();const i=t.relatedTarget;this._element===i||this._element.contains(i)||this._maybeScheduleHide()}_setListeners(){fe.on(this._element,to,(t=>this._onInteraction(t,!0))),fe.on(this._element,eo,(t=>this._onInteraction(t,!1))),fe.on(this._element,io,(t=>this._onInteraction(t,!0))),fe.on(this._element,no,(t=>this._onInteraction(t,!1)))}_clearTimeout(){clearTimeout(this._timeout),this._timeout=null}static jQueryInterface(t){return this.each((function(){const e=po.getOrCreateInstance(this,t);if("string"==typeof t){if(void 0===e[t])throw new TypeError(`No method named "${t}"`);e[t](this)}}))}}function mo(t){"loading"!=document.readyState?t():document.addEventListener("DOMContentLoaded",t)}Ee(po),Qt(po),mo((function(){[].slice.call(document.querySelectorAll('[data-bs-toggle="tooltip"]')).map((function(t){return new fs(t,{delay:{show:500,hide:100}})}))})),mo((function(){document.getElementById("pst-back-to-top").addEventListener("click",(function(){document.body.scrollTop=0,document.documentElement.scrollTop=0}))})),mo((function(){var t=document.getElementById("pst-back-to-top"),e=document.getElementsByClassName("bd-header")[0].getBoundingClientRect();window.addEventListener("scroll",(function(){this.oldScroll>this.scrollY&&this.scrollY>e.bottom?t.style.display="block":t.style.display="none",this.oldScroll=this.scrollY}))})),window.bootstrap=i})(); +//# sourceMappingURL=bootstrap.js.map \ No newline at end of file diff --git a/_static/scripts/bootstrap.js.LICENSE.txt b/_static/scripts/bootstrap.js.LICENSE.txt new file mode 100644 index 0000000..28755c2 --- /dev/null +++ b/_static/scripts/bootstrap.js.LICENSE.txt @@ -0,0 +1,5 @@ +/*! + * Bootstrap v5.3.3 (https://getbootstrap.com/) + * Copyright 2011-2024 The Bootstrap Authors (https://github.com/twbs/bootstrap/graphs/contributors) + * Licensed under MIT (https://github.com/twbs/bootstrap/blob/main/LICENSE) + */ diff --git a/_static/scripts/bootstrap.js.map b/_static/scripts/bootstrap.js.map new file mode 100644 index 0000000..4a3502a --- /dev/null +++ b/_static/scripts/bootstrap.js.map @@ -0,0 +1 @@ +{"version":3,"file":"scripts/bootstrap.js","mappings":"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","sources":["webpack://pydata_sphinx_theme/webpack/bootstrap","webpack://pydata_sphinx_theme/webpack/runtime/define property getters","webpack://pydata_sphinx_theme/webpack/runtime/hasOwnProperty shorthand","webpack://pydata_sphinx_theme/webpack/runtime/make namespace object","webpack://pydata_sphinx_theme/./node_modules/@popperjs/core/lib/enums.js","webpack://pydata_sphinx_theme/./node_modules/@popperjs/core/lib/dom-utils/getNodeName.js","webpack://pydata_sphinx_theme/./node_modules/@popperjs/core/lib/dom-utils/getWindow.js","webpack://pydata_sphinx_theme/./node_modules/@popperjs/core/lib/dom-utils/instanceOf.js","webpack://pydata_sphinx_theme/./node_modules/@popperjs/core/lib/modifiers/applyStyles.js","webpack://pydata_sphinx_theme/./node_modules/@popperjs/core/lib/utils/getBasePlacement.js","webpack://pydata_sphinx_theme/./node_modules/@popperjs/core/lib/utils/math.js","webpack://pydata_sphinx_theme/./node_modules/@popperjs/core/lib/utils/userAgent.js","webpack://pydata_sphinx_theme/./node_modules/@popperjs/core/lib/dom-utils/isLayoutViewport.js","webpack://pydata_sphinx_theme/./node_modules/@popperjs/core/lib/dom-utils/getBoundingClientRect.js","webpack://pydata_sphinx_theme/./node_modules/@popperjs/core/lib/dom-utils/getLayoutRect.js","webpack://pydata_sphinx_theme/./node_modules/@popperjs/core/lib/dom-utils/contains.js","webpack://pydata_sphinx_theme/./node_modules/@popperjs/core/lib/dom-utils/getComputedStyle.js","webpack://pydata_sphinx_theme/./node_modules/@popperjs/core/lib/dom-utils/isTableElement.js","webpack://pydata_sphinx_theme/./node_modules/@popperjs/core/lib/dom-utils/getDocumentElement.js","webpack://pydata_sphinx_theme/./node_modules/@popperjs/core/lib/dom-utils/getParentNode.js","webpack://pydata_sphinx_theme/./node_modules/@popperjs/core/lib/dom-utils/getOffsetParent.js","webpack://pydata_sphinx_theme/./node_modules/@popperjs/core/lib/utils/getMainAxisFromPlacement.js","webpack://pydata_sphinx_theme/./node_modules/@popperjs/core/lib/utils/within.js","webpack://pydata_sphinx_theme/./node_modules/@popperjs/core/lib/utils/mergePaddingObject.js","webpack://pydata_sphinx_theme/./node_modules/@popperjs/core/lib/utils/getFreshSideObject.js","webpack://pydata_sphinx_theme/./node_modules/@popperjs/core/lib/utils/expandToHashMap.js","webpack://pydata_sphinx_theme/./node_modules/@popperjs/core/lib/modifiers/arrow.js","webpack://pydata_sphinx_theme/./node_modules/@popperjs/core/lib/utils/getVariation.js","webpack://pydata_sphinx_theme/./node_modules/@popperjs/core/lib/modifiers/computeStyles.js","webpack://pydata_sphinx_theme/./node_modules/@popperjs/core/lib/modifiers/eventListeners.js","webpack://pydata_sphinx_theme/./node_modules/@popperjs/core/lib/utils/getOppositePlacement.js","webpack://pydata_sphinx_theme/./node_modules/@popperjs/core/lib/utils/getOppositeVariationPlacement.js","webpack://pydata_sphinx_theme/./node_modules/@popperjs/core/lib/dom-utils/getWindowScroll.js","webpack://pydata_sphinx_theme/./node_modules/@popperjs/core/lib/dom-utils/getWindowScrollBarX.js","webpack://pydata_sphinx_theme/./node_modules/@popperjs/core/lib/dom-utils/isScrollParent.js","webpack://pydata_sphinx_theme/./node_modules/@popperjs/core/lib/dom-utils/getScrollParent.js","webpack://pydata_sphinx_theme/./node_modules/@popperjs/core/lib/dom-utils/listScrollParents.js","webpack://pydata_sphinx_theme/./node_modules/@popperjs/core/lib/utils/rectToClientRect.js","webpack://pydata_sphinx_theme/./node_modules/@popperjs/core/lib/dom-utils/getClippingRect.js","webpack://pydata_sphinx_theme/./node_modules/@popperjs/core/lib/dom-utils/getViewportRect.js","webpack://pydata_sphinx_theme/./node_modules/@popperjs/core/lib/dom-utils/getDocumentRect.js","webpack://pydata_sphinx_theme/./node_modules/@popperjs/core/lib/utils/computeOffsets.js","webpack://pydata_sphinx_theme/./node_modules/@popperjs/core/lib/utils/detectOverflow.js","webpack://pydata_sphinx_theme/./node_modules/@popperjs/core/lib/modifiers/flip.js","webpack://pydata_sphinx_theme/./node_modules/@popperjs/core/lib/utils/computeAutoPlacement.js","webpack://pydata_sphinx_theme/./node_modules/@popperjs/core/lib/modifiers/hide.js","webpack://pydata_sphinx_theme/./node_modules/@popperjs/core/lib/modifiers/offset.js","webpack://pydata_sphinx_theme/./node_modules/@popperjs/core/lib/modifiers/popperOffsets.js","webpack://pydata_sphinx_theme/./node_modules/@popperjs/core/lib/modifiers/preventOverflow.js","webpack://pydata_sphinx_theme/./node_modules/@popperjs/core/lib/utils/getAltAxis.js","webpack://pydata_sphinx_theme/./node_modules/@popperjs/core/lib/dom-utils/getCompositeRect.js","webpack://pydata_sphinx_theme/./node_modules/@popperjs/core/lib/dom-utils/getNodeScroll.js","webpack://pydata_sphinx_theme/./node_modules/@popperjs/core/lib/dom-utils/getHTMLElementScroll.js","webpack://pydata_sphinx_theme/./node_modules/@popperjs/core/lib/utils/orderModifiers.js","webpack://pydata_sphinx_theme/./node_modules/@popperjs/core/lib/createPopper.js","webpack://pydata_sphinx_theme/./node_modules/@popperjs/core/lib/utils/debounce.js","webpack://pydata_sphinx_theme/./node_modules/@popperjs/core/lib/utils/mergeByName.js","webpack://pydata_sphinx_theme/./node_modules/@popperjs/core/lib/popper.js","webpack://pydata_sphinx_theme/./node_modules/@popperjs/core/lib/popper-lite.js","webpack://pydata_sphinx_theme/./node_modules/bootstrap/dist/js/bootstrap.esm.js","webpack://pydata_sphinx_theme/./src/pydata_sphinx_theme/assets/scripts/mixin.js","webpack://pydata_sphinx_theme/./src/pydata_sphinx_theme/assets/scripts/bootstrap.js"],"sourcesContent":["// The require scope\nvar __webpack_require__ = {};\n\n","// define getter functions for harmony exports\n__webpack_require__.d = (exports, definition) => {\n\tfor(var key in definition) {\n\t\tif(__webpack_require__.o(definition, key) && !__webpack_require__.o(exports, key)) {\n\t\t\tObject.defineProperty(exports, key, { enumerable: true, get: definition[key] });\n\t\t}\n\t}\n};","__webpack_require__.o = (obj, prop) => (Object.prototype.hasOwnProperty.call(obj, prop))","// define __esModule on exports\n__webpack_require__.r = (exports) => {\n\tif(typeof Symbol !== 'undefined' && Symbol.toStringTag) {\n\t\tObject.defineProperty(exports, Symbol.toStringTag, { value: 'Module' });\n\t}\n\tObject.defineProperty(exports, '__esModule', { value: true });\n};","export var top = 'top';\nexport var bottom = 'bottom';\nexport var right = 'right';\nexport var left = 'left';\nexport var auto = 'auto';\nexport var basePlacements = [top, bottom, right, left];\nexport var start = 'start';\nexport var end = 'end';\nexport var clippingParents = 'clippingParents';\nexport var viewport = 'viewport';\nexport var popper = 'popper';\nexport var reference = 'reference';\nexport var variationPlacements = /*#__PURE__*/basePlacements.reduce(function (acc, placement) {\n return acc.concat([placement + \"-\" + start, placement + \"-\" + end]);\n}, []);\nexport var placements = /*#__PURE__*/[].concat(basePlacements, [auto]).reduce(function (acc, placement) {\n return acc.concat([placement, placement + \"-\" + start, placement + \"-\" + end]);\n}, []); // modifiers that need to read the DOM\n\nexport var beforeRead = 'beforeRead';\nexport var read = 'read';\nexport var afterRead = 'afterRead'; // pure-logic modifiers\n\nexport var beforeMain = 'beforeMain';\nexport var main = 'main';\nexport var afterMain = 'afterMain'; // modifier with the purpose to write to the DOM (or write into a framework state)\n\nexport var beforeWrite = 'beforeWrite';\nexport var write = 'write';\nexport var afterWrite = 'afterWrite';\nexport var modifierPhases = [beforeRead, read, afterRead, beforeMain, main, afterMain, beforeWrite, write, afterWrite];","export default function getNodeName(element) {\n return element ? (element.nodeName || '').toLowerCase() : null;\n}","export default function getWindow(node) {\n if (node == null) {\n return window;\n }\n\n if (node.toString() !== '[object Window]') {\n var ownerDocument = node.ownerDocument;\n return ownerDocument ? ownerDocument.defaultView || window : window;\n }\n\n return node;\n}","import getWindow from \"./getWindow.js\";\n\nfunction isElement(node) {\n var OwnElement = getWindow(node).Element;\n return node instanceof OwnElement || node instanceof Element;\n}\n\nfunction isHTMLElement(node) {\n var OwnElement = getWindow(node).HTMLElement;\n return node instanceof OwnElement || node instanceof HTMLElement;\n}\n\nfunction isShadowRoot(node) {\n // IE 11 has no ShadowRoot\n if (typeof ShadowRoot === 'undefined') {\n return false;\n }\n\n var OwnElement = getWindow(node).ShadowRoot;\n return node instanceof OwnElement || node instanceof ShadowRoot;\n}\n\nexport { isElement, isHTMLElement, isShadowRoot };","import getNodeName from \"../dom-utils/getNodeName.js\";\nimport { isHTMLElement } from \"../dom-utils/instanceOf.js\"; // This modifier takes the styles prepared by the `computeStyles` modifier\n// and applies them to the HTMLElements such as popper and arrow\n\nfunction applyStyles(_ref) {\n var state = _ref.state;\n Object.keys(state.elements).forEach(function (name) {\n var style = state.styles[name] || {};\n var attributes = state.attributes[name] || {};\n var element = state.elements[name]; // arrow is optional + virtual elements\n\n if (!isHTMLElement(element) || !getNodeName(element)) {\n return;\n } // Flow doesn't support to extend this property, but it's the most\n // effective way to apply styles to an HTMLElement\n // $FlowFixMe[cannot-write]\n\n\n Object.assign(element.style, style);\n Object.keys(attributes).forEach(function (name) {\n var value = attributes[name];\n\n if (value === false) {\n element.removeAttribute(name);\n } else {\n element.setAttribute(name, value === true ? '' : value);\n }\n });\n });\n}\n\nfunction effect(_ref2) {\n var state = _ref2.state;\n var initialStyles = {\n popper: {\n position: state.options.strategy,\n left: '0',\n top: '0',\n margin: '0'\n },\n arrow: {\n position: 'absolute'\n },\n reference: {}\n };\n Object.assign(state.elements.popper.style, initialStyles.popper);\n state.styles = initialStyles;\n\n if (state.elements.arrow) {\n Object.assign(state.elements.arrow.style, initialStyles.arrow);\n }\n\n return function () {\n Object.keys(state.elements).forEach(function (name) {\n var element = state.elements[name];\n var attributes = state.attributes[name] || {};\n var styleProperties = Object.keys(state.styles.hasOwnProperty(name) ? state.styles[name] : initialStyles[name]); // Set all values to an empty string to unset them\n\n var style = styleProperties.reduce(function (style, property) {\n style[property] = '';\n return style;\n }, {}); // arrow is optional + virtual elements\n\n if (!isHTMLElement(element) || !getNodeName(element)) {\n return;\n }\n\n Object.assign(element.style, style);\n Object.keys(attributes).forEach(function (attribute) {\n element.removeAttribute(attribute);\n });\n });\n };\n} // eslint-disable-next-line import/no-unused-modules\n\n\nexport default {\n name: 'applyStyles',\n enabled: true,\n phase: 'write',\n fn: applyStyles,\n effect: effect,\n requires: ['computeStyles']\n};","import { auto } from \"../enums.js\";\nexport default function getBasePlacement(placement) {\n return placement.split('-')[0];\n}","export var max = Math.max;\nexport var min = Math.min;\nexport var round = Math.round;","export default function getUAString() {\n var uaData = navigator.userAgentData;\n\n if (uaData != null && uaData.brands && Array.isArray(uaData.brands)) {\n return uaData.brands.map(function (item) {\n return item.brand + \"/\" + item.version;\n }).join(' ');\n }\n\n return navigator.userAgent;\n}","import getUAString from \"../utils/userAgent.js\";\nexport default function isLayoutViewport() {\n return !/^((?!chrome|android).)*safari/i.test(getUAString());\n}","import { isElement, isHTMLElement } from \"./instanceOf.js\";\nimport { round } from \"../utils/math.js\";\nimport getWindow from \"./getWindow.js\";\nimport isLayoutViewport from \"./isLayoutViewport.js\";\nexport default function getBoundingClientRect(element, includeScale, isFixedStrategy) {\n if (includeScale === void 0) {\n includeScale = false;\n }\n\n if (isFixedStrategy === void 0) {\n isFixedStrategy = false;\n }\n\n var clientRect = element.getBoundingClientRect();\n var scaleX = 1;\n var scaleY = 1;\n\n if (includeScale && isHTMLElement(element)) {\n scaleX = element.offsetWidth > 0 ? round(clientRect.width) / element.offsetWidth || 1 : 1;\n scaleY = element.offsetHeight > 0 ? round(clientRect.height) / element.offsetHeight || 1 : 1;\n }\n\n var _ref = isElement(element) ? getWindow(element) : window,\n visualViewport = _ref.visualViewport;\n\n var addVisualOffsets = !isLayoutViewport() && isFixedStrategy;\n var x = (clientRect.left + (addVisualOffsets && visualViewport ? visualViewport.offsetLeft : 0)) / scaleX;\n var y = (clientRect.top + (addVisualOffsets && visualViewport ? visualViewport.offsetTop : 0)) / scaleY;\n var width = clientRect.width / scaleX;\n var height = clientRect.height / scaleY;\n return {\n width: width,\n height: height,\n top: y,\n right: x + width,\n bottom: y + height,\n left: x,\n x: x,\n y: y\n };\n}","import getBoundingClientRect from \"./getBoundingClientRect.js\"; // Returns the layout rect of an element relative to its offsetParent. Layout\n// means it doesn't take into account transforms.\n\nexport default function getLayoutRect(element) {\n var clientRect = getBoundingClientRect(element); // Use the clientRect sizes if it's not been transformed.\n // Fixes https://github.com/popperjs/popper-core/issues/1223\n\n var width = element.offsetWidth;\n var height = element.offsetHeight;\n\n if (Math.abs(clientRect.width - width) <= 1) {\n width = clientRect.width;\n }\n\n if (Math.abs(clientRect.height - height) <= 1) {\n height = clientRect.height;\n }\n\n return {\n x: element.offsetLeft,\n y: element.offsetTop,\n width: width,\n height: height\n };\n}","import { isShadowRoot } from \"./instanceOf.js\";\nexport default function contains(parent, child) {\n var rootNode = child.getRootNode && child.getRootNode(); // First, attempt with faster native method\n\n if (parent.contains(child)) {\n return true;\n } // then fallback to custom implementation with Shadow DOM support\n else if (rootNode && isShadowRoot(rootNode)) {\n var next = child;\n\n do {\n if (next && parent.isSameNode(next)) {\n return true;\n } // $FlowFixMe[prop-missing]: need a better way to handle this...\n\n\n next = next.parentNode || next.host;\n } while (next);\n } // Give up, the result is false\n\n\n return false;\n}","import getWindow from \"./getWindow.js\";\nexport default function getComputedStyle(element) {\n return getWindow(element).getComputedStyle(element);\n}","import getNodeName from \"./getNodeName.js\";\nexport default function isTableElement(element) {\n return ['table', 'td', 'th'].indexOf(getNodeName(element)) >= 0;\n}","import { isElement } from \"./instanceOf.js\";\nexport default function getDocumentElement(element) {\n // $FlowFixMe[incompatible-return]: assume body is always available\n return ((isElement(element) ? element.ownerDocument : // $FlowFixMe[prop-missing]\n element.document) || window.document).documentElement;\n}","import getNodeName from \"./getNodeName.js\";\nimport getDocumentElement from \"./getDocumentElement.js\";\nimport { isShadowRoot } from \"./instanceOf.js\";\nexport default function getParentNode(element) {\n if (getNodeName(element) === 'html') {\n return element;\n }\n\n return (// this is a quicker (but less type safe) way to save quite some bytes from the bundle\n // $FlowFixMe[incompatible-return]\n // $FlowFixMe[prop-missing]\n element.assignedSlot || // step into the shadow DOM of the parent of a slotted node\n element.parentNode || ( // DOM Element detected\n isShadowRoot(element) ? element.host : null) || // ShadowRoot detected\n // $FlowFixMe[incompatible-call]: HTMLElement is a Node\n getDocumentElement(element) // fallback\n\n );\n}","import getWindow from \"./getWindow.js\";\nimport getNodeName from \"./getNodeName.js\";\nimport getComputedStyle from \"./getComputedStyle.js\";\nimport { isHTMLElement, isShadowRoot } from \"./instanceOf.js\";\nimport isTableElement from \"./isTableElement.js\";\nimport getParentNode from \"./getParentNode.js\";\nimport getUAString from \"../utils/userAgent.js\";\n\nfunction getTrueOffsetParent(element) {\n if (!isHTMLElement(element) || // https://github.com/popperjs/popper-core/issues/837\n getComputedStyle(element).position === 'fixed') {\n return null;\n }\n\n return element.offsetParent;\n} // `.offsetParent` reports `null` for fixed elements, while absolute elements\n// return the containing block\n\n\nfunction getContainingBlock(element) {\n var isFirefox = /firefox/i.test(getUAString());\n var isIE = /Trident/i.test(getUAString());\n\n if (isIE && isHTMLElement(element)) {\n // In IE 9, 10 and 11 fixed elements containing block is always established by the viewport\n var elementCss = getComputedStyle(element);\n\n if (elementCss.position === 'fixed') {\n return null;\n }\n }\n\n var currentNode = getParentNode(element);\n\n if (isShadowRoot(currentNode)) {\n currentNode = currentNode.host;\n }\n\n while (isHTMLElement(currentNode) && ['html', 'body'].indexOf(getNodeName(currentNode)) < 0) {\n var css = getComputedStyle(currentNode); // This is non-exhaustive but covers the most common CSS properties that\n // create a containing block.\n // https://developer.mozilla.org/en-US/docs/Web/CSS/Containing_block#identifying_the_containing_block\n\n if (css.transform !== 'none' || css.perspective !== 'none' || css.contain === 'paint' || ['transform', 'perspective'].indexOf(css.willChange) !== -1 || isFirefox && css.willChange === 'filter' || isFirefox && css.filter && css.filter !== 'none') {\n return currentNode;\n } else {\n currentNode = currentNode.parentNode;\n }\n }\n\n return null;\n} // Gets the closest ancestor positioned element. Handles some edge cases,\n// such as table ancestors and cross browser bugs.\n\n\nexport default function getOffsetParent(element) {\n var window = getWindow(element);\n var offsetParent = getTrueOffsetParent(element);\n\n while (offsetParent && isTableElement(offsetParent) && getComputedStyle(offsetParent).position === 'static') {\n offsetParent = getTrueOffsetParent(offsetParent);\n }\n\n if (offsetParent && (getNodeName(offsetParent) === 'html' || getNodeName(offsetParent) === 'body' && getComputedStyle(offsetParent).position === 'static')) {\n return window;\n }\n\n return offsetParent || getContainingBlock(element) || window;\n}","export default function getMainAxisFromPlacement(placement) {\n return ['top', 'bottom'].indexOf(placement) >= 0 ? 'x' : 'y';\n}","import { max as mathMax, min as mathMin } from \"./math.js\";\nexport function within(min, value, max) {\n return mathMax(min, mathMin(value, max));\n}\nexport function withinMaxClamp(min, value, max) {\n var v = within(min, value, max);\n return v > max ? max : v;\n}","import getFreshSideObject from \"./getFreshSideObject.js\";\nexport default function mergePaddingObject(paddingObject) {\n return Object.assign({}, getFreshSideObject(), paddingObject);\n}","export default function getFreshSideObject() {\n return {\n top: 0,\n right: 0,\n bottom: 0,\n left: 0\n };\n}","export default function expandToHashMap(value, keys) {\n return keys.reduce(function (hashMap, key) {\n hashMap[key] = value;\n return hashMap;\n }, {});\n}","import getBasePlacement from \"../utils/getBasePlacement.js\";\nimport getLayoutRect from \"../dom-utils/getLayoutRect.js\";\nimport contains from \"../dom-utils/contains.js\";\nimport getOffsetParent from \"../dom-utils/getOffsetParent.js\";\nimport getMainAxisFromPlacement from \"../utils/getMainAxisFromPlacement.js\";\nimport { within } from \"../utils/within.js\";\nimport mergePaddingObject from \"../utils/mergePaddingObject.js\";\nimport expandToHashMap from \"../utils/expandToHashMap.js\";\nimport { left, right, basePlacements, top, bottom } from \"../enums.js\"; // eslint-disable-next-line import/no-unused-modules\n\nvar toPaddingObject = function toPaddingObject(padding, state) {\n padding = typeof padding === 'function' ? padding(Object.assign({}, state.rects, {\n placement: state.placement\n })) : padding;\n return mergePaddingObject(typeof padding !== 'number' ? padding : expandToHashMap(padding, basePlacements));\n};\n\nfunction arrow(_ref) {\n var _state$modifiersData$;\n\n var state = _ref.state,\n name = _ref.name,\n options = _ref.options;\n var arrowElement = state.elements.arrow;\n var popperOffsets = state.modifiersData.popperOffsets;\n var basePlacement = getBasePlacement(state.placement);\n var axis = getMainAxisFromPlacement(basePlacement);\n var isVertical = [left, right].indexOf(basePlacement) >= 0;\n var len = isVertical ? 'height' : 'width';\n\n if (!arrowElement || !popperOffsets) {\n return;\n }\n\n var paddingObject = toPaddingObject(options.padding, state);\n var arrowRect = getLayoutRect(arrowElement);\n var minProp = axis === 'y' ? top : left;\n var maxProp = axis === 'y' ? bottom : right;\n var endDiff = state.rects.reference[len] + state.rects.reference[axis] - popperOffsets[axis] - state.rects.popper[len];\n var startDiff = popperOffsets[axis] - state.rects.reference[axis];\n var arrowOffsetParent = getOffsetParent(arrowElement);\n var clientSize = arrowOffsetParent ? axis === 'y' ? arrowOffsetParent.clientHeight || 0 : arrowOffsetParent.clientWidth || 0 : 0;\n var centerToReference = endDiff / 2 - startDiff / 2; // Make sure the arrow doesn't overflow the popper if the center point is\n // outside of the popper bounds\n\n var min = paddingObject[minProp];\n var max = clientSize - arrowRect[len] - paddingObject[maxProp];\n var center = clientSize / 2 - arrowRect[len] / 2 + centerToReference;\n var offset = within(min, center, max); // Prevents breaking syntax highlighting...\n\n var axisProp = axis;\n state.modifiersData[name] = (_state$modifiersData$ = {}, _state$modifiersData$[axisProp] = offset, _state$modifiersData$.centerOffset = offset - center, _state$modifiersData$);\n}\n\nfunction effect(_ref2) {\n var state = _ref2.state,\n options = _ref2.options;\n var _options$element = options.element,\n arrowElement = _options$element === void 0 ? '[data-popper-arrow]' : _options$element;\n\n if (arrowElement == null) {\n return;\n } // CSS selector\n\n\n if (typeof arrowElement === 'string') {\n arrowElement = state.elements.popper.querySelector(arrowElement);\n\n if (!arrowElement) {\n return;\n }\n }\n\n if (!contains(state.elements.popper, arrowElement)) {\n return;\n }\n\n state.elements.arrow = arrowElement;\n} // eslint-disable-next-line import/no-unused-modules\n\n\nexport default {\n name: 'arrow',\n enabled: true,\n phase: 'main',\n fn: arrow,\n effect: effect,\n requires: ['popperOffsets'],\n requiresIfExists: ['preventOverflow']\n};","export default function getVariation(placement) {\n return placement.split('-')[1];\n}","import { top, left, right, bottom, end } from \"../enums.js\";\nimport getOffsetParent from \"../dom-utils/getOffsetParent.js\";\nimport getWindow from \"../dom-utils/getWindow.js\";\nimport getDocumentElement from \"../dom-utils/getDocumentElement.js\";\nimport getComputedStyle from \"../dom-utils/getComputedStyle.js\";\nimport getBasePlacement from \"../utils/getBasePlacement.js\";\nimport getVariation from \"../utils/getVariation.js\";\nimport { round } from \"../utils/math.js\"; // eslint-disable-next-line import/no-unused-modules\n\nvar unsetSides = {\n top: 'auto',\n right: 'auto',\n bottom: 'auto',\n left: 'auto'\n}; // Round the offsets to the nearest suitable subpixel based on the DPR.\n// Zooming can change the DPR, but it seems to report a value that will\n// cleanly divide the values into the appropriate subpixels.\n\nfunction roundOffsetsByDPR(_ref, win) {\n var x = _ref.x,\n y = _ref.y;\n var dpr = win.devicePixelRatio || 1;\n return {\n x: round(x * dpr) / dpr || 0,\n y: round(y * dpr) / dpr || 0\n };\n}\n\nexport function mapToStyles(_ref2) {\n var _Object$assign2;\n\n var popper = _ref2.popper,\n popperRect = _ref2.popperRect,\n placement = _ref2.placement,\n variation = _ref2.variation,\n offsets = _ref2.offsets,\n position = _ref2.position,\n gpuAcceleration = _ref2.gpuAcceleration,\n adaptive = _ref2.adaptive,\n roundOffsets = _ref2.roundOffsets,\n isFixed = _ref2.isFixed;\n var _offsets$x = offsets.x,\n x = _offsets$x === void 0 ? 0 : _offsets$x,\n _offsets$y = offsets.y,\n y = _offsets$y === void 0 ? 0 : _offsets$y;\n\n var _ref3 = typeof roundOffsets === 'function' ? roundOffsets({\n x: x,\n y: y\n }) : {\n x: x,\n y: y\n };\n\n x = _ref3.x;\n y = _ref3.y;\n var hasX = offsets.hasOwnProperty('x');\n var hasY = offsets.hasOwnProperty('y');\n var sideX = left;\n var sideY = top;\n var win = window;\n\n if (adaptive) {\n var offsetParent = getOffsetParent(popper);\n var heightProp = 'clientHeight';\n var widthProp = 'clientWidth';\n\n if (offsetParent === getWindow(popper)) {\n offsetParent = getDocumentElement(popper);\n\n if (getComputedStyle(offsetParent).position !== 'static' && position === 'absolute') {\n heightProp = 'scrollHeight';\n widthProp = 'scrollWidth';\n }\n } // $FlowFixMe[incompatible-cast]: force type refinement, we compare offsetParent with window above, but Flow doesn't detect it\n\n\n offsetParent = offsetParent;\n\n if (placement === top || (placement === left || placement === right) && variation === end) {\n sideY = bottom;\n var offsetY = isFixed && offsetParent === win && win.visualViewport ? win.visualViewport.height : // $FlowFixMe[prop-missing]\n offsetParent[heightProp];\n y -= offsetY - popperRect.height;\n y *= gpuAcceleration ? 1 : -1;\n }\n\n if (placement === left || (placement === top || placement === bottom) && variation === end) {\n sideX = right;\n var offsetX = isFixed && offsetParent === win && win.visualViewport ? win.visualViewport.width : // $FlowFixMe[prop-missing]\n offsetParent[widthProp];\n x -= offsetX - popperRect.width;\n x *= gpuAcceleration ? 1 : -1;\n }\n }\n\n var commonStyles = Object.assign({\n position: position\n }, adaptive && unsetSides);\n\n var _ref4 = roundOffsets === true ? roundOffsetsByDPR({\n x: x,\n y: y\n }, getWindow(popper)) : {\n x: x,\n y: y\n };\n\n x = _ref4.x;\n y = _ref4.y;\n\n if (gpuAcceleration) {\n var _Object$assign;\n\n return Object.assign({}, commonStyles, (_Object$assign = {}, _Object$assign[sideY] = hasY ? '0' : '', _Object$assign[sideX] = hasX ? '0' : '', _Object$assign.transform = (win.devicePixelRatio || 1) <= 1 ? \"translate(\" + x + \"px, \" + y + \"px)\" : \"translate3d(\" + x + \"px, \" + y + \"px, 0)\", _Object$assign));\n }\n\n return Object.assign({}, commonStyles, (_Object$assign2 = {}, _Object$assign2[sideY] = hasY ? y + \"px\" : '', _Object$assign2[sideX] = hasX ? x + \"px\" : '', _Object$assign2.transform = '', _Object$assign2));\n}\n\nfunction computeStyles(_ref5) {\n var state = _ref5.state,\n options = _ref5.options;\n var _options$gpuAccelerat = options.gpuAcceleration,\n gpuAcceleration = _options$gpuAccelerat === void 0 ? true : _options$gpuAccelerat,\n _options$adaptive = options.adaptive,\n adaptive = _options$adaptive === void 0 ? true : _options$adaptive,\n _options$roundOffsets = options.roundOffsets,\n roundOffsets = _options$roundOffsets === void 0 ? true : _options$roundOffsets;\n var commonStyles = {\n placement: getBasePlacement(state.placement),\n variation: getVariation(state.placement),\n popper: state.elements.popper,\n popperRect: state.rects.popper,\n gpuAcceleration: gpuAcceleration,\n isFixed: state.options.strategy === 'fixed'\n };\n\n if (state.modifiersData.popperOffsets != null) {\n state.styles.popper = Object.assign({}, state.styles.popper, mapToStyles(Object.assign({}, commonStyles, {\n offsets: state.modifiersData.popperOffsets,\n position: state.options.strategy,\n adaptive: adaptive,\n roundOffsets: roundOffsets\n })));\n }\n\n if (state.modifiersData.arrow != null) {\n state.styles.arrow = Object.assign({}, state.styles.arrow, mapToStyles(Object.assign({}, commonStyles, {\n offsets: state.modifiersData.arrow,\n position: 'absolute',\n adaptive: false,\n roundOffsets: roundOffsets\n })));\n }\n\n state.attributes.popper = Object.assign({}, state.attributes.popper, {\n 'data-popper-placement': state.placement\n });\n} // eslint-disable-next-line import/no-unused-modules\n\n\nexport default {\n name: 'computeStyles',\n enabled: true,\n phase: 'beforeWrite',\n fn: computeStyles,\n data: {}\n};","import getWindow from \"../dom-utils/getWindow.js\"; // eslint-disable-next-line import/no-unused-modules\n\nvar passive = {\n passive: true\n};\n\nfunction effect(_ref) {\n var state = _ref.state,\n instance = _ref.instance,\n options = _ref.options;\n var _options$scroll = options.scroll,\n scroll = _options$scroll === void 0 ? true : _options$scroll,\n _options$resize = options.resize,\n resize = _options$resize === void 0 ? true : _options$resize;\n var window = getWindow(state.elements.popper);\n var scrollParents = [].concat(state.scrollParents.reference, state.scrollParents.popper);\n\n if (scroll) {\n scrollParents.forEach(function (scrollParent) {\n scrollParent.addEventListener('scroll', instance.update, passive);\n });\n }\n\n if (resize) {\n window.addEventListener('resize', instance.update, passive);\n }\n\n return function () {\n if (scroll) {\n scrollParents.forEach(function (scrollParent) {\n scrollParent.removeEventListener('scroll', instance.update, passive);\n });\n }\n\n if (resize) {\n window.removeEventListener('resize', instance.update, passive);\n }\n };\n} // eslint-disable-next-line import/no-unused-modules\n\n\nexport default {\n name: 'eventListeners',\n enabled: true,\n phase: 'write',\n fn: function fn() {},\n effect: effect,\n data: {}\n};","var hash = {\n left: 'right',\n right: 'left',\n bottom: 'top',\n top: 'bottom'\n};\nexport default function getOppositePlacement(placement) {\n return placement.replace(/left|right|bottom|top/g, function (matched) {\n return hash[matched];\n });\n}","var hash = {\n start: 'end',\n end: 'start'\n};\nexport default function getOppositeVariationPlacement(placement) {\n return placement.replace(/start|end/g, function (matched) {\n return hash[matched];\n });\n}","import getWindow from \"./getWindow.js\";\nexport default function getWindowScroll(node) {\n var win = getWindow(node);\n var scrollLeft = win.pageXOffset;\n var scrollTop = win.pageYOffset;\n return {\n scrollLeft: scrollLeft,\n scrollTop: scrollTop\n };\n}","import getBoundingClientRect from \"./getBoundingClientRect.js\";\nimport getDocumentElement from \"./getDocumentElement.js\";\nimport getWindowScroll from \"./getWindowScroll.js\";\nexport default function getWindowScrollBarX(element) {\n // If has a CSS width greater than the viewport, then this will be\n // incorrect for RTL.\n // Popper 1 is broken in this case and never had a bug report so let's assume\n // it's not an issue. I don't think anyone ever specifies width on \n // anyway.\n // Browsers where the left scrollbar doesn't cause an issue report `0` for\n // this (e.g. Edge 2019, IE11, Safari)\n return getBoundingClientRect(getDocumentElement(element)).left + getWindowScroll(element).scrollLeft;\n}","import getComputedStyle from \"./getComputedStyle.js\";\nexport default function isScrollParent(element) {\n // Firefox wants us to check `-x` and `-y` variations as well\n var _getComputedStyle = getComputedStyle(element),\n overflow = _getComputedStyle.overflow,\n overflowX = _getComputedStyle.overflowX,\n overflowY = _getComputedStyle.overflowY;\n\n return /auto|scroll|overlay|hidden/.test(overflow + overflowY + overflowX);\n}","import getParentNode from \"./getParentNode.js\";\nimport isScrollParent from \"./isScrollParent.js\";\nimport getNodeName from \"./getNodeName.js\";\nimport { isHTMLElement } from \"./instanceOf.js\";\nexport default function getScrollParent(node) {\n if (['html', 'body', '#document'].indexOf(getNodeName(node)) >= 0) {\n // $FlowFixMe[incompatible-return]: assume body is always available\n return node.ownerDocument.body;\n }\n\n if (isHTMLElement(node) && isScrollParent(node)) {\n return node;\n }\n\n return getScrollParent(getParentNode(node));\n}","import getScrollParent from \"./getScrollParent.js\";\nimport getParentNode from \"./getParentNode.js\";\nimport getWindow from \"./getWindow.js\";\nimport isScrollParent from \"./isScrollParent.js\";\n/*\ngiven a DOM element, return the list of all scroll parents, up the list of ancesors\nuntil we get to the top window object. This list is what we attach scroll listeners\nto, because if any of these parent elements scroll, we'll need to re-calculate the\nreference element's position.\n*/\n\nexport default function listScrollParents(element, list) {\n var _element$ownerDocumen;\n\n if (list === void 0) {\n list = [];\n }\n\n var scrollParent = getScrollParent(element);\n var isBody = scrollParent === ((_element$ownerDocumen = element.ownerDocument) == null ? void 0 : _element$ownerDocumen.body);\n var win = getWindow(scrollParent);\n var target = isBody ? [win].concat(win.visualViewport || [], isScrollParent(scrollParent) ? scrollParent : []) : scrollParent;\n var updatedList = list.concat(target);\n return isBody ? updatedList : // $FlowFixMe[incompatible-call]: isBody tells us target will be an HTMLElement here\n updatedList.concat(listScrollParents(getParentNode(target)));\n}","export default function rectToClientRect(rect) {\n return Object.assign({}, rect, {\n left: rect.x,\n top: rect.y,\n right: rect.x + rect.width,\n bottom: rect.y + rect.height\n });\n}","import { viewport } from \"../enums.js\";\nimport getViewportRect from \"./getViewportRect.js\";\nimport getDocumentRect from \"./getDocumentRect.js\";\nimport listScrollParents from \"./listScrollParents.js\";\nimport getOffsetParent from \"./getOffsetParent.js\";\nimport getDocumentElement from \"./getDocumentElement.js\";\nimport getComputedStyle from \"./getComputedStyle.js\";\nimport { isElement, isHTMLElement } from \"./instanceOf.js\";\nimport getBoundingClientRect from \"./getBoundingClientRect.js\";\nimport getParentNode from \"./getParentNode.js\";\nimport contains from \"./contains.js\";\nimport getNodeName from \"./getNodeName.js\";\nimport rectToClientRect from \"../utils/rectToClientRect.js\";\nimport { max, min } from \"../utils/math.js\";\n\nfunction getInnerBoundingClientRect(element, strategy) {\n var rect = getBoundingClientRect(element, false, strategy === 'fixed');\n rect.top = rect.top + element.clientTop;\n rect.left = rect.left + element.clientLeft;\n rect.bottom = rect.top + element.clientHeight;\n rect.right = rect.left + element.clientWidth;\n rect.width = element.clientWidth;\n rect.height = element.clientHeight;\n rect.x = rect.left;\n rect.y = rect.top;\n return rect;\n}\n\nfunction getClientRectFromMixedType(element, clippingParent, strategy) {\n return clippingParent === viewport ? rectToClientRect(getViewportRect(element, strategy)) : isElement(clippingParent) ? getInnerBoundingClientRect(clippingParent, strategy) : rectToClientRect(getDocumentRect(getDocumentElement(element)));\n} // A \"clipping parent\" is an overflowable container with the characteristic of\n// clipping (or hiding) overflowing elements with a position different from\n// `initial`\n\n\nfunction getClippingParents(element) {\n var clippingParents = listScrollParents(getParentNode(element));\n var canEscapeClipping = ['absolute', 'fixed'].indexOf(getComputedStyle(element).position) >= 0;\n var clipperElement = canEscapeClipping && isHTMLElement(element) ? getOffsetParent(element) : element;\n\n if (!isElement(clipperElement)) {\n return [];\n } // $FlowFixMe[incompatible-return]: https://github.com/facebook/flow/issues/1414\n\n\n return clippingParents.filter(function (clippingParent) {\n return isElement(clippingParent) && contains(clippingParent, clipperElement) && getNodeName(clippingParent) !== 'body';\n });\n} // Gets the maximum area that the element is visible in due to any number of\n// clipping parents\n\n\nexport default function getClippingRect(element, boundary, rootBoundary, strategy) {\n var mainClippingParents = boundary === 'clippingParents' ? getClippingParents(element) : [].concat(boundary);\n var clippingParents = [].concat(mainClippingParents, [rootBoundary]);\n var firstClippingParent = clippingParents[0];\n var clippingRect = clippingParents.reduce(function (accRect, clippingParent) {\n var rect = getClientRectFromMixedType(element, clippingParent, strategy);\n accRect.top = max(rect.top, accRect.top);\n accRect.right = min(rect.right, accRect.right);\n accRect.bottom = min(rect.bottom, accRect.bottom);\n accRect.left = max(rect.left, accRect.left);\n return accRect;\n }, getClientRectFromMixedType(element, firstClippingParent, strategy));\n clippingRect.width = clippingRect.right - clippingRect.left;\n clippingRect.height = clippingRect.bottom - clippingRect.top;\n clippingRect.x = clippingRect.left;\n clippingRect.y = clippingRect.top;\n return clippingRect;\n}","import getWindow from \"./getWindow.js\";\nimport getDocumentElement from \"./getDocumentElement.js\";\nimport getWindowScrollBarX from \"./getWindowScrollBarX.js\";\nimport isLayoutViewport from \"./isLayoutViewport.js\";\nexport default function getViewportRect(element, strategy) {\n var win = getWindow(element);\n var html = getDocumentElement(element);\n var visualViewport = win.visualViewport;\n var width = html.clientWidth;\n var height = html.clientHeight;\n var x = 0;\n var y = 0;\n\n if (visualViewport) {\n width = visualViewport.width;\n height = visualViewport.height;\n var layoutViewport = isLayoutViewport();\n\n if (layoutViewport || !layoutViewport && strategy === 'fixed') {\n x = visualViewport.offsetLeft;\n y = visualViewport.offsetTop;\n }\n }\n\n return {\n width: width,\n height: height,\n x: x + getWindowScrollBarX(element),\n y: y\n };\n}","import getDocumentElement from \"./getDocumentElement.js\";\nimport getComputedStyle from \"./getComputedStyle.js\";\nimport getWindowScrollBarX from \"./getWindowScrollBarX.js\";\nimport getWindowScroll from \"./getWindowScroll.js\";\nimport { max } from \"../utils/math.js\"; // Gets the entire size of the scrollable document area, even extending outside\n// of the `` and `` rect bounds if horizontally scrollable\n\nexport default function getDocumentRect(element) {\n var _element$ownerDocumen;\n\n var html = getDocumentElement(element);\n var winScroll = getWindowScroll(element);\n var body = (_element$ownerDocumen = element.ownerDocument) == null ? void 0 : _element$ownerDocumen.body;\n var width = max(html.scrollWidth, html.clientWidth, body ? body.scrollWidth : 0, body ? body.clientWidth : 0);\n var height = max(html.scrollHeight, html.clientHeight, body ? body.scrollHeight : 0, body ? body.clientHeight : 0);\n var x = -winScroll.scrollLeft + getWindowScrollBarX(element);\n var y = -winScroll.scrollTop;\n\n if (getComputedStyle(body || html).direction === 'rtl') {\n x += max(html.clientWidth, body ? body.clientWidth : 0) - width;\n }\n\n return {\n width: width,\n height: height,\n x: x,\n y: y\n };\n}","import getBasePlacement from \"./getBasePlacement.js\";\nimport getVariation from \"./getVariation.js\";\nimport getMainAxisFromPlacement from \"./getMainAxisFromPlacement.js\";\nimport { top, right, bottom, left, start, end } from \"../enums.js\";\nexport default function computeOffsets(_ref) {\n var reference = _ref.reference,\n element = _ref.element,\n placement = _ref.placement;\n var basePlacement = placement ? getBasePlacement(placement) : null;\n var variation = placement ? getVariation(placement) : null;\n var commonX = reference.x + reference.width / 2 - element.width / 2;\n var commonY = reference.y + reference.height / 2 - element.height / 2;\n var offsets;\n\n switch (basePlacement) {\n case top:\n offsets = {\n x: commonX,\n y: reference.y - element.height\n };\n break;\n\n case bottom:\n offsets = {\n x: commonX,\n y: reference.y + reference.height\n };\n break;\n\n case right:\n offsets = {\n x: reference.x + reference.width,\n y: commonY\n };\n break;\n\n case left:\n offsets = {\n x: reference.x - element.width,\n y: commonY\n };\n break;\n\n default:\n offsets = {\n x: reference.x,\n y: reference.y\n };\n }\n\n var mainAxis = basePlacement ? getMainAxisFromPlacement(basePlacement) : null;\n\n if (mainAxis != null) {\n var len = mainAxis === 'y' ? 'height' : 'width';\n\n switch (variation) {\n case start:\n offsets[mainAxis] = offsets[mainAxis] - (reference[len] / 2 - element[len] / 2);\n break;\n\n case end:\n offsets[mainAxis] = offsets[mainAxis] + (reference[len] / 2 - element[len] / 2);\n break;\n\n default:\n }\n }\n\n return offsets;\n}","import getClippingRect from \"../dom-utils/getClippingRect.js\";\nimport getDocumentElement from \"../dom-utils/getDocumentElement.js\";\nimport getBoundingClientRect from \"../dom-utils/getBoundingClientRect.js\";\nimport computeOffsets from \"./computeOffsets.js\";\nimport rectToClientRect from \"./rectToClientRect.js\";\nimport { clippingParents, reference, popper, bottom, top, right, basePlacements, viewport } from \"../enums.js\";\nimport { isElement } from \"../dom-utils/instanceOf.js\";\nimport mergePaddingObject from \"./mergePaddingObject.js\";\nimport expandToHashMap from \"./expandToHashMap.js\"; // eslint-disable-next-line import/no-unused-modules\n\nexport default function detectOverflow(state, options) {\n if (options === void 0) {\n options = {};\n }\n\n var _options = options,\n _options$placement = _options.placement,\n placement = _options$placement === void 0 ? state.placement : _options$placement,\n _options$strategy = _options.strategy,\n strategy = _options$strategy === void 0 ? state.strategy : _options$strategy,\n _options$boundary = _options.boundary,\n boundary = _options$boundary === void 0 ? clippingParents : _options$boundary,\n _options$rootBoundary = _options.rootBoundary,\n rootBoundary = _options$rootBoundary === void 0 ? viewport : _options$rootBoundary,\n _options$elementConte = _options.elementContext,\n elementContext = _options$elementConte === void 0 ? popper : _options$elementConte,\n _options$altBoundary = _options.altBoundary,\n altBoundary = _options$altBoundary === void 0 ? false : _options$altBoundary,\n _options$padding = _options.padding,\n padding = _options$padding === void 0 ? 0 : _options$padding;\n var paddingObject = mergePaddingObject(typeof padding !== 'number' ? padding : expandToHashMap(padding, basePlacements));\n var altContext = elementContext === popper ? reference : popper;\n var popperRect = state.rects.popper;\n var element = state.elements[altBoundary ? altContext : elementContext];\n var clippingClientRect = getClippingRect(isElement(element) ? element : element.contextElement || getDocumentElement(state.elements.popper), boundary, rootBoundary, strategy);\n var referenceClientRect = getBoundingClientRect(state.elements.reference);\n var popperOffsets = computeOffsets({\n reference: referenceClientRect,\n element: popperRect,\n strategy: 'absolute',\n placement: placement\n });\n var popperClientRect = rectToClientRect(Object.assign({}, popperRect, popperOffsets));\n var elementClientRect = elementContext === popper ? popperClientRect : referenceClientRect; // positive = overflowing the clipping rect\n // 0 or negative = within the clipping rect\n\n var overflowOffsets = {\n top: clippingClientRect.top - elementClientRect.top + paddingObject.top,\n bottom: elementClientRect.bottom - clippingClientRect.bottom + paddingObject.bottom,\n left: clippingClientRect.left - elementClientRect.left + paddingObject.left,\n right: elementClientRect.right - clippingClientRect.right + paddingObject.right\n };\n var offsetData = state.modifiersData.offset; // Offsets can be applied only to the popper element\n\n if (elementContext === popper && offsetData) {\n var offset = offsetData[placement];\n Object.keys(overflowOffsets).forEach(function (key) {\n var multiply = [right, bottom].indexOf(key) >= 0 ? 1 : -1;\n var axis = [top, bottom].indexOf(key) >= 0 ? 'y' : 'x';\n overflowOffsets[key] += offset[axis] * multiply;\n });\n }\n\n return overflowOffsets;\n}","import getOppositePlacement from \"../utils/getOppositePlacement.js\";\nimport getBasePlacement from \"../utils/getBasePlacement.js\";\nimport getOppositeVariationPlacement from \"../utils/getOppositeVariationPlacement.js\";\nimport detectOverflow from \"../utils/detectOverflow.js\";\nimport computeAutoPlacement from \"../utils/computeAutoPlacement.js\";\nimport { bottom, top, start, right, left, auto } from \"../enums.js\";\nimport getVariation from \"../utils/getVariation.js\"; // eslint-disable-next-line import/no-unused-modules\n\nfunction getExpandedFallbackPlacements(placement) {\n if (getBasePlacement(placement) === auto) {\n return [];\n }\n\n var oppositePlacement = getOppositePlacement(placement);\n return [getOppositeVariationPlacement(placement), oppositePlacement, getOppositeVariationPlacement(oppositePlacement)];\n}\n\nfunction flip(_ref) {\n var state = _ref.state,\n options = _ref.options,\n name = _ref.name;\n\n if (state.modifiersData[name]._skip) {\n return;\n }\n\n var _options$mainAxis = options.mainAxis,\n checkMainAxis = _options$mainAxis === void 0 ? true : _options$mainAxis,\n _options$altAxis = options.altAxis,\n checkAltAxis = _options$altAxis === void 0 ? true : _options$altAxis,\n specifiedFallbackPlacements = options.fallbackPlacements,\n padding = options.padding,\n boundary = options.boundary,\n rootBoundary = options.rootBoundary,\n altBoundary = options.altBoundary,\n _options$flipVariatio = options.flipVariations,\n flipVariations = _options$flipVariatio === void 0 ? true : _options$flipVariatio,\n allowedAutoPlacements = options.allowedAutoPlacements;\n var preferredPlacement = state.options.placement;\n var basePlacement = getBasePlacement(preferredPlacement);\n var isBasePlacement = basePlacement === preferredPlacement;\n var fallbackPlacements = specifiedFallbackPlacements || (isBasePlacement || !flipVariations ? [getOppositePlacement(preferredPlacement)] : getExpandedFallbackPlacements(preferredPlacement));\n var placements = [preferredPlacement].concat(fallbackPlacements).reduce(function (acc, placement) {\n return acc.concat(getBasePlacement(placement) === auto ? computeAutoPlacement(state, {\n placement: placement,\n boundary: boundary,\n rootBoundary: rootBoundary,\n padding: padding,\n flipVariations: flipVariations,\n allowedAutoPlacements: allowedAutoPlacements\n }) : placement);\n }, []);\n var referenceRect = state.rects.reference;\n var popperRect = state.rects.popper;\n var checksMap = new Map();\n var makeFallbackChecks = true;\n var firstFittingPlacement = placements[0];\n\n for (var i = 0; i < placements.length; i++) {\n var placement = placements[i];\n\n var _basePlacement = getBasePlacement(placement);\n\n var isStartVariation = getVariation(placement) === start;\n var isVertical = [top, bottom].indexOf(_basePlacement) >= 0;\n var len = isVertical ? 'width' : 'height';\n var overflow = detectOverflow(state, {\n placement: placement,\n boundary: boundary,\n rootBoundary: rootBoundary,\n altBoundary: altBoundary,\n padding: padding\n });\n var mainVariationSide = isVertical ? isStartVariation ? right : left : isStartVariation ? bottom : top;\n\n if (referenceRect[len] > popperRect[len]) {\n mainVariationSide = getOppositePlacement(mainVariationSide);\n }\n\n var altVariationSide = getOppositePlacement(mainVariationSide);\n var checks = [];\n\n if (checkMainAxis) {\n checks.push(overflow[_basePlacement] <= 0);\n }\n\n if (checkAltAxis) {\n checks.push(overflow[mainVariationSide] <= 0, overflow[altVariationSide] <= 0);\n }\n\n if (checks.every(function (check) {\n return check;\n })) {\n firstFittingPlacement = placement;\n makeFallbackChecks = false;\n break;\n }\n\n checksMap.set(placement, checks);\n }\n\n if (makeFallbackChecks) {\n // `2` may be desired in some cases – research later\n var numberOfChecks = flipVariations ? 3 : 1;\n\n var _loop = function _loop(_i) {\n var fittingPlacement = placements.find(function (placement) {\n var checks = checksMap.get(placement);\n\n if (checks) {\n return checks.slice(0, _i).every(function (check) {\n return check;\n });\n }\n });\n\n if (fittingPlacement) {\n firstFittingPlacement = fittingPlacement;\n return \"break\";\n }\n };\n\n for (var _i = numberOfChecks; _i > 0; _i--) {\n var _ret = _loop(_i);\n\n if (_ret === \"break\") break;\n }\n }\n\n if (state.placement !== firstFittingPlacement) {\n state.modifiersData[name]._skip = true;\n state.placement = firstFittingPlacement;\n state.reset = true;\n }\n} // eslint-disable-next-line import/no-unused-modules\n\n\nexport default {\n name: 'flip',\n enabled: true,\n phase: 'main',\n fn: flip,\n requiresIfExists: ['offset'],\n data: {\n _skip: false\n }\n};","import getVariation from \"./getVariation.js\";\nimport { variationPlacements, basePlacements, placements as allPlacements } from \"../enums.js\";\nimport detectOverflow from \"./detectOverflow.js\";\nimport getBasePlacement from \"./getBasePlacement.js\";\nexport default function computeAutoPlacement(state, options) {\n if (options === void 0) {\n options = {};\n }\n\n var _options = options,\n placement = _options.placement,\n boundary = _options.boundary,\n rootBoundary = _options.rootBoundary,\n padding = _options.padding,\n flipVariations = _options.flipVariations,\n _options$allowedAutoP = _options.allowedAutoPlacements,\n allowedAutoPlacements = _options$allowedAutoP === void 0 ? allPlacements : _options$allowedAutoP;\n var variation = getVariation(placement);\n var placements = variation ? flipVariations ? variationPlacements : variationPlacements.filter(function (placement) {\n return getVariation(placement) === variation;\n }) : basePlacements;\n var allowedPlacements = placements.filter(function (placement) {\n return allowedAutoPlacements.indexOf(placement) >= 0;\n });\n\n if (allowedPlacements.length === 0) {\n allowedPlacements = placements;\n } // $FlowFixMe[incompatible-type]: Flow seems to have problems with two array unions...\n\n\n var overflows = allowedPlacements.reduce(function (acc, placement) {\n acc[placement] = detectOverflow(state, {\n placement: placement,\n boundary: boundary,\n rootBoundary: rootBoundary,\n padding: padding\n })[getBasePlacement(placement)];\n return acc;\n }, {});\n return Object.keys(overflows).sort(function (a, b) {\n return overflows[a] - overflows[b];\n });\n}","import { top, bottom, left, right } from \"../enums.js\";\nimport detectOverflow from \"../utils/detectOverflow.js\";\n\nfunction getSideOffsets(overflow, rect, preventedOffsets) {\n if (preventedOffsets === void 0) {\n preventedOffsets = {\n x: 0,\n y: 0\n };\n }\n\n return {\n top: overflow.top - rect.height - preventedOffsets.y,\n right: overflow.right - rect.width + preventedOffsets.x,\n bottom: overflow.bottom - rect.height + preventedOffsets.y,\n left: overflow.left - rect.width - preventedOffsets.x\n };\n}\n\nfunction isAnySideFullyClipped(overflow) {\n return [top, right, bottom, left].some(function (side) {\n return overflow[side] >= 0;\n });\n}\n\nfunction hide(_ref) {\n var state = _ref.state,\n name = _ref.name;\n var referenceRect = state.rects.reference;\n var popperRect = state.rects.popper;\n var preventedOffsets = state.modifiersData.preventOverflow;\n var referenceOverflow = detectOverflow(state, {\n elementContext: 'reference'\n });\n var popperAltOverflow = detectOverflow(state, {\n altBoundary: true\n });\n var referenceClippingOffsets = getSideOffsets(referenceOverflow, referenceRect);\n var popperEscapeOffsets = getSideOffsets(popperAltOverflow, popperRect, preventedOffsets);\n var isReferenceHidden = isAnySideFullyClipped(referenceClippingOffsets);\n var hasPopperEscaped = isAnySideFullyClipped(popperEscapeOffsets);\n state.modifiersData[name] = {\n referenceClippingOffsets: referenceClippingOffsets,\n popperEscapeOffsets: popperEscapeOffsets,\n isReferenceHidden: isReferenceHidden,\n hasPopperEscaped: hasPopperEscaped\n };\n state.attributes.popper = Object.assign({}, state.attributes.popper, {\n 'data-popper-reference-hidden': isReferenceHidden,\n 'data-popper-escaped': hasPopperEscaped\n });\n} // eslint-disable-next-line import/no-unused-modules\n\n\nexport default {\n name: 'hide',\n enabled: true,\n phase: 'main',\n requiresIfExists: ['preventOverflow'],\n fn: hide\n};","import getBasePlacement from \"../utils/getBasePlacement.js\";\nimport { top, left, right, placements } from \"../enums.js\"; // eslint-disable-next-line import/no-unused-modules\n\nexport function distanceAndSkiddingToXY(placement, rects, offset) {\n var basePlacement = getBasePlacement(placement);\n var invertDistance = [left, top].indexOf(basePlacement) >= 0 ? -1 : 1;\n\n var _ref = typeof offset === 'function' ? offset(Object.assign({}, rects, {\n placement: placement\n })) : offset,\n skidding = _ref[0],\n distance = _ref[1];\n\n skidding = skidding || 0;\n distance = (distance || 0) * invertDistance;\n return [left, right].indexOf(basePlacement) >= 0 ? {\n x: distance,\n y: skidding\n } : {\n x: skidding,\n y: distance\n };\n}\n\nfunction offset(_ref2) {\n var state = _ref2.state,\n options = _ref2.options,\n name = _ref2.name;\n var _options$offset = options.offset,\n offset = _options$offset === void 0 ? [0, 0] : _options$offset;\n var data = placements.reduce(function (acc, placement) {\n acc[placement] = distanceAndSkiddingToXY(placement, state.rects, offset);\n return acc;\n }, {});\n var _data$state$placement = data[state.placement],\n x = _data$state$placement.x,\n y = _data$state$placement.y;\n\n if (state.modifiersData.popperOffsets != null) {\n state.modifiersData.popperOffsets.x += x;\n state.modifiersData.popperOffsets.y += y;\n }\n\n state.modifiersData[name] = data;\n} // eslint-disable-next-line import/no-unused-modules\n\n\nexport default {\n name: 'offset',\n enabled: true,\n phase: 'main',\n requires: ['popperOffsets'],\n fn: offset\n};","import computeOffsets from \"../utils/computeOffsets.js\";\n\nfunction popperOffsets(_ref) {\n var state = _ref.state,\n name = _ref.name;\n // Offsets are the actual position the popper needs to have to be\n // properly positioned near its reference element\n // This is the most basic placement, and will be adjusted by\n // the modifiers in the next step\n state.modifiersData[name] = computeOffsets({\n reference: state.rects.reference,\n element: state.rects.popper,\n strategy: 'absolute',\n placement: state.placement\n });\n} // eslint-disable-next-line import/no-unused-modules\n\n\nexport default {\n name: 'popperOffsets',\n enabled: true,\n phase: 'read',\n fn: popperOffsets,\n data: {}\n};","import { top, left, right, bottom, start } from \"../enums.js\";\nimport getBasePlacement from \"../utils/getBasePlacement.js\";\nimport getMainAxisFromPlacement from \"../utils/getMainAxisFromPlacement.js\";\nimport getAltAxis from \"../utils/getAltAxis.js\";\nimport { within, withinMaxClamp } from \"../utils/within.js\";\nimport getLayoutRect from \"../dom-utils/getLayoutRect.js\";\nimport getOffsetParent from \"../dom-utils/getOffsetParent.js\";\nimport detectOverflow from \"../utils/detectOverflow.js\";\nimport getVariation from \"../utils/getVariation.js\";\nimport getFreshSideObject from \"../utils/getFreshSideObject.js\";\nimport { min as mathMin, max as mathMax } from \"../utils/math.js\";\n\nfunction preventOverflow(_ref) {\n var state = _ref.state,\n options = _ref.options,\n name = _ref.name;\n var _options$mainAxis = options.mainAxis,\n checkMainAxis = _options$mainAxis === void 0 ? true : _options$mainAxis,\n _options$altAxis = options.altAxis,\n checkAltAxis = _options$altAxis === void 0 ? false : _options$altAxis,\n boundary = options.boundary,\n rootBoundary = options.rootBoundary,\n altBoundary = options.altBoundary,\n padding = options.padding,\n _options$tether = options.tether,\n tether = _options$tether === void 0 ? true : _options$tether,\n _options$tetherOffset = options.tetherOffset,\n tetherOffset = _options$tetherOffset === void 0 ? 0 : _options$tetherOffset;\n var overflow = detectOverflow(state, {\n boundary: boundary,\n rootBoundary: rootBoundary,\n padding: padding,\n altBoundary: altBoundary\n });\n var basePlacement = getBasePlacement(state.placement);\n var variation = getVariation(state.placement);\n var isBasePlacement = !variation;\n var mainAxis = getMainAxisFromPlacement(basePlacement);\n var altAxis = getAltAxis(mainAxis);\n var popperOffsets = state.modifiersData.popperOffsets;\n var referenceRect = state.rects.reference;\n var popperRect = state.rects.popper;\n var tetherOffsetValue = typeof tetherOffset === 'function' ? tetherOffset(Object.assign({}, state.rects, {\n placement: state.placement\n })) : tetherOffset;\n var normalizedTetherOffsetValue = typeof tetherOffsetValue === 'number' ? {\n mainAxis: tetherOffsetValue,\n altAxis: tetherOffsetValue\n } : Object.assign({\n mainAxis: 0,\n altAxis: 0\n }, tetherOffsetValue);\n var offsetModifierState = state.modifiersData.offset ? state.modifiersData.offset[state.placement] : null;\n var data = {\n x: 0,\n y: 0\n };\n\n if (!popperOffsets) {\n return;\n }\n\n if (checkMainAxis) {\n var _offsetModifierState$;\n\n var mainSide = mainAxis === 'y' ? top : left;\n var altSide = mainAxis === 'y' ? bottom : right;\n var len = mainAxis === 'y' ? 'height' : 'width';\n var offset = popperOffsets[mainAxis];\n var min = offset + overflow[mainSide];\n var max = offset - overflow[altSide];\n var additive = tether ? -popperRect[len] / 2 : 0;\n var minLen = variation === start ? referenceRect[len] : popperRect[len];\n var maxLen = variation === start ? -popperRect[len] : -referenceRect[len]; // We need to include the arrow in the calculation so the arrow doesn't go\n // outside the reference bounds\n\n var arrowElement = state.elements.arrow;\n var arrowRect = tether && arrowElement ? getLayoutRect(arrowElement) : {\n width: 0,\n height: 0\n };\n var arrowPaddingObject = state.modifiersData['arrow#persistent'] ? state.modifiersData['arrow#persistent'].padding : getFreshSideObject();\n var arrowPaddingMin = arrowPaddingObject[mainSide];\n var arrowPaddingMax = arrowPaddingObject[altSide]; // If the reference length is smaller than the arrow length, we don't want\n // to include its full size in the calculation. If the reference is small\n // and near the edge of a boundary, the popper can overflow even if the\n // reference is not overflowing as well (e.g. virtual elements with no\n // width or height)\n\n var arrowLen = within(0, referenceRect[len], arrowRect[len]);\n var minOffset = isBasePlacement ? referenceRect[len] / 2 - additive - arrowLen - arrowPaddingMin - normalizedTetherOffsetValue.mainAxis : minLen - arrowLen - arrowPaddingMin - normalizedTetherOffsetValue.mainAxis;\n var maxOffset = isBasePlacement ? -referenceRect[len] / 2 + additive + arrowLen + arrowPaddingMax + normalizedTetherOffsetValue.mainAxis : maxLen + arrowLen + arrowPaddingMax + normalizedTetherOffsetValue.mainAxis;\n var arrowOffsetParent = state.elements.arrow && getOffsetParent(state.elements.arrow);\n var clientOffset = arrowOffsetParent ? mainAxis === 'y' ? arrowOffsetParent.clientTop || 0 : arrowOffsetParent.clientLeft || 0 : 0;\n var offsetModifierValue = (_offsetModifierState$ = offsetModifierState == null ? void 0 : offsetModifierState[mainAxis]) != null ? _offsetModifierState$ : 0;\n var tetherMin = offset + minOffset - offsetModifierValue - clientOffset;\n var tetherMax = offset + maxOffset - offsetModifierValue;\n var preventedOffset = within(tether ? mathMin(min, tetherMin) : min, offset, tether ? mathMax(max, tetherMax) : max);\n popperOffsets[mainAxis] = preventedOffset;\n data[mainAxis] = preventedOffset - offset;\n }\n\n if (checkAltAxis) {\n var _offsetModifierState$2;\n\n var _mainSide = mainAxis === 'x' ? top : left;\n\n var _altSide = mainAxis === 'x' ? bottom : right;\n\n var _offset = popperOffsets[altAxis];\n\n var _len = altAxis === 'y' ? 'height' : 'width';\n\n var _min = _offset + overflow[_mainSide];\n\n var _max = _offset - overflow[_altSide];\n\n var isOriginSide = [top, left].indexOf(basePlacement) !== -1;\n\n var _offsetModifierValue = (_offsetModifierState$2 = offsetModifierState == null ? void 0 : offsetModifierState[altAxis]) != null ? _offsetModifierState$2 : 0;\n\n var _tetherMin = isOriginSide ? _min : _offset - referenceRect[_len] - popperRect[_len] - _offsetModifierValue + normalizedTetherOffsetValue.altAxis;\n\n var _tetherMax = isOriginSide ? _offset + referenceRect[_len] + popperRect[_len] - _offsetModifierValue - normalizedTetherOffsetValue.altAxis : _max;\n\n var _preventedOffset = tether && isOriginSide ? withinMaxClamp(_tetherMin, _offset, _tetherMax) : within(tether ? _tetherMin : _min, _offset, tether ? _tetherMax : _max);\n\n popperOffsets[altAxis] = _preventedOffset;\n data[altAxis] = _preventedOffset - _offset;\n }\n\n state.modifiersData[name] = data;\n} // eslint-disable-next-line import/no-unused-modules\n\n\nexport default {\n name: 'preventOverflow',\n enabled: true,\n phase: 'main',\n fn: preventOverflow,\n requiresIfExists: ['offset']\n};","export default function getAltAxis(axis) {\n return axis === 'x' ? 'y' : 'x';\n}","import getBoundingClientRect from \"./getBoundingClientRect.js\";\nimport getNodeScroll from \"./getNodeScroll.js\";\nimport getNodeName from \"./getNodeName.js\";\nimport { isHTMLElement } from \"./instanceOf.js\";\nimport getWindowScrollBarX from \"./getWindowScrollBarX.js\";\nimport getDocumentElement from \"./getDocumentElement.js\";\nimport isScrollParent from \"./isScrollParent.js\";\nimport { round } from \"../utils/math.js\";\n\nfunction isElementScaled(element) {\n var rect = element.getBoundingClientRect();\n var scaleX = round(rect.width) / element.offsetWidth || 1;\n var scaleY = round(rect.height) / element.offsetHeight || 1;\n return scaleX !== 1 || scaleY !== 1;\n} // Returns the composite rect of an element relative to its offsetParent.\n// Composite means it takes into account transforms as well as layout.\n\n\nexport default function getCompositeRect(elementOrVirtualElement, offsetParent, isFixed) {\n if (isFixed === void 0) {\n isFixed = false;\n }\n\n var isOffsetParentAnElement = isHTMLElement(offsetParent);\n var offsetParentIsScaled = isHTMLElement(offsetParent) && isElementScaled(offsetParent);\n var documentElement = getDocumentElement(offsetParent);\n var rect = getBoundingClientRect(elementOrVirtualElement, offsetParentIsScaled, isFixed);\n var scroll = {\n scrollLeft: 0,\n scrollTop: 0\n };\n var offsets = {\n x: 0,\n y: 0\n };\n\n if (isOffsetParentAnElement || !isOffsetParentAnElement && !isFixed) {\n if (getNodeName(offsetParent) !== 'body' || // https://github.com/popperjs/popper-core/issues/1078\n isScrollParent(documentElement)) {\n scroll = getNodeScroll(offsetParent);\n }\n\n if (isHTMLElement(offsetParent)) {\n offsets = getBoundingClientRect(offsetParent, true);\n offsets.x += offsetParent.clientLeft;\n offsets.y += offsetParent.clientTop;\n } else if (documentElement) {\n offsets.x = getWindowScrollBarX(documentElement);\n }\n }\n\n return {\n x: rect.left + scroll.scrollLeft - offsets.x,\n y: rect.top + scroll.scrollTop - offsets.y,\n width: rect.width,\n height: rect.height\n };\n}","import getWindowScroll from \"./getWindowScroll.js\";\nimport getWindow from \"./getWindow.js\";\nimport { isHTMLElement } from \"./instanceOf.js\";\nimport getHTMLElementScroll from \"./getHTMLElementScroll.js\";\nexport default function getNodeScroll(node) {\n if (node === getWindow(node) || !isHTMLElement(node)) {\n return getWindowScroll(node);\n } else {\n return getHTMLElementScroll(node);\n }\n}","export default function getHTMLElementScroll(element) {\n return {\n scrollLeft: element.scrollLeft,\n scrollTop: element.scrollTop\n };\n}","import { modifierPhases } from \"../enums.js\"; // source: https://stackoverflow.com/questions/49875255\n\nfunction order(modifiers) {\n var map = new Map();\n var visited = new Set();\n var result = [];\n modifiers.forEach(function (modifier) {\n map.set(modifier.name, modifier);\n }); // On visiting object, check for its dependencies and visit them recursively\n\n function sort(modifier) {\n visited.add(modifier.name);\n var requires = [].concat(modifier.requires || [], modifier.requiresIfExists || []);\n requires.forEach(function (dep) {\n if (!visited.has(dep)) {\n var depModifier = map.get(dep);\n\n if (depModifier) {\n sort(depModifier);\n }\n }\n });\n result.push(modifier);\n }\n\n modifiers.forEach(function (modifier) {\n if (!visited.has(modifier.name)) {\n // check for visited object\n sort(modifier);\n }\n });\n return result;\n}\n\nexport default function orderModifiers(modifiers) {\n // order based on dependencies\n var orderedModifiers = order(modifiers); // order based on phase\n\n return modifierPhases.reduce(function (acc, phase) {\n return acc.concat(orderedModifiers.filter(function (modifier) {\n return modifier.phase === phase;\n }));\n }, []);\n}","import getCompositeRect from \"./dom-utils/getCompositeRect.js\";\nimport getLayoutRect from \"./dom-utils/getLayoutRect.js\";\nimport listScrollParents from \"./dom-utils/listScrollParents.js\";\nimport getOffsetParent from \"./dom-utils/getOffsetParent.js\";\nimport orderModifiers from \"./utils/orderModifiers.js\";\nimport debounce from \"./utils/debounce.js\";\nimport mergeByName from \"./utils/mergeByName.js\";\nimport detectOverflow from \"./utils/detectOverflow.js\";\nimport { isElement } from \"./dom-utils/instanceOf.js\";\nvar DEFAULT_OPTIONS = {\n placement: 'bottom',\n modifiers: [],\n strategy: 'absolute'\n};\n\nfunction areValidElements() {\n for (var _len = arguments.length, args = new Array(_len), _key = 0; _key < _len; _key++) {\n args[_key] = arguments[_key];\n }\n\n return !args.some(function (element) {\n return !(element && typeof element.getBoundingClientRect === 'function');\n });\n}\n\nexport function popperGenerator(generatorOptions) {\n if (generatorOptions === void 0) {\n generatorOptions = {};\n }\n\n var _generatorOptions = generatorOptions,\n _generatorOptions$def = _generatorOptions.defaultModifiers,\n defaultModifiers = _generatorOptions$def === void 0 ? [] : _generatorOptions$def,\n _generatorOptions$def2 = _generatorOptions.defaultOptions,\n defaultOptions = _generatorOptions$def2 === void 0 ? DEFAULT_OPTIONS : _generatorOptions$def2;\n return function createPopper(reference, popper, options) {\n if (options === void 0) {\n options = defaultOptions;\n }\n\n var state = {\n placement: 'bottom',\n orderedModifiers: [],\n options: Object.assign({}, DEFAULT_OPTIONS, defaultOptions),\n modifiersData: {},\n elements: {\n reference: reference,\n popper: popper\n },\n attributes: {},\n styles: {}\n };\n var effectCleanupFns = [];\n var isDestroyed = false;\n var instance = {\n state: state,\n setOptions: function setOptions(setOptionsAction) {\n var options = typeof setOptionsAction === 'function' ? setOptionsAction(state.options) : setOptionsAction;\n cleanupModifierEffects();\n state.options = Object.assign({}, defaultOptions, state.options, options);\n state.scrollParents = {\n reference: isElement(reference) ? listScrollParents(reference) : reference.contextElement ? listScrollParents(reference.contextElement) : [],\n popper: listScrollParents(popper)\n }; // Orders the modifiers based on their dependencies and `phase`\n // properties\n\n var orderedModifiers = orderModifiers(mergeByName([].concat(defaultModifiers, state.options.modifiers))); // Strip out disabled modifiers\n\n state.orderedModifiers = orderedModifiers.filter(function (m) {\n return m.enabled;\n });\n runModifierEffects();\n return instance.update();\n },\n // Sync update – it will always be executed, even if not necessary. This\n // is useful for low frequency updates where sync behavior simplifies the\n // logic.\n // For high frequency updates (e.g. `resize` and `scroll` events), always\n // prefer the async Popper#update method\n forceUpdate: function forceUpdate() {\n if (isDestroyed) {\n return;\n }\n\n var _state$elements = state.elements,\n reference = _state$elements.reference,\n popper = _state$elements.popper; // Don't proceed if `reference` or `popper` are not valid elements\n // anymore\n\n if (!areValidElements(reference, popper)) {\n return;\n } // Store the reference and popper rects to be read by modifiers\n\n\n state.rects = {\n reference: getCompositeRect(reference, getOffsetParent(popper), state.options.strategy === 'fixed'),\n popper: getLayoutRect(popper)\n }; // Modifiers have the ability to reset the current update cycle. The\n // most common use case for this is the `flip` modifier changing the\n // placement, which then needs to re-run all the modifiers, because the\n // logic was previously ran for the previous placement and is therefore\n // stale/incorrect\n\n state.reset = false;\n state.placement = state.options.placement; // On each update cycle, the `modifiersData` property for each modifier\n // is filled with the initial data specified by the modifier. This means\n // it doesn't persist and is fresh on each update.\n // To ensure persistent data, use `${name}#persistent`\n\n state.orderedModifiers.forEach(function (modifier) {\n return state.modifiersData[modifier.name] = Object.assign({}, modifier.data);\n });\n\n for (var index = 0; index < state.orderedModifiers.length; index++) {\n if (state.reset === true) {\n state.reset = false;\n index = -1;\n continue;\n }\n\n var _state$orderedModifie = state.orderedModifiers[index],\n fn = _state$orderedModifie.fn,\n _state$orderedModifie2 = _state$orderedModifie.options,\n _options = _state$orderedModifie2 === void 0 ? {} : _state$orderedModifie2,\n name = _state$orderedModifie.name;\n\n if (typeof fn === 'function') {\n state = fn({\n state: state,\n options: _options,\n name: name,\n instance: instance\n }) || state;\n }\n }\n },\n // Async and optimistically optimized update – it will not be executed if\n // not necessary (debounced to run at most once-per-tick)\n update: debounce(function () {\n return new Promise(function (resolve) {\n instance.forceUpdate();\n resolve(state);\n });\n }),\n destroy: function destroy() {\n cleanupModifierEffects();\n isDestroyed = true;\n }\n };\n\n if (!areValidElements(reference, popper)) {\n return instance;\n }\n\n instance.setOptions(options).then(function (state) {\n if (!isDestroyed && options.onFirstUpdate) {\n options.onFirstUpdate(state);\n }\n }); // Modifiers have the ability to execute arbitrary code before the first\n // update cycle runs. They will be executed in the same order as the update\n // cycle. This is useful when a modifier adds some persistent data that\n // other modifiers need to use, but the modifier is run after the dependent\n // one.\n\n function runModifierEffects() {\n state.orderedModifiers.forEach(function (_ref) {\n var name = _ref.name,\n _ref$options = _ref.options,\n options = _ref$options === void 0 ? {} : _ref$options,\n effect = _ref.effect;\n\n if (typeof effect === 'function') {\n var cleanupFn = effect({\n state: state,\n name: name,\n instance: instance,\n options: options\n });\n\n var noopFn = function noopFn() {};\n\n effectCleanupFns.push(cleanupFn || noopFn);\n }\n });\n }\n\n function cleanupModifierEffects() {\n effectCleanupFns.forEach(function (fn) {\n return fn();\n });\n effectCleanupFns = [];\n }\n\n return instance;\n };\n}\nexport var createPopper = /*#__PURE__*/popperGenerator(); // eslint-disable-next-line import/no-unused-modules\n\nexport { detectOverflow };","export default function debounce(fn) {\n var pending;\n return function () {\n if (!pending) {\n pending = new Promise(function (resolve) {\n Promise.resolve().then(function () {\n pending = undefined;\n resolve(fn());\n });\n });\n }\n\n return pending;\n };\n}","export default function mergeByName(modifiers) {\n var merged = modifiers.reduce(function (merged, current) {\n var existing = merged[current.name];\n merged[current.name] = existing ? Object.assign({}, existing, current, {\n options: Object.assign({}, existing.options, current.options),\n data: Object.assign({}, existing.data, current.data)\n }) : current;\n return merged;\n }, {}); // IE11 does not support Object.values\n\n return Object.keys(merged).map(function (key) {\n return merged[key];\n });\n}","import { popperGenerator, detectOverflow } from \"./createPopper.js\";\nimport eventListeners from \"./modifiers/eventListeners.js\";\nimport popperOffsets from \"./modifiers/popperOffsets.js\";\nimport computeStyles from \"./modifiers/computeStyles.js\";\nimport applyStyles from \"./modifiers/applyStyles.js\";\nimport offset from \"./modifiers/offset.js\";\nimport flip from \"./modifiers/flip.js\";\nimport preventOverflow from \"./modifiers/preventOverflow.js\";\nimport arrow from \"./modifiers/arrow.js\";\nimport hide from \"./modifiers/hide.js\";\nvar defaultModifiers = [eventListeners, popperOffsets, computeStyles, applyStyles, offset, flip, preventOverflow, arrow, hide];\nvar createPopper = /*#__PURE__*/popperGenerator({\n defaultModifiers: defaultModifiers\n}); // eslint-disable-next-line import/no-unused-modules\n\nexport { createPopper, popperGenerator, defaultModifiers, detectOverflow }; // eslint-disable-next-line import/no-unused-modules\n\nexport { createPopper as createPopperLite } from \"./popper-lite.js\"; // eslint-disable-next-line import/no-unused-modules\n\nexport * from \"./modifiers/index.js\";","import { popperGenerator, detectOverflow } from \"./createPopper.js\";\nimport eventListeners from \"./modifiers/eventListeners.js\";\nimport popperOffsets from \"./modifiers/popperOffsets.js\";\nimport computeStyles from \"./modifiers/computeStyles.js\";\nimport applyStyles from \"./modifiers/applyStyles.js\";\nvar defaultModifiers = [eventListeners, popperOffsets, computeStyles, applyStyles];\nvar createPopper = /*#__PURE__*/popperGenerator({\n defaultModifiers: defaultModifiers\n}); // eslint-disable-next-line import/no-unused-modules\n\nexport { createPopper, popperGenerator, defaultModifiers, detectOverflow };","/*!\n * Bootstrap v5.3.3 (https://getbootstrap.com/)\n * Copyright 2011-2024 The Bootstrap Authors (https://github.com/twbs/bootstrap/graphs/contributors)\n * Licensed under MIT (https://github.com/twbs/bootstrap/blob/main/LICENSE)\n */\nimport * as Popper from '@popperjs/core';\n\n/**\n * --------------------------------------------------------------------------\n * Bootstrap dom/data.js\n * Licensed under MIT (https://github.com/twbs/bootstrap/blob/main/LICENSE)\n * --------------------------------------------------------------------------\n */\n\n/**\n * Constants\n */\n\nconst elementMap = new Map();\nconst Data = {\n set(element, key, instance) {\n if (!elementMap.has(element)) {\n elementMap.set(element, new Map());\n }\n const instanceMap = elementMap.get(element);\n\n // make it clear we only want one instance per element\n // can be removed later when multiple key/instances are fine to be used\n if (!instanceMap.has(key) && instanceMap.size !== 0) {\n // eslint-disable-next-line no-console\n console.error(`Bootstrap doesn't allow more than one instance per element. Bound instance: ${Array.from(instanceMap.keys())[0]}.`);\n return;\n }\n instanceMap.set(key, instance);\n },\n get(element, key) {\n if (elementMap.has(element)) {\n return elementMap.get(element).get(key) || null;\n }\n return null;\n },\n remove(element, key) {\n if (!elementMap.has(element)) {\n return;\n }\n const instanceMap = elementMap.get(element);\n instanceMap.delete(key);\n\n // free up element references if there are no instances left for an element\n if (instanceMap.size === 0) {\n elementMap.delete(element);\n }\n }\n};\n\n/**\n * --------------------------------------------------------------------------\n * Bootstrap util/index.js\n * Licensed under MIT (https://github.com/twbs/bootstrap/blob/main/LICENSE)\n * --------------------------------------------------------------------------\n */\n\nconst MAX_UID = 1000000;\nconst MILLISECONDS_MULTIPLIER = 1000;\nconst TRANSITION_END = 'transitionend';\n\n/**\n * Properly escape IDs selectors to handle weird IDs\n * @param {string} selector\n * @returns {string}\n */\nconst parseSelector = selector => {\n if (selector && window.CSS && window.CSS.escape) {\n // document.querySelector needs escaping to handle IDs (html5+) containing for instance /\n selector = selector.replace(/#([^\\s\"#']+)/g, (match, id) => `#${CSS.escape(id)}`);\n }\n return selector;\n};\n\n// Shout-out Angus Croll (https://goo.gl/pxwQGp)\nconst toType = object => {\n if (object === null || object === undefined) {\n return `${object}`;\n }\n return Object.prototype.toString.call(object).match(/\\s([a-z]+)/i)[1].toLowerCase();\n};\n\n/**\n * Public Util API\n */\n\nconst getUID = prefix => {\n do {\n prefix += Math.floor(Math.random() * MAX_UID);\n } while (document.getElementById(prefix));\n return prefix;\n};\nconst getTransitionDurationFromElement = element => {\n if (!element) {\n return 0;\n }\n\n // Get transition-duration of the element\n let {\n transitionDuration,\n transitionDelay\n } = window.getComputedStyle(element);\n const floatTransitionDuration = Number.parseFloat(transitionDuration);\n const floatTransitionDelay = Number.parseFloat(transitionDelay);\n\n // Return 0 if element or transition duration is not found\n if (!floatTransitionDuration && !floatTransitionDelay) {\n return 0;\n }\n\n // If multiple durations are defined, take the first\n transitionDuration = transitionDuration.split(',')[0];\n transitionDelay = transitionDelay.split(',')[0];\n return (Number.parseFloat(transitionDuration) + Number.parseFloat(transitionDelay)) * MILLISECONDS_MULTIPLIER;\n};\nconst triggerTransitionEnd = element => {\n element.dispatchEvent(new Event(TRANSITION_END));\n};\nconst isElement = object => {\n if (!object || typeof object !== 'object') {\n return false;\n }\n if (typeof object.jquery !== 'undefined') {\n object = object[0];\n }\n return typeof object.nodeType !== 'undefined';\n};\nconst getElement = object => {\n // it's a jQuery object or a node element\n if (isElement(object)) {\n return object.jquery ? object[0] : object;\n }\n if (typeof object === 'string' && object.length > 0) {\n return document.querySelector(parseSelector(object));\n }\n return null;\n};\nconst isVisible = element => {\n if (!isElement(element) || element.getClientRects().length === 0) {\n return false;\n }\n const elementIsVisible = getComputedStyle(element).getPropertyValue('visibility') === 'visible';\n // Handle `details` element as its content may falsie appear visible when it is closed\n const closedDetails = element.closest('details:not([open])');\n if (!closedDetails) {\n return elementIsVisible;\n }\n if (closedDetails !== element) {\n const summary = element.closest('summary');\n if (summary && summary.parentNode !== closedDetails) {\n return false;\n }\n if (summary === null) {\n return false;\n }\n }\n return elementIsVisible;\n};\nconst isDisabled = element => {\n if (!element || element.nodeType !== Node.ELEMENT_NODE) {\n return true;\n }\n if (element.classList.contains('disabled')) {\n return true;\n }\n if (typeof element.disabled !== 'undefined') {\n return element.disabled;\n }\n return element.hasAttribute('disabled') && element.getAttribute('disabled') !== 'false';\n};\nconst findShadowRoot = element => {\n if (!document.documentElement.attachShadow) {\n return null;\n }\n\n // Can find the shadow root otherwise it'll return the document\n if (typeof element.getRootNode === 'function') {\n const root = element.getRootNode();\n return root instanceof ShadowRoot ? root : null;\n }\n if (element instanceof ShadowRoot) {\n return element;\n }\n\n // when we don't find a shadow root\n if (!element.parentNode) {\n return null;\n }\n return findShadowRoot(element.parentNode);\n};\nconst noop = () => {};\n\n/**\n * Trick to restart an element's animation\n *\n * @param {HTMLElement} element\n * @return void\n *\n * @see https://www.charistheo.io/blog/2021/02/restart-a-css-animation-with-javascript/#restarting-a-css-animation\n */\nconst reflow = element => {\n element.offsetHeight; // eslint-disable-line no-unused-expressions\n};\nconst getjQuery = () => {\n if (window.jQuery && !document.body.hasAttribute('data-bs-no-jquery')) {\n return window.jQuery;\n }\n return null;\n};\nconst DOMContentLoadedCallbacks = [];\nconst onDOMContentLoaded = callback => {\n if (document.readyState === 'loading') {\n // add listener on the first call when the document is in loading state\n if (!DOMContentLoadedCallbacks.length) {\n document.addEventListener('DOMContentLoaded', () => {\n for (const callback of DOMContentLoadedCallbacks) {\n callback();\n }\n });\n }\n DOMContentLoadedCallbacks.push(callback);\n } else {\n callback();\n }\n};\nconst isRTL = () => document.documentElement.dir === 'rtl';\nconst defineJQueryPlugin = plugin => {\n onDOMContentLoaded(() => {\n const $ = getjQuery();\n /* istanbul ignore if */\n if ($) {\n const name = plugin.NAME;\n const JQUERY_NO_CONFLICT = $.fn[name];\n $.fn[name] = plugin.jQueryInterface;\n $.fn[name].Constructor = plugin;\n $.fn[name].noConflict = () => {\n $.fn[name] = JQUERY_NO_CONFLICT;\n return plugin.jQueryInterface;\n };\n }\n });\n};\nconst execute = (possibleCallback, args = [], defaultValue = possibleCallback) => {\n return typeof possibleCallback === 'function' ? possibleCallback(...args) : defaultValue;\n};\nconst executeAfterTransition = (callback, transitionElement, waitForTransition = true) => {\n if (!waitForTransition) {\n execute(callback);\n return;\n }\n const durationPadding = 5;\n const emulatedDuration = getTransitionDurationFromElement(transitionElement) + durationPadding;\n let called = false;\n const handler = ({\n target\n }) => {\n if (target !== transitionElement) {\n return;\n }\n called = true;\n transitionElement.removeEventListener(TRANSITION_END, handler);\n execute(callback);\n };\n transitionElement.addEventListener(TRANSITION_END, handler);\n setTimeout(() => {\n if (!called) {\n triggerTransitionEnd(transitionElement);\n }\n }, emulatedDuration);\n};\n\n/**\n * Return the previous/next element of a list.\n *\n * @param {array} list The list of elements\n * @param activeElement The active element\n * @param shouldGetNext Choose to get next or previous element\n * @param isCycleAllowed\n * @return {Element|elem} The proper element\n */\nconst getNextActiveElement = (list, activeElement, shouldGetNext, isCycleAllowed) => {\n const listLength = list.length;\n let index = list.indexOf(activeElement);\n\n // if the element does not exist in the list return an element\n // depending on the direction and if cycle is allowed\n if (index === -1) {\n return !shouldGetNext && isCycleAllowed ? list[listLength - 1] : list[0];\n }\n index += shouldGetNext ? 1 : -1;\n if (isCycleAllowed) {\n index = (index + listLength) % listLength;\n }\n return list[Math.max(0, Math.min(index, listLength - 1))];\n};\n\n/**\n * --------------------------------------------------------------------------\n * Bootstrap dom/event-handler.js\n * Licensed under MIT (https://github.com/twbs/bootstrap/blob/main/LICENSE)\n * --------------------------------------------------------------------------\n */\n\n\n/**\n * Constants\n */\n\nconst namespaceRegex = /[^.]*(?=\\..*)\\.|.*/;\nconst stripNameRegex = /\\..*/;\nconst stripUidRegex = /::\\d+$/;\nconst eventRegistry = {}; // Events storage\nlet uidEvent = 1;\nconst customEvents = {\n mouseenter: 'mouseover',\n mouseleave: 'mouseout'\n};\nconst nativeEvents = new Set(['click', 'dblclick', 'mouseup', 'mousedown', 'contextmenu', 'mousewheel', 'DOMMouseScroll', 'mouseover', 'mouseout', 'mousemove', 'selectstart', 'selectend', 'keydown', 'keypress', 'keyup', 'orientationchange', 'touchstart', 'touchmove', 'touchend', 'touchcancel', 'pointerdown', 'pointermove', 'pointerup', 'pointerleave', 'pointercancel', 'gesturestart', 'gesturechange', 'gestureend', 'focus', 'blur', 'change', 'reset', 'select', 'submit', 'focusin', 'focusout', 'load', 'unload', 'beforeunload', 'resize', 'move', 'DOMContentLoaded', 'readystatechange', 'error', 'abort', 'scroll']);\n\n/**\n * Private methods\n */\n\nfunction makeEventUid(element, uid) {\n return uid && `${uid}::${uidEvent++}` || element.uidEvent || uidEvent++;\n}\nfunction getElementEvents(element) {\n const uid = makeEventUid(element);\n element.uidEvent = uid;\n eventRegistry[uid] = eventRegistry[uid] || {};\n return eventRegistry[uid];\n}\nfunction bootstrapHandler(element, fn) {\n return function handler(event) {\n hydrateObj(event, {\n delegateTarget: element\n });\n if (handler.oneOff) {\n EventHandler.off(element, event.type, fn);\n }\n return fn.apply(element, [event]);\n };\n}\nfunction bootstrapDelegationHandler(element, selector, fn) {\n return function handler(event) {\n const domElements = element.querySelectorAll(selector);\n for (let {\n target\n } = event; target && target !== this; target = target.parentNode) {\n for (const domElement of domElements) {\n if (domElement !== target) {\n continue;\n }\n hydrateObj(event, {\n delegateTarget: target\n });\n if (handler.oneOff) {\n EventHandler.off(element, event.type, selector, fn);\n }\n return fn.apply(target, [event]);\n }\n }\n };\n}\nfunction findHandler(events, callable, delegationSelector = null) {\n return Object.values(events).find(event => event.callable === callable && event.delegationSelector === delegationSelector);\n}\nfunction normalizeParameters(originalTypeEvent, handler, delegationFunction) {\n const isDelegated = typeof handler === 'string';\n // TODO: tooltip passes `false` instead of selector, so we need to check\n const callable = isDelegated ? delegationFunction : handler || delegationFunction;\n let typeEvent = getTypeEvent(originalTypeEvent);\n if (!nativeEvents.has(typeEvent)) {\n typeEvent = originalTypeEvent;\n }\n return [isDelegated, callable, typeEvent];\n}\nfunction addHandler(element, originalTypeEvent, handler, delegationFunction, oneOff) {\n if (typeof originalTypeEvent !== 'string' || !element) {\n return;\n }\n let [isDelegated, callable, typeEvent] = normalizeParameters(originalTypeEvent, handler, delegationFunction);\n\n // in case of mouseenter or mouseleave wrap the handler within a function that checks for its DOM position\n // this prevents the handler from being dispatched the same way as mouseover or mouseout does\n if (originalTypeEvent in customEvents) {\n const wrapFunction = fn => {\n return function (event) {\n if (!event.relatedTarget || event.relatedTarget !== event.delegateTarget && !event.delegateTarget.contains(event.relatedTarget)) {\n return fn.call(this, event);\n }\n };\n };\n callable = wrapFunction(callable);\n }\n const events = getElementEvents(element);\n const handlers = events[typeEvent] || (events[typeEvent] = {});\n const previousFunction = findHandler(handlers, callable, isDelegated ? handler : null);\n if (previousFunction) {\n previousFunction.oneOff = previousFunction.oneOff && oneOff;\n return;\n }\n const uid = makeEventUid(callable, originalTypeEvent.replace(namespaceRegex, ''));\n const fn = isDelegated ? bootstrapDelegationHandler(element, handler, callable) : bootstrapHandler(element, callable);\n fn.delegationSelector = isDelegated ? handler : null;\n fn.callable = callable;\n fn.oneOff = oneOff;\n fn.uidEvent = uid;\n handlers[uid] = fn;\n element.addEventListener(typeEvent, fn, isDelegated);\n}\nfunction removeHandler(element, events, typeEvent, handler, delegationSelector) {\n const fn = findHandler(events[typeEvent], handler, delegationSelector);\n if (!fn) {\n return;\n }\n element.removeEventListener(typeEvent, fn, Boolean(delegationSelector));\n delete events[typeEvent][fn.uidEvent];\n}\nfunction removeNamespacedHandlers(element, events, typeEvent, namespace) {\n const storeElementEvent = events[typeEvent] || {};\n for (const [handlerKey, event] of Object.entries(storeElementEvent)) {\n if (handlerKey.includes(namespace)) {\n removeHandler(element, events, typeEvent, event.callable, event.delegationSelector);\n }\n }\n}\nfunction getTypeEvent(event) {\n // allow to get the native events from namespaced events ('click.bs.button' --> 'click')\n event = event.replace(stripNameRegex, '');\n return customEvents[event] || event;\n}\nconst EventHandler = {\n on(element, event, handler, delegationFunction) {\n addHandler(element, event, handler, delegationFunction, false);\n },\n one(element, event, handler, delegationFunction) {\n addHandler(element, event, handler, delegationFunction, true);\n },\n off(element, originalTypeEvent, handler, delegationFunction) {\n if (typeof originalTypeEvent !== 'string' || !element) {\n return;\n }\n const [isDelegated, callable, typeEvent] = normalizeParameters(originalTypeEvent, handler, delegationFunction);\n const inNamespace = typeEvent !== originalTypeEvent;\n const events = getElementEvents(element);\n const storeElementEvent = events[typeEvent] || {};\n const isNamespace = originalTypeEvent.startsWith('.');\n if (typeof callable !== 'undefined') {\n // Simplest case: handler is passed, remove that listener ONLY.\n if (!Object.keys(storeElementEvent).length) {\n return;\n }\n removeHandler(element, events, typeEvent, callable, isDelegated ? handler : null);\n return;\n }\n if (isNamespace) {\n for (const elementEvent of Object.keys(events)) {\n removeNamespacedHandlers(element, events, elementEvent, originalTypeEvent.slice(1));\n }\n }\n for (const [keyHandlers, event] of Object.entries(storeElementEvent)) {\n const handlerKey = keyHandlers.replace(stripUidRegex, '');\n if (!inNamespace || originalTypeEvent.includes(handlerKey)) {\n removeHandler(element, events, typeEvent, event.callable, event.delegationSelector);\n }\n }\n },\n trigger(element, event, args) {\n if (typeof event !== 'string' || !element) {\n return null;\n }\n const $ = getjQuery();\n const typeEvent = getTypeEvent(event);\n const inNamespace = event !== typeEvent;\n let jQueryEvent = null;\n let bubbles = true;\n let nativeDispatch = true;\n let defaultPrevented = false;\n if (inNamespace && $) {\n jQueryEvent = $.Event(event, args);\n $(element).trigger(jQueryEvent);\n bubbles = !jQueryEvent.isPropagationStopped();\n nativeDispatch = !jQueryEvent.isImmediatePropagationStopped();\n defaultPrevented = jQueryEvent.isDefaultPrevented();\n }\n const evt = hydrateObj(new Event(event, {\n bubbles,\n cancelable: true\n }), args);\n if (defaultPrevented) {\n evt.preventDefault();\n }\n if (nativeDispatch) {\n element.dispatchEvent(evt);\n }\n if (evt.defaultPrevented && jQueryEvent) {\n jQueryEvent.preventDefault();\n }\n return evt;\n }\n};\nfunction hydrateObj(obj, meta = {}) {\n for (const [key, value] of Object.entries(meta)) {\n try {\n obj[key] = value;\n } catch (_unused) {\n Object.defineProperty(obj, key, {\n configurable: true,\n get() {\n return value;\n }\n });\n }\n }\n return obj;\n}\n\n/**\n * --------------------------------------------------------------------------\n * Bootstrap dom/manipulator.js\n * Licensed under MIT (https://github.com/twbs/bootstrap/blob/main/LICENSE)\n * --------------------------------------------------------------------------\n */\n\nfunction normalizeData(value) {\n if (value === 'true') {\n return true;\n }\n if (value === 'false') {\n return false;\n }\n if (value === Number(value).toString()) {\n return Number(value);\n }\n if (value === '' || value === 'null') {\n return null;\n }\n if (typeof value !== 'string') {\n return value;\n }\n try {\n return JSON.parse(decodeURIComponent(value));\n } catch (_unused) {\n return value;\n }\n}\nfunction normalizeDataKey(key) {\n return key.replace(/[A-Z]/g, chr => `-${chr.toLowerCase()}`);\n}\nconst Manipulator = {\n setDataAttribute(element, key, value) {\n element.setAttribute(`data-bs-${normalizeDataKey(key)}`, value);\n },\n removeDataAttribute(element, key) {\n element.removeAttribute(`data-bs-${normalizeDataKey(key)}`);\n },\n getDataAttributes(element) {\n if (!element) {\n return {};\n }\n const attributes = {};\n const bsKeys = Object.keys(element.dataset).filter(key => key.startsWith('bs') && !key.startsWith('bsConfig'));\n for (const key of bsKeys) {\n let pureKey = key.replace(/^bs/, '');\n pureKey = pureKey.charAt(0).toLowerCase() + pureKey.slice(1, pureKey.length);\n attributes[pureKey] = normalizeData(element.dataset[key]);\n }\n return attributes;\n },\n getDataAttribute(element, key) {\n return normalizeData(element.getAttribute(`data-bs-${normalizeDataKey(key)}`));\n }\n};\n\n/**\n * --------------------------------------------------------------------------\n * Bootstrap util/config.js\n * Licensed under MIT (https://github.com/twbs/bootstrap/blob/main/LICENSE)\n * --------------------------------------------------------------------------\n */\n\n\n/**\n * Class definition\n */\n\nclass Config {\n // Getters\n static get Default() {\n return {};\n }\n static get DefaultType() {\n return {};\n }\n static get NAME() {\n throw new Error('You have to implement the static method \"NAME\", for each component!');\n }\n _getConfig(config) {\n config = this._mergeConfigObj(config);\n config = this._configAfterMerge(config);\n this._typeCheckConfig(config);\n return config;\n }\n _configAfterMerge(config) {\n return config;\n }\n _mergeConfigObj(config, element) {\n const jsonConfig = isElement(element) ? Manipulator.getDataAttribute(element, 'config') : {}; // try to parse\n\n return {\n ...this.constructor.Default,\n ...(typeof jsonConfig === 'object' ? jsonConfig : {}),\n ...(isElement(element) ? Manipulator.getDataAttributes(element) : {}),\n ...(typeof config === 'object' ? config : {})\n };\n }\n _typeCheckConfig(config, configTypes = this.constructor.DefaultType) {\n for (const [property, expectedTypes] of Object.entries(configTypes)) {\n const value = config[property];\n const valueType = isElement(value) ? 'element' : toType(value);\n if (!new RegExp(expectedTypes).test(valueType)) {\n throw new TypeError(`${this.constructor.NAME.toUpperCase()}: Option \"${property}\" provided type \"${valueType}\" but expected type \"${expectedTypes}\".`);\n }\n }\n }\n}\n\n/**\n * --------------------------------------------------------------------------\n * Bootstrap base-component.js\n * Licensed under MIT (https://github.com/twbs/bootstrap/blob/main/LICENSE)\n * --------------------------------------------------------------------------\n */\n\n\n/**\n * Constants\n */\n\nconst VERSION = '5.3.3';\n\n/**\n * Class definition\n */\n\nclass BaseComponent extends Config {\n constructor(element, config) {\n super();\n element = getElement(element);\n if (!element) {\n return;\n }\n this._element = element;\n this._config = this._getConfig(config);\n Data.set(this._element, this.constructor.DATA_KEY, this);\n }\n\n // Public\n dispose() {\n Data.remove(this._element, this.constructor.DATA_KEY);\n EventHandler.off(this._element, this.constructor.EVENT_KEY);\n for (const propertyName of Object.getOwnPropertyNames(this)) {\n this[propertyName] = null;\n }\n }\n _queueCallback(callback, element, isAnimated = true) {\n executeAfterTransition(callback, element, isAnimated);\n }\n _getConfig(config) {\n config = this._mergeConfigObj(config, this._element);\n config = this._configAfterMerge(config);\n this._typeCheckConfig(config);\n return config;\n }\n\n // Static\n static getInstance(element) {\n return Data.get(getElement(element), this.DATA_KEY);\n }\n static getOrCreateInstance(element, config = {}) {\n return this.getInstance(element) || new this(element, typeof config === 'object' ? config : null);\n }\n static get VERSION() {\n return VERSION;\n }\n static get DATA_KEY() {\n return `bs.${this.NAME}`;\n }\n static get EVENT_KEY() {\n return `.${this.DATA_KEY}`;\n }\n static eventName(name) {\n return `${name}${this.EVENT_KEY}`;\n }\n}\n\n/**\n * --------------------------------------------------------------------------\n * Bootstrap dom/selector-engine.js\n * Licensed under MIT (https://github.com/twbs/bootstrap/blob/main/LICENSE)\n * --------------------------------------------------------------------------\n */\n\nconst getSelector = element => {\n let selector = element.getAttribute('data-bs-target');\n if (!selector || selector === '#') {\n let hrefAttribute = element.getAttribute('href');\n\n // The only valid content that could double as a selector are IDs or classes,\n // so everything starting with `#` or `.`. If a \"real\" URL is used as the selector,\n // `document.querySelector` will rightfully complain it is invalid.\n // See https://github.com/twbs/bootstrap/issues/32273\n if (!hrefAttribute || !hrefAttribute.includes('#') && !hrefAttribute.startsWith('.')) {\n return null;\n }\n\n // Just in case some CMS puts out a full URL with the anchor appended\n if (hrefAttribute.includes('#') && !hrefAttribute.startsWith('#')) {\n hrefAttribute = `#${hrefAttribute.split('#')[1]}`;\n }\n selector = hrefAttribute && hrefAttribute !== '#' ? hrefAttribute.trim() : null;\n }\n return selector ? selector.split(',').map(sel => parseSelector(sel)).join(',') : null;\n};\nconst SelectorEngine = {\n find(selector, element = document.documentElement) {\n return [].concat(...Element.prototype.querySelectorAll.call(element, selector));\n },\n findOne(selector, element = document.documentElement) {\n return Element.prototype.querySelector.call(element, selector);\n },\n children(element, selector) {\n return [].concat(...element.children).filter(child => child.matches(selector));\n },\n parents(element, selector) {\n const parents = [];\n let ancestor = element.parentNode.closest(selector);\n while (ancestor) {\n parents.push(ancestor);\n ancestor = ancestor.parentNode.closest(selector);\n }\n return parents;\n },\n prev(element, selector) {\n let previous = element.previousElementSibling;\n while (previous) {\n if (previous.matches(selector)) {\n return [previous];\n }\n previous = previous.previousElementSibling;\n }\n return [];\n },\n // TODO: this is now unused; remove later along with prev()\n next(element, selector) {\n let next = element.nextElementSibling;\n while (next) {\n if (next.matches(selector)) {\n return [next];\n }\n next = next.nextElementSibling;\n }\n return [];\n },\n focusableChildren(element) {\n const focusables = ['a', 'button', 'input', 'textarea', 'select', 'details', '[tabindex]', '[contenteditable=\"true\"]'].map(selector => `${selector}:not([tabindex^=\"-\"])`).join(',');\n return this.find(focusables, element).filter(el => !isDisabled(el) && isVisible(el));\n },\n getSelectorFromElement(element) {\n const selector = getSelector(element);\n if (selector) {\n return SelectorEngine.findOne(selector) ? selector : null;\n }\n return null;\n },\n getElementFromSelector(element) {\n const selector = getSelector(element);\n return selector ? SelectorEngine.findOne(selector) : null;\n },\n getMultipleElementsFromSelector(element) {\n const selector = getSelector(element);\n return selector ? SelectorEngine.find(selector) : [];\n }\n};\n\n/**\n * --------------------------------------------------------------------------\n * Bootstrap util/component-functions.js\n * Licensed under MIT (https://github.com/twbs/bootstrap/blob/main/LICENSE)\n * --------------------------------------------------------------------------\n */\n\nconst enableDismissTrigger = (component, method = 'hide') => {\n const clickEvent = `click.dismiss${component.EVENT_KEY}`;\n const name = component.NAME;\n EventHandler.on(document, clickEvent, `[data-bs-dismiss=\"${name}\"]`, function (event) {\n if (['A', 'AREA'].includes(this.tagName)) {\n event.preventDefault();\n }\n if (isDisabled(this)) {\n return;\n }\n const target = SelectorEngine.getElementFromSelector(this) || this.closest(`.${name}`);\n const instance = component.getOrCreateInstance(target);\n\n // Method argument is left, for Alert and only, as it doesn't implement the 'hide' method\n instance[method]();\n });\n};\n\n/**\n * --------------------------------------------------------------------------\n * Bootstrap alert.js\n * Licensed under MIT (https://github.com/twbs/bootstrap/blob/main/LICENSE)\n * --------------------------------------------------------------------------\n */\n\n\n/**\n * Constants\n */\n\nconst NAME$f = 'alert';\nconst DATA_KEY$a = 'bs.alert';\nconst EVENT_KEY$b = `.${DATA_KEY$a}`;\nconst EVENT_CLOSE = `close${EVENT_KEY$b}`;\nconst EVENT_CLOSED = `closed${EVENT_KEY$b}`;\nconst CLASS_NAME_FADE$5 = 'fade';\nconst CLASS_NAME_SHOW$8 = 'show';\n\n/**\n * Class definition\n */\n\nclass Alert extends BaseComponent {\n // Getters\n static get NAME() {\n return NAME$f;\n }\n\n // Public\n close() {\n const closeEvent = EventHandler.trigger(this._element, EVENT_CLOSE);\n if (closeEvent.defaultPrevented) {\n return;\n }\n this._element.classList.remove(CLASS_NAME_SHOW$8);\n const isAnimated = this._element.classList.contains(CLASS_NAME_FADE$5);\n this._queueCallback(() => this._destroyElement(), this._element, isAnimated);\n }\n\n // Private\n _destroyElement() {\n this._element.remove();\n EventHandler.trigger(this._element, EVENT_CLOSED);\n this.dispose();\n }\n\n // Static\n static jQueryInterface(config) {\n return this.each(function () {\n const data = Alert.getOrCreateInstance(this);\n if (typeof config !== 'string') {\n return;\n }\n if (data[config] === undefined || config.startsWith('_') || config === 'constructor') {\n throw new TypeError(`No method named \"${config}\"`);\n }\n data[config](this);\n });\n }\n}\n\n/**\n * Data API implementation\n */\n\nenableDismissTrigger(Alert, 'close');\n\n/**\n * jQuery\n */\n\ndefineJQueryPlugin(Alert);\n\n/**\n * --------------------------------------------------------------------------\n * Bootstrap button.js\n * Licensed under MIT (https://github.com/twbs/bootstrap/blob/main/LICENSE)\n * --------------------------------------------------------------------------\n */\n\n\n/**\n * Constants\n */\n\nconst NAME$e = 'button';\nconst DATA_KEY$9 = 'bs.button';\nconst EVENT_KEY$a = `.${DATA_KEY$9}`;\nconst DATA_API_KEY$6 = '.data-api';\nconst CLASS_NAME_ACTIVE$3 = 'active';\nconst SELECTOR_DATA_TOGGLE$5 = '[data-bs-toggle=\"button\"]';\nconst EVENT_CLICK_DATA_API$6 = `click${EVENT_KEY$a}${DATA_API_KEY$6}`;\n\n/**\n * Class definition\n */\n\nclass Button extends BaseComponent {\n // Getters\n static get NAME() {\n return NAME$e;\n }\n\n // Public\n toggle() {\n // Toggle class and sync the `aria-pressed` attribute with the return value of the `.toggle()` method\n this._element.setAttribute('aria-pressed', this._element.classList.toggle(CLASS_NAME_ACTIVE$3));\n }\n\n // Static\n static jQueryInterface(config) {\n return this.each(function () {\n const data = Button.getOrCreateInstance(this);\n if (config === 'toggle') {\n data[config]();\n }\n });\n }\n}\n\n/**\n * Data API implementation\n */\n\nEventHandler.on(document, EVENT_CLICK_DATA_API$6, SELECTOR_DATA_TOGGLE$5, event => {\n event.preventDefault();\n const button = event.target.closest(SELECTOR_DATA_TOGGLE$5);\n const data = Button.getOrCreateInstance(button);\n data.toggle();\n});\n\n/**\n * jQuery\n */\n\ndefineJQueryPlugin(Button);\n\n/**\n * --------------------------------------------------------------------------\n * Bootstrap util/swipe.js\n * Licensed under MIT (https://github.com/twbs/bootstrap/blob/main/LICENSE)\n * --------------------------------------------------------------------------\n */\n\n\n/**\n * Constants\n */\n\nconst NAME$d = 'swipe';\nconst EVENT_KEY$9 = '.bs.swipe';\nconst EVENT_TOUCHSTART = `touchstart${EVENT_KEY$9}`;\nconst EVENT_TOUCHMOVE = `touchmove${EVENT_KEY$9}`;\nconst EVENT_TOUCHEND = `touchend${EVENT_KEY$9}`;\nconst EVENT_POINTERDOWN = `pointerdown${EVENT_KEY$9}`;\nconst EVENT_POINTERUP = `pointerup${EVENT_KEY$9}`;\nconst POINTER_TYPE_TOUCH = 'touch';\nconst POINTER_TYPE_PEN = 'pen';\nconst CLASS_NAME_POINTER_EVENT = 'pointer-event';\nconst SWIPE_THRESHOLD = 40;\nconst Default$c = {\n endCallback: null,\n leftCallback: null,\n rightCallback: null\n};\nconst DefaultType$c = {\n endCallback: '(function|null)',\n leftCallback: '(function|null)',\n rightCallback: '(function|null)'\n};\n\n/**\n * Class definition\n */\n\nclass Swipe extends Config {\n constructor(element, config) {\n super();\n this._element = element;\n if (!element || !Swipe.isSupported()) {\n return;\n }\n this._config = this._getConfig(config);\n this._deltaX = 0;\n this._supportPointerEvents = Boolean(window.PointerEvent);\n this._initEvents();\n }\n\n // Getters\n static get Default() {\n return Default$c;\n }\n static get DefaultType() {\n return DefaultType$c;\n }\n static get NAME() {\n return NAME$d;\n }\n\n // Public\n dispose() {\n EventHandler.off(this._element, EVENT_KEY$9);\n }\n\n // Private\n _start(event) {\n if (!this._supportPointerEvents) {\n this._deltaX = event.touches[0].clientX;\n return;\n }\n if (this._eventIsPointerPenTouch(event)) {\n this._deltaX = event.clientX;\n }\n }\n _end(event) {\n if (this._eventIsPointerPenTouch(event)) {\n this._deltaX = event.clientX - this._deltaX;\n }\n this._handleSwipe();\n execute(this._config.endCallback);\n }\n _move(event) {\n this._deltaX = event.touches && event.touches.length > 1 ? 0 : event.touches[0].clientX - this._deltaX;\n }\n _handleSwipe() {\n const absDeltaX = Math.abs(this._deltaX);\n if (absDeltaX <= SWIPE_THRESHOLD) {\n return;\n }\n const direction = absDeltaX / this._deltaX;\n this._deltaX = 0;\n if (!direction) {\n return;\n }\n execute(direction > 0 ? this._config.rightCallback : this._config.leftCallback);\n }\n _initEvents() {\n if (this._supportPointerEvents) {\n EventHandler.on(this._element, EVENT_POINTERDOWN, event => this._start(event));\n EventHandler.on(this._element, EVENT_POINTERUP, event => this._end(event));\n this._element.classList.add(CLASS_NAME_POINTER_EVENT);\n } else {\n EventHandler.on(this._element, EVENT_TOUCHSTART, event => this._start(event));\n EventHandler.on(this._element, EVENT_TOUCHMOVE, event => this._move(event));\n EventHandler.on(this._element, EVENT_TOUCHEND, event => this._end(event));\n }\n }\n _eventIsPointerPenTouch(event) {\n return this._supportPointerEvents && (event.pointerType === POINTER_TYPE_PEN || event.pointerType === POINTER_TYPE_TOUCH);\n }\n\n // Static\n static isSupported() {\n return 'ontouchstart' in document.documentElement || navigator.maxTouchPoints > 0;\n }\n}\n\n/**\n * --------------------------------------------------------------------------\n * Bootstrap carousel.js\n * Licensed under MIT (https://github.com/twbs/bootstrap/blob/main/LICENSE)\n * --------------------------------------------------------------------------\n */\n\n\n/**\n * Constants\n */\n\nconst NAME$c = 'carousel';\nconst DATA_KEY$8 = 'bs.carousel';\nconst EVENT_KEY$8 = `.${DATA_KEY$8}`;\nconst DATA_API_KEY$5 = '.data-api';\nconst ARROW_LEFT_KEY$1 = 'ArrowLeft';\nconst ARROW_RIGHT_KEY$1 = 'ArrowRight';\nconst TOUCHEVENT_COMPAT_WAIT = 500; // Time for mouse compat events to fire after touch\n\nconst ORDER_NEXT = 'next';\nconst ORDER_PREV = 'prev';\nconst DIRECTION_LEFT = 'left';\nconst DIRECTION_RIGHT = 'right';\nconst EVENT_SLIDE = `slide${EVENT_KEY$8}`;\nconst EVENT_SLID = `slid${EVENT_KEY$8}`;\nconst EVENT_KEYDOWN$1 = `keydown${EVENT_KEY$8}`;\nconst EVENT_MOUSEENTER$1 = `mouseenter${EVENT_KEY$8}`;\nconst EVENT_MOUSELEAVE$1 = `mouseleave${EVENT_KEY$8}`;\nconst EVENT_DRAG_START = `dragstart${EVENT_KEY$8}`;\nconst EVENT_LOAD_DATA_API$3 = `load${EVENT_KEY$8}${DATA_API_KEY$5}`;\nconst EVENT_CLICK_DATA_API$5 = `click${EVENT_KEY$8}${DATA_API_KEY$5}`;\nconst CLASS_NAME_CAROUSEL = 'carousel';\nconst CLASS_NAME_ACTIVE$2 = 'active';\nconst CLASS_NAME_SLIDE = 'slide';\nconst CLASS_NAME_END = 'carousel-item-end';\nconst CLASS_NAME_START = 'carousel-item-start';\nconst CLASS_NAME_NEXT = 'carousel-item-next';\nconst CLASS_NAME_PREV = 'carousel-item-prev';\nconst SELECTOR_ACTIVE = '.active';\nconst SELECTOR_ITEM = '.carousel-item';\nconst SELECTOR_ACTIVE_ITEM = SELECTOR_ACTIVE + SELECTOR_ITEM;\nconst SELECTOR_ITEM_IMG = '.carousel-item img';\nconst SELECTOR_INDICATORS = '.carousel-indicators';\nconst SELECTOR_DATA_SLIDE = '[data-bs-slide], [data-bs-slide-to]';\nconst SELECTOR_DATA_RIDE = '[data-bs-ride=\"carousel\"]';\nconst KEY_TO_DIRECTION = {\n [ARROW_LEFT_KEY$1]: DIRECTION_RIGHT,\n [ARROW_RIGHT_KEY$1]: DIRECTION_LEFT\n};\nconst Default$b = {\n interval: 5000,\n keyboard: true,\n pause: 'hover',\n ride: false,\n touch: true,\n wrap: true\n};\nconst DefaultType$b = {\n interval: '(number|boolean)',\n // TODO:v6 remove boolean support\n keyboard: 'boolean',\n pause: '(string|boolean)',\n ride: '(boolean|string)',\n touch: 'boolean',\n wrap: 'boolean'\n};\n\n/**\n * Class definition\n */\n\nclass Carousel extends BaseComponent {\n constructor(element, config) {\n super(element, config);\n this._interval = null;\n this._activeElement = null;\n this._isSliding = false;\n this.touchTimeout = null;\n this._swipeHelper = null;\n this._indicatorsElement = SelectorEngine.findOne(SELECTOR_INDICATORS, this._element);\n this._addEventListeners();\n if (this._config.ride === CLASS_NAME_CAROUSEL) {\n this.cycle();\n }\n }\n\n // Getters\n static get Default() {\n return Default$b;\n }\n static get DefaultType() {\n return DefaultType$b;\n }\n static get NAME() {\n return NAME$c;\n }\n\n // Public\n next() {\n this._slide(ORDER_NEXT);\n }\n nextWhenVisible() {\n // FIXME TODO use `document.visibilityState`\n // Don't call next when the page isn't visible\n // or the carousel or its parent isn't visible\n if (!document.hidden && isVisible(this._element)) {\n this.next();\n }\n }\n prev() {\n this._slide(ORDER_PREV);\n }\n pause() {\n if (this._isSliding) {\n triggerTransitionEnd(this._element);\n }\n this._clearInterval();\n }\n cycle() {\n this._clearInterval();\n this._updateInterval();\n this._interval = setInterval(() => this.nextWhenVisible(), this._config.interval);\n }\n _maybeEnableCycle() {\n if (!this._config.ride) {\n return;\n }\n if (this._isSliding) {\n EventHandler.one(this._element, EVENT_SLID, () => this.cycle());\n return;\n }\n this.cycle();\n }\n to(index) {\n const items = this._getItems();\n if (index > items.length - 1 || index < 0) {\n return;\n }\n if (this._isSliding) {\n EventHandler.one(this._element, EVENT_SLID, () => this.to(index));\n return;\n }\n const activeIndex = this._getItemIndex(this._getActive());\n if (activeIndex === index) {\n return;\n }\n const order = index > activeIndex ? ORDER_NEXT : ORDER_PREV;\n this._slide(order, items[index]);\n }\n dispose() {\n if (this._swipeHelper) {\n this._swipeHelper.dispose();\n }\n super.dispose();\n }\n\n // Private\n _configAfterMerge(config) {\n config.defaultInterval = config.interval;\n return config;\n }\n _addEventListeners() {\n if (this._config.keyboard) {\n EventHandler.on(this._element, EVENT_KEYDOWN$1, event => this._keydown(event));\n }\n if (this._config.pause === 'hover') {\n EventHandler.on(this._element, EVENT_MOUSEENTER$1, () => this.pause());\n EventHandler.on(this._element, EVENT_MOUSELEAVE$1, () => this._maybeEnableCycle());\n }\n if (this._config.touch && Swipe.isSupported()) {\n this._addTouchEventListeners();\n }\n }\n _addTouchEventListeners() {\n for (const img of SelectorEngine.find(SELECTOR_ITEM_IMG, this._element)) {\n EventHandler.on(img, EVENT_DRAG_START, event => event.preventDefault());\n }\n const endCallBack = () => {\n if (this._config.pause !== 'hover') {\n return;\n }\n\n // If it's a touch-enabled device, mouseenter/leave are fired as\n // part of the mouse compatibility events on first tap - the carousel\n // would stop cycling until user tapped out of it;\n // here, we listen for touchend, explicitly pause the carousel\n // (as if it's the second time we tap on it, mouseenter compat event\n // is NOT fired) and after a timeout (to allow for mouse compatibility\n // events to fire) we explicitly restart cycling\n\n this.pause();\n if (this.touchTimeout) {\n clearTimeout(this.touchTimeout);\n }\n this.touchTimeout = setTimeout(() => this._maybeEnableCycle(), TOUCHEVENT_COMPAT_WAIT + this._config.interval);\n };\n const swipeConfig = {\n leftCallback: () => this._slide(this._directionToOrder(DIRECTION_LEFT)),\n rightCallback: () => this._slide(this._directionToOrder(DIRECTION_RIGHT)),\n endCallback: endCallBack\n };\n this._swipeHelper = new Swipe(this._element, swipeConfig);\n }\n _keydown(event) {\n if (/input|textarea/i.test(event.target.tagName)) {\n return;\n }\n const direction = KEY_TO_DIRECTION[event.key];\n if (direction) {\n event.preventDefault();\n this._slide(this._directionToOrder(direction));\n }\n }\n _getItemIndex(element) {\n return this._getItems().indexOf(element);\n }\n _setActiveIndicatorElement(index) {\n if (!this._indicatorsElement) {\n return;\n }\n const activeIndicator = SelectorEngine.findOne(SELECTOR_ACTIVE, this._indicatorsElement);\n activeIndicator.classList.remove(CLASS_NAME_ACTIVE$2);\n activeIndicator.removeAttribute('aria-current');\n const newActiveIndicator = SelectorEngine.findOne(`[data-bs-slide-to=\"${index}\"]`, this._indicatorsElement);\n if (newActiveIndicator) {\n newActiveIndicator.classList.add(CLASS_NAME_ACTIVE$2);\n newActiveIndicator.setAttribute('aria-current', 'true');\n }\n }\n _updateInterval() {\n const element = this._activeElement || this._getActive();\n if (!element) {\n return;\n }\n const elementInterval = Number.parseInt(element.getAttribute('data-bs-interval'), 10);\n this._config.interval = elementInterval || this._config.defaultInterval;\n }\n _slide(order, element = null) {\n if (this._isSliding) {\n return;\n }\n const activeElement = this._getActive();\n const isNext = order === ORDER_NEXT;\n const nextElement = element || getNextActiveElement(this._getItems(), activeElement, isNext, this._config.wrap);\n if (nextElement === activeElement) {\n return;\n }\n const nextElementIndex = this._getItemIndex(nextElement);\n const triggerEvent = eventName => {\n return EventHandler.trigger(this._element, eventName, {\n relatedTarget: nextElement,\n direction: this._orderToDirection(order),\n from: this._getItemIndex(activeElement),\n to: nextElementIndex\n });\n };\n const slideEvent = triggerEvent(EVENT_SLIDE);\n if (slideEvent.defaultPrevented) {\n return;\n }\n if (!activeElement || !nextElement) {\n // Some weirdness is happening, so we bail\n // TODO: change tests that use empty divs to avoid this check\n return;\n }\n const isCycling = Boolean(this._interval);\n this.pause();\n this._isSliding = true;\n this._setActiveIndicatorElement(nextElementIndex);\n this._activeElement = nextElement;\n const directionalClassName = isNext ? CLASS_NAME_START : CLASS_NAME_END;\n const orderClassName = isNext ? CLASS_NAME_NEXT : CLASS_NAME_PREV;\n nextElement.classList.add(orderClassName);\n reflow(nextElement);\n activeElement.classList.add(directionalClassName);\n nextElement.classList.add(directionalClassName);\n const completeCallBack = () => {\n nextElement.classList.remove(directionalClassName, orderClassName);\n nextElement.classList.add(CLASS_NAME_ACTIVE$2);\n activeElement.classList.remove(CLASS_NAME_ACTIVE$2, orderClassName, directionalClassName);\n this._isSliding = false;\n triggerEvent(EVENT_SLID);\n };\n this._queueCallback(completeCallBack, activeElement, this._isAnimated());\n if (isCycling) {\n this.cycle();\n }\n }\n _isAnimated() {\n return this._element.classList.contains(CLASS_NAME_SLIDE);\n }\n _getActive() {\n return SelectorEngine.findOne(SELECTOR_ACTIVE_ITEM, this._element);\n }\n _getItems() {\n return SelectorEngine.find(SELECTOR_ITEM, this._element);\n }\n _clearInterval() {\n if (this._interval) {\n clearInterval(this._interval);\n this._interval = null;\n }\n }\n _directionToOrder(direction) {\n if (isRTL()) {\n return direction === DIRECTION_LEFT ? ORDER_PREV : ORDER_NEXT;\n }\n return direction === DIRECTION_LEFT ? ORDER_NEXT : ORDER_PREV;\n }\n _orderToDirection(order) {\n if (isRTL()) {\n return order === ORDER_PREV ? DIRECTION_LEFT : DIRECTION_RIGHT;\n }\n return order === ORDER_PREV ? DIRECTION_RIGHT : DIRECTION_LEFT;\n }\n\n // Static\n static jQueryInterface(config) {\n return this.each(function () {\n const data = Carousel.getOrCreateInstance(this, config);\n if (typeof config === 'number') {\n data.to(config);\n return;\n }\n if (typeof config === 'string') {\n if (data[config] === undefined || config.startsWith('_') || config === 'constructor') {\n throw new TypeError(`No method named \"${config}\"`);\n }\n data[config]();\n }\n });\n }\n}\n\n/**\n * Data API implementation\n */\n\nEventHandler.on(document, EVENT_CLICK_DATA_API$5, SELECTOR_DATA_SLIDE, function (event) {\n const target = SelectorEngine.getElementFromSelector(this);\n if (!target || !target.classList.contains(CLASS_NAME_CAROUSEL)) {\n return;\n }\n event.preventDefault();\n const carousel = Carousel.getOrCreateInstance(target);\n const slideIndex = this.getAttribute('data-bs-slide-to');\n if (slideIndex) {\n carousel.to(slideIndex);\n carousel._maybeEnableCycle();\n return;\n }\n if (Manipulator.getDataAttribute(this, 'slide') === 'next') {\n carousel.next();\n carousel._maybeEnableCycle();\n return;\n }\n carousel.prev();\n carousel._maybeEnableCycle();\n});\nEventHandler.on(window, EVENT_LOAD_DATA_API$3, () => {\n const carousels = SelectorEngine.find(SELECTOR_DATA_RIDE);\n for (const carousel of carousels) {\n Carousel.getOrCreateInstance(carousel);\n }\n});\n\n/**\n * jQuery\n */\n\ndefineJQueryPlugin(Carousel);\n\n/**\n * --------------------------------------------------------------------------\n * Bootstrap collapse.js\n * Licensed under MIT (https://github.com/twbs/bootstrap/blob/main/LICENSE)\n * --------------------------------------------------------------------------\n */\n\n\n/**\n * Constants\n */\n\nconst NAME$b = 'collapse';\nconst DATA_KEY$7 = 'bs.collapse';\nconst EVENT_KEY$7 = `.${DATA_KEY$7}`;\nconst DATA_API_KEY$4 = '.data-api';\nconst EVENT_SHOW$6 = `show${EVENT_KEY$7}`;\nconst EVENT_SHOWN$6 = `shown${EVENT_KEY$7}`;\nconst EVENT_HIDE$6 = `hide${EVENT_KEY$7}`;\nconst EVENT_HIDDEN$6 = `hidden${EVENT_KEY$7}`;\nconst EVENT_CLICK_DATA_API$4 = `click${EVENT_KEY$7}${DATA_API_KEY$4}`;\nconst CLASS_NAME_SHOW$7 = 'show';\nconst CLASS_NAME_COLLAPSE = 'collapse';\nconst CLASS_NAME_COLLAPSING = 'collapsing';\nconst CLASS_NAME_COLLAPSED = 'collapsed';\nconst CLASS_NAME_DEEPER_CHILDREN = `:scope .${CLASS_NAME_COLLAPSE} .${CLASS_NAME_COLLAPSE}`;\nconst CLASS_NAME_HORIZONTAL = 'collapse-horizontal';\nconst WIDTH = 'width';\nconst HEIGHT = 'height';\nconst SELECTOR_ACTIVES = '.collapse.show, .collapse.collapsing';\nconst SELECTOR_DATA_TOGGLE$4 = '[data-bs-toggle=\"collapse\"]';\nconst Default$a = {\n parent: null,\n toggle: true\n};\nconst DefaultType$a = {\n parent: '(null|element)',\n toggle: 'boolean'\n};\n\n/**\n * Class definition\n */\n\nclass Collapse extends BaseComponent {\n constructor(element, config) {\n super(element, config);\n this._isTransitioning = false;\n this._triggerArray = [];\n const toggleList = SelectorEngine.find(SELECTOR_DATA_TOGGLE$4);\n for (const elem of toggleList) {\n const selector = SelectorEngine.getSelectorFromElement(elem);\n const filterElement = SelectorEngine.find(selector).filter(foundElement => foundElement === this._element);\n if (selector !== null && filterElement.length) {\n this._triggerArray.push(elem);\n }\n }\n this._initializeChildren();\n if (!this._config.parent) {\n this._addAriaAndCollapsedClass(this._triggerArray, this._isShown());\n }\n if (this._config.toggle) {\n this.toggle();\n }\n }\n\n // Getters\n static get Default() {\n return Default$a;\n }\n static get DefaultType() {\n return DefaultType$a;\n }\n static get NAME() {\n return NAME$b;\n }\n\n // Public\n toggle() {\n if (this._isShown()) {\n this.hide();\n } else {\n this.show();\n }\n }\n show() {\n if (this._isTransitioning || this._isShown()) {\n return;\n }\n let activeChildren = [];\n\n // find active children\n if (this._config.parent) {\n activeChildren = this._getFirstLevelChildren(SELECTOR_ACTIVES).filter(element => element !== this._element).map(element => Collapse.getOrCreateInstance(element, {\n toggle: false\n }));\n }\n if (activeChildren.length && activeChildren[0]._isTransitioning) {\n return;\n }\n const startEvent = EventHandler.trigger(this._element, EVENT_SHOW$6);\n if (startEvent.defaultPrevented) {\n return;\n }\n for (const activeInstance of activeChildren) {\n activeInstance.hide();\n }\n const dimension = this._getDimension();\n this._element.classList.remove(CLASS_NAME_COLLAPSE);\n this._element.classList.add(CLASS_NAME_COLLAPSING);\n this._element.style[dimension] = 0;\n this._addAriaAndCollapsedClass(this._triggerArray, true);\n this._isTransitioning = true;\n const complete = () => {\n this._isTransitioning = false;\n this._element.classList.remove(CLASS_NAME_COLLAPSING);\n this._element.classList.add(CLASS_NAME_COLLAPSE, CLASS_NAME_SHOW$7);\n this._element.style[dimension] = '';\n EventHandler.trigger(this._element, EVENT_SHOWN$6);\n };\n const capitalizedDimension = dimension[0].toUpperCase() + dimension.slice(1);\n const scrollSize = `scroll${capitalizedDimension}`;\n this._queueCallback(complete, this._element, true);\n this._element.style[dimension] = `${this._element[scrollSize]}px`;\n }\n hide() {\n if (this._isTransitioning || !this._isShown()) {\n return;\n }\n const startEvent = EventHandler.trigger(this._element, EVENT_HIDE$6);\n if (startEvent.defaultPrevented) {\n return;\n }\n const dimension = this._getDimension();\n this._element.style[dimension] = `${this._element.getBoundingClientRect()[dimension]}px`;\n reflow(this._element);\n this._element.classList.add(CLASS_NAME_COLLAPSING);\n this._element.classList.remove(CLASS_NAME_COLLAPSE, CLASS_NAME_SHOW$7);\n for (const trigger of this._triggerArray) {\n const element = SelectorEngine.getElementFromSelector(trigger);\n if (element && !this._isShown(element)) {\n this._addAriaAndCollapsedClass([trigger], false);\n }\n }\n this._isTransitioning = true;\n const complete = () => {\n this._isTransitioning = false;\n this._element.classList.remove(CLASS_NAME_COLLAPSING);\n this._element.classList.add(CLASS_NAME_COLLAPSE);\n EventHandler.trigger(this._element, EVENT_HIDDEN$6);\n };\n this._element.style[dimension] = '';\n this._queueCallback(complete, this._element, true);\n }\n _isShown(element = this._element) {\n return element.classList.contains(CLASS_NAME_SHOW$7);\n }\n\n // Private\n _configAfterMerge(config) {\n config.toggle = Boolean(config.toggle); // Coerce string values\n config.parent = getElement(config.parent);\n return config;\n }\n _getDimension() {\n return this._element.classList.contains(CLASS_NAME_HORIZONTAL) ? WIDTH : HEIGHT;\n }\n _initializeChildren() {\n if (!this._config.parent) {\n return;\n }\n const children = this._getFirstLevelChildren(SELECTOR_DATA_TOGGLE$4);\n for (const element of children) {\n const selected = SelectorEngine.getElementFromSelector(element);\n if (selected) {\n this._addAriaAndCollapsedClass([element], this._isShown(selected));\n }\n }\n }\n _getFirstLevelChildren(selector) {\n const children = SelectorEngine.find(CLASS_NAME_DEEPER_CHILDREN, this._config.parent);\n // remove children if greater depth\n return SelectorEngine.find(selector, this._config.parent).filter(element => !children.includes(element));\n }\n _addAriaAndCollapsedClass(triggerArray, isOpen) {\n if (!triggerArray.length) {\n return;\n }\n for (const element of triggerArray) {\n element.classList.toggle(CLASS_NAME_COLLAPSED, !isOpen);\n element.setAttribute('aria-expanded', isOpen);\n }\n }\n\n // Static\n static jQueryInterface(config) {\n const _config = {};\n if (typeof config === 'string' && /show|hide/.test(config)) {\n _config.toggle = false;\n }\n return this.each(function () {\n const data = Collapse.getOrCreateInstance(this, _config);\n if (typeof config === 'string') {\n if (typeof data[config] === 'undefined') {\n throw new TypeError(`No method named \"${config}\"`);\n }\n data[config]();\n }\n });\n }\n}\n\n/**\n * Data API implementation\n */\n\nEventHandler.on(document, EVENT_CLICK_DATA_API$4, SELECTOR_DATA_TOGGLE$4, function (event) {\n // preventDefault only for elements (which change the URL) not inside the collapsible element\n if (event.target.tagName === 'A' || event.delegateTarget && event.delegateTarget.tagName === 'A') {\n event.preventDefault();\n }\n for (const element of SelectorEngine.getMultipleElementsFromSelector(this)) {\n Collapse.getOrCreateInstance(element, {\n toggle: false\n }).toggle();\n }\n});\n\n/**\n * jQuery\n */\n\ndefineJQueryPlugin(Collapse);\n\n/**\n * --------------------------------------------------------------------------\n * Bootstrap dropdown.js\n * Licensed under MIT (https://github.com/twbs/bootstrap/blob/main/LICENSE)\n * --------------------------------------------------------------------------\n */\n\n\n/**\n * Constants\n */\n\nconst NAME$a = 'dropdown';\nconst DATA_KEY$6 = 'bs.dropdown';\nconst EVENT_KEY$6 = `.${DATA_KEY$6}`;\nconst DATA_API_KEY$3 = '.data-api';\nconst ESCAPE_KEY$2 = 'Escape';\nconst TAB_KEY$1 = 'Tab';\nconst ARROW_UP_KEY$1 = 'ArrowUp';\nconst ARROW_DOWN_KEY$1 = 'ArrowDown';\nconst RIGHT_MOUSE_BUTTON = 2; // MouseEvent.button value for the secondary button, usually the right button\n\nconst EVENT_HIDE$5 = `hide${EVENT_KEY$6}`;\nconst EVENT_HIDDEN$5 = `hidden${EVENT_KEY$6}`;\nconst EVENT_SHOW$5 = `show${EVENT_KEY$6}`;\nconst EVENT_SHOWN$5 = `shown${EVENT_KEY$6}`;\nconst EVENT_CLICK_DATA_API$3 = `click${EVENT_KEY$6}${DATA_API_KEY$3}`;\nconst EVENT_KEYDOWN_DATA_API = `keydown${EVENT_KEY$6}${DATA_API_KEY$3}`;\nconst EVENT_KEYUP_DATA_API = `keyup${EVENT_KEY$6}${DATA_API_KEY$3}`;\nconst CLASS_NAME_SHOW$6 = 'show';\nconst CLASS_NAME_DROPUP = 'dropup';\nconst CLASS_NAME_DROPEND = 'dropend';\nconst CLASS_NAME_DROPSTART = 'dropstart';\nconst CLASS_NAME_DROPUP_CENTER = 'dropup-center';\nconst CLASS_NAME_DROPDOWN_CENTER = 'dropdown-center';\nconst SELECTOR_DATA_TOGGLE$3 = '[data-bs-toggle=\"dropdown\"]:not(.disabled):not(:disabled)';\nconst SELECTOR_DATA_TOGGLE_SHOWN = `${SELECTOR_DATA_TOGGLE$3}.${CLASS_NAME_SHOW$6}`;\nconst SELECTOR_MENU = '.dropdown-menu';\nconst SELECTOR_NAVBAR = '.navbar';\nconst SELECTOR_NAVBAR_NAV = '.navbar-nav';\nconst SELECTOR_VISIBLE_ITEMS = '.dropdown-menu .dropdown-item:not(.disabled):not(:disabled)';\nconst PLACEMENT_TOP = isRTL() ? 'top-end' : 'top-start';\nconst PLACEMENT_TOPEND = isRTL() ? 'top-start' : 'top-end';\nconst PLACEMENT_BOTTOM = isRTL() ? 'bottom-end' : 'bottom-start';\nconst PLACEMENT_BOTTOMEND = isRTL() ? 'bottom-start' : 'bottom-end';\nconst PLACEMENT_RIGHT = isRTL() ? 'left-start' : 'right-start';\nconst PLACEMENT_LEFT = isRTL() ? 'right-start' : 'left-start';\nconst PLACEMENT_TOPCENTER = 'top';\nconst PLACEMENT_BOTTOMCENTER = 'bottom';\nconst Default$9 = {\n autoClose: true,\n boundary: 'clippingParents',\n display: 'dynamic',\n offset: [0, 2],\n popperConfig: null,\n reference: 'toggle'\n};\nconst DefaultType$9 = {\n autoClose: '(boolean|string)',\n boundary: '(string|element)',\n display: 'string',\n offset: '(array|string|function)',\n popperConfig: '(null|object|function)',\n reference: '(string|element|object)'\n};\n\n/**\n * Class definition\n */\n\nclass Dropdown extends BaseComponent {\n constructor(element, config) {\n super(element, config);\n this._popper = null;\n this._parent = this._element.parentNode; // dropdown wrapper\n // TODO: v6 revert #37011 & change markup https://getbootstrap.com/docs/5.3/forms/input-group/\n this._menu = SelectorEngine.next(this._element, SELECTOR_MENU)[0] || SelectorEngine.prev(this._element, SELECTOR_MENU)[0] || SelectorEngine.findOne(SELECTOR_MENU, this._parent);\n this._inNavbar = this._detectNavbar();\n }\n\n // Getters\n static get Default() {\n return Default$9;\n }\n static get DefaultType() {\n return DefaultType$9;\n }\n static get NAME() {\n return NAME$a;\n }\n\n // Public\n toggle() {\n return this._isShown() ? this.hide() : this.show();\n }\n show() {\n if (isDisabled(this._element) || this._isShown()) {\n return;\n }\n const relatedTarget = {\n relatedTarget: this._element\n };\n const showEvent = EventHandler.trigger(this._element, EVENT_SHOW$5, relatedTarget);\n if (showEvent.defaultPrevented) {\n return;\n }\n this._createPopper();\n\n // If this is a touch-enabled device we add extra\n // empty mouseover listeners to the body's immediate children;\n // only needed because of broken event delegation on iOS\n // https://www.quirksmode.org/blog/archives/2014/02/mouse_event_bub.html\n if ('ontouchstart' in document.documentElement && !this._parent.closest(SELECTOR_NAVBAR_NAV)) {\n for (const element of [].concat(...document.body.children)) {\n EventHandler.on(element, 'mouseover', noop);\n }\n }\n this._element.focus();\n this._element.setAttribute('aria-expanded', true);\n this._menu.classList.add(CLASS_NAME_SHOW$6);\n this._element.classList.add(CLASS_NAME_SHOW$6);\n EventHandler.trigger(this._element, EVENT_SHOWN$5, relatedTarget);\n }\n hide() {\n if (isDisabled(this._element) || !this._isShown()) {\n return;\n }\n const relatedTarget = {\n relatedTarget: this._element\n };\n this._completeHide(relatedTarget);\n }\n dispose() {\n if (this._popper) {\n this._popper.destroy();\n }\n super.dispose();\n }\n update() {\n this._inNavbar = this._detectNavbar();\n if (this._popper) {\n this._popper.update();\n }\n }\n\n // Private\n _completeHide(relatedTarget) {\n const hideEvent = EventHandler.trigger(this._element, EVENT_HIDE$5, relatedTarget);\n if (hideEvent.defaultPrevented) {\n return;\n }\n\n // If this is a touch-enabled device we remove the extra\n // empty mouseover listeners we added for iOS support\n if ('ontouchstart' in document.documentElement) {\n for (const element of [].concat(...document.body.children)) {\n EventHandler.off(element, 'mouseover', noop);\n }\n }\n if (this._popper) {\n this._popper.destroy();\n }\n this._menu.classList.remove(CLASS_NAME_SHOW$6);\n this._element.classList.remove(CLASS_NAME_SHOW$6);\n this._element.setAttribute('aria-expanded', 'false');\n Manipulator.removeDataAttribute(this._menu, 'popper');\n EventHandler.trigger(this._element, EVENT_HIDDEN$5, relatedTarget);\n }\n _getConfig(config) {\n config = super._getConfig(config);\n if (typeof config.reference === 'object' && !isElement(config.reference) && typeof config.reference.getBoundingClientRect !== 'function') {\n // Popper virtual elements require a getBoundingClientRect method\n throw new TypeError(`${NAME$a.toUpperCase()}: Option \"reference\" provided type \"object\" without a required \"getBoundingClientRect\" method.`);\n }\n return config;\n }\n _createPopper() {\n if (typeof Popper === 'undefined') {\n throw new TypeError('Bootstrap\\'s dropdowns require Popper (https://popper.js.org)');\n }\n let referenceElement = this._element;\n if (this._config.reference === 'parent') {\n referenceElement = this._parent;\n } else if (isElement(this._config.reference)) {\n referenceElement = getElement(this._config.reference);\n } else if (typeof this._config.reference === 'object') {\n referenceElement = this._config.reference;\n }\n const popperConfig = this._getPopperConfig();\n this._popper = Popper.createPopper(referenceElement, this._menu, popperConfig);\n }\n _isShown() {\n return this._menu.classList.contains(CLASS_NAME_SHOW$6);\n }\n _getPlacement() {\n const parentDropdown = this._parent;\n if (parentDropdown.classList.contains(CLASS_NAME_DROPEND)) {\n return PLACEMENT_RIGHT;\n }\n if (parentDropdown.classList.contains(CLASS_NAME_DROPSTART)) {\n return PLACEMENT_LEFT;\n }\n if (parentDropdown.classList.contains(CLASS_NAME_DROPUP_CENTER)) {\n return PLACEMENT_TOPCENTER;\n }\n if (parentDropdown.classList.contains(CLASS_NAME_DROPDOWN_CENTER)) {\n return PLACEMENT_BOTTOMCENTER;\n }\n\n // We need to trim the value because custom properties can also include spaces\n const isEnd = getComputedStyle(this._menu).getPropertyValue('--bs-position').trim() === 'end';\n if (parentDropdown.classList.contains(CLASS_NAME_DROPUP)) {\n return isEnd ? PLACEMENT_TOPEND : PLACEMENT_TOP;\n }\n return isEnd ? PLACEMENT_BOTTOMEND : PLACEMENT_BOTTOM;\n }\n _detectNavbar() {\n return this._element.closest(SELECTOR_NAVBAR) !== null;\n }\n _getOffset() {\n const {\n offset\n } = this._config;\n if (typeof offset === 'string') {\n return offset.split(',').map(value => Number.parseInt(value, 10));\n }\n if (typeof offset === 'function') {\n return popperData => offset(popperData, this._element);\n }\n return offset;\n }\n _getPopperConfig() {\n const defaultBsPopperConfig = {\n placement: this._getPlacement(),\n modifiers: [{\n name: 'preventOverflow',\n options: {\n boundary: this._config.boundary\n }\n }, {\n name: 'offset',\n options: {\n offset: this._getOffset()\n }\n }]\n };\n\n // Disable Popper if we have a static display or Dropdown is in Navbar\n if (this._inNavbar || this._config.display === 'static') {\n Manipulator.setDataAttribute(this._menu, 'popper', 'static'); // TODO: v6 remove\n defaultBsPopperConfig.modifiers = [{\n name: 'applyStyles',\n enabled: false\n }];\n }\n return {\n ...defaultBsPopperConfig,\n ...execute(this._config.popperConfig, [defaultBsPopperConfig])\n };\n }\n _selectMenuItem({\n key,\n target\n }) {\n const items = SelectorEngine.find(SELECTOR_VISIBLE_ITEMS, this._menu).filter(element => isVisible(element));\n if (!items.length) {\n return;\n }\n\n // if target isn't included in items (e.g. when expanding the dropdown)\n // allow cycling to get the last item in case key equals ARROW_UP_KEY\n getNextActiveElement(items, target, key === ARROW_DOWN_KEY$1, !items.includes(target)).focus();\n }\n\n // Static\n static jQueryInterface(config) {\n return this.each(function () {\n const data = Dropdown.getOrCreateInstance(this, config);\n if (typeof config !== 'string') {\n return;\n }\n if (typeof data[config] === 'undefined') {\n throw new TypeError(`No method named \"${config}\"`);\n }\n data[config]();\n });\n }\n static clearMenus(event) {\n if (event.button === RIGHT_MOUSE_BUTTON || event.type === 'keyup' && event.key !== TAB_KEY$1) {\n return;\n }\n const openToggles = SelectorEngine.find(SELECTOR_DATA_TOGGLE_SHOWN);\n for (const toggle of openToggles) {\n const context = Dropdown.getInstance(toggle);\n if (!context || context._config.autoClose === false) {\n continue;\n }\n const composedPath = event.composedPath();\n const isMenuTarget = composedPath.includes(context._menu);\n if (composedPath.includes(context._element) || context._config.autoClose === 'inside' && !isMenuTarget || context._config.autoClose === 'outside' && isMenuTarget) {\n continue;\n }\n\n // Tab navigation through the dropdown menu or events from contained inputs shouldn't close the menu\n if (context._menu.contains(event.target) && (event.type === 'keyup' && event.key === TAB_KEY$1 || /input|select|option|textarea|form/i.test(event.target.tagName))) {\n continue;\n }\n const relatedTarget = {\n relatedTarget: context._element\n };\n if (event.type === 'click') {\n relatedTarget.clickEvent = event;\n }\n context._completeHide(relatedTarget);\n }\n }\n static dataApiKeydownHandler(event) {\n // If not an UP | DOWN | ESCAPE key => not a dropdown command\n // If input/textarea && if key is other than ESCAPE => not a dropdown command\n\n const isInput = /input|textarea/i.test(event.target.tagName);\n const isEscapeEvent = event.key === ESCAPE_KEY$2;\n const isUpOrDownEvent = [ARROW_UP_KEY$1, ARROW_DOWN_KEY$1].includes(event.key);\n if (!isUpOrDownEvent && !isEscapeEvent) {\n return;\n }\n if (isInput && !isEscapeEvent) {\n return;\n }\n event.preventDefault();\n\n // TODO: v6 revert #37011 & change markup https://getbootstrap.com/docs/5.3/forms/input-group/\n const getToggleButton = this.matches(SELECTOR_DATA_TOGGLE$3) ? this : SelectorEngine.prev(this, SELECTOR_DATA_TOGGLE$3)[0] || SelectorEngine.next(this, SELECTOR_DATA_TOGGLE$3)[0] || SelectorEngine.findOne(SELECTOR_DATA_TOGGLE$3, event.delegateTarget.parentNode);\n const instance = Dropdown.getOrCreateInstance(getToggleButton);\n if (isUpOrDownEvent) {\n event.stopPropagation();\n instance.show();\n instance._selectMenuItem(event);\n return;\n }\n if (instance._isShown()) {\n // else is escape and we check if it is shown\n event.stopPropagation();\n instance.hide();\n getToggleButton.focus();\n }\n }\n}\n\n/**\n * Data API implementation\n */\n\nEventHandler.on(document, EVENT_KEYDOWN_DATA_API, SELECTOR_DATA_TOGGLE$3, Dropdown.dataApiKeydownHandler);\nEventHandler.on(document, EVENT_KEYDOWN_DATA_API, SELECTOR_MENU, Dropdown.dataApiKeydownHandler);\nEventHandler.on(document, EVENT_CLICK_DATA_API$3, Dropdown.clearMenus);\nEventHandler.on(document, EVENT_KEYUP_DATA_API, Dropdown.clearMenus);\nEventHandler.on(document, EVENT_CLICK_DATA_API$3, SELECTOR_DATA_TOGGLE$3, function (event) {\n event.preventDefault();\n Dropdown.getOrCreateInstance(this).toggle();\n});\n\n/**\n * jQuery\n */\n\ndefineJQueryPlugin(Dropdown);\n\n/**\n * --------------------------------------------------------------------------\n * Bootstrap util/backdrop.js\n * Licensed under MIT (https://github.com/twbs/bootstrap/blob/main/LICENSE)\n * --------------------------------------------------------------------------\n */\n\n\n/**\n * Constants\n */\n\nconst NAME$9 = 'backdrop';\nconst CLASS_NAME_FADE$4 = 'fade';\nconst CLASS_NAME_SHOW$5 = 'show';\nconst EVENT_MOUSEDOWN = `mousedown.bs.${NAME$9}`;\nconst Default$8 = {\n className: 'modal-backdrop',\n clickCallback: null,\n isAnimated: false,\n isVisible: true,\n // if false, we use the backdrop helper without adding any element to the dom\n rootElement: 'body' // give the choice to place backdrop under different elements\n};\nconst DefaultType$8 = {\n className: 'string',\n clickCallback: '(function|null)',\n isAnimated: 'boolean',\n isVisible: 'boolean',\n rootElement: '(element|string)'\n};\n\n/**\n * Class definition\n */\n\nclass Backdrop extends Config {\n constructor(config) {\n super();\n this._config = this._getConfig(config);\n this._isAppended = false;\n this._element = null;\n }\n\n // Getters\n static get Default() {\n return Default$8;\n }\n static get DefaultType() {\n return DefaultType$8;\n }\n static get NAME() {\n return NAME$9;\n }\n\n // Public\n show(callback) {\n if (!this._config.isVisible) {\n execute(callback);\n return;\n }\n this._append();\n const element = this._getElement();\n if (this._config.isAnimated) {\n reflow(element);\n }\n element.classList.add(CLASS_NAME_SHOW$5);\n this._emulateAnimation(() => {\n execute(callback);\n });\n }\n hide(callback) {\n if (!this._config.isVisible) {\n execute(callback);\n return;\n }\n this._getElement().classList.remove(CLASS_NAME_SHOW$5);\n this._emulateAnimation(() => {\n this.dispose();\n execute(callback);\n });\n }\n dispose() {\n if (!this._isAppended) {\n return;\n }\n EventHandler.off(this._element, EVENT_MOUSEDOWN);\n this._element.remove();\n this._isAppended = false;\n }\n\n // Private\n _getElement() {\n if (!this._element) {\n const backdrop = document.createElement('div');\n backdrop.className = this._config.className;\n if (this._config.isAnimated) {\n backdrop.classList.add(CLASS_NAME_FADE$4);\n }\n this._element = backdrop;\n }\n return this._element;\n }\n _configAfterMerge(config) {\n // use getElement() with the default \"body\" to get a fresh Element on each instantiation\n config.rootElement = getElement(config.rootElement);\n return config;\n }\n _append() {\n if (this._isAppended) {\n return;\n }\n const element = this._getElement();\n this._config.rootElement.append(element);\n EventHandler.on(element, EVENT_MOUSEDOWN, () => {\n execute(this._config.clickCallback);\n });\n this._isAppended = true;\n }\n _emulateAnimation(callback) {\n executeAfterTransition(callback, this._getElement(), this._config.isAnimated);\n }\n}\n\n/**\n * --------------------------------------------------------------------------\n * Bootstrap util/focustrap.js\n * Licensed under MIT (https://github.com/twbs/bootstrap/blob/main/LICENSE)\n * --------------------------------------------------------------------------\n */\n\n\n/**\n * Constants\n */\n\nconst NAME$8 = 'focustrap';\nconst DATA_KEY$5 = 'bs.focustrap';\nconst EVENT_KEY$5 = `.${DATA_KEY$5}`;\nconst EVENT_FOCUSIN$2 = `focusin${EVENT_KEY$5}`;\nconst EVENT_KEYDOWN_TAB = `keydown.tab${EVENT_KEY$5}`;\nconst TAB_KEY = 'Tab';\nconst TAB_NAV_FORWARD = 'forward';\nconst TAB_NAV_BACKWARD = 'backward';\nconst Default$7 = {\n autofocus: true,\n trapElement: null // The element to trap focus inside of\n};\nconst DefaultType$7 = {\n autofocus: 'boolean',\n trapElement: 'element'\n};\n\n/**\n * Class definition\n */\n\nclass FocusTrap extends Config {\n constructor(config) {\n super();\n this._config = this._getConfig(config);\n this._isActive = false;\n this._lastTabNavDirection = null;\n }\n\n // Getters\n static get Default() {\n return Default$7;\n }\n static get DefaultType() {\n return DefaultType$7;\n }\n static get NAME() {\n return NAME$8;\n }\n\n // Public\n activate() {\n if (this._isActive) {\n return;\n }\n if (this._config.autofocus) {\n this._config.trapElement.focus();\n }\n EventHandler.off(document, EVENT_KEY$5); // guard against infinite focus loop\n EventHandler.on(document, EVENT_FOCUSIN$2, event => this._handleFocusin(event));\n EventHandler.on(document, EVENT_KEYDOWN_TAB, event => this._handleKeydown(event));\n this._isActive = true;\n }\n deactivate() {\n if (!this._isActive) {\n return;\n }\n this._isActive = false;\n EventHandler.off(document, EVENT_KEY$5);\n }\n\n // Private\n _handleFocusin(event) {\n const {\n trapElement\n } = this._config;\n if (event.target === document || event.target === trapElement || trapElement.contains(event.target)) {\n return;\n }\n const elements = SelectorEngine.focusableChildren(trapElement);\n if (elements.length === 0) {\n trapElement.focus();\n } else if (this._lastTabNavDirection === TAB_NAV_BACKWARD) {\n elements[elements.length - 1].focus();\n } else {\n elements[0].focus();\n }\n }\n _handleKeydown(event) {\n if (event.key !== TAB_KEY) {\n return;\n }\n this._lastTabNavDirection = event.shiftKey ? TAB_NAV_BACKWARD : TAB_NAV_FORWARD;\n }\n}\n\n/**\n * --------------------------------------------------------------------------\n * Bootstrap util/scrollBar.js\n * Licensed under MIT (https://github.com/twbs/bootstrap/blob/main/LICENSE)\n * --------------------------------------------------------------------------\n */\n\n\n/**\n * Constants\n */\n\nconst SELECTOR_FIXED_CONTENT = '.fixed-top, .fixed-bottom, .is-fixed, .sticky-top';\nconst SELECTOR_STICKY_CONTENT = '.sticky-top';\nconst PROPERTY_PADDING = 'padding-right';\nconst PROPERTY_MARGIN = 'margin-right';\n\n/**\n * Class definition\n */\n\nclass ScrollBarHelper {\n constructor() {\n this._element = document.body;\n }\n\n // Public\n getWidth() {\n // https://developer.mozilla.org/en-US/docs/Web/API/Window/innerWidth#usage_notes\n const documentWidth = document.documentElement.clientWidth;\n return Math.abs(window.innerWidth - documentWidth);\n }\n hide() {\n const width = this.getWidth();\n this._disableOverFlow();\n // give padding to element to balance the hidden scrollbar width\n this._setElementAttributes(this._element, PROPERTY_PADDING, calculatedValue => calculatedValue + width);\n // trick: We adjust positive paddingRight and negative marginRight to sticky-top elements to keep showing fullwidth\n this._setElementAttributes(SELECTOR_FIXED_CONTENT, PROPERTY_PADDING, calculatedValue => calculatedValue + width);\n this._setElementAttributes(SELECTOR_STICKY_CONTENT, PROPERTY_MARGIN, calculatedValue => calculatedValue - width);\n }\n reset() {\n this._resetElementAttributes(this._element, 'overflow');\n this._resetElementAttributes(this._element, PROPERTY_PADDING);\n this._resetElementAttributes(SELECTOR_FIXED_CONTENT, PROPERTY_PADDING);\n this._resetElementAttributes(SELECTOR_STICKY_CONTENT, PROPERTY_MARGIN);\n }\n isOverflowing() {\n return this.getWidth() > 0;\n }\n\n // Private\n _disableOverFlow() {\n this._saveInitialAttribute(this._element, 'overflow');\n this._element.style.overflow = 'hidden';\n }\n _setElementAttributes(selector, styleProperty, callback) {\n const scrollbarWidth = this.getWidth();\n const manipulationCallBack = element => {\n if (element !== this._element && window.innerWidth > element.clientWidth + scrollbarWidth) {\n return;\n }\n this._saveInitialAttribute(element, styleProperty);\n const calculatedValue = window.getComputedStyle(element).getPropertyValue(styleProperty);\n element.style.setProperty(styleProperty, `${callback(Number.parseFloat(calculatedValue))}px`);\n };\n this._applyManipulationCallback(selector, manipulationCallBack);\n }\n _saveInitialAttribute(element, styleProperty) {\n const actualValue = element.style.getPropertyValue(styleProperty);\n if (actualValue) {\n Manipulator.setDataAttribute(element, styleProperty, actualValue);\n }\n }\n _resetElementAttributes(selector, styleProperty) {\n const manipulationCallBack = element => {\n const value = Manipulator.getDataAttribute(element, styleProperty);\n // We only want to remove the property if the value is `null`; the value can also be zero\n if (value === null) {\n element.style.removeProperty(styleProperty);\n return;\n }\n Manipulator.removeDataAttribute(element, styleProperty);\n element.style.setProperty(styleProperty, value);\n };\n this._applyManipulationCallback(selector, manipulationCallBack);\n }\n _applyManipulationCallback(selector, callBack) {\n if (isElement(selector)) {\n callBack(selector);\n return;\n }\n for (const sel of SelectorEngine.find(selector, this._element)) {\n callBack(sel);\n }\n }\n}\n\n/**\n * --------------------------------------------------------------------------\n * Bootstrap modal.js\n * Licensed under MIT (https://github.com/twbs/bootstrap/blob/main/LICENSE)\n * --------------------------------------------------------------------------\n */\n\n\n/**\n * Constants\n */\n\nconst NAME$7 = 'modal';\nconst DATA_KEY$4 = 'bs.modal';\nconst EVENT_KEY$4 = `.${DATA_KEY$4}`;\nconst DATA_API_KEY$2 = '.data-api';\nconst ESCAPE_KEY$1 = 'Escape';\nconst EVENT_HIDE$4 = `hide${EVENT_KEY$4}`;\nconst EVENT_HIDE_PREVENTED$1 = `hidePrevented${EVENT_KEY$4}`;\nconst EVENT_HIDDEN$4 = `hidden${EVENT_KEY$4}`;\nconst EVENT_SHOW$4 = `show${EVENT_KEY$4}`;\nconst EVENT_SHOWN$4 = `shown${EVENT_KEY$4}`;\nconst EVENT_RESIZE$1 = `resize${EVENT_KEY$4}`;\nconst EVENT_CLICK_DISMISS = `click.dismiss${EVENT_KEY$4}`;\nconst EVENT_MOUSEDOWN_DISMISS = `mousedown.dismiss${EVENT_KEY$4}`;\nconst EVENT_KEYDOWN_DISMISS$1 = `keydown.dismiss${EVENT_KEY$4}`;\nconst EVENT_CLICK_DATA_API$2 = `click${EVENT_KEY$4}${DATA_API_KEY$2}`;\nconst CLASS_NAME_OPEN = 'modal-open';\nconst CLASS_NAME_FADE$3 = 'fade';\nconst CLASS_NAME_SHOW$4 = 'show';\nconst CLASS_NAME_STATIC = 'modal-static';\nconst OPEN_SELECTOR$1 = '.modal.show';\nconst SELECTOR_DIALOG = '.modal-dialog';\nconst SELECTOR_MODAL_BODY = '.modal-body';\nconst SELECTOR_DATA_TOGGLE$2 = '[data-bs-toggle=\"modal\"]';\nconst Default$6 = {\n backdrop: true,\n focus: true,\n keyboard: true\n};\nconst DefaultType$6 = {\n backdrop: '(boolean|string)',\n focus: 'boolean',\n keyboard: 'boolean'\n};\n\n/**\n * Class definition\n */\n\nclass Modal extends BaseComponent {\n constructor(element, config) {\n super(element, config);\n this._dialog = SelectorEngine.findOne(SELECTOR_DIALOG, this._element);\n this._backdrop = this._initializeBackDrop();\n this._focustrap = this._initializeFocusTrap();\n this._isShown = false;\n this._isTransitioning = false;\n this._scrollBar = new ScrollBarHelper();\n this._addEventListeners();\n }\n\n // Getters\n static get Default() {\n return Default$6;\n }\n static get DefaultType() {\n return DefaultType$6;\n }\n static get NAME() {\n return NAME$7;\n }\n\n // Public\n toggle(relatedTarget) {\n return this._isShown ? this.hide() : this.show(relatedTarget);\n }\n show(relatedTarget) {\n if (this._isShown || this._isTransitioning) {\n return;\n }\n const showEvent = EventHandler.trigger(this._element, EVENT_SHOW$4, {\n relatedTarget\n });\n if (showEvent.defaultPrevented) {\n return;\n }\n this._isShown = true;\n this._isTransitioning = true;\n this._scrollBar.hide();\n document.body.classList.add(CLASS_NAME_OPEN);\n this._adjustDialog();\n this._backdrop.show(() => this._showElement(relatedTarget));\n }\n hide() {\n if (!this._isShown || this._isTransitioning) {\n return;\n }\n const hideEvent = EventHandler.trigger(this._element, EVENT_HIDE$4);\n if (hideEvent.defaultPrevented) {\n return;\n }\n this._isShown = false;\n this._isTransitioning = true;\n this._focustrap.deactivate();\n this._element.classList.remove(CLASS_NAME_SHOW$4);\n this._queueCallback(() => this._hideModal(), this._element, this._isAnimated());\n }\n dispose() {\n EventHandler.off(window, EVENT_KEY$4);\n EventHandler.off(this._dialog, EVENT_KEY$4);\n this._backdrop.dispose();\n this._focustrap.deactivate();\n super.dispose();\n }\n handleUpdate() {\n this._adjustDialog();\n }\n\n // Private\n _initializeBackDrop() {\n return new Backdrop({\n isVisible: Boolean(this._config.backdrop),\n // 'static' option will be translated to true, and booleans will keep their value,\n isAnimated: this._isAnimated()\n });\n }\n _initializeFocusTrap() {\n return new FocusTrap({\n trapElement: this._element\n });\n }\n _showElement(relatedTarget) {\n // try to append dynamic modal\n if (!document.body.contains(this._element)) {\n document.body.append(this._element);\n }\n this._element.style.display = 'block';\n this._element.removeAttribute('aria-hidden');\n this._element.setAttribute('aria-modal', true);\n this._element.setAttribute('role', 'dialog');\n this._element.scrollTop = 0;\n const modalBody = SelectorEngine.findOne(SELECTOR_MODAL_BODY, this._dialog);\n if (modalBody) {\n modalBody.scrollTop = 0;\n }\n reflow(this._element);\n this._element.classList.add(CLASS_NAME_SHOW$4);\n const transitionComplete = () => {\n if (this._config.focus) {\n this._focustrap.activate();\n }\n this._isTransitioning = false;\n EventHandler.trigger(this._element, EVENT_SHOWN$4, {\n relatedTarget\n });\n };\n this._queueCallback(transitionComplete, this._dialog, this._isAnimated());\n }\n _addEventListeners() {\n EventHandler.on(this._element, EVENT_KEYDOWN_DISMISS$1, event => {\n if (event.key !== ESCAPE_KEY$1) {\n return;\n }\n if (this._config.keyboard) {\n this.hide();\n return;\n }\n this._triggerBackdropTransition();\n });\n EventHandler.on(window, EVENT_RESIZE$1, () => {\n if (this._isShown && !this._isTransitioning) {\n this._adjustDialog();\n }\n });\n EventHandler.on(this._element, EVENT_MOUSEDOWN_DISMISS, event => {\n // a bad trick to segregate clicks that may start inside dialog but end outside, and avoid listen to scrollbar clicks\n EventHandler.one(this._element, EVENT_CLICK_DISMISS, event2 => {\n if (this._element !== event.target || this._element !== event2.target) {\n return;\n }\n if (this._config.backdrop === 'static') {\n this._triggerBackdropTransition();\n return;\n }\n if (this._config.backdrop) {\n this.hide();\n }\n });\n });\n }\n _hideModal() {\n this._element.style.display = 'none';\n this._element.setAttribute('aria-hidden', true);\n this._element.removeAttribute('aria-modal');\n this._element.removeAttribute('role');\n this._isTransitioning = false;\n this._backdrop.hide(() => {\n document.body.classList.remove(CLASS_NAME_OPEN);\n this._resetAdjustments();\n this._scrollBar.reset();\n EventHandler.trigger(this._element, EVENT_HIDDEN$4);\n });\n }\n _isAnimated() {\n return this._element.classList.contains(CLASS_NAME_FADE$3);\n }\n _triggerBackdropTransition() {\n const hideEvent = EventHandler.trigger(this._element, EVENT_HIDE_PREVENTED$1);\n if (hideEvent.defaultPrevented) {\n return;\n }\n const isModalOverflowing = this._element.scrollHeight > document.documentElement.clientHeight;\n const initialOverflowY = this._element.style.overflowY;\n // return if the following background transition hasn't yet completed\n if (initialOverflowY === 'hidden' || this._element.classList.contains(CLASS_NAME_STATIC)) {\n return;\n }\n if (!isModalOverflowing) {\n this._element.style.overflowY = 'hidden';\n }\n this._element.classList.add(CLASS_NAME_STATIC);\n this._queueCallback(() => {\n this._element.classList.remove(CLASS_NAME_STATIC);\n this._queueCallback(() => {\n this._element.style.overflowY = initialOverflowY;\n }, this._dialog);\n }, this._dialog);\n this._element.focus();\n }\n\n /**\n * The following methods are used to handle overflowing modals\n */\n\n _adjustDialog() {\n const isModalOverflowing = this._element.scrollHeight > document.documentElement.clientHeight;\n const scrollbarWidth = this._scrollBar.getWidth();\n const isBodyOverflowing = scrollbarWidth > 0;\n if (isBodyOverflowing && !isModalOverflowing) {\n const property = isRTL() ? 'paddingLeft' : 'paddingRight';\n this._element.style[property] = `${scrollbarWidth}px`;\n }\n if (!isBodyOverflowing && isModalOverflowing) {\n const property = isRTL() ? 'paddingRight' : 'paddingLeft';\n this._element.style[property] = `${scrollbarWidth}px`;\n }\n }\n _resetAdjustments() {\n this._element.style.paddingLeft = '';\n this._element.style.paddingRight = '';\n }\n\n // Static\n static jQueryInterface(config, relatedTarget) {\n return this.each(function () {\n const data = Modal.getOrCreateInstance(this, config);\n if (typeof config !== 'string') {\n return;\n }\n if (typeof data[config] === 'undefined') {\n throw new TypeError(`No method named \"${config}\"`);\n }\n data[config](relatedTarget);\n });\n }\n}\n\n/**\n * Data API implementation\n */\n\nEventHandler.on(document, EVENT_CLICK_DATA_API$2, SELECTOR_DATA_TOGGLE$2, function (event) {\n const target = SelectorEngine.getElementFromSelector(this);\n if (['A', 'AREA'].includes(this.tagName)) {\n event.preventDefault();\n }\n EventHandler.one(target, EVENT_SHOW$4, showEvent => {\n if (showEvent.defaultPrevented) {\n // only register focus restorer if modal will actually get shown\n return;\n }\n EventHandler.one(target, EVENT_HIDDEN$4, () => {\n if (isVisible(this)) {\n this.focus();\n }\n });\n });\n\n // avoid conflict when clicking modal toggler while another one is open\n const alreadyOpen = SelectorEngine.findOne(OPEN_SELECTOR$1);\n if (alreadyOpen) {\n Modal.getInstance(alreadyOpen).hide();\n }\n const data = Modal.getOrCreateInstance(target);\n data.toggle(this);\n});\nenableDismissTrigger(Modal);\n\n/**\n * jQuery\n */\n\ndefineJQueryPlugin(Modal);\n\n/**\n * --------------------------------------------------------------------------\n * Bootstrap offcanvas.js\n * Licensed under MIT (https://github.com/twbs/bootstrap/blob/main/LICENSE)\n * --------------------------------------------------------------------------\n */\n\n\n/**\n * Constants\n */\n\nconst NAME$6 = 'offcanvas';\nconst DATA_KEY$3 = 'bs.offcanvas';\nconst EVENT_KEY$3 = `.${DATA_KEY$3}`;\nconst DATA_API_KEY$1 = '.data-api';\nconst EVENT_LOAD_DATA_API$2 = `load${EVENT_KEY$3}${DATA_API_KEY$1}`;\nconst ESCAPE_KEY = 'Escape';\nconst CLASS_NAME_SHOW$3 = 'show';\nconst CLASS_NAME_SHOWING$1 = 'showing';\nconst CLASS_NAME_HIDING = 'hiding';\nconst CLASS_NAME_BACKDROP = 'offcanvas-backdrop';\nconst OPEN_SELECTOR = '.offcanvas.show';\nconst EVENT_SHOW$3 = `show${EVENT_KEY$3}`;\nconst EVENT_SHOWN$3 = `shown${EVENT_KEY$3}`;\nconst EVENT_HIDE$3 = `hide${EVENT_KEY$3}`;\nconst EVENT_HIDE_PREVENTED = `hidePrevented${EVENT_KEY$3}`;\nconst EVENT_HIDDEN$3 = `hidden${EVENT_KEY$3}`;\nconst EVENT_RESIZE = `resize${EVENT_KEY$3}`;\nconst EVENT_CLICK_DATA_API$1 = `click${EVENT_KEY$3}${DATA_API_KEY$1}`;\nconst EVENT_KEYDOWN_DISMISS = `keydown.dismiss${EVENT_KEY$3}`;\nconst SELECTOR_DATA_TOGGLE$1 = '[data-bs-toggle=\"offcanvas\"]';\nconst Default$5 = {\n backdrop: true,\n keyboard: true,\n scroll: false\n};\nconst DefaultType$5 = {\n backdrop: '(boolean|string)',\n keyboard: 'boolean',\n scroll: 'boolean'\n};\n\n/**\n * Class definition\n */\n\nclass Offcanvas extends BaseComponent {\n constructor(element, config) {\n super(element, config);\n this._isShown = false;\n this._backdrop = this._initializeBackDrop();\n this._focustrap = this._initializeFocusTrap();\n this._addEventListeners();\n }\n\n // Getters\n static get Default() {\n return Default$5;\n }\n static get DefaultType() {\n return DefaultType$5;\n }\n static get NAME() {\n return NAME$6;\n }\n\n // Public\n toggle(relatedTarget) {\n return this._isShown ? this.hide() : this.show(relatedTarget);\n }\n show(relatedTarget) {\n if (this._isShown) {\n return;\n }\n const showEvent = EventHandler.trigger(this._element, EVENT_SHOW$3, {\n relatedTarget\n });\n if (showEvent.defaultPrevented) {\n return;\n }\n this._isShown = true;\n this._backdrop.show();\n if (!this._config.scroll) {\n new ScrollBarHelper().hide();\n }\n this._element.setAttribute('aria-modal', true);\n this._element.setAttribute('role', 'dialog');\n this._element.classList.add(CLASS_NAME_SHOWING$1);\n const completeCallBack = () => {\n if (!this._config.scroll || this._config.backdrop) {\n this._focustrap.activate();\n }\n this._element.classList.add(CLASS_NAME_SHOW$3);\n this._element.classList.remove(CLASS_NAME_SHOWING$1);\n EventHandler.trigger(this._element, EVENT_SHOWN$3, {\n relatedTarget\n });\n };\n this._queueCallback(completeCallBack, this._element, true);\n }\n hide() {\n if (!this._isShown) {\n return;\n }\n const hideEvent = EventHandler.trigger(this._element, EVENT_HIDE$3);\n if (hideEvent.defaultPrevented) {\n return;\n }\n this._focustrap.deactivate();\n this._element.blur();\n this._isShown = false;\n this._element.classList.add(CLASS_NAME_HIDING);\n this._backdrop.hide();\n const completeCallback = () => {\n this._element.classList.remove(CLASS_NAME_SHOW$3, CLASS_NAME_HIDING);\n this._element.removeAttribute('aria-modal');\n this._element.removeAttribute('role');\n if (!this._config.scroll) {\n new ScrollBarHelper().reset();\n }\n EventHandler.trigger(this._element, EVENT_HIDDEN$3);\n };\n this._queueCallback(completeCallback, this._element, true);\n }\n dispose() {\n this._backdrop.dispose();\n this._focustrap.deactivate();\n super.dispose();\n }\n\n // Private\n _initializeBackDrop() {\n const clickCallback = () => {\n if (this._config.backdrop === 'static') {\n EventHandler.trigger(this._element, EVENT_HIDE_PREVENTED);\n return;\n }\n this.hide();\n };\n\n // 'static' option will be translated to true, and booleans will keep their value\n const isVisible = Boolean(this._config.backdrop);\n return new Backdrop({\n className: CLASS_NAME_BACKDROP,\n isVisible,\n isAnimated: true,\n rootElement: this._element.parentNode,\n clickCallback: isVisible ? clickCallback : null\n });\n }\n _initializeFocusTrap() {\n return new FocusTrap({\n trapElement: this._element\n });\n }\n _addEventListeners() {\n EventHandler.on(this._element, EVENT_KEYDOWN_DISMISS, event => {\n if (event.key !== ESCAPE_KEY) {\n return;\n }\n if (this._config.keyboard) {\n this.hide();\n return;\n }\n EventHandler.trigger(this._element, EVENT_HIDE_PREVENTED);\n });\n }\n\n // Static\n static jQueryInterface(config) {\n return this.each(function () {\n const data = Offcanvas.getOrCreateInstance(this, config);\n if (typeof config !== 'string') {\n return;\n }\n if (data[config] === undefined || config.startsWith('_') || config === 'constructor') {\n throw new TypeError(`No method named \"${config}\"`);\n }\n data[config](this);\n });\n }\n}\n\n/**\n * Data API implementation\n */\n\nEventHandler.on(document, EVENT_CLICK_DATA_API$1, SELECTOR_DATA_TOGGLE$1, function (event) {\n const target = SelectorEngine.getElementFromSelector(this);\n if (['A', 'AREA'].includes(this.tagName)) {\n event.preventDefault();\n }\n if (isDisabled(this)) {\n return;\n }\n EventHandler.one(target, EVENT_HIDDEN$3, () => {\n // focus on trigger when it is closed\n if (isVisible(this)) {\n this.focus();\n }\n });\n\n // avoid conflict when clicking a toggler of an offcanvas, while another is open\n const alreadyOpen = SelectorEngine.findOne(OPEN_SELECTOR);\n if (alreadyOpen && alreadyOpen !== target) {\n Offcanvas.getInstance(alreadyOpen).hide();\n }\n const data = Offcanvas.getOrCreateInstance(target);\n data.toggle(this);\n});\nEventHandler.on(window, EVENT_LOAD_DATA_API$2, () => {\n for (const selector of SelectorEngine.find(OPEN_SELECTOR)) {\n Offcanvas.getOrCreateInstance(selector).show();\n }\n});\nEventHandler.on(window, EVENT_RESIZE, () => {\n for (const element of SelectorEngine.find('[aria-modal][class*=show][class*=offcanvas-]')) {\n if (getComputedStyle(element).position !== 'fixed') {\n Offcanvas.getOrCreateInstance(element).hide();\n }\n }\n});\nenableDismissTrigger(Offcanvas);\n\n/**\n * jQuery\n */\n\ndefineJQueryPlugin(Offcanvas);\n\n/**\n * --------------------------------------------------------------------------\n * Bootstrap util/sanitizer.js\n * Licensed under MIT (https://github.com/twbs/bootstrap/blob/main/LICENSE)\n * --------------------------------------------------------------------------\n */\n\n// js-docs-start allow-list\nconst ARIA_ATTRIBUTE_PATTERN = /^aria-[\\w-]*$/i;\nconst DefaultAllowlist = {\n // Global attributes allowed on any supplied element below.\n '*': ['class', 'dir', 'id', 'lang', 'role', ARIA_ATTRIBUTE_PATTERN],\n a: ['target', 'href', 'title', 'rel'],\n area: [],\n b: [],\n br: [],\n col: [],\n code: [],\n dd: [],\n div: [],\n dl: [],\n dt: [],\n em: [],\n hr: [],\n h1: [],\n h2: [],\n h3: [],\n h4: [],\n h5: [],\n h6: [],\n i: [],\n img: ['src', 'srcset', 'alt', 'title', 'width', 'height'],\n li: [],\n ol: [],\n p: [],\n pre: [],\n s: [],\n small: [],\n span: [],\n sub: [],\n sup: [],\n strong: [],\n u: [],\n ul: []\n};\n// js-docs-end allow-list\n\nconst uriAttributes = new Set(['background', 'cite', 'href', 'itemtype', 'longdesc', 'poster', 'src', 'xlink:href']);\n\n/**\n * A pattern that recognizes URLs that are safe wrt. XSS in URL navigation\n * contexts.\n *\n * Shout-out to Angular https://github.com/angular/angular/blob/15.2.8/packages/core/src/sanitization/url_sanitizer.ts#L38\n */\n// eslint-disable-next-line unicorn/better-regex\nconst SAFE_URL_PATTERN = /^(?!javascript:)(?:[a-z0-9+.-]+:|[^&:/?#]*(?:[/?#]|$))/i;\nconst allowedAttribute = (attribute, allowedAttributeList) => {\n const attributeName = attribute.nodeName.toLowerCase();\n if (allowedAttributeList.includes(attributeName)) {\n if (uriAttributes.has(attributeName)) {\n return Boolean(SAFE_URL_PATTERN.test(attribute.nodeValue));\n }\n return true;\n }\n\n // Check if a regular expression validates the attribute.\n return allowedAttributeList.filter(attributeRegex => attributeRegex instanceof RegExp).some(regex => regex.test(attributeName));\n};\nfunction sanitizeHtml(unsafeHtml, allowList, sanitizeFunction) {\n if (!unsafeHtml.length) {\n return unsafeHtml;\n }\n if (sanitizeFunction && typeof sanitizeFunction === 'function') {\n return sanitizeFunction(unsafeHtml);\n }\n const domParser = new window.DOMParser();\n const createdDocument = domParser.parseFromString(unsafeHtml, 'text/html');\n const elements = [].concat(...createdDocument.body.querySelectorAll('*'));\n for (const element of elements) {\n const elementName = element.nodeName.toLowerCase();\n if (!Object.keys(allowList).includes(elementName)) {\n element.remove();\n continue;\n }\n const attributeList = [].concat(...element.attributes);\n const allowedAttributes = [].concat(allowList['*'] || [], allowList[elementName] || []);\n for (const attribute of attributeList) {\n if (!allowedAttribute(attribute, allowedAttributes)) {\n element.removeAttribute(attribute.nodeName);\n }\n }\n }\n return createdDocument.body.innerHTML;\n}\n\n/**\n * --------------------------------------------------------------------------\n * Bootstrap util/template-factory.js\n * Licensed under MIT (https://github.com/twbs/bootstrap/blob/main/LICENSE)\n * --------------------------------------------------------------------------\n */\n\n\n/**\n * Constants\n */\n\nconst NAME$5 = 'TemplateFactory';\nconst Default$4 = {\n allowList: DefaultAllowlist,\n content: {},\n // { selector : text , selector2 : text2 , }\n extraClass: '',\n html: false,\n sanitize: true,\n sanitizeFn: null,\n template: '
'\n};\nconst DefaultType$4 = {\n allowList: 'object',\n content: 'object',\n extraClass: '(string|function)',\n html: 'boolean',\n sanitize: 'boolean',\n sanitizeFn: '(null|function)',\n template: 'string'\n};\nconst DefaultContentType = {\n entry: '(string|element|function|null)',\n selector: '(string|element)'\n};\n\n/**\n * Class definition\n */\n\nclass TemplateFactory extends Config {\n constructor(config) {\n super();\n this._config = this._getConfig(config);\n }\n\n // Getters\n static get Default() {\n return Default$4;\n }\n static get DefaultType() {\n return DefaultType$4;\n }\n static get NAME() {\n return NAME$5;\n }\n\n // Public\n getContent() {\n return Object.values(this._config.content).map(config => this._resolvePossibleFunction(config)).filter(Boolean);\n }\n hasContent() {\n return this.getContent().length > 0;\n }\n changeContent(content) {\n this._checkContent(content);\n this._config.content = {\n ...this._config.content,\n ...content\n };\n return this;\n }\n toHtml() {\n const templateWrapper = document.createElement('div');\n templateWrapper.innerHTML = this._maybeSanitize(this._config.template);\n for (const [selector, text] of Object.entries(this._config.content)) {\n this._setContent(templateWrapper, text, selector);\n }\n const template = templateWrapper.children[0];\n const extraClass = this._resolvePossibleFunction(this._config.extraClass);\n if (extraClass) {\n template.classList.add(...extraClass.split(' '));\n }\n return template;\n }\n\n // Private\n _typeCheckConfig(config) {\n super._typeCheckConfig(config);\n this._checkContent(config.content);\n }\n _checkContent(arg) {\n for (const [selector, content] of Object.entries(arg)) {\n super._typeCheckConfig({\n selector,\n entry: content\n }, DefaultContentType);\n }\n }\n _setContent(template, content, selector) {\n const templateElement = SelectorEngine.findOne(selector, template);\n if (!templateElement) {\n return;\n }\n content = this._resolvePossibleFunction(content);\n if (!content) {\n templateElement.remove();\n return;\n }\n if (isElement(content)) {\n this._putElementInTemplate(getElement(content), templateElement);\n return;\n }\n if (this._config.html) {\n templateElement.innerHTML = this._maybeSanitize(content);\n return;\n }\n templateElement.textContent = content;\n }\n _maybeSanitize(arg) {\n return this._config.sanitize ? sanitizeHtml(arg, this._config.allowList, this._config.sanitizeFn) : arg;\n }\n _resolvePossibleFunction(arg) {\n return execute(arg, [this]);\n }\n _putElementInTemplate(element, templateElement) {\n if (this._config.html) {\n templateElement.innerHTML = '';\n templateElement.append(element);\n return;\n }\n templateElement.textContent = element.textContent;\n }\n}\n\n/**\n * --------------------------------------------------------------------------\n * Bootstrap tooltip.js\n * Licensed under MIT (https://github.com/twbs/bootstrap/blob/main/LICENSE)\n * --------------------------------------------------------------------------\n */\n\n\n/**\n * Constants\n */\n\nconst NAME$4 = 'tooltip';\nconst DISALLOWED_ATTRIBUTES = new Set(['sanitize', 'allowList', 'sanitizeFn']);\nconst CLASS_NAME_FADE$2 = 'fade';\nconst CLASS_NAME_MODAL = 'modal';\nconst CLASS_NAME_SHOW$2 = 'show';\nconst SELECTOR_TOOLTIP_INNER = '.tooltip-inner';\nconst SELECTOR_MODAL = `.${CLASS_NAME_MODAL}`;\nconst EVENT_MODAL_HIDE = 'hide.bs.modal';\nconst TRIGGER_HOVER = 'hover';\nconst TRIGGER_FOCUS = 'focus';\nconst TRIGGER_CLICK = 'click';\nconst TRIGGER_MANUAL = 'manual';\nconst EVENT_HIDE$2 = 'hide';\nconst EVENT_HIDDEN$2 = 'hidden';\nconst EVENT_SHOW$2 = 'show';\nconst EVENT_SHOWN$2 = 'shown';\nconst EVENT_INSERTED = 'inserted';\nconst EVENT_CLICK$1 = 'click';\nconst EVENT_FOCUSIN$1 = 'focusin';\nconst EVENT_FOCUSOUT$1 = 'focusout';\nconst EVENT_MOUSEENTER = 'mouseenter';\nconst EVENT_MOUSELEAVE = 'mouseleave';\nconst AttachmentMap = {\n AUTO: 'auto',\n TOP: 'top',\n RIGHT: isRTL() ? 'left' : 'right',\n BOTTOM: 'bottom',\n LEFT: isRTL() ? 'right' : 'left'\n};\nconst Default$3 = {\n allowList: DefaultAllowlist,\n animation: true,\n boundary: 'clippingParents',\n container: false,\n customClass: '',\n delay: 0,\n fallbackPlacements: ['top', 'right', 'bottom', 'left'],\n html: false,\n offset: [0, 6],\n placement: 'top',\n popperConfig: null,\n sanitize: true,\n sanitizeFn: null,\n selector: false,\n template: '
' + '
' + '
' + '
',\n title: '',\n trigger: 'hover focus'\n};\nconst DefaultType$3 = {\n allowList: 'object',\n animation: 'boolean',\n boundary: '(string|element)',\n container: '(string|element|boolean)',\n customClass: '(string|function)',\n delay: '(number|object)',\n fallbackPlacements: 'array',\n html: 'boolean',\n offset: '(array|string|function)',\n placement: '(string|function)',\n popperConfig: '(null|object|function)',\n sanitize: 'boolean',\n sanitizeFn: '(null|function)',\n selector: '(string|boolean)',\n template: 'string',\n title: '(string|element|function)',\n trigger: 'string'\n};\n\n/**\n * Class definition\n */\n\nclass Tooltip extends BaseComponent {\n constructor(element, config) {\n if (typeof Popper === 'undefined') {\n throw new TypeError('Bootstrap\\'s tooltips require Popper (https://popper.js.org)');\n }\n super(element, config);\n\n // Private\n this._isEnabled = true;\n this._timeout = 0;\n this._isHovered = null;\n this._activeTrigger = {};\n this._popper = null;\n this._templateFactory = null;\n this._newContent = null;\n\n // Protected\n this.tip = null;\n this._setListeners();\n if (!this._config.selector) {\n this._fixTitle();\n }\n }\n\n // Getters\n static get Default() {\n return Default$3;\n }\n static get DefaultType() {\n return DefaultType$3;\n }\n static get NAME() {\n return NAME$4;\n }\n\n // Public\n enable() {\n this._isEnabled = true;\n }\n disable() {\n this._isEnabled = false;\n }\n toggleEnabled() {\n this._isEnabled = !this._isEnabled;\n }\n toggle() {\n if (!this._isEnabled) {\n return;\n }\n this._activeTrigger.click = !this._activeTrigger.click;\n if (this._isShown()) {\n this._leave();\n return;\n }\n this._enter();\n }\n dispose() {\n clearTimeout(this._timeout);\n EventHandler.off(this._element.closest(SELECTOR_MODAL), EVENT_MODAL_HIDE, this._hideModalHandler);\n if (this._element.getAttribute('data-bs-original-title')) {\n this._element.setAttribute('title', this._element.getAttribute('data-bs-original-title'));\n }\n this._disposePopper();\n super.dispose();\n }\n show() {\n if (this._element.style.display === 'none') {\n throw new Error('Please use show on visible elements');\n }\n if (!(this._isWithContent() && this._isEnabled)) {\n return;\n }\n const showEvent = EventHandler.trigger(this._element, this.constructor.eventName(EVENT_SHOW$2));\n const shadowRoot = findShadowRoot(this._element);\n const isInTheDom = (shadowRoot || this._element.ownerDocument.documentElement).contains(this._element);\n if (showEvent.defaultPrevented || !isInTheDom) {\n return;\n }\n\n // TODO: v6 remove this or make it optional\n this._disposePopper();\n const tip = this._getTipElement();\n this._element.setAttribute('aria-describedby', tip.getAttribute('id'));\n const {\n container\n } = this._config;\n if (!this._element.ownerDocument.documentElement.contains(this.tip)) {\n container.append(tip);\n EventHandler.trigger(this._element, this.constructor.eventName(EVENT_INSERTED));\n }\n this._popper = this._createPopper(tip);\n tip.classList.add(CLASS_NAME_SHOW$2);\n\n // If this is a touch-enabled device we add extra\n // empty mouseover listeners to the body's immediate children;\n // only needed because of broken event delegation on iOS\n // https://www.quirksmode.org/blog/archives/2014/02/mouse_event_bub.html\n if ('ontouchstart' in document.documentElement) {\n for (const element of [].concat(...document.body.children)) {\n EventHandler.on(element, 'mouseover', noop);\n }\n }\n const complete = () => {\n EventHandler.trigger(this._element, this.constructor.eventName(EVENT_SHOWN$2));\n if (this._isHovered === false) {\n this._leave();\n }\n this._isHovered = false;\n };\n this._queueCallback(complete, this.tip, this._isAnimated());\n }\n hide() {\n if (!this._isShown()) {\n return;\n }\n const hideEvent = EventHandler.trigger(this._element, this.constructor.eventName(EVENT_HIDE$2));\n if (hideEvent.defaultPrevented) {\n return;\n }\n const tip = this._getTipElement();\n tip.classList.remove(CLASS_NAME_SHOW$2);\n\n // If this is a touch-enabled device we remove the extra\n // empty mouseover listeners we added for iOS support\n if ('ontouchstart' in document.documentElement) {\n for (const element of [].concat(...document.body.children)) {\n EventHandler.off(element, 'mouseover', noop);\n }\n }\n this._activeTrigger[TRIGGER_CLICK] = false;\n this._activeTrigger[TRIGGER_FOCUS] = false;\n this._activeTrigger[TRIGGER_HOVER] = false;\n this._isHovered = null; // it is a trick to support manual triggering\n\n const complete = () => {\n if (this._isWithActiveTrigger()) {\n return;\n }\n if (!this._isHovered) {\n this._disposePopper();\n }\n this._element.removeAttribute('aria-describedby');\n EventHandler.trigger(this._element, this.constructor.eventName(EVENT_HIDDEN$2));\n };\n this._queueCallback(complete, this.tip, this._isAnimated());\n }\n update() {\n if (this._popper) {\n this._popper.update();\n }\n }\n\n // Protected\n _isWithContent() {\n return Boolean(this._getTitle());\n }\n _getTipElement() {\n if (!this.tip) {\n this.tip = this._createTipElement(this._newContent || this._getContentForTemplate());\n }\n return this.tip;\n }\n _createTipElement(content) {\n const tip = this._getTemplateFactory(content).toHtml();\n\n // TODO: remove this check in v6\n if (!tip) {\n return null;\n }\n tip.classList.remove(CLASS_NAME_FADE$2, CLASS_NAME_SHOW$2);\n // TODO: v6 the following can be achieved with CSS only\n tip.classList.add(`bs-${this.constructor.NAME}-auto`);\n const tipId = getUID(this.constructor.NAME).toString();\n tip.setAttribute('id', tipId);\n if (this._isAnimated()) {\n tip.classList.add(CLASS_NAME_FADE$2);\n }\n return tip;\n }\n setContent(content) {\n this._newContent = content;\n if (this._isShown()) {\n this._disposePopper();\n this.show();\n }\n }\n _getTemplateFactory(content) {\n if (this._templateFactory) {\n this._templateFactory.changeContent(content);\n } else {\n this._templateFactory = new TemplateFactory({\n ...this._config,\n // the `content` var has to be after `this._config`\n // to override config.content in case of popover\n content,\n extraClass: this._resolvePossibleFunction(this._config.customClass)\n });\n }\n return this._templateFactory;\n }\n _getContentForTemplate() {\n return {\n [SELECTOR_TOOLTIP_INNER]: this._getTitle()\n };\n }\n _getTitle() {\n return this._resolvePossibleFunction(this._config.title) || this._element.getAttribute('data-bs-original-title');\n }\n\n // Private\n _initializeOnDelegatedTarget(event) {\n return this.constructor.getOrCreateInstance(event.delegateTarget, this._getDelegateConfig());\n }\n _isAnimated() {\n return this._config.animation || this.tip && this.tip.classList.contains(CLASS_NAME_FADE$2);\n }\n _isShown() {\n return this.tip && this.tip.classList.contains(CLASS_NAME_SHOW$2);\n }\n _createPopper(tip) {\n const placement = execute(this._config.placement, [this, tip, this._element]);\n const attachment = AttachmentMap[placement.toUpperCase()];\n return Popper.createPopper(this._element, tip, this._getPopperConfig(attachment));\n }\n _getOffset() {\n const {\n offset\n } = this._config;\n if (typeof offset === 'string') {\n return offset.split(',').map(value => Number.parseInt(value, 10));\n }\n if (typeof offset === 'function') {\n return popperData => offset(popperData, this._element);\n }\n return offset;\n }\n _resolvePossibleFunction(arg) {\n return execute(arg, [this._element]);\n }\n _getPopperConfig(attachment) {\n const defaultBsPopperConfig = {\n placement: attachment,\n modifiers: [{\n name: 'flip',\n options: {\n fallbackPlacements: this._config.fallbackPlacements\n }\n }, {\n name: 'offset',\n options: {\n offset: this._getOffset()\n }\n }, {\n name: 'preventOverflow',\n options: {\n boundary: this._config.boundary\n }\n }, {\n name: 'arrow',\n options: {\n element: `.${this.constructor.NAME}-arrow`\n }\n }, {\n name: 'preSetPlacement',\n enabled: true,\n phase: 'beforeMain',\n fn: data => {\n // Pre-set Popper's placement attribute in order to read the arrow sizes properly.\n // Otherwise, Popper mixes up the width and height dimensions since the initial arrow style is for top placement\n this._getTipElement().setAttribute('data-popper-placement', data.state.placement);\n }\n }]\n };\n return {\n ...defaultBsPopperConfig,\n ...execute(this._config.popperConfig, [defaultBsPopperConfig])\n };\n }\n _setListeners() {\n const triggers = this._config.trigger.split(' ');\n for (const trigger of triggers) {\n if (trigger === 'click') {\n EventHandler.on(this._element, this.constructor.eventName(EVENT_CLICK$1), this._config.selector, event => {\n const context = this._initializeOnDelegatedTarget(event);\n context.toggle();\n });\n } else if (trigger !== TRIGGER_MANUAL) {\n const eventIn = trigger === TRIGGER_HOVER ? this.constructor.eventName(EVENT_MOUSEENTER) : this.constructor.eventName(EVENT_FOCUSIN$1);\n const eventOut = trigger === TRIGGER_HOVER ? this.constructor.eventName(EVENT_MOUSELEAVE) : this.constructor.eventName(EVENT_FOCUSOUT$1);\n EventHandler.on(this._element, eventIn, this._config.selector, event => {\n const context = this._initializeOnDelegatedTarget(event);\n context._activeTrigger[event.type === 'focusin' ? TRIGGER_FOCUS : TRIGGER_HOVER] = true;\n context._enter();\n });\n EventHandler.on(this._element, eventOut, this._config.selector, event => {\n const context = this._initializeOnDelegatedTarget(event);\n context._activeTrigger[event.type === 'focusout' ? TRIGGER_FOCUS : TRIGGER_HOVER] = context._element.contains(event.relatedTarget);\n context._leave();\n });\n }\n }\n this._hideModalHandler = () => {\n if (this._element) {\n this.hide();\n }\n };\n EventHandler.on(this._element.closest(SELECTOR_MODAL), EVENT_MODAL_HIDE, this._hideModalHandler);\n }\n _fixTitle() {\n const title = this._element.getAttribute('title');\n if (!title) {\n return;\n }\n if (!this._element.getAttribute('aria-label') && !this._element.textContent.trim()) {\n this._element.setAttribute('aria-label', title);\n }\n this._element.setAttribute('data-bs-original-title', title); // DO NOT USE IT. Is only for backwards compatibility\n this._element.removeAttribute('title');\n }\n _enter() {\n if (this._isShown() || this._isHovered) {\n this._isHovered = true;\n return;\n }\n this._isHovered = true;\n this._setTimeout(() => {\n if (this._isHovered) {\n this.show();\n }\n }, this._config.delay.show);\n }\n _leave() {\n if (this._isWithActiveTrigger()) {\n return;\n }\n this._isHovered = false;\n this._setTimeout(() => {\n if (!this._isHovered) {\n this.hide();\n }\n }, this._config.delay.hide);\n }\n _setTimeout(handler, timeout) {\n clearTimeout(this._timeout);\n this._timeout = setTimeout(handler, timeout);\n }\n _isWithActiveTrigger() {\n return Object.values(this._activeTrigger).includes(true);\n }\n _getConfig(config) {\n const dataAttributes = Manipulator.getDataAttributes(this._element);\n for (const dataAttribute of Object.keys(dataAttributes)) {\n if (DISALLOWED_ATTRIBUTES.has(dataAttribute)) {\n delete dataAttributes[dataAttribute];\n }\n }\n config = {\n ...dataAttributes,\n ...(typeof config === 'object' && config ? config : {})\n };\n config = this._mergeConfigObj(config);\n config = this._configAfterMerge(config);\n this._typeCheckConfig(config);\n return config;\n }\n _configAfterMerge(config) {\n config.container = config.container === false ? document.body : getElement(config.container);\n if (typeof config.delay === 'number') {\n config.delay = {\n show: config.delay,\n hide: config.delay\n };\n }\n if (typeof config.title === 'number') {\n config.title = config.title.toString();\n }\n if (typeof config.content === 'number') {\n config.content = config.content.toString();\n }\n return config;\n }\n _getDelegateConfig() {\n const config = {};\n for (const [key, value] of Object.entries(this._config)) {\n if (this.constructor.Default[key] !== value) {\n config[key] = value;\n }\n }\n config.selector = false;\n config.trigger = 'manual';\n\n // In the future can be replaced with:\n // const keysWithDifferentValues = Object.entries(this._config).filter(entry => this.constructor.Default[entry[0]] !== this._config[entry[0]])\n // `Object.fromEntries(keysWithDifferentValues)`\n return config;\n }\n _disposePopper() {\n if (this._popper) {\n this._popper.destroy();\n this._popper = null;\n }\n if (this.tip) {\n this.tip.remove();\n this.tip = null;\n }\n }\n\n // Static\n static jQueryInterface(config) {\n return this.each(function () {\n const data = Tooltip.getOrCreateInstance(this, config);\n if (typeof config !== 'string') {\n return;\n }\n if (typeof data[config] === 'undefined') {\n throw new TypeError(`No method named \"${config}\"`);\n }\n data[config]();\n });\n }\n}\n\n/**\n * jQuery\n */\n\ndefineJQueryPlugin(Tooltip);\n\n/**\n * --------------------------------------------------------------------------\n * Bootstrap popover.js\n * Licensed under MIT (https://github.com/twbs/bootstrap/blob/main/LICENSE)\n * --------------------------------------------------------------------------\n */\n\n\n/**\n * Constants\n */\n\nconst NAME$3 = 'popover';\nconst SELECTOR_TITLE = '.popover-header';\nconst SELECTOR_CONTENT = '.popover-body';\nconst Default$2 = {\n ...Tooltip.Default,\n content: '',\n offset: [0, 8],\n placement: 'right',\n template: '
' + '
' + '

' + '
' + '
',\n trigger: 'click'\n};\nconst DefaultType$2 = {\n ...Tooltip.DefaultType,\n content: '(null|string|element|function)'\n};\n\n/**\n * Class definition\n */\n\nclass Popover extends Tooltip {\n // Getters\n static get Default() {\n return Default$2;\n }\n static get DefaultType() {\n return DefaultType$2;\n }\n static get NAME() {\n return NAME$3;\n }\n\n // Overrides\n _isWithContent() {\n return this._getTitle() || this._getContent();\n }\n\n // Private\n _getContentForTemplate() {\n return {\n [SELECTOR_TITLE]: this._getTitle(),\n [SELECTOR_CONTENT]: this._getContent()\n };\n }\n _getContent() {\n return this._resolvePossibleFunction(this._config.content);\n }\n\n // Static\n static jQueryInterface(config) {\n return this.each(function () {\n const data = Popover.getOrCreateInstance(this, config);\n if (typeof config !== 'string') {\n return;\n }\n if (typeof data[config] === 'undefined') {\n throw new TypeError(`No method named \"${config}\"`);\n }\n data[config]();\n });\n }\n}\n\n/**\n * jQuery\n */\n\ndefineJQueryPlugin(Popover);\n\n/**\n * --------------------------------------------------------------------------\n * Bootstrap scrollspy.js\n * Licensed under MIT (https://github.com/twbs/bootstrap/blob/main/LICENSE)\n * --------------------------------------------------------------------------\n */\n\n\n/**\n * Constants\n */\n\nconst NAME$2 = 'scrollspy';\nconst DATA_KEY$2 = 'bs.scrollspy';\nconst EVENT_KEY$2 = `.${DATA_KEY$2}`;\nconst DATA_API_KEY = '.data-api';\nconst EVENT_ACTIVATE = `activate${EVENT_KEY$2}`;\nconst EVENT_CLICK = `click${EVENT_KEY$2}`;\nconst EVENT_LOAD_DATA_API$1 = `load${EVENT_KEY$2}${DATA_API_KEY}`;\nconst CLASS_NAME_DROPDOWN_ITEM = 'dropdown-item';\nconst CLASS_NAME_ACTIVE$1 = 'active';\nconst SELECTOR_DATA_SPY = '[data-bs-spy=\"scroll\"]';\nconst SELECTOR_TARGET_LINKS = '[href]';\nconst SELECTOR_NAV_LIST_GROUP = '.nav, .list-group';\nconst SELECTOR_NAV_LINKS = '.nav-link';\nconst SELECTOR_NAV_ITEMS = '.nav-item';\nconst SELECTOR_LIST_ITEMS = '.list-group-item';\nconst SELECTOR_LINK_ITEMS = `${SELECTOR_NAV_LINKS}, ${SELECTOR_NAV_ITEMS} > ${SELECTOR_NAV_LINKS}, ${SELECTOR_LIST_ITEMS}`;\nconst SELECTOR_DROPDOWN = '.dropdown';\nconst SELECTOR_DROPDOWN_TOGGLE$1 = '.dropdown-toggle';\nconst Default$1 = {\n offset: null,\n // TODO: v6 @deprecated, keep it for backwards compatibility reasons\n rootMargin: '0px 0px -25%',\n smoothScroll: false,\n target: null,\n threshold: [0.1, 0.5, 1]\n};\nconst DefaultType$1 = {\n offset: '(number|null)',\n // TODO v6 @deprecated, keep it for backwards compatibility reasons\n rootMargin: 'string',\n smoothScroll: 'boolean',\n target: 'element',\n threshold: 'array'\n};\n\n/**\n * Class definition\n */\n\nclass ScrollSpy extends BaseComponent {\n constructor(element, config) {\n super(element, config);\n\n // this._element is the observablesContainer and config.target the menu links wrapper\n this._targetLinks = new Map();\n this._observableSections = new Map();\n this._rootElement = getComputedStyle(this._element).overflowY === 'visible' ? null : this._element;\n this._activeTarget = null;\n this._observer = null;\n this._previousScrollData = {\n visibleEntryTop: 0,\n parentScrollTop: 0\n };\n this.refresh(); // initialize\n }\n\n // Getters\n static get Default() {\n return Default$1;\n }\n static get DefaultType() {\n return DefaultType$1;\n }\n static get NAME() {\n return NAME$2;\n }\n\n // Public\n refresh() {\n this._initializeTargetsAndObservables();\n this._maybeEnableSmoothScroll();\n if (this._observer) {\n this._observer.disconnect();\n } else {\n this._observer = this._getNewObserver();\n }\n for (const section of this._observableSections.values()) {\n this._observer.observe(section);\n }\n }\n dispose() {\n this._observer.disconnect();\n super.dispose();\n }\n\n // Private\n _configAfterMerge(config) {\n // TODO: on v6 target should be given explicitly & remove the {target: 'ss-target'} case\n config.target = getElement(config.target) || document.body;\n\n // TODO: v6 Only for backwards compatibility reasons. Use rootMargin only\n config.rootMargin = config.offset ? `${config.offset}px 0px -30%` : config.rootMargin;\n if (typeof config.threshold === 'string') {\n config.threshold = config.threshold.split(',').map(value => Number.parseFloat(value));\n }\n return config;\n }\n _maybeEnableSmoothScroll() {\n if (!this._config.smoothScroll) {\n return;\n }\n\n // unregister any previous listeners\n EventHandler.off(this._config.target, EVENT_CLICK);\n EventHandler.on(this._config.target, EVENT_CLICK, SELECTOR_TARGET_LINKS, event => {\n const observableSection = this._observableSections.get(event.target.hash);\n if (observableSection) {\n event.preventDefault();\n const root = this._rootElement || window;\n const height = observableSection.offsetTop - this._element.offsetTop;\n if (root.scrollTo) {\n root.scrollTo({\n top: height,\n behavior: 'smooth'\n });\n return;\n }\n\n // Chrome 60 doesn't support `scrollTo`\n root.scrollTop = height;\n }\n });\n }\n _getNewObserver() {\n const options = {\n root: this._rootElement,\n threshold: this._config.threshold,\n rootMargin: this._config.rootMargin\n };\n return new IntersectionObserver(entries => this._observerCallback(entries), options);\n }\n\n // The logic of selection\n _observerCallback(entries) {\n const targetElement = entry => this._targetLinks.get(`#${entry.target.id}`);\n const activate = entry => {\n this._previousScrollData.visibleEntryTop = entry.target.offsetTop;\n this._process(targetElement(entry));\n };\n const parentScrollTop = (this._rootElement || document.documentElement).scrollTop;\n const userScrollsDown = parentScrollTop >= this._previousScrollData.parentScrollTop;\n this._previousScrollData.parentScrollTop = parentScrollTop;\n for (const entry of entries) {\n if (!entry.isIntersecting) {\n this._activeTarget = null;\n this._clearActiveClass(targetElement(entry));\n continue;\n }\n const entryIsLowerThanPrevious = entry.target.offsetTop >= this._previousScrollData.visibleEntryTop;\n // if we are scrolling down, pick the bigger offsetTop\n if (userScrollsDown && entryIsLowerThanPrevious) {\n activate(entry);\n // if parent isn't scrolled, let's keep the first visible item, breaking the iteration\n if (!parentScrollTop) {\n return;\n }\n continue;\n }\n\n // if we are scrolling up, pick the smallest offsetTop\n if (!userScrollsDown && !entryIsLowerThanPrevious) {\n activate(entry);\n }\n }\n }\n _initializeTargetsAndObservables() {\n this._targetLinks = new Map();\n this._observableSections = new Map();\n const targetLinks = SelectorEngine.find(SELECTOR_TARGET_LINKS, this._config.target);\n for (const anchor of targetLinks) {\n // ensure that the anchor has an id and is not disabled\n if (!anchor.hash || isDisabled(anchor)) {\n continue;\n }\n const observableSection = SelectorEngine.findOne(decodeURI(anchor.hash), this._element);\n\n // ensure that the observableSection exists & is visible\n if (isVisible(observableSection)) {\n this._targetLinks.set(decodeURI(anchor.hash), anchor);\n this._observableSections.set(anchor.hash, observableSection);\n }\n }\n }\n _process(target) {\n if (this._activeTarget === target) {\n return;\n }\n this._clearActiveClass(this._config.target);\n this._activeTarget = target;\n target.classList.add(CLASS_NAME_ACTIVE$1);\n this._activateParents(target);\n EventHandler.trigger(this._element, EVENT_ACTIVATE, {\n relatedTarget: target\n });\n }\n _activateParents(target) {\n // Activate dropdown parents\n if (target.classList.contains(CLASS_NAME_DROPDOWN_ITEM)) {\n SelectorEngine.findOne(SELECTOR_DROPDOWN_TOGGLE$1, target.closest(SELECTOR_DROPDOWN)).classList.add(CLASS_NAME_ACTIVE$1);\n return;\n }\n for (const listGroup of SelectorEngine.parents(target, SELECTOR_NAV_LIST_GROUP)) {\n // Set triggered links parents as active\n // With both