diff --git a/.buildinfo b/.buildinfo new file mode 100644 index 000000000..0ec16831a --- /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: 6b460b237c9c882bfec7031d96dea6cf +tags: 645f666f9bcd5a90fca523b33c5a78b7 diff --git a/.nojekyll b/.nojekyll new file mode 100644 index 000000000..e69de29bb diff --git a/_images/2021-05-09_10-32.jpg b/_images/2021-05-09_10-32.jpg new file mode 100644 index 000000000..d16a02951 Binary files /dev/null and b/_images/2021-05-09_10-32.jpg differ diff --git a/_images/2021-05-09_10-36.jpg b/_images/2021-05-09_10-36.jpg new file mode 100644 index 000000000..db2876cb3 Binary files /dev/null and b/_images/2021-05-09_10-36.jpg differ diff --git a/_images/2021-05-09_10-50.jpg b/_images/2021-05-09_10-50.jpg new file mode 100644 index 000000000..cd12ca5d9 Binary files /dev/null and b/_images/2021-05-09_10-50.jpg differ diff --git a/_images/2021-05-09_17-36.jpg b/_images/2021-05-09_17-36.jpg new file mode 100644 index 000000000..b5581088b Binary files /dev/null and b/_images/2021-05-09_17-36.jpg differ diff --git a/_images/2021-05-09_17-38.jpg b/_images/2021-05-09_17-38.jpg new file mode 100644 index 000000000..60c056e85 Binary files /dev/null and b/_images/2021-05-09_17-38.jpg differ diff --git a/_images/2021-05-09_17-44.jpg b/_images/2021-05-09_17-44.jpg new file mode 100644 index 000000000..83b414d0f Binary files /dev/null and b/_images/2021-05-09_17-44.jpg differ diff --git a/_images/2021-05-09_17-57.jpg b/_images/2021-05-09_17-57.jpg new file mode 100644 index 000000000..b9dd0947e Binary files /dev/null and b/_images/2021-05-09_17-57.jpg differ diff --git a/_images/2021-05-09_21-21.jpg b/_images/2021-05-09_21-21.jpg new file mode 100644 index 000000000..1b27de5f0 Binary files /dev/null and b/_images/2021-05-09_21-21.jpg differ diff --git a/_images/2021-05-09_21-46.jpg b/_images/2021-05-09_21-46.jpg new file mode 100644 index 000000000..903bd8e92 Binary files /dev/null and b/_images/2021-05-09_21-46.jpg differ diff --git a/_images/2021-05-09_22-09.jpg b/_images/2021-05-09_22-09.jpg new file mode 100644 index 000000000..5b3a1fcd2 Binary files /dev/null and b/_images/2021-05-09_22-09.jpg differ diff --git a/_images/2021-05-09_22-39.jpg b/_images/2021-05-09_22-39.jpg new file mode 100644 index 000000000..d02b6668f Binary files /dev/null and b/_images/2021-05-09_22-39.jpg differ diff --git a/_images/2021-05-10_22-35.jpg b/_images/2021-05-10_22-35.jpg new file mode 100644 index 000000000..9c3650585 Binary files /dev/null and b/_images/2021-05-10_22-35.jpg differ diff --git a/_images/2021-05-10_22-59.jpg b/_images/2021-05-10_22-59.jpg new file mode 100644 index 000000000..ca184bc6c Binary files /dev/null and b/_images/2021-05-10_22-59.jpg differ diff --git a/_images/2021-05-12_18-18.jpg b/_images/2021-05-12_18-18.jpg new file mode 100644 index 000000000..5ee9dac5a Binary files /dev/null and b/_images/2021-05-12_18-18.jpg differ diff --git a/_images/2021-05-12_18-51.jpg b/_images/2021-05-12_18-51.jpg new file mode 100644 index 000000000..afb248858 Binary files /dev/null and b/_images/2021-05-12_18-51.jpg differ diff --git a/_images/2021-05-12_19-12.jpg b/_images/2021-05-12_19-12.jpg new file mode 100644 index 000000000..cda80cf9d Binary files /dev/null and b/_images/2021-05-12_19-12.jpg differ diff --git a/_images/2021-05-12_20-06.jpg b/_images/2021-05-12_20-06.jpg new file mode 100644 index 000000000..b34855edf Binary files /dev/null and b/_images/2021-05-12_20-06.jpg differ diff --git a/_images/2021-05-12_20-10.jpg b/_images/2021-05-12_20-10.jpg new file mode 100644 index 000000000..04d6402f1 Binary files /dev/null and b/_images/2021-05-12_20-10.jpg differ diff --git a/_images/2021-05-12_21-03.jpg b/_images/2021-05-12_21-03.jpg new file mode 100644 index 000000000..384e2ba0d Binary files /dev/null and b/_images/2021-05-12_21-03.jpg differ diff --git a/_images/2021-05-13_20-46.jpg b/_images/2021-05-13_20-46.jpg new file mode 100644 index 000000000..d3448b877 Binary files /dev/null and b/_images/2021-05-13_20-46.jpg differ diff --git a/_images/2021-05-17_11-50.jpg b/_images/2021-05-17_11-50.jpg new file mode 100644 index 000000000..3fb35b9c0 Binary files /dev/null and b/_images/2021-05-17_11-50.jpg differ diff --git a/_images/crosslayer_intro.jpg b/_images/crosslayer_intro.jpg new file mode 100644 index 000000000..41349a91a Binary files /dev/null and b/_images/crosslayer_intro.jpg differ diff --git a/_images/examples_01_Introduction_20_0.png b/_images/examples_01_Introduction_20_0.png new file mode 100644 index 000000000..bc5a73d9e Binary files /dev/null and b/_images/examples_01_Introduction_20_0.png differ diff --git a/_images/examples_02_Intro_to_Physical_Architecture_API_5_0.png b/_images/examples_02_Intro_to_Physical_Architecture_API_5_0.png new file mode 100644 index 000000000..1af694078 Binary files /dev/null and b/_images/examples_02_Intro_to_Physical_Architecture_API_5_0.png differ diff --git a/_images/examples_05_Introduction_to_Libraries_9_1.png b/_images/examples_05_Introduction_to_Libraries_9_1.png new file mode 100644 index 000000000..7c4f5bbfd Binary files /dev/null and b/_images/examples_05_Introduction_to_Libraries_9_1.png differ diff --git a/_images/examples_07_Code_Generation_4_1.png b/_images/examples_07_Code_Generation_4_1.png new file mode 100644 index 000000000..a71101b3b Binary files /dev/null and b/_images/examples_07_Code_Generation_4_1.png differ diff --git a/_images/examples_09_Context_Diagrams_11_0.png b/_images/examples_09_Context_Diagrams_11_0.png new file mode 100644 index 000000000..06f236a1a Binary files /dev/null and b/_images/examples_09_Context_Diagrams_11_0.png differ diff --git a/_images/examples_09_Context_Diagrams_13_1.png b/_images/examples_09_Context_Diagrams_13_1.png new file mode 100644 index 000000000..1ee589c1a Binary files /dev/null and b/_images/examples_09_Context_Diagrams_13_1.png differ diff --git a/_images/examples_09_Context_Diagrams_13_11.png b/_images/examples_09_Context_Diagrams_13_11.png new file mode 100644 index 000000000..4f5284027 Binary files /dev/null and b/_images/examples_09_Context_Diagrams_13_11.png differ diff --git a/_images/examples_09_Context_Diagrams_13_13.png b/_images/examples_09_Context_Diagrams_13_13.png new file mode 100644 index 000000000..aa4ea2d58 Binary files /dev/null and b/_images/examples_09_Context_Diagrams_13_13.png differ diff --git a/_images/examples_09_Context_Diagrams_13_15.png b/_images/examples_09_Context_Diagrams_13_15.png new file mode 100644 index 000000000..6df2d2917 Binary files /dev/null and b/_images/examples_09_Context_Diagrams_13_15.png differ diff --git a/_images/examples_09_Context_Diagrams_13_17.png b/_images/examples_09_Context_Diagrams_13_17.png new file mode 100644 index 000000000..a3872594a Binary files /dev/null and b/_images/examples_09_Context_Diagrams_13_17.png differ diff --git a/_images/examples_09_Context_Diagrams_13_19.png b/_images/examples_09_Context_Diagrams_13_19.png new file mode 100644 index 000000000..54c6ac4be Binary files /dev/null and b/_images/examples_09_Context_Diagrams_13_19.png differ diff --git a/_images/examples_09_Context_Diagrams_13_21.png b/_images/examples_09_Context_Diagrams_13_21.png new file mode 100644 index 000000000..1cdedf9eb Binary files /dev/null and b/_images/examples_09_Context_Diagrams_13_21.png differ diff --git a/_images/examples_09_Context_Diagrams_13_23.png b/_images/examples_09_Context_Diagrams_13_23.png new file mode 100644 index 000000000..b159da47d Binary files /dev/null and b/_images/examples_09_Context_Diagrams_13_23.png differ diff --git a/_images/examples_09_Context_Diagrams_13_25.png b/_images/examples_09_Context_Diagrams_13_25.png new file mode 100644 index 000000000..7b5532208 Binary files /dev/null and b/_images/examples_09_Context_Diagrams_13_25.png differ diff --git a/_images/examples_09_Context_Diagrams_13_27.png b/_images/examples_09_Context_Diagrams_13_27.png new file mode 100644 index 000000000..91783e099 Binary files /dev/null and b/_images/examples_09_Context_Diagrams_13_27.png differ diff --git a/_images/examples_09_Context_Diagrams_13_29.png b/_images/examples_09_Context_Diagrams_13_29.png new file mode 100644 index 000000000..1e7bcae77 Binary files /dev/null and b/_images/examples_09_Context_Diagrams_13_29.png differ diff --git a/_images/examples_09_Context_Diagrams_13_3.png b/_images/examples_09_Context_Diagrams_13_3.png new file mode 100644 index 000000000..ed84f840c Binary files /dev/null and b/_images/examples_09_Context_Diagrams_13_3.png differ diff --git a/_images/examples_09_Context_Diagrams_13_5.png b/_images/examples_09_Context_Diagrams_13_5.png new file mode 100644 index 000000000..8794aec3d Binary files /dev/null and b/_images/examples_09_Context_Diagrams_13_5.png differ diff --git a/_images/examples_09_Context_Diagrams_13_7.png b/_images/examples_09_Context_Diagrams_13_7.png new file mode 100644 index 000000000..194b0adb0 Binary files /dev/null and b/_images/examples_09_Context_Diagrams_13_7.png differ diff --git a/_images/examples_09_Context_Diagrams_13_9.png b/_images/examples_09_Context_Diagrams_13_9.png new file mode 100644 index 000000000..b39924f90 Binary files /dev/null and b/_images/examples_09_Context_Diagrams_13_9.png differ diff --git a/_images/examples_09_Context_Diagrams_15_1.png b/_images/examples_09_Context_Diagrams_15_1.png new file mode 100644 index 000000000..51b50d001 Binary files /dev/null and b/_images/examples_09_Context_Diagrams_15_1.png differ diff --git a/_images/examples_09_Context_Diagrams_15_11.png b/_images/examples_09_Context_Diagrams_15_11.png new file mode 100644 index 000000000..d02a330b9 Binary files /dev/null and b/_images/examples_09_Context_Diagrams_15_11.png differ diff --git a/_images/examples_09_Context_Diagrams_15_3.png b/_images/examples_09_Context_Diagrams_15_3.png new file mode 100644 index 000000000..865475c31 Binary files /dev/null and b/_images/examples_09_Context_Diagrams_15_3.png differ diff --git a/_images/examples_09_Context_Diagrams_15_5.png b/_images/examples_09_Context_Diagrams_15_5.png new file mode 100644 index 000000000..240be14fc Binary files /dev/null and b/_images/examples_09_Context_Diagrams_15_5.png differ diff --git a/_images/examples_09_Context_Diagrams_15_7.png b/_images/examples_09_Context_Diagrams_15_7.png new file mode 100644 index 000000000..d7ba6baf3 Binary files /dev/null and b/_images/examples_09_Context_Diagrams_15_7.png differ diff --git a/_images/examples_09_Context_Diagrams_15_9.png b/_images/examples_09_Context_Diagrams_15_9.png new file mode 100644 index 000000000..a54d463e2 Binary files /dev/null and b/_images/examples_09_Context_Diagrams_15_9.png differ diff --git a/_images/examples_09_Context_Diagrams_6_0.png b/_images/examples_09_Context_Diagrams_6_0.png new file mode 100644 index 000000000..75fc267ab Binary files /dev/null and b/_images/examples_09_Context_Diagrams_6_0.png differ diff --git a/_images/examples_09_Context_Diagrams_8_0.png b/_images/examples_09_Context_Diagrams_8_0.png new file mode 100644 index 000000000..23012d7a4 Binary files /dev/null and b/_images/examples_09_Context_Diagrams_8_0.png differ diff --git a/_images/examples_10_Declarative_Modeling_12_0.png b/_images/examples_10_Declarative_Modeling_12_0.png new file mode 100644 index 000000000..123b8c9c0 Binary files /dev/null and b/_images/examples_10_Declarative_Modeling_12_0.png differ diff --git a/_images/examples_11_Complexity_Assessment_2_1.svg b/_images/examples_11_Complexity_Assessment_2_1.svg new file mode 100644 index 000000000..27079911d --- /dev/null +++ b/_images/examples_11_Complexity_Assessment_2_1.svg @@ -0,0 +1,39 @@ + + + +180 +objects + + +31% + +16% + +27% + +27% + +18 +diagrams + + +39% + +22% + +28% + +11% + + + +Operational Analysis + +System Analysis + +Logical Architecture + +Physical Architecture + + + \ No newline at end of file diff --git a/_images/harrys_wand.png b/_images/harrys_wand.png new file mode 100644 index 000000000..effece64b Binary files /dev/null and b/_images/harrys_wand.png differ diff --git a/_images/waypoints.png b/_images/waypoints.png new file mode 100644 index 000000000..fd021c986 Binary files /dev/null and b/_images/waypoints.png differ diff --git a/_sources/code/capellambse.aird.rst.txt b/_sources/code/capellambse.aird.rst.txt new file mode 100644 index 000000000..36d1aa945 --- /dev/null +++ b/_sources/code/capellambse.aird.rst.txt @@ -0,0 +1,18 @@ +capellambse.aird package +======================== + +.. automodule:: capellambse.aird + :members: + :undoc-members: + :show-inheritance: + +Submodules +---------- + +capellambse.aird.diagram module +------------------------------- + +.. automodule:: capellambse.aird.diagram + :members: + :undoc-members: + :show-inheritance: diff --git a/_sources/code/capellambse.diagram.rst.txt b/_sources/code/capellambse.diagram.rst.txt new file mode 100644 index 000000000..bf6ec497b --- /dev/null +++ b/_sources/code/capellambse.diagram.rst.txt @@ -0,0 +1,18 @@ +capellambse.diagram package +=========================== + +.. automodule:: capellambse.diagram + :members: + :undoc-members: + :show-inheritance: + +Submodules +---------- + +capellambse.diagram.capstyle module +----------------------------------- + +.. automodule:: capellambse.diagram.capstyle + :members: + :undoc-members: + :show-inheritance: diff --git a/_sources/code/capellambse.extensions.metrics.rst.txt b/_sources/code/capellambse.extensions.metrics.rst.txt new file mode 100644 index 000000000..a81d6a574 --- /dev/null +++ b/_sources/code/capellambse.extensions.metrics.rst.txt @@ -0,0 +1,26 @@ +capellambse.extensions.metrics package +====================================== + +.. automodule:: capellambse.extensions.metrics + :members: + :undoc-members: + :show-inheritance: + +Submodules +---------- + +capellambse.extensions.metrics.collector module +----------------------------------------------- + +.. automodule:: capellambse.extensions.metrics.collector + :members: + :undoc-members: + :show-inheritance: + +capellambse.extensions.metrics.composer module +---------------------------------------------- + +.. automodule:: capellambse.extensions.metrics.composer + :members: + :undoc-members: + :show-inheritance: diff --git a/_sources/code/capellambse.extensions.reqif.rst.txt b/_sources/code/capellambse.extensions.reqif.rst.txt new file mode 100644 index 000000000..0cd2d7236 --- /dev/null +++ b/_sources/code/capellambse.extensions.reqif.rst.txt @@ -0,0 +1,26 @@ +capellambse.extensions.reqif package +==================================== + +.. automodule:: capellambse.extensions.reqif + :members: + :undoc-members: + :show-inheritance: + +Submodules +---------- + +capellambse.extensions.reqif.elements module +-------------------------------------------- + +.. automodule:: capellambse.extensions.reqif.elements + :members: + :undoc-members: + :show-inheritance: + +capellambse.extensions.reqif.exporter module +-------------------------------------------- + +.. automodule:: capellambse.extensions.reqif.exporter + :members: + :undoc-members: + :show-inheritance: diff --git a/_sources/code/capellambse.extensions.rst.txt b/_sources/code/capellambse.extensions.rst.txt new file mode 100644 index 000000000..3f574cde4 --- /dev/null +++ b/_sources/code/capellambse.extensions.rst.txt @@ -0,0 +1,35 @@ +capellambse.extensions package +============================== + +.. automodule:: capellambse.extensions + :members: + :undoc-members: + :show-inheritance: + +Subpackages +----------- + +.. toctree:: + :maxdepth: 4 + + capellambse.extensions.metrics + capellambse.extensions.reqif + +Submodules +---------- + +capellambse.extensions.filtering module +--------------------------------------- + +.. automodule:: capellambse.extensions.filtering + :members: + :undoc-members: + :show-inheritance: + +capellambse.extensions.pvmt module +---------------------------------- + +.. automodule:: capellambse.extensions.pvmt + :members: + :undoc-members: + :show-inheritance: diff --git a/_sources/code/capellambse.filehandler.rst.txt b/_sources/code/capellambse.filehandler.rst.txt new file mode 100644 index 000000000..4bf8a4e8d --- /dev/null +++ b/_sources/code/capellambse.filehandler.rst.txt @@ -0,0 +1,66 @@ +capellambse.filehandler package +=============================== + +.. automodule:: capellambse.filehandler + :members: + :undoc-members: + :show-inheritance: + +Submodules +---------- + +capellambse.filehandler.abc module +---------------------------------- + +.. automodule:: capellambse.filehandler.abc + :members: + :undoc-members: + :show-inheritance: + +capellambse.filehandler.git module +---------------------------------- + +.. automodule:: capellambse.filehandler.git + :members: + :undoc-members: + :show-inheritance: + +capellambse.filehandler.git\_askpass module +------------------------------------------- + +.. automodule:: capellambse.filehandler.git_askpass + :members: + :undoc-members: + :show-inheritance: + +capellambse.filehandler.gitlab\_artifacts module +------------------------------------------------ + +.. automodule:: capellambse.filehandler.gitlab_artifacts + :members: + :undoc-members: + :show-inheritance: + +capellambse.filehandler.http module +----------------------------------- + +.. automodule:: capellambse.filehandler.http + :members: + :undoc-members: + :show-inheritance: + +capellambse.filehandler.local module +------------------------------------ + +.. automodule:: capellambse.filehandler.local + :members: + :undoc-members: + :show-inheritance: + +capellambse.filehandler.memory module +------------------------------------- + +.. automodule:: capellambse.filehandler.memory + :members: + :undoc-members: + :show-inheritance: diff --git a/_sources/code/capellambse.loader.rst.txt b/_sources/code/capellambse.loader.rst.txt new file mode 100644 index 000000000..299b73166 --- /dev/null +++ b/_sources/code/capellambse.loader.rst.txt @@ -0,0 +1,50 @@ +capellambse.loader package +========================== + +.. automodule:: capellambse.loader + :members: + :undoc-members: + :show-inheritance: + +Submodules +---------- + +capellambse.loader.core module +------------------------------ + +.. automodule:: capellambse.loader.core + :members: + :undoc-members: + :show-inheritance: + +capellambse.loader.exs module +----------------------------- + +.. automodule:: capellambse.loader.exs + :members: + :undoc-members: + :show-inheritance: + +capellambse.loader.filehandler module +------------------------------------- + +.. automodule:: capellambse.loader.filehandler + :members: + :undoc-members: + :show-inheritance: + +capellambse.loader.modelinfo module +----------------------------------- + +.. automodule:: capellambse.loader.modelinfo + :members: + :undoc-members: + :show-inheritance: + +capellambse.loader.xmltools module +---------------------------------- + +.. automodule:: capellambse.loader.xmltools + :members: + :undoc-members: + :show-inheritance: diff --git a/_sources/code/capellambse.model.common.rst.txt b/_sources/code/capellambse.model.common.rst.txt new file mode 100644 index 000000000..a6f7b8311 --- /dev/null +++ b/_sources/code/capellambse.model.common.rst.txt @@ -0,0 +1,34 @@ +capellambse.model.common package +================================ + +.. automodule:: capellambse.model.common + :members: + :undoc-members: + :show-inheritance: + +Submodules +---------- + +capellambse.model.common.accessors module +----------------------------------------- + +.. automodule:: capellambse.model.common.accessors + :members: + :undoc-members: + :show-inheritance: + +capellambse.model.common.element module +--------------------------------------- + +.. automodule:: capellambse.model.common.element + :members: + :undoc-members: + :show-inheritance: + +capellambse.model.common.properties module +------------------------------------------ + +.. automodule:: capellambse.model.common.properties + :members: + :undoc-members: + :show-inheritance: diff --git a/_sources/code/capellambse.model.crosslayer.information.rst.txt b/_sources/code/capellambse.model.crosslayer.information.rst.txt new file mode 100644 index 000000000..30846a88a --- /dev/null +++ b/_sources/code/capellambse.model.crosslayer.information.rst.txt @@ -0,0 +1,26 @@ +capellambse.model.crosslayer.information package +================================================ + +.. automodule:: capellambse.model.crosslayer.information + :members: + :undoc-members: + :show-inheritance: + +Submodules +---------- + +capellambse.model.crosslayer.information.datatype module +-------------------------------------------------------- + +.. automodule:: capellambse.model.crosslayer.information.datatype + :members: + :undoc-members: + :show-inheritance: + +capellambse.model.crosslayer.information.datavalue module +--------------------------------------------------------- + +.. automodule:: capellambse.model.crosslayer.information.datavalue + :members: + :undoc-members: + :show-inheritance: diff --git a/_sources/code/capellambse.model.crosslayer.rst.txt b/_sources/code/capellambse.model.crosslayer.rst.txt new file mode 100644 index 000000000..20a2f2367 --- /dev/null +++ b/_sources/code/capellambse.model.crosslayer.rst.txt @@ -0,0 +1,66 @@ +capellambse.model.crosslayer package +==================================== + +.. automodule:: capellambse.model.crosslayer + :members: + :undoc-members: + :show-inheritance: + +Subpackages +----------- + +.. toctree:: + :maxdepth: 4 + + capellambse.model.crosslayer.information + +Submodules +---------- + +capellambse.model.crosslayer.capellacommon module +------------------------------------------------- + +.. automodule:: capellambse.model.crosslayer.capellacommon + :members: + :undoc-members: + :show-inheritance: + +capellambse.model.crosslayer.capellacore module +----------------------------------------------- + +.. automodule:: capellambse.model.crosslayer.capellacore + :members: + :undoc-members: + :show-inheritance: + +capellambse.model.crosslayer.cs module +-------------------------------------- + +.. automodule:: capellambse.model.crosslayer.cs + :members: + :undoc-members: + :show-inheritance: + +capellambse.model.crosslayer.fa module +-------------------------------------- + +.. automodule:: capellambse.model.crosslayer.fa + :members: + :undoc-members: + :show-inheritance: + +capellambse.model.crosslayer.interaction module +----------------------------------------------- + +.. automodule:: capellambse.model.crosslayer.interaction + :members: + :undoc-members: + :show-inheritance: + +capellambse.model.crosslayer.modellingcore module +------------------------------------------------- + +.. automodule:: capellambse.model.crosslayer.modellingcore + :members: + :undoc-members: + :show-inheritance: diff --git a/_sources/code/capellambse.model.layers.rst.txt b/_sources/code/capellambse.model.layers.rst.txt new file mode 100644 index 000000000..f07ab131d --- /dev/null +++ b/_sources/code/capellambse.model.layers.rst.txt @@ -0,0 +1,42 @@ +capellambse.model.layers package +================================ + +.. automodule:: capellambse.model.layers + :members: + :undoc-members: + :show-inheritance: + +Submodules +---------- + +capellambse.model.layers.ctx module +----------------------------------- + +.. automodule:: capellambse.model.layers.ctx + :members: + :undoc-members: + :show-inheritance: + +capellambse.model.layers.la module +---------------------------------- + +.. automodule:: capellambse.model.layers.la + :members: + :undoc-members: + :show-inheritance: + +capellambse.model.layers.oa module +---------------------------------- + +.. automodule:: capellambse.model.layers.oa + :members: + :undoc-members: + :show-inheritance: + +capellambse.model.layers.pa module +---------------------------------- + +.. automodule:: capellambse.model.layers.pa + :members: + :undoc-members: + :show-inheritance: diff --git a/_sources/code/capellambse.model.rst.txt b/_sources/code/capellambse.model.rst.txt new file mode 100644 index 000000000..ccce537f1 --- /dev/null +++ b/_sources/code/capellambse.model.rst.txt @@ -0,0 +1,36 @@ +capellambse.model package +========================= + +.. automodule:: capellambse.model + :members: + :undoc-members: + :show-inheritance: + +Subpackages +----------- + +.. toctree:: + :maxdepth: 4 + + capellambse.model.common + capellambse.model.crosslayer + capellambse.model.layers + +Submodules +---------- + +capellambse.model.diagram module +-------------------------------- + +.. automodule:: capellambse.model.diagram + :members: + :undoc-members: + :show-inheritance: + +capellambse.model.modeltypes module +----------------------------------- + +.. automodule:: capellambse.model.modeltypes + :members: + :undoc-members: + :show-inheritance: diff --git a/_sources/code/capellambse.pvmt.rst.txt b/_sources/code/capellambse.pvmt.rst.txt new file mode 100644 index 000000000..c52cb12f2 --- /dev/null +++ b/_sources/code/capellambse.pvmt.rst.txt @@ -0,0 +1,50 @@ +capellambse.pvmt package +======================== + +.. automodule:: capellambse.pvmt + :members: + :undoc-members: + :show-inheritance: + +Submodules +---------- + +capellambse.pvmt.core module +---------------------------- + +.. automodule:: capellambse.pvmt.core + :members: + :undoc-members: + :show-inheritance: + +capellambse.pvmt.exceptions module +---------------------------------- + +.. automodule:: capellambse.pvmt.exceptions + :members: + :undoc-members: + :show-inheritance: + +capellambse.pvmt.model module +----------------------------- + +.. automodule:: capellambse.pvmt.model + :members: + :undoc-members: + :show-inheritance: + +capellambse.pvmt.types module +----------------------------- + +.. automodule:: capellambse.pvmt.types + :members: + :undoc-members: + :show-inheritance: + +capellambse.pvmt.validation module +---------------------------------- + +.. automodule:: capellambse.pvmt.validation + :members: + :undoc-members: + :show-inheritance: diff --git a/_sources/code/capellambse.rst.txt b/_sources/code/capellambse.rst.txt new file mode 100644 index 000000000..83c1eec45 --- /dev/null +++ b/_sources/code/capellambse.rst.txt @@ -0,0 +1,65 @@ +capellambse package +=================== + +.. automodule:: capellambse + :members: + :undoc-members: + :show-inheritance: + +Subpackages +----------- + +.. toctree:: + :maxdepth: 4 + + capellambse.aird + capellambse.diagram + capellambse.extensions + capellambse.filehandler + capellambse.loader + capellambse.model + capellambse.pvmt + capellambse.svg + +Submodules +---------- + +capellambse.auditing module +--------------------------- + +.. automodule:: capellambse.auditing + :members: + :undoc-members: + :show-inheritance: + +capellambse.cli\_helpers module +------------------------------- + +.. automodule:: capellambse.cli_helpers + :members: + :undoc-members: + :show-inheritance: + +capellambse.decl module +----------------------- + +.. automodule:: capellambse.decl + :members: + :undoc-members: + :show-inheritance: + +capellambse.diagram\_cache module +--------------------------------- + +.. automodule:: capellambse.diagram_cache + :members: + :undoc-members: + :show-inheritance: + +capellambse.helpers module +-------------------------- + +.. automodule:: capellambse.helpers + :members: + :undoc-members: + :show-inheritance: diff --git a/_sources/code/capellambse.svg.rst.txt b/_sources/code/capellambse.svg.rst.txt new file mode 100644 index 000000000..e8e2069f9 --- /dev/null +++ b/_sources/code/capellambse.svg.rst.txt @@ -0,0 +1,58 @@ +capellambse.svg package +======================= + +.. automodule:: capellambse.svg + :members: + :undoc-members: + :show-inheritance: + +Submodules +---------- + +capellambse.svg.decorations module +---------------------------------- + +.. automodule:: capellambse.svg.decorations + :members: + :undoc-members: + :show-inheritance: + +capellambse.svg.drawing module +------------------------------ + +.. automodule:: capellambse.svg.drawing + :members: + :undoc-members: + :show-inheritance: + +capellambse.svg.generate module +------------------------------- + +.. automodule:: capellambse.svg.generate + :members: + :undoc-members: + :show-inheritance: + +capellambse.svg.helpers module +------------------------------ + +.. automodule:: capellambse.svg.helpers + :members: + :undoc-members: + :show-inheritance: + +capellambse.svg.style module +---------------------------- + +.. automodule:: capellambse.svg.style + :members: + :undoc-members: + :show-inheritance: + +capellambse.svg.symbols module +------------------------------ + +.. automodule:: capellambse.svg.symbols + :members: + :undoc-members: + :show-inheritance: diff --git a/_sources/code/modules.rst.txt b/_sources/code/modules.rst.txt new file mode 100644 index 000000000..34442a353 --- /dev/null +++ b/_sources/code/modules.rst.txt @@ -0,0 +1,7 @@ +py-capellambse +============== + +.. toctree:: + :maxdepth: 4 + + capellambse diff --git a/_sources/development/developing-docs.rst.txt b/_sources/development/developing-docs.rst.txt new file mode 100644 index 000000000..23412e95e --- /dev/null +++ b/_sources/development/developing-docs.rst.txt @@ -0,0 +1,23 @@ +.. + SPDX-FileCopyrightText: Copyright DB Netz AG and the capellambse contributors + SPDX-License-Identifier: Apache-2.0 + +************************* +Documentation development +************************* + +The following command deletes previous built documentation and derives +docs out of code: + +.. code:: bash + + make -C docs apidoc + +The following command builds the docs: + +.. code:: bash + + make -C docs html + +The resulting documentation build should be available in `docs/build/html`, +entry point is `index.html` diff --git a/_sources/development/how-to-explore-capella-mm.rst.txt b/_sources/development/how-to-explore-capella-mm.rst.txt new file mode 100644 index 000000000..73ce1fdfc --- /dev/null +++ b/_sources/development/how-to-explore-capella-mm.rst.txt @@ -0,0 +1,225 @@ +.. + SPDX-FileCopyrightText: Copyright DB Netz AG and the capellambse contributors + SPDX-License-Identifier: Apache-2.0 + +********************************* +How to explore Capella meta-model +********************************* + +Initially the API design was mostly based on our understanding of XML files, +however we soon moved on into exploring the Capella meta-model. Here is a short +summary of how we do it. + +Getting the meta-model sources +############################## + +First of all we need to get the data - the most straight-forward way is to +clone the Capella source code: + +.. code-block:: bash + + mkdir capella-mm && cd capella-mm + git clone https://github.com/eclipse/capella.git + +As there are quite a few files we may want to discover the packages of interest +using ``find``: + +- ``find -name '*.ecore' -o -name '*.genmodel'`` lists all the ECore models and + Generation files and adds Generation viewpoint for related ecore +- ``find -name '*.odesign'`` lists all the Sirius definitions (if you like to + see how the diagram look-and-feel is defined) + +You could open all the related projects right where they are but I'd prefer to +have a clean sandbox. + +.. code-block:: bash + + mkdir capella-metamodel + cp -r capella/common/plugins/org.polarsys.capella.common.data.activity.gen capella-metamodel + cp -r capella/common/plugins/org.polarsys.capella.common.data.behavior.gen capella-metamodel + cp -r capella/common/plugins/org.polarsys.capella.common.data.core.gen capella-metamodel + cp -r capella/common/plugins/org.polarsys.capella.common.libraries.gen capella-metamodel + cp -r capella/common/plugins/org.polarsys.capella.common.re.gen capella-metamodel + cp -r capella/core/plugins/org.polarsys.capella.core.data.gen capella-metamodel + cp -r capella/releng/cdo/plugins/org.polarsys.kitalpha.emde.model.cdo capella-metamodel + cp -r capella/tests/plugins/org.polarsys.capella.test.diagram.layout.ju capella-metamodel + +Now we have all of the meta-model dependencies in one place and can open it as +a single Eclipse project. You could open those with any Eclipse that has EMF +(like the Modeling bundle) but I'd advise using `Capella Studio`__ (open the +link and search for "Studio") for the best experience - it is provisioned with +all the extras that help you visualize / review the meta-model. + +__ https://www.eclipse.org/capella/download.html + +.. image:: ../_static/img/2021-05-09_10-32.jpg +.. image:: ../_static/img/2021-05-09_10-36.jpg + +After import Eclipse may highlight some projects with error or warning sign - +that's fine, we'll let it go for now. + +Before we dive into visualizations, let's have a quick look around. The core of +Capella metamodel is provided by ``org.polarsys.capella.common.data.core.gen`` +project. There you'll find the definitions of the "Capella Layers" and the +enabling "EnginereengConcerns" components. For example, the ``SystemAnalysis`` +is provided by ``ContextArchitecture.ecore``. + +.. image:: ../_static/img/2021-05-09_10-50.jpg + +Even though the default tree-view (triggered by double-click on a ``.core``) is +already giving us something there are much better ways to review the models. +We'll talk about that in the `Visualizing ECores`__ chapter. But first we +should have a look into the package structure and interdependencies. + +__ #visualizing-ecores + +Quick intro to package structure +################################ + +If you are unfamiliar with ECore you might be wondering what are those ECore +files anyways. We could say that they act as UML Packages. Every package +contains ontology elements (or UML Classes) and "local" element relationships +(Associations). The packages depend on each-other as elements frequently build +on top of each-other via Generalization relationship. Thats one of the best +things about Capella metamodel - it is defined by UML (or well, structural +subset, but still pretty cool)! + +So, from using Capella we know our model layers - Operational Analysis, System +Analysis, Logical Architecture, Physical Architecture and EPBS. We can locate +the corresponding ECores pretty quick, but what are all the other packages +about? To answer that question we should visualize the package dependency - the +quickest way to do so is to `create a new representation file`__. And after +that follow the steps in `create package dependency overview`__. The result may +look like what we have below: + +__ #create-new-representations-file +__ #package-dependency-overview + +.. image:: ../_static/img/2021-05-09_17-38.jpg + +And even though there are not too many packages in the meta-model, when we +visualize the inter-package dependency it may be a bit difficult to understand. +By the way, the figure above `is also available in SVG`__ - you can open it in +a new tab and zoom-in if you like or scroll down for a simplified one (but +opinionated). + +__ core-pkg-deps-raw.svg + +It almost feels like everything depends on everything but that isn't true +really. We could change perspective and look for the dependencies of an +end-user exposed package, such as Operational Analysis (oa). I'll use Papyrus +to visualize that: + +.. image:: ../_static/img/2021-05-09_17-44.jpg + +When we add all the Capella layers to that picture things get a bit more +interesting. I added some artificial grouping on top of the existing packages +that will help us later on - the artificial groups are: ``CapellaLayers`` - +packages that cover the layers we used to see in the tool; ``CrossLayer`` - +packages that define ontology and patterns that we see in almost every layer; +``Enablers`` - the low level ontology that enables every element. + +.. image:: ../_static/img/2021-05-09_21-21.jpg + +To make a further point on ``CrossLayer``, lets have a closer look at a +Function. We know that Functions are quite similar in how they look and feel +across Capella layers and have a lot in common with OperationalActivity. When +we open ``LogicalArchitecture.ecore`` we see it defines a ``LogicalFunction`` +as a specialization of ``AbstractFunction`` that comes from +``FunctionalAnalysis.ecore``. There is a very nice feature provided by the +`Ecore Tools`__ (that is also included in Capella Studio) - Class Inheritance +view (there is also a tiny `how-to use it below`__). We'll use that to +visualize the way the functions are made. + +__ https://www.eclipse.org/ecoretools/overview.html +__ # + +.. image:: ../_static/img/2021-05-09_22-09.jpg + +Just to be on a safe side I've done the above exercise for +``OperationalActivity``, ``SystemFunction``, ``LogicalFunction`` and +``PhysicalFunction`` - the inheritance tree is exactly the same. I've done this +check for a few other familiar ontology elements and got same result. I think +this gives a feeling for what the ``CrossLayer`` is about - to me that's the +place where most of the magic happens. And this is how the relationship between +the ``CapellaLayers`` and ``CrossLayer`` looks like when we de-noise it a bit: + +.. image:: ../_static/img/2021-05-09_22-39.jpg + +It's been a lengthy chain of thought and to finish on a hopefully useful +visualization - lets have a look at the Class contexts of some things that we +use most frequently + +Visualizations of some frequently used ontology elements +######################################################## + +Below you'll find some quick visualizations for frequently used ontology +elements, grouped by CrossLayer package + +Functional Analysis +******************* + +.. image:: ../_static/img/2021-05-10_22-35.jpg + +.. image:: ../_static/img/2021-05-17_11-50.jpg + +.. image:: ../_static/img/2021-05-10_22-59.jpg + +.. image:: ../_static/img/2021-05-12_21-03.jpg + +State Machines +************** + +.. image:: ../_static/img/2021-05-12_18-18.jpg + +.. image:: ../_static/img/2021-05-13_20-46.jpg + +The figure above is somewhat a "treasure map" to everything related to state +machines. It is made in a semi-automatic way with the help of ECore Tools and +ELK + +Composite Structure +******************* + +.. image:: ../_static/img/2021-05-12_20-10.jpg + +.. image:: ../_static/img/2021-05-12_18-51.jpg + +You may also want to have a look at the Block context below as it defines some +other useful things that a Component (or LogicalComponent) can do. + +.. image:: ../_static/img/2021-05-12_20-06.jpg + +.. image:: ../_static/img/2021-05-12_19-12.jpg + +Appendix: Visualizing ECores +############################ + +If you are new to CapellaStudio and Ecore, here are some practical hints for +how to get stuff done, ignore the below otherwise: + +Create new representations file +******************************* + +To start playing with visualizations we need a new representations file +(.aird). It is pretty easy to get there but just in case, there is figure below +to guide you through that. + +.. image:: ../_static/img/2021-05-09_17-36.jpg + +Package dependency overview +*************************** + +To create a package dependency overview for all packages you may follow the +guidance in the figure below: + +.. image:: ../_static/img/2021-05-09_17-57.jpg + +Visualizing class inheritance +***************************** + +There is a very nice feature that allows given a class to show all of its +super-classes (generalizations) and specializations. The figure below gives you +some hints for how to use it: + +.. image:: ../_static/img/2021-05-09_21-46.jpg diff --git a/_sources/development/low-level-api.rst.txt b/_sources/development/low-level-api.rst.txt new file mode 100644 index 000000000..ea12faf1b --- /dev/null +++ b/_sources/development/low-level-api.rst.txt @@ -0,0 +1,307 @@ +.. + SPDX-FileCopyrightText: Copyright DB Netz AG and the capellambse contributors + SPDX-License-Identifier: Apache-2.0 + +************* +Low-level API +************* + +.. py:currentmodule:: capellambse.loader.core + +The high level :py:class:`~capellambse.model.MelodyModel`-based API is largely +manually designed, and therefore sometimes does not cover all interesting +objects to a usable level. While we are constantly working on improving the +situation, it's also possible to use the low-level API based directly on the +XML files in order to temporarily work around these shortcomings. This +documentation sheds some light on the inner workings of this low-level API. + +In order to effectively work with it, you need to understand the basics of XML. +It also helps to be familiar with LXML_, which is used to parse and manipulate +the XML trees in memory. + +Unfortunately it's not possible to use LXML's built-in XML serializer. It +produces different whitespace in the XML tree, which confuses Capella's XML +diff-merge algorithm. This is why |project| ships with a custom serializer that +produces the same output format as Capella. It resides in the +:py:mod:`capellambse.loader.exs` module. + +.. _LXML: https://lxml.de/ + +The MelodyLoader object +======================= + +While the main object of interest for the high-level API is the +:py:class:`capellambse.model.MelodyModel` class, for the low-level API it is +the :py:class:`capellambse.loader.core.MelodyLoader`. It offers numerous +methods to search elements, resolve references, ensure model integrity during +certain modifications, and many more. + +The following sections categorize and document the various methods. + +The ``MelodyLoader`` closely works together with its auxiliary class +:py:class:`~capellambse.loader.core.ModelFile`. However, the ``ModelFile`` +mainly plays a role while loading or saving a model from/to disk (or other data +stores), and isn't used much when interacting with an already loaded model. + +.. _api-level-shift: + +Shifting between API levels +=========================== + +High to low-level shift +----------------------- + +Every model object (i.e. instance of ``GenericElement`` or one of its +subclasses) has an attribute ``_element``, which holds a reference to the +corresponding :py:class:`lxml.etree._Element` instance. The low-level API works +directly with these ``_Element`` instances. + +The ``MelodyModel`` object stores a reference to the +:py:class:`~capellambse.loader.core.MelodyLoader` instance. + +Low to high-level shift +----------------------- + +The GenericElement class offers the +:py:meth:`~capellambse.model.common.element.GenericElement.from_model` class +method, which takes a ``MelodyModel`` instance and a low-level LXML +``_Element`` as arguments and constructs a high-level API proxy object from +them. This is the way "back up" to the high-level API. + +.. note:: + + Always call ``from_model`` on the base ``GenericElement`` class, not on its + subclasses. The base class automatically searches for the correct subclass + to instantiate, based on the ``xsi:type`` of the passed XML element. Calling + the method on a subclass directly may inadvertently cause the wrong class to + be picked. + +.. code-block:: python + + >>> myfunc = model.search("LogicalFunction")[0] + >>> el = myfunc._element + >>> el + + >>> from capellambse.model import GenericElement + >>> high_el = GenericElement.from_model(model, el) + >>> high_el == myfunc + True + +When working with multiple objects, it can be desirable to directly construct a +high-level :py:class:`~capellambse.model.common.element.ElementList` with them. +The ElementList constructor works similar to ``GenericElement.from_model``, but +it takes a list of elements instead of only a single one. + +.. code-block:: python + + >>> mycomp = model.search("LogicalComponent")[0] + >>> children = mycomp._element.getchildren() + >>> len(children) + 7 + >>> mylist = ElementList(model, children) + >>> mylist + [0] + [1] + [2] + [3] + [4] + [5] + [6] + +Moving along the XML tree +========================= + +In most simple cases, you can use the standard LXML methods in order to select +parent, child and sibling elements. + +.. code-block:: python + + >>> myfunc = model.search("LogicalFunction")[3] + >>> myfunc._element.getparent() + + >>> myfunc._element.getchildren() + [] + >>> myfunc._element.getprevious() + + >>> myfunc._element.getnext() + + +These elements and lists of elements can then be fed into +``GenericElement.from_model`` or the ``ElementList`` constructor respectively +in order to :ref:`return to the high-level API `. + +Capella models support fragmentation into multiple files, which results in +multiple XML trees being loaded into memory. This makes it difficult to +traverse up and down the hierarchy, because in theory every element can be a +fragment boundary – in this case, it does not have a physical parent element, +and ``getparent()`` will return ``None``. A call to ``getchildren()`` or +similar on the (logical) parent element will yield a placeholder which only +contains a reference to the real element, but does not hold any other +information. + +``MelodyLoader`` provides methods to traverse upwards or downwards in the +model's XML tree, while also taking into account fragment boundaries and the +aforementioned placeholder elements. + +.. class:: MelodyLoader + :noindex: + + .. automethod:: iterancestors + :noindex: + .. automethod:: iterchildren_xt + :noindex: + .. automethod:: iterdescendants + :noindex: + .. automethod:: iterdescendants_xt + :noindex: + +Resolving references +==================== + +You will often encounter attributes that contain references to other elements. + +The ``MelodyLoader`` provides the following methods to work with references: + +.. class:: MelodyLoader + :noindex: + + .. automethod:: follow_link + :noindex: + .. automethod:: follow_links + :noindex: + .. automethod:: create_link + :noindex: + +Finding elements elsewhere +========================== + +The low-level API implements the fundamentals for looking up model objects or +finding them by their type. The following methods are involved in these +operations: + +.. class:: MelodyLoader + :noindex: + + .. automethod:: iterall + :noindex: + .. automethod:: iterall_xt + :noindex: + .. automethod:: xpath + :noindex: + .. automethod:: xpath2 + :noindex: + +Manipulating objects +==================== + +.. warning:: + + The low-level API by itself does not do any consistency or validity checks + when modifying a model. Therefore it is very easy to break a model using it, + which can be very hard to recover from. Proceed with caution. + +As ``GenericElement`` instances are simply wrappers around the raw XML +elements, any changes to their attributes are directly reflected by changes to +the attributes or children of the underlying XML element and vice versa. This +means that no special care needs to be taken to keep the high-level and +low-level parts of the API synchronized. + +In many cases, the attribute names of the high-level API match those in the +XML, with the difference that the former uses ``snake_case`` naming (as is +conventional in the Python world), while the latter uses ``camelCase`` naming. +This example shows how the name of a function is accessed and modified using +the low-level API: + +.. code-block:: python + + >>> myfunc = model.search("LogicalFunction")[3] + >>> myfunc.name + 'defend the surrounding area against Intruders' + >>> myfunc._element.attrib["name"] + 'defend the surrounding area against Intruders' + >>> myfunc._element.attrib["name"] = "My Function" + >>> myfunc.name + 'My Function' + +Be aware that the XML usually does not explicitly store attributes that are set +to their default value (as defined by the meta model). In addition to that, the +high-level API often offers convenience shortcuts and reverse lookups that are +not directly reflected by XML attributes. Without at the detailed definitions, +it can therefore be difficult to infer the correct attributes for the low-level +API objects. + +Creating and deleting objects +============================= + +.. warning:: + + Creating or deleting objects through the low-level API is highly + discouraged, as it bears a very high risk of breaking the model. It's + unlikely that we can support you with any breakage that you encounter as a + result of using the low-level API. + + If you need access to model elements and relations that are not yet covered + by our high-level API, please consider contributing and extending it instead + – it's probably easier anyway. ;) + +The ID cache +------------ + +In order to provide instantaneous access to any model element via its UUID, the +MelodyLoader maintains a hashmap containing all UUIDs. This hashmap needs to be +updated when inserting or removing elements in the tree. The following methods +take care of that: + +.. class:: MelodyLoader + :noindex: + + .. automethod:: idcache_index + :noindex: + .. automethod:: idcache_remove + :noindex: + .. automethod:: idcache_rebuild + :noindex: + +Creating objects +---------------- + +Creating a new object with the low-level API is a rather complex process. The +``MelodyLoader`` does provide some basic integrity checks, but most of the +meta-model-aware logic is implemented within the high-level API. + +Before creating a new object, you need to generate and reserve a UUID for it. +This is done using the ``generate_uuid`` method. ``new_uuid`` provides a +context manager around it, which automatically cleans up the model in case +anything went wrong. It also checks that the UUID was properly registered with +the ID cache (see below). It is therefore highly recommended to use +``new_uuid`` over directly calling ``generate_uuid``. Note that even when using +``new_uuid``, you still need to manually call ``idcache_index`` on the newly +inserted element. + +.. class:: MelodyLoader + :noindex: + + .. automethod:: generate_uuid + :noindex: + .. automethod:: new_uuid + :noindex: + +Deleting objects +---------------- + +Inversely to creating new ones, when deleting an object from the XML tree it +also needs to be removed from the ID cache. This is done by calling +``idcache_remove`` (see above) on the element to be removed. Afterwards, delete +the element from its parent using the standard LXML API. + +Saving modifications +==================== + +The ``MelodyLoader`` provides the same ``save()`` method as the high-level +``MelodyModel``. + +.. class:: MelodyLoader + :noindex: + + .. automethod:: save + :noindex: diff --git a/_sources/development/repl.rst.txt b/_sources/development/repl.rst.txt new file mode 100644 index 000000000..f7d479089 --- /dev/null +++ b/_sources/development/repl.rst.txt @@ -0,0 +1,16 @@ +.. + SPDX-FileCopyrightText: Copyright DB Netz AG and the capellambse contributors + SPDX-License-Identifier: Apache-2.0 + +The |project| REPL +================== + +.. automodule:: capellambse.repl + +.. sphinx_argparse_cli:: + :module: capellambse.repl + :func: main + :hook: + :prog: capellambse/repl.py + :title: Capellambse Repl + :group_title_prefix: diff --git a/_sources/examples/01 Introduction.ipynb.txt b/_sources/examples/01 Introduction.ipynb.txt new file mode 100644 index 000000000..983c7661c --- /dev/null +++ b/_sources/examples/01 Introduction.ipynb.txt @@ -0,0 +1,607 @@ +{ + "cells": [ + { + "cell_type": "markdown", + "id": "1a1fe414", + "metadata": {}, + "source": [ + "# Introduction\n", + "\n", + "Welcome to the py-capella-mbse Showcase notebook. This notebook will show you some basic (and not so basic) things that you can get done using this library. For more advanced features have a look around the nearby notebooks.\n", + "\n", + "The below code loads the library and one of the test models:" + ] + }, + { + "cell_type": "code", + "execution_count": 1, + "id": "3e28d1c9", + "metadata": {}, + "outputs": [ + { + "data": { + "text/plain": [ + "" + ] + }, + "execution_count": 1, + "metadata": {}, + "output_type": "execute_result" + } + ], + "source": [ + "import capellambse\n", + "path_to_model = \"../../../tests/data/melodymodel/5_0/Melody Model Test.aird\"\n", + "model = capellambse.MelodyModel(path_to_model, jupyter_untrusted=True)\n", + "model" + ] + }, + { + "cell_type": "markdown", + "id": "48ce0c1e", + "metadata": {}, + "source": [ + "The additional `jupyter_untrusted` parameter is currently needed to render diagrams correctly. In untrusted notebooks, all CSS elements (and JavaScript too) are removed for security reasons - however, capellambse-generated SVGs use a stylesheet to make the implementation a bit cleaner.\n", + "\n", + "Having said that, lets go to the first practical example of working with the library!" + ] + }, + { + "cell_type": "markdown", + "id": "7e68b88c-b4bc-4c20-a39f-48094c0eabdd", + "metadata": { + "tags": [] + }, + "source": [ + "## Example 1: Actor functions\n", + "\n", + "The below code will print every Actor available in the Logical Architecture layer" + ] + }, + { + "cell_type": "code", + "execution_count": 2, + "id": "abcd8693", + "metadata": {}, + "outputs": [ + { + "name": "stdout", + "output_type": "stream", + "text": [ + "Multiport\n", + "Prof. S. Snape\n", + "Voldemort\n", + "R. Weasley\n", + "Prof. A. P. W. B. Dumbledore\n", + "Harry J. Potter\n" + ] + } + ], + "source": [ + "for actor in model.la.all_actors:\n", + " print(actor.name)" + ] + }, + { + "cell_type": "markdown", + "id": "4ddbf9be", + "metadata": {}, + "source": [ + "but we could also \"zoom-in\" to an actor of interest:" + ] + }, + { + "cell_type": "code", + "execution_count": 3, + "id": "56bb4b44", + "metadata": { + "scrolled": true + }, + "outputs": [ + { + "data": { + "text/html": [ + "

Prof. S. Snape (org.polarsys.capella.core.data.la:LogicalComponent)

allocated_functions
  1. LogicalFunction "Teaching" (a7acb298-d14b-4707-a419-fea272434541)
  2. LogicalFunction "maintain a layer of defense for the Sorcerer's Stone" (4a2a7f3c-d223-4d44-94a7-50dd2906a70c)
applied_property_value_groups
  1. PropertyValueGroup "DarkMagic.Power" (2a480409-57d1-46f8-a0ce-e574706a9a7c)
  2. PropertyValueGroup "DarkMagic.Power Level" (b1d7453b-69ab-4d81-ab8b-1e48b5870340)
applied_property_values

(Empty list)

components

(Empty list)

constraints

(Empty list)

context_diagramContext of Prof. S. Snape (uuid: 6f463eed-c77b-4568-8078-beec0536f243_context)
description

Good guy and teacher of brewing arts.

\n", + "
diagrams

(Empty list)

exchanges

(Empty list)

filtering_criteria

(Empty list)

functions
  1. LogicalFunction "Teaching" (a7acb298-d14b-4707-a419-fea272434541)
  2. LogicalFunction "maintain a layer of defense for the Sorcerer's Stone" (4a2a7f3c-d223-4d44-94a7-50dd2906a70c)
is_abstractFalse
is_actorTrue
is_humanTrue
nameProf. S. Snape
ownerLogicalComponentPkg "Structure" (84c0978d-9a32-4f5b-8013-5b0b6adbfd73)
parentLogicalComponentPkg "Structure" (84c0978d-9a32-4f5b-8013-5b0b6adbfd73)
partsBackreference to Part - omitted: can be slow to compute. Display this property directly to show.
physical_links

(Empty list)

physical_paths

(Empty list)

physical_ports

(Empty list)

ports
  1. ComponentPort "CP 1" (b4e39757-b0fd-41ff-a7b8-c9fc36de2ca9)
progress_statusNOT_SET
property_value_groups
  1. PropertyValueGroup "DarkMagic.Power" (2a480409-57d1-46f8-a0ce-e574706a9a7c)
  2. PropertyValueGroup "DarkMagic.Power Level" (b1d7453b-69ab-4d81-ab8b-1e48b5870340)
property_values

(Empty list)

pvmt<capellambse.extensions.pvmt.PropertyValueProxy object at 0x7f496cb3f010>
realized_components

(Empty list)

realized_system_components

(Empty list)

realizing_componentsBackreference to - omitted: can be slow to compute. Display this property directly to show.
realizing_physical_componentsBackreference to PhysicalComponent - omitted: can be slow to compute. Display this property directly to show.
related_exchangesBackreference to ComponentExchange - omitted: can be slow to compute. Display this property directly to show.
requirements

(Empty list)

state_machines

(Empty list)

summaryNone
traces

(Empty list)

uuid6f463eed-c77b-4568-8078-beec0536f243
xtypeorg.polarsys.capella.core.data.la:LogicalComponent
" + ], + "text/plain": [ + "\n", + ".allocated_functions = [0] \n", + " [1] \n", + ".applied_property_value_groups = [0] \n", + " [1] \n", + ".applied_property_values = []\n", + ".components = []\n", + ".constraints = []\n", + ".context_diagram = \n", + ".description = Markup('

Good guy and teacher of brewing arts.

\\n')\n", + ".diagrams = []\n", + ".exchanges = []\n", + ".filtering_criteria = []\n", + ".functions = [0] \n", + " [1] \n", + ".is_abstract = False\n", + ".is_actor = True\n", + ".is_human = True\n", + ".name = 'Prof. S. Snape'\n", + ".owner = \n", + ".parent = \n", + ".parts = ... # backreference to Part - omitted: can be slow to compute\n", + ".physical_links = []\n", + ".physical_paths = []\n", + ".physical_ports = []\n", + ".ports = [0] \n", + ".progress_status = 'NOT_SET'\n", + ".property_value_groups = [0] \n", + " [1] \n", + ".property_values = []\n", + ".pvmt = \n", + ".realized_components = []\n", + ".realized_system_components = []\n", + ".realizing_components = ... # backreference to - omitted: can be slow to compute\n", + ".realizing_physical_components = ... # backreference to PhysicalComponent - omitted: can be slow to compute\n", + ".related_exchanges = ... # backreference to ComponentExchange - omitted: can be slow to compute\n", + ".requirements = []\n", + ".state_machines = []\n", + ".summary = None\n", + ".traces = []\n", + ".uuid = '6f463eed-c77b-4568-8078-beec0536f243'\n", + ".xtype = 'org.polarsys.capella.core.data.la:LogicalComponent'" + ] + }, + "execution_count": 3, + "metadata": {}, + "output_type": "execute_result" + } + ], + "source": [ + "model.la.all_actors.by_name(\"Prof. S. Snape\")" + ] + }, + { + "cell_type": "markdown", + "id": "30e17ac3", + "metadata": {}, + "source": [ + "We can also turn the above data into a table, for example \"actor function allocation\", using `pandas`.\n", + "\n", + "For this, we first make sure pandas itself is installed, as well as an extension we'll use later." + ] + }, + { + "cell_type": "code", + "execution_count": null, + "id": "400e483d-eca7-4fdd-a9e0-71467e1af8d8", + "metadata": {}, + "outputs": [], + "source": [ + "%pip install -q pandas openpyxl" + ] + }, + { + "cell_type": "markdown", + "id": "acbe1247-e1a1-4da8-a95a-7b41f4836937", + "metadata": {}, + "source": [ + "Now we can use it together with `capellambse`:" + ] + }, + { + "cell_type": "code", + "execution_count": 4, + "id": "833220d0", + "metadata": {}, + "outputs": [ + { + "data": { + "text/html": [ + "
\n", + "\n", + "\n", + " \n", + " \n", + " \n", + " \n", + " \n", + " \n", + " \n", + " \n", + " \n", + " \n", + " \n", + " \n", + " \n", + " \n", + " \n", + " \n", + " \n", + " \n", + " \n", + " \n", + " \n", + " \n", + " \n", + " \n", + " \n", + " \n", + " \n", + " \n", + " \n", + " \n", + " \n", + " \n", + " \n", + " \n", + " \n", + " \n", + " \n", + " \n", + " \n", + "
actorfunctions
0MultiportLAF 1
1Prof. S. SnapeTeaching; maintain a layer of defense for the ...
2Voldemortno functions assigned
3R. Weasleyassist Harry; break school rules
4Prof. A. P. W. B. Dumbledoremanage the school; advise Harry
5Harry J. Potterkill He Who Must Not Be Named
\n", + "
" + ], + "text/plain": [ + " actor \\\n", + "0 Multiport \n", + "1 Prof. S. Snape \n", + "2 Voldemort \n", + "3 R. Weasley \n", + "4 Prof. A. P. W. B. Dumbledore \n", + "5 Harry J. Potter \n", + "\n", + " functions \n", + "0 LAF 1 \n", + "1 Teaching; maintain a layer of defense for the ... \n", + "2 no functions assigned \n", + "3 assist Harry; break school rules \n", + "4 manage the school; advise Harry \n", + "5 kill He Who Must Not Be Named " + ] + }, + "execution_count": 4, + "metadata": {}, + "output_type": "execute_result" + } + ], + "source": [ + "import pandas as pd\n", + "\n", + "data = []\n", + "for actor in model.la.all_actors:\n", + " actor_functions = \"; \".join([function.name for function in actor.functions] or [\"no functions assigned\"])\n", + " data.append(dict(actor=actor.name, functions=actor_functions))\n", + "df = pd.DataFrame(data)\n", + "df" + ] + }, + { + "cell_type": "markdown", + "id": "6e04c482", + "metadata": {}, + "source": [ + "and any `pandas.DataFrame` can always be turned into an Excel Spreadsheet, just like that:" + ] + }, + { + "cell_type": "code", + "execution_count": 5, + "id": "10af24a2", + "metadata": {}, + "outputs": [], + "source": [ + "df.to_excel(\"01_intro_actor_functions.xlsx\")" + ] + }, + { + "cell_type": "markdown", + "id": "5370bc56", + "metadata": { + "tags": [] + }, + "source": [ + "you can check the resulting file in the folder next to this notebook (right after you run the above cell)\n", + "\n", + "Now that we've seen the basics, lets do something visually cool." + ] + }, + { + "cell_type": "markdown", + "id": "5afcdfdd-79e0-4e07-b321-d92a11f1d082", + "metadata": { + "tags": [] + }, + "source": [ + "## Example 2: working with diagrams\n", + "\n", + "The below code will find some diagrams for us." + ] + }, + { + "cell_type": "code", + "execution_count": 6, + "id": "eb5f8747", + "metadata": {}, + "outputs": [ + { + "name": "stdout", + "output_type": "stream", + "text": [ + "[LAB] Wizzard Education\n", + "[LAB] Test Component Port Filter\n", + "[LAB] Hidden Wizzard Education\n" + ] + } + ], + "source": [ + "for diagram in model.la.diagrams.by_type('LAB'):\n", + " print(diagram.name)" + ] + }, + { + "cell_type": "markdown", + "id": "66609218", + "metadata": {}, + "source": [ + "We can analyze which model objects are shown in a particular diagram." + ] + }, + { + "cell_type": "code", + "execution_count": 7, + "id": "71bf090e", + "metadata": { + "scrolled": true + }, + "outputs": [ + { + "data": { + "text/html": [ + "
  1. Part "Hogwarts" (101ffa60-f8a2-4ea2-a0d8-d10910ceac06)
  2. LogicalFunction "produce Great Wizards" (0e71a0d3-0a18-4671-bba0-71b5f88f95dd)
  3. LogicalFunction "protect Students against the Death Eaters" (264fb47d-67b7-4bdc-8d06-8a0e5139edbf)
  4. Part "Campus" (a3194240-cd17-4998-8f8b-785233487ec3)
  5. Part "School" (018a8ae9-8e8e-4aea-8191-4abf844a79e3)
  6. LogicalFunction "educate Wizards" (957c5799-1d4a-4ac0-b5de-33a65bf1519c)
  7. Part "Whomping Willow" (1188fc31-789b-424f-a2d4-06791873a351)
  8. LogicalFunction "defend the surrounding area against Intruders" (7f2936ab-0b54-4e92-9f0c-85a9f0981959)
  9. Part "Prof. A. P. W. B. Dumbledore" (4c1f2b5d-0641-42c7-911f-7a42928580b8)
  10. LogicalFunction "manage the school" (f708bc29-d69f-42a0-90cc-11fc01054cd0)
  11. LogicalFunction "advise Harry" (beaf5ba4-8fa9-4342-911f-0266bb29be45)
  12. Part "Prof. S. Snape" (ccbad61a-39dc-4af8-8199-3fee30de2f1d)
  13. LogicalFunction "Teaching" (a7acb298-d14b-4707-a419-fea272434541)
  14. LogicalFunction "maintain a layer of defense for the Sorcerer's Stone" (4a2a7f3c-d223-4d44-94a7-50dd2906a70c)
  15. ComponentExchange "Headmaster Responsibilities" (c0bc49e1-8043-4418-8c0a-de6c6b749eab)
  16. ComponentExchange "Teacher Responsibilities" (9cbdd233-aff5-47dd-9bef-9be1277c77c3)
  17. Part "Harry J. Potter" (26543596-7646-4d81-8f15-c4e01ec930a7)
  18. LogicalFunction "kill He Who Must Not Be Named" (aa9931e3-116c-461e-8215-6b9fdbdd4a1b)
  19. Part "R. Weasley" (4d31caaf-210e-4bdf-982e-cdecbc80c947)
  20. LogicalFunction "assist Harry" (c1a42acc-1f53-42bb-8404-77a5c08c414b)
  21. LogicalFunction "break school rules" (edbd1ad4-31c0-4d53-b856-3ffa60e0e99b)
  22. ComponentExchange "Punishment" (85a1fb20-38ea-4d77-acd7-90a8c44dc695)
  23. FunctionalExchange "wizardry" (6545a77d-d224-4662-a5b2-3c016b78e33d)
  24. PortAllocation "" (a4e2bf11-0705-4f20-bf73-5fa5519954f7)
  25. PortAllocation "" (7d31ab50-63d6-46bb-bf53-1716175beae3)
  26. PortAllocation "" (317715cb-376c-4df6-a32c-433e3c081f8d)
  27. ComponentExchange "Learning" (3b3fc202-be5c-49ae-bf2f-1d61daf3bb50)
  28. PortAllocation "" (c1019d06-f376-48e3-832e-634a8ec59463)
  29. PortAllocation "" (4cbdf5fd-7268-470b-9811-b62ad67fded1)
  30. FunctionalExchange "assistance" (241f3901-11f0-4b00-a903-ed158cce73de)
  31. FunctionalExchange "friendship" (1bbb9b2d-517c-4f77-a35c-b3aa3f9422b8)
  32. PortAllocation "" (18fa81ee-8b16-4815-86ea-0c287ace43d8)
  33. ComponentExchange "Help for Harry" (d8655737-39ab-4482-a934-ee847c7ff6bd)
  34. FunctionalExchange "punish" (96a0cf4c-adfe-4490-92d1-bcf75ee77004)
  35. FunctionalExchange "educate & mature" (09efaeb7-2d50-40ed-a4da-46afcb9ca7a1)
  36. FunctionalExchange "Knowledge" (b1a817bc-40a9-4fc4-b62c-8dea4aa28915)
  37. PortAllocation "" (dda7a62a-f25f-46d8-8f05-867c616914c1)
  38. PortAllocation "" (fee1fff5-d751-401b-bb3c-2114a74f0c8a)
  39. PortAllocation "" (0f6e1aa0-942a-40a9-930f-c7df34b9d8eb)
  40. ComponentExchange "Care" (c31491db-817d-44b3-a27c-67e9cc1e06a2)
  41. PortAllocation "" (98760017-b3a3-46ca-b1ef-87eee9ea1600)
  42. PortAllocation "" (74bd0ab3-6a28-4025-822e-90201445a56e)
  43. PortAllocation "" (3ed5ae4f-8a4e-4690-9088-655990a1b77b)
  44. PortAllocation "" (299b98b8-8716-4dbc-bc7e-4b9349778c26)
  45. PortAllocation "" (14cabdd9-c36f-4e01-ad09-110f906ad725)
  46. PortAllocation "" (6d882e28-4208-41d0-b8a5-3a19e1805a34)
  47. FunctionalExchange "educate" (cdc69c5e-ddd8-4e59-8b99-f510400650aa)
  48. PortAllocation "" (c90bb30d-e36b-46a3-a3a1-e39fdcb519be)
" + ], + "text/plain": [ + "[0] \n", + "[1] \n", + "[2] \n", + "[3] \n", + "[4] \n", + "[5] \n", + "[6] \n", + "[7] \n", + "[8] \n", + "[9] \n", + "[10] \n", + "[11] \n", + "[12] \n", + "[13] \n", + "[14] \n", + "[15] \n", + "[16] \n", + "[17] \n", + "[18] \n", + "[19] \n", + "[20] \n", + "[21] \n", + "[22] \n", + "[23] \n", + "[24] \n", + "[25] \n", + "[26] \n", + "[27] \n", + "[28] \n", + "[29] \n", + "[30] \n", + "[31] \n", + "[32] \n", + "[33] \n", + "[34] \n", + "[35] \n", + "[36] \n", + "[37] \n", + "[38] \n", + "[39] \n", + "[40] \n", + "[41] \n", + "[42] \n", + "[43] \n", + "[44] \n", + "[45] \n", + "[46] \n", + "[47] " + ] + }, + "execution_count": 7, + "metadata": {}, + "output_type": "execute_result" + } + ], + "source": [ + "diagram = model.diagrams.by_name('[LAB] Wizzard Education')\n", + "diagram.nodes" + ] + }, + { + "cell_type": "markdown", + "id": "eaaffccd", + "metadata": {}, + "source": [ + "And again there are warnings - there are quite a few visual filters in Capella and we are not handling all of those yet but mostly those that are used in our projects. The filter coverage will eventally improve, stay tuned." + ] + }, + { + "cell_type": "markdown", + "id": "9ec8f512-6e78-46fe-9dce-ed6fbfc8ad7a", + "metadata": {}, + "source": [ + "And finally, you can display the diagram right in the notebook." + ] + }, + { + "cell_type": "code", + "execution_count": 8, + "id": "cc09fd22", + "metadata": {}, + "outputs": [ + { + "data": { + "image/png": "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", + "text/plain": [ + "" + ] + }, + "execution_count": 8, + "metadata": {}, + "output_type": "execute_result" + } + ], + "source": [ + "diagram" + ] + }, + { + "cell_type": "markdown", + "id": "f678a3ee-ac7e-4ce4-a8e2-0d41c612800e", + "metadata": {}, + "source": [ + "We use SVG diagrams a lot since they look great in documentation, are zoomable and really light-weight. To make integrating them into a pipeline easier, we also support some derived formats, which you can access using `.as_` style attributes. With some additional dependencies set up (see the README), capellambse can also automatically convert these images to PNG format." + ] + }, + { + "cell_type": "code", + "execution_count": 9, + "id": "b5699e75-2e4c-4b8e-81af-51be00ff6dc3", + "metadata": {}, + "outputs": [ + { + "name": "stdout", + "output_type": "stream", + "text": [ + "` tag, using the above `data:` URI as `src`\n", + "print(diagram.as_png[:10], \"...\") # A raw PNG byte stream, which can be written to a `.png` file" + ] + }, + { + "cell_type": "markdown", + "id": "21e0b9c2", + "metadata": {}, + "source": [ + "Lets now try something else - we check if function port has any protocols (state machines) underneath:" + ] + }, + { + "cell_type": "code", + "execution_count": 10, + "id": "cf91c71a", + "metadata": {}, + "outputs": [ + { + "data": { + "text/html": [ + "
  1. StateMachine "FaultStates" (06cefb2b-534e-4453-9aba-fe53329197ad)
" + ], + "text/plain": [ + "[0] " + ] + }, + "execution_count": 10, + "metadata": {}, + "output_type": "execute_result" + } + ], + "source": [ + "fnc = model.la.all_functions.by_name(\"defend the surrounding area against Intruders\")\n", + "stms = fnc.outputs[0].state_machines\n", + "stms" + ] + }, + { + "cell_type": "markdown", + "id": "dc8069df", + "metadata": {}, + "source": [ + "and we can also check what states it could have:" + ] + }, + { + "cell_type": "code", + "execution_count": 11, + "id": "b5751a0d", + "metadata": {}, + "outputs": [ + { + "data": { + "text/html": [ + "
  1. State "normal defence" (e494e247-efce-4258-9cc6-fd799dbb0adf)
  2. State "erroneous defence" (81f3de46-4596-41b0-8569-c3c21161a2f6)
  3. State "no defence" (5b6a03d8-0ef9-4b2b-9a50-a745f490d663)
" + ], + "text/plain": [ + "[0] \n", + "[1] \n", + "[2] " + ] + }, + "execution_count": 11, + "metadata": {}, + "output_type": "execute_result" + } + ], + "source": [ + "stms[0].regions[0].states" + ] + }, + { + "cell_type": "markdown", + "id": "86e23011", + "metadata": {}, + "source": [ + "This concludes our introduction. There is a lot more you can do with the library - feel free to explore the examples collection or create an issue to ask for a specific use-case example and you may see it around pretty soon." + ] + } + ], + "metadata": { + "kernelspec": { + "display_name": "Python 3 (ipykernel)", + "language": "python", + "name": "python3" + }, + "language_info": { + "codemirror_mode": { + "name": "ipython", + "version": 3 + }, + "file_extension": ".py", + "mimetype": "text/x-python", + "name": "python", + "nbconvert_exporter": "python", + "pygments_lexer": "ipython3", + "version": "3.11.5" + }, + "vscode": { + "interpreter": { + "hash": "c5ea7dc634d8047a259e5b898f154d237fbe6934b444b1a949475949608d751e" + } + } + }, + "nbformat": 4, + "nbformat_minor": 5 +} diff --git a/_sources/examples/02 Intro to Physical Architecture API.ipynb.txt b/_sources/examples/02 Intro to Physical Architecture API.ipynb.txt new file mode 100644 index 000000000..f358835dd --- /dev/null +++ b/_sources/examples/02 Intro to Physical Architecture API.ipynb.txt @@ -0,0 +1,726 @@ +{ + "cells": [ + { + "cell_type": "markdown", + "id": "b02a2030", + "metadata": {}, + "source": [ + "# Introduction to Physical Architecture API\n", + "\n", + "**Note**: In this notebook we will use `pandas` dataframes library to construct and visualize tables, **if you don't have pandas installed** in the current environment you may want to do so by running the cell below." + ] + }, + { + "cell_type": "code", + "execution_count": null, + "id": "4c181cfb", + "metadata": { + "editable": true, + "slideshow": { + "slide_type": "" + }, + "tags": [] + }, + "outputs": [], + "source": [ + "%pip install pandas" + ] + }, + { + "cell_type": "markdown", + "id": "3d39f975", + "metadata": {}, + "source": [ + "The cell below loads our test model so we could play with it and silences warnings." + ] + }, + { + "cell_type": "code", + "execution_count": 1, + "id": "1386d2f7", + "metadata": { + "scrolled": true + }, + "outputs": [], + "source": [ + "import capellambse\n", + "import logging\n", + "import pandas as pd\n", + "\n", + "logging.getLogger().setLevel(logging.CRITICAL)\n", + "model = capellambse.MelodyModel(\n", + " \"../../../tests/data/melodymodel/5_0/Melody Model Test.aird\",\n", + " jupyter_untrusted=True,\n", + ")" + ] + }, + { + "cell_type": "markdown", + "id": "9f73eb7e", + "metadata": {}, + "source": [ + "but before we jump into code, lets have a look first at Capella metamodel concerning the Physical Architecture layer (PA).\n", + "\n", + "Things in PA are very similar to what we see in SysML when it comes to `ibd`s (Internal Block Diagrams) - the boxes we see on those are `Part`s that are instanciated from `Block` objects. Same happens in Capella - the boxes we see on `PAB` diagrams are `Parts` that were instanciated from `PhysicalComponent`s. Here also comes the very special difference of Capella - unless you explicitly enable **part re-use**, `PhysicalComponent` will always have only one `Part`. This is the default behavior of Capella. \n", + "\n", + "Our API should support both cases but at the moment we don't use models with **part re-use** enabled in production yet and so don't test the library against this case. Yet we do implement Parts and support many parts - one component relationship model.\n", + "\n", + "One more issue to mention - rendering PA diagrams outside of Capella was never a high priority so the resulting representations of PABs rendered without Capella are not very accurate at the moment. We hope to improve it soon though. If you still do want to see how it looks like when we render it right now - uncomment the `# diagram` in the cell below" + ] + }, + { + "cell_type": "code", + "execution_count": 2, + "id": "8b555976", + "metadata": {}, + "outputs": [ + { + "data": { + "image/png": "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", + "text/plain": [ + "" + ] + }, + "execution_count": 2, + "metadata": {}, + "output_type": "execute_result" + } + ], + "source": [ + "diagram = model.pa.diagrams.by_name(\"[PAB] A sample vehicle arch\")\n", + "diagram" + ] + }, + { + "cell_type": "markdown", + "id": "1c479ca3", + "metadata": {}, + "source": [ + "## Example 1: List components that are visible on a diagram\n", + "\n", + "To start, let's get all parts on that diagram and turn them into PhysicalComponents." + ] + }, + { + "cell_type": "code", + "execution_count": 3, + "id": "711574aa", + "metadata": { + "scrolled": true + }, + "outputs": [ + { + "data": { + "text/html": [ + "
  1. PhysicalComponent "Vehicle" (b327d900-abd2-4138-a111-9ff0684739d8)
  2. PhysicalComponent "Equipment compartment" (3d68852d-fcc0-452c-af12-a2fbe22f81fa)
  3. PhysicalComponent "Server" (9137f463-7497-40c2-b20a-897158fdba9a)
  4. PhysicalComponent "Compute Card 1" (63be604e-883e-41ea-9023-fc74f29906fe)
  5. PhysicalComponent "Card 1 OS" (7b188ad0-0d82-4b2c-9913-45292e537871)
  6. PhysicalComponent "Camera Driver SWC" (74067f56-33bf-47f5-bb8b-f3604097f653)
  7. PhysicalComponent "App 1 SWC" (b80a6fcc-8d35-4675-a2e6-60efcbd61e27)
  8. PhysicalComponent "Cooling Fan" (65e82f3f-c5b7-44c1-bfea-8e20bb0230be)
  9. PhysicalComponent "Compute Card 2" (3a982128-3281-4d37-8838-a6058b7a25d9)
  10. PhysicalComponent "Card 2 OS" (09e19313-c824-467f-9fb5-95ed8b4e2d51)
  11. PhysicalComponent "App 2 SWC" (ca5af12c-5259-4844-aaac-9ca9f84aa90b)
  12. PhysicalComponent "Network Switch" (b51ccc6f-5f96-4e28-b90e-72463a3b50cf)
  13. PhysicalComponent "Switch Firmware" (c78b5d7c-be0c-4ed4-9d12-d447cb39304e)
  14. PhysicalComponent "Switch Configuration" (23c47b69-7352-481d-be88-498fb351adbe)
  15. PhysicalComponent "Sensor compartment" (3f416925-9d8a-4e9c-99f3-e912efb23d2f)
  16. PhysicalComponent "Camera Assembly" (5bfc516b-c20d-4007-9a38-5ba0e889d0a4)
  17. PhysicalComponent "Camera Firmware" (db2d86d7-48ee-478b-a6fc-d6387ab0032e)
" + ], + "text/plain": [ + "[0] \n", + "[1] \n", + "[2] \n", + "[3] \n", + "[4] \n", + "[5] \n", + "[6] \n", + "[7] \n", + "[8] \n", + "[9] \n", + "[10] \n", + "[11] \n", + "[12] \n", + "[13] \n", + "[14] \n", + "[15] \n", + "[16] " + ] + }, + "execution_count": 3, + "metadata": {}, + "output_type": "execute_result" + } + ], + "source": [ + "components_on_diagram = diagram.nodes.by_type(\"Part\").map(\"type\")\n", + "components_on_diagram" + ] + }, + { + "cell_type": "markdown", + "id": "aab38c3e", + "metadata": {}, + "source": [ + "We could also get all components across the entire PA layer by doing `model.pa.all_components`, but we will not go there in this example.\n", + "\n", + "We can review any single component from that list above:" + ] + }, + { + "cell_type": "code", + "execution_count": 4, + "id": "21ef5c4d", + "metadata": {}, + "outputs": [ + { + "data": { + "text/html": [ + "

Cooling Fan (org.polarsys.capella.core.data.pa:PhysicalComponent)

allocated_functions

(Empty list)

applied_property_value_groups

(Empty list)

applied_property_values

(Empty list)

components

(Empty list)

constraints

(Empty list)

context_diagramContext of Cooling Fan (uuid: 65e82f3f-c5b7-44c1-bfea-8e20bb0230be_context)
deployed_components

(Empty list)

deploying_componentsBackreference to PhysicalComponent - omitted: can be slow to compute. Display this property directly to show.
description
diagrams

(Empty list)

exchanges

(Empty list)

filtering_criteria

(Empty list)

functions

(Empty list)

is_abstractFalse
is_actorFalse
is_humanFalse
kind<Kind.HARDWARE: 2>
nameCooling Fan
nature<Nature.NODE: 1>
owned_components

(Empty list)

ownerPhysicalComponent "Compute Card 1" (63be604e-883e-41ea-9023-fc74f29906fe)
parentPhysicalComponent "Compute Card 1" (63be604e-883e-41ea-9023-fc74f29906fe)
partsBackreference to Part - omitted: can be slow to compute. Display this property directly to show.
physical_links

(Empty list)

physical_paths

(Empty list)

physical_ports

(Empty list)

ports

(Empty list)

progress_statusNOT_SET
property_value_groups

(Empty list)

property_values

(Empty list)

pvmt<capellambse.extensions.pvmt.PropertyValueProxy object at 0x7fc7990cc790>
realized_components

(Empty list)

realized_logical_components

(Empty list)

realizing_componentsBackreference to - omitted: can be slow to compute. Display this property directly to show.
related_exchangesBackreference to ComponentExchange - omitted: can be slow to compute. Display this property directly to show.
requirements

(Empty list)

state_machines

(Empty list)

summaryNone
traces

(Empty list)

uuid65e82f3f-c5b7-44c1-bfea-8e20bb0230be
xtypeorg.polarsys.capella.core.data.pa:PhysicalComponent
" + ], + "text/plain": [ + "\n", + ".allocated_functions = []\n", + ".applied_property_value_groups = []\n", + ".applied_property_values = []\n", + ".components = []\n", + ".constraints = []\n", + ".context_diagram = \n", + ".deployed_components = []\n", + ".deploying_components = ... # backreference to PhysicalComponent - omitted: can be slow to compute\n", + ".description = ''\n", + ".diagrams = []\n", + ".exchanges = []\n", + ".filtering_criteria = []\n", + ".functions = []\n", + ".is_abstract = False\n", + ".is_actor = False\n", + ".is_human = False\n", + ".kind = \n", + ".name = 'Cooling Fan'\n", + ".nature = \n", + ".owned_components = []\n", + ".owner = \n", + ".parent = \n", + ".parts = ... # backreference to Part - omitted: can be slow to compute\n", + ".physical_links = []\n", + ".physical_paths = []\n", + ".physical_ports = []\n", + ".ports = []\n", + ".progress_status = 'NOT_SET'\n", + ".property_value_groups = []\n", + ".property_values = []\n", + ".pvmt = \n", + ".realized_components = []\n", + ".realized_logical_components = []\n", + ".realizing_components = ... # backreference to - omitted: can be slow to compute\n", + ".related_exchanges = ... # backreference to ComponentExchange - omitted: can be slow to compute\n", + ".requirements = []\n", + ".state_machines = []\n", + ".summary = None\n", + ".traces = []\n", + ".uuid = '65e82f3f-c5b7-44c1-bfea-8e20bb0230be'\n", + ".xtype = 'org.polarsys.capella.core.data.pa:PhysicalComponent'" + ] + }, + "execution_count": 4, + "metadata": {}, + "output_type": "execute_result" + } + ], + "source": [ + "components_on_diagram[7]" + ] + }, + { + "cell_type": "markdown", + "id": "22343924", + "metadata": {}, + "source": [ + "However we may need just a few of those attributes in a view.\n", + "\n", + "Now that we have a list of components lets collect some of the attributes of interest in a table. To keep it simple we'll introduce an attribute extractor function that will turn fields of interest into a nice dictionary." + ] + }, + { + "cell_type": "code", + "execution_count": 5, + "id": "63ff286c", + "metadata": {}, + "outputs": [], + "source": [ + "def extract_attrs_of_interest(component):\n", + " return dict(\n", + " name=component.name,\n", + " nature=component.nature,\n", + " kind=component.kind,\n", + " components=\"; \".join([cmp.name for cmp in component.components])\n", + " )" + ] + }, + { + "cell_type": "markdown", + "id": "55d509d0", + "metadata": {}, + "source": [ + "We can then apply that extractor function to our list of components and use `pandas` to display it for us in a tabular form:" + ] + }, + { + "cell_type": "code", + "execution_count": 6, + "id": "6d8ff25c", + "metadata": {}, + "outputs": [ + { + "data": { + "text/html": [ + "
\n", + "\n", + "\n", + " \n", + " \n", + " \n", + " \n", + " \n", + " \n", + " \n", + " \n", + " \n", + " \n", + " \n", + " \n", + " \n", + " \n", + " \n", + " \n", + " \n", + " \n", + " \n", + " \n", + " \n", + " \n", + " \n", + " \n", + " \n", + " \n", + " \n", + " \n", + " \n", + " \n", + " \n", + " \n", + " \n", + " \n", + " \n", + " \n", + " \n", + " \n", + " \n", + " \n", + " \n", + " \n", + " \n", + " \n", + " \n", + " \n", + " \n", + " \n", + " \n", + " \n", + " \n", + " \n", + " \n", + " \n", + " \n", + " \n", + " \n", + " \n", + " \n", + " \n", + " \n", + " \n", + " \n", + " \n", + " \n", + " \n", + " \n", + " \n", + " \n", + " \n", + " \n", + " \n", + " \n", + " \n", + " \n", + " \n", + " \n", + " \n", + " \n", + " \n", + " \n", + " \n", + " \n", + " \n", + " \n", + " \n", + " \n", + " \n", + " \n", + " \n", + " \n", + " \n", + " \n", + " \n", + " \n", + " \n", + " \n", + " \n", + " \n", + " \n", + " \n", + " \n", + " \n", + " \n", + " \n", + " \n", + " \n", + " \n", + " \n", + " \n", + " \n", + " \n", + " \n", + " \n", + " \n", + " \n", + " \n", + " \n", + " \n", + " \n", + " \n", + " \n", + " \n", + " \n", + " \n", + " \n", + " \n", + " \n", + " \n", + " \n", + "
namenaturekindcomponents
0VehicleNODESOFTWARE_DEPLOYMENT_UNITEquipment compartment; Sensor compartment
1Equipment compartmentNODEFACILITIESServer; Network Switch
2ServerNODEHARDWARECompute Card 1; Compute Card 2
3Compute Card 1NODEHARDWARECard 1 OS; Cooling Fan
4Card 1 OSBEHAVIORSERVICESCamera Driver SWC; App 1 SWC
5Camera Driver SWCBEHAVIORSOFTWARE
6App 1 SWCBEHAVIORSOFTWARE
7Cooling FanNODEHARDWARE
8Compute Card 2NODEHARDWARECard 2 OS
9Card 2 OSBEHAVIORSOFTWAREApp 2 SWC
10App 2 SWCBEHAVIORSOFTWARE
11Network SwitchNODEHARDWARESwitch Firmware; Switch Configuration
12Switch FirmwareBEHAVIORHARDWARE_COMPUTER
13Switch ConfigurationBEHAVIORDATA
14Sensor compartmentNODEFACILITIESCamera Assembly
15Camera AssemblyNODEHARDWARECamera Firmware
16Camera FirmwareBEHAVIORSOFTWARE_EXECUTION_UNIT
\n", + "
" + ], + "text/plain": [ + " name nature kind \\\n", + "0 Vehicle NODE SOFTWARE_DEPLOYMENT_UNIT \n", + "1 Equipment compartment NODE FACILITIES \n", + "2 Server NODE HARDWARE \n", + "3 Compute Card 1 NODE HARDWARE \n", + "4 Card 1 OS BEHAVIOR SERVICES \n", + "5 Camera Driver SWC BEHAVIOR SOFTWARE \n", + "6 App 1 SWC BEHAVIOR SOFTWARE \n", + "7 Cooling Fan NODE HARDWARE \n", + "8 Compute Card 2 NODE HARDWARE \n", + "9 Card 2 OS BEHAVIOR SOFTWARE \n", + "10 App 2 SWC BEHAVIOR SOFTWARE \n", + "11 Network Switch NODE HARDWARE \n", + "12 Switch Firmware BEHAVIOR HARDWARE_COMPUTER \n", + "13 Switch Configuration BEHAVIOR DATA \n", + "14 Sensor compartment NODE FACILITIES \n", + "15 Camera Assembly NODE HARDWARE \n", + "16 Camera Firmware BEHAVIOR SOFTWARE_EXECUTION_UNIT \n", + "\n", + " components \n", + "0 Equipment compartment; Sensor compartment \n", + "1 Server; Network Switch \n", + "2 Compute Card 1; Compute Card 2 \n", + "3 Card 1 OS; Cooling Fan \n", + "4 Camera Driver SWC; App 1 SWC \n", + "5 \n", + "6 \n", + "7 \n", + "8 Card 2 OS \n", + "9 App 2 SWC \n", + "10 \n", + "11 Switch Firmware; Switch Configuration \n", + "12 \n", + "13 \n", + "14 Camera Assembly \n", + "15 Camera Firmware \n", + "16 " + ] + }, + "execution_count": 6, + "metadata": {}, + "output_type": "execute_result" + } + ], + "source": [ + "pd.DataFrame(list(map(extract_attrs_of_interest, components_on_diagram)))" + ] + }, + { + "cell_type": "markdown", + "id": "91a9ab15", + "metadata": {}, + "source": [ + "## Example 2: Create HW-SW allocation table\n", + "\n", + "Now let's assume that components of \"Node\" nature are hardware things and \"Behavior\" corresponds to software components. Assuming that, let's identify leaf hardware components (lowest replaceable units), and for each of those indicate which software components they have.\n", + "\n", + "To get there, let's first filter out the list of components of \"Node\" nature that have at least one subcomponent of \"Behavior\" nature. We'll use the `.filter` method of our object list. To use that method we'll need to provide a function or lambda that determines whether an object should be selected. We can also make use of the implicit \"truthiness\" of non-empty list attributes:" + ] + }, + { + "cell_type": "code", + "execution_count": 7, + "id": "01644c9d", + "metadata": {}, + "outputs": [ + { + "data": { + "text/html": [ + "
  1. PhysicalComponent "Sub PC" (793e6da2-d019-4716-a5c5-af8ad550ca5e)
  2. PhysicalComponent "Deploy Sub PC" (8a6c6ec9-095d-4d8b-9728-69bc79af5f27)
  3. PhysicalComponent "Compute Card 1" (63be604e-883e-41ea-9023-fc74f29906fe)
  4. PhysicalComponent "Compute Card 2" (3a982128-3281-4d37-8838-a6058b7a25d9)
  5. PhysicalComponent "Network Switch" (b51ccc6f-5f96-4e28-b90e-72463a3b50cf)
  6. PhysicalComponent "Camera Assembly" (5bfc516b-c20d-4007-9a38-5ba0e889d0a4)
  7. PhysicalComponent "Computer" (b14ff190-9198-4d05-95db-b121c11e9f17)
  8. PhysicalComponent "ISP Network" (221cef9f-0582-419c-b67e-96ddb678dd4c)
" + ], + "text/plain": [ + "[0] \n", + "[1] \n", + "[2] \n", + "[3] \n", + "[4] \n", + "[5] \n", + "[6] \n", + "[7] " + ] + }, + "execution_count": 7, + "metadata": {}, + "output_type": "execute_result" + } + ], + "source": [ + "cmps = model.pa.all_components[1:].by_nature(\"NODE\")\n", + "# Note: the above [1:] assumes that 0th component is the root component and is not of interest (+ see issue #41)\n", + "cmps_with_sw = cmps.filter(lambda i: i.components.by_nature(\"BEHAVIOR\"))\n", + "cmps_with_sw" + ] + }, + { + "cell_type": "markdown", + "id": "4286bd9d", + "metadata": {}, + "source": [ + "For sure we could come to the above list filtering by `kind` attribute, however in practice not all projects strictly use the `kind` attribute / it needs manual setting and maintenance and therefore is a bit less reliable.\n", + "\n", + "Our next stop is to list the SW components of those HW components. As SW components may be nested (i.e. apps on OS or partitions, etc.) we would simply \"flatten\" that hierarchy. We can create function `get_sw_components` that would recursively crawl down the SW components tree and give us back a flat list and test it on one component." + ] + }, + { + "cell_type": "code", + "execution_count": 8, + "id": "70abf6f9", + "metadata": {}, + "outputs": [ + { + "data": { + "text/html": [ + "
  1. PhysicalComponent "Card 1 OS" (7b188ad0-0d82-4b2c-9913-45292e537871)
  2. PhysicalComponent "Camera Driver SWC" (74067f56-33bf-47f5-bb8b-f3604097f653)
  3. PhysicalComponent "App 1 SWC" (b80a6fcc-8d35-4675-a2e6-60efcbd61e27)
" + ], + "text/plain": [ + "[0] \n", + "[1] \n", + "[2] " + ] + }, + "execution_count": 8, + "metadata": {}, + "output_type": "execute_result" + } + ], + "source": [ + "def get_sw_components(sw_component):\n", + " subcmp = sw_component.components.by_nature(\"BEHAVIOR\")\n", + " for cmp in subcmp:\n", + " subcmp += get_sw_components(cmp)\n", + " return subcmp\n", + "\n", + "get_sw_components(cmps_with_sw.by_name(\"Compute Card 1\"))" + ] + }, + { + "cell_type": "markdown", + "id": "8086d62c", + "metadata": {}, + "source": [ + "Let's apply the `get_sw_components` function to the complete list of components with SW, `cmps_with_sw`, and turn the results into a table. In this table we'll serialize SW components into a semicolon-separated string:" + ] + }, + { + "cell_type": "code", + "execution_count": 9, + "id": "faa00eb0", + "metadata": {}, + "outputs": [ + { + "data": { + "text/html": [ + "
\n", + "\n", + "\n", + " \n", + " \n", + " \n", + " \n", + " \n", + " \n", + " \n", + " \n", + " \n", + " \n", + " \n", + " \n", + " \n", + " \n", + " \n", + " \n", + " \n", + " \n", + " \n", + " \n", + " \n", + " \n", + " \n", + " \n", + " \n", + " \n", + " \n", + " \n", + " \n", + " \n", + " \n", + " \n", + " \n", + " \n", + " \n", + " \n", + " \n", + " \n", + " \n", + " \n", + " \n", + " \n", + " \n", + " \n", + " \n", + " \n", + " \n", + " \n", + " \n", + "
hardware_componentdeployed_sw_components
0Sub PCPC 16
1Deploy Sub PCPC 17
2Compute Card 1Card 1 OS; Camera Driver SWC; App 1 SWC
3Compute Card 2Card 2 OS; App 2 SWC
4Network SwitchSwitch Firmware; Switch Configuration
5Camera AssemblyCamera Firmware
6ComputerMail client
7ISP NetworkMail server
\n", + "
" + ], + "text/plain": [ + " hardware_component deployed_sw_components\n", + "0 Sub PC PC 16\n", + "1 Deploy Sub PC PC 17\n", + "2 Compute Card 1 Card 1 OS; Camera Driver SWC; App 1 SWC\n", + "3 Compute Card 2 Card 2 OS; App 2 SWC\n", + "4 Network Switch Switch Firmware; Switch Configuration\n", + "5 Camera Assembly Camera Firmware\n", + "6 Computer Mail client\n", + "7 ISP Network Mail server" + ] + }, + "execution_count": 9, + "metadata": {}, + "output_type": "execute_result" + } + ], + "source": [ + "def describe_hw_component_sw_allocations(hw_component):\n", + " return dict(\n", + " hardware_component=hw_component.name,\n", + " deployed_sw_components=\"; \".join(i.name for i in get_sw_components(hw_component))\n", + " )\n", + "\n", + "df = pd.DataFrame(list(map(describe_hw_component_sw_allocations, cmps_with_sw)))\n", + "df" + ] + }, + { + "cell_type": "markdown", + "id": "851b0540", + "metadata": {}, + "source": [ + "To complete the picture, we could also list all HW components that don't have software (so that a sanity check could be performed):" + ] + }, + { + "cell_type": "code", + "execution_count": 10, + "id": "e48d4998", + "metadata": {}, + "outputs": [ + { + "data": { + "text/html": [ + "
  1. PhysicalComponent "PC 1" (8a6d68c8-ac3d-4654-a07e-ada7adeed09f)
  2. PhysicalComponent "PC 3" (f5d7980d-e1e9-4515-8bb0-be7e80ac5839)
  3. PhysicalComponent "Vehicle" (a2c7f619-b38a-4b92-94a5-cbaa631badfc)
  4. PhysicalComponent "Vehicle" (b327d900-abd2-4138-a111-9ff0684739d8)
  5. PhysicalComponent "Equipment compartment" (3d68852d-fcc0-452c-af12-a2fbe22f81fa)
  6. PhysicalComponent "Server" (9137f463-7497-40c2-b20a-897158fdba9a)
  7. PhysicalComponent "Cooling Fan" (65e82f3f-c5b7-44c1-bfea-8e20bb0230be)
  8. PhysicalComponent "Sensor compartment" (3f416925-9d8a-4e9c-99f3-e912efb23d2f)
  9. PhysicalComponent "Router" (69bfe48d-78f0-4b1f-89c0-917dc2339ff9)
  10. PhysicalComponent "PA 1" (a0847e9c-8b82-407d-8143-e908e2db97a1)
" + ], + "text/plain": [ + "[0] \n", + "[1] \n", + "[2] \n", + "[3] \n", + "[4] \n", + "[5] \n", + "[6] \n", + "[7] \n", + "[8] \n", + "[9] " + ] + }, + "execution_count": 10, + "metadata": {}, + "output_type": "execute_result" + } + ], + "source": [ + "cmps_without_sw = cmps.filter(lambda i: not i.components.by_nature(\"BEHAVIOR\"))\n", + "cmps_without_sw" + ] + } + ], + "metadata": { + "kernelspec": { + "display_name": "Python 3 (ipykernel)", + "language": "python", + "name": "python3" + }, + "language_info": { + "codemirror_mode": { + "name": "ipython", + "version": 3 + }, + "file_extension": ".py", + "mimetype": "text/x-python", + "name": "python", + "nbconvert_exporter": "python", + "pygments_lexer": "ipython3", + "version": "3.11.5" + } + }, + "nbformat": 4, + "nbformat_minor": 5 +} diff --git a/_sources/examples/03 Data Values.ipynb.txt b/_sources/examples/03 Data Values.ipynb.txt new file mode 100644 index 000000000..78dd996c9 --- /dev/null +++ b/_sources/examples/03 Data Values.ipynb.txt @@ -0,0 +1,601 @@ +{ + "cells": [ + { + "cell_type": "markdown", + "id": "1a1fe414", + "metadata": {}, + "source": [ + "# Data Types and Data Values\n", + "\n", + "This Jupyter notebook demonstrates how Data Values in Capella can be handled.\n", + "First, let's load the model again..." + ] + }, + { + "cell_type": "code", + "execution_count": 1, + "id": "3e28d1c9", + "metadata": {}, + "outputs": [ + { + "name": "stderr", + "output_type": "stream", + "text": [ + "Cannot load PVMT extension: ValueError: Provided model does not have a PropertyValuePkg\n", + "Property values are not available in this model\n" + ] + }, + { + "data": { + "text/plain": [ + "" + ] + }, + "execution_count": 1, + "metadata": {}, + "output_type": "execute_result" + } + ], + "source": [ + "import capellambse\n", + "path_to_model = \"../../../tests/data/melodymodel/5_2/Melody Model Test.aird\"\n", + "model = capellambse.MelodyModel(path_to_model, jupyter_untrusted=True)\n", + "model" + ] + }, + { + "cell_type": "markdown", + "id": "48ce0c1e", + "metadata": {}, + "source": [ + "As explained in the notebook 01, please ignore the warning about PVMT missing above." + ] + }, + { + "cell_type": "markdown", + "id": "7e68b88c-b4bc-4c20-a39f-48094c0eabdd", + "metadata": { + "tags": [] + }, + "source": [ + "## Example 1: Look into the Data package of the Logical Architecture\n", + "\n", + "Let's have a look into the data package on the Logical Architecture. It works the same with the other architectures, just replace the `oa` accordingly. We can see the defined classes, collections, enuemrations, and so on." + ] + }, + { + "cell_type": "code", + "execution_count": 2, + "id": "da8b86b7", + "metadata": {}, + "outputs": [ + { + "data": { + "text/html": [ + "

Data (org.polarsys.capella.core.data.information:DataPkg)

applied_property_value_groups

(Empty list)

applied_property_values

(Empty list)

classes
  1. Class "Wand" (c710f1c2-ede6-444e-9e2b-0ff30d7fd040)
  2. Class "Class 2" (1adf8097-18f9-474e-b136-6c845fc6d9e9)
  3. Class "Branch" (2b34c799-769c-42f2-8a1b-4533dba209a0)
collections

(Empty list)

complex_values

(Empty list)

constraints

(Empty list)

description
diagrams
  1. [CDB] Harry's Wand (uuid: _kqdwsF9REe2rko4oG1H6IQ)
enumerations
  1. Enumeration "Wand Core" (546cd75a-c7ac-4e07-9d2d-8a1f93d82419)
  2. Enumeration "Wand Wood" (60314ce6-bc96-4b57-8965-7187241148ae)
filtering_criteria

(Empty list)

nameData
packages
  1. DataPkg "Wand Objects" (880af86d-6fac-4fba-a559-2fffd036fa9a)
parentLogicalArchitecture "Logical Architecture" (853cb005-cba0-489b-8fe3-bb694ad4543b)
progress_statusNOT_SET
property_value_groups

(Empty list)

property_values

(Empty list)

requirements

(Empty list)

summaryNone
traces

(Empty list)

unions

(Empty list)

uuid39e99d4a-a32c-4b70-b4b6-d03fec612e17
xtypeorg.polarsys.capella.core.data.information:DataPkg
" + ], + "text/plain": [ + "\n", + ".applied_property_value_groups = []\n", + ".applied_property_values = []\n", + ".classes = [0] \n", + " [1] \n", + " [2] \n", + ".collections = []\n", + ".complex_values = []\n", + ".constraints = []\n", + ".description = ''\n", + ".diagrams = [0] \n", + ".enumerations = [0] \n", + " [1] \n", + ".filtering_criteria = []\n", + ".name = 'Data'\n", + ".packages = [0] \n", + ".parent = \n", + ".progress_status = 'NOT_SET'\n", + ".property_value_groups = []\n", + ".property_values = []\n", + ".requirements = []\n", + ".summary = None\n", + ".traces = []\n", + ".unions = []\n", + ".uuid = '39e99d4a-a32c-4b70-b4b6-d03fec612e17'\n", + ".xtype = 'org.polarsys.capella.core.data.information:DataPkg'" + ] + }, + "execution_count": 2, + "metadata": {}, + "output_type": "execute_result" + } + ], + "source": [ + "model.la.data_package" + ] + }, + { + "cell_type": "markdown", + "id": "e01263c2", + "metadata": {}, + "source": [ + "For Enumerations we can see the Literals assigned to it. We can see both the literals that have been inherited by the specialized super class, and the literals that are defined within this model element (own_literals)." + ] + }, + { + "cell_type": "code", + "execution_count": 3, + "id": "abcd8693", + "metadata": {}, + "outputs": [ + { + "data": { + "text/html": [ + "

Wand Core (org.polarsys.capella.core.data.information.datatype:Enumeration)

applied_property_value_groups

(Empty list)

applied_property_values

(Empty list)

constraints

(Empty list)

description
diagrams

(Empty list)

filtering_criteria

(Empty list)

literals
  1. EnumerationLiteral "Unicorn Hair" (79263437-b45d-410d-a264-8aa28d7574d1)
  2. EnumerationLiteral "Dragon Heartstring" (6bb9876c-f3a7-4d59-a6d1-819372368fa0)
  3. EnumerationLiteral "Pheonix Feather" (492fd9ca-88cb-4e9d-b92e-df14a1c1543b)
  4. EnumerationLiteral "Thestral Tail-Hair" (1e73d13b-1c26-4537-834d-e467f993befe)
nameWand Core
owned_literals
  1. EnumerationLiteral "Unicorn Hair" (79263437-b45d-410d-a264-8aa28d7574d1)
  2. EnumerationLiteral "Dragon Heartstring" (6bb9876c-f3a7-4d59-a6d1-819372368fa0)
  3. EnumerationLiteral "Pheonix Feather" (492fd9ca-88cb-4e9d-b92e-df14a1c1543b)
  4. EnumerationLiteral "Thestral Tail-Hair" (1e73d13b-1c26-4537-834d-e467f993befe)
parentDataPkg "Data" (39e99d4a-a32c-4b70-b4b6-d03fec612e17)
progress_statusNOT_SET
property_value_groups

(Empty list)

property_values

(Empty list)

requirements

(Empty list)

subBackreference to Enumeration - omitted: can be slow to compute. Display this property directly to show.
summaryNone
superNone
traces

(Empty list)

uuid546cd75a-c7ac-4e07-9d2d-8a1f93d82419
xtypeorg.polarsys.capella.core.data.information.datatype:Enumeration
" + ], + "text/plain": [ + "\n", + ".applied_property_value_groups = []\n", + ".applied_property_values = []\n", + ".constraints = []\n", + ".description = ''\n", + ".diagrams = []\n", + ".filtering_criteria = []\n", + ".literals = [0] \n", + " [1] \n", + " [2] \n", + " [3] \n", + ".name = 'Wand Core'\n", + ".owned_literals = [0] \n", + " [1] \n", + " [2] \n", + " [3] \n", + ".parent = \n", + ".progress_status = 'NOT_SET'\n", + ".property_value_groups = []\n", + ".property_values = []\n", + ".requirements = []\n", + ".sub = ... # backreference to Enumeration - omitted: can be slow to compute\n", + ".summary = None\n", + ".super = None\n", + ".traces = []\n", + ".uuid = '546cd75a-c7ac-4e07-9d2d-8a1f93d82419'\n", + ".xtype = 'org.polarsys.capella.core.data.information.datatype:Enumeration'" + ] + }, + "execution_count": 3, + "metadata": {}, + "output_type": "execute_result" + } + ], + "source": [ + "model.la.data_package.enumerations[0]" + ] + }, + { + "cell_type": "markdown", + "id": "dfcbe9cf", + "metadata": {}, + "source": [ + "Let's do the same for a class. Again, we can see the properties of the super class and the properties of the own model element.\n", + "\n", + "![Harry's Wand](../_static/img/harrys_wand.png)" + ] + }, + { + "cell_type": "code", + "execution_count": 4, + "id": "a968821f", + "metadata": {}, + "outputs": [ + { + "data": { + "text/html": [ + "

Wand (org.polarsys.capella.core.data.information:Class)

applied_property_value_groups

(Empty list)

applied_property_values

(Empty list)

constraints

(Empty list)

description
diagrams

(Empty list)

filtering_criteria

(Empty list)

is_abstractFalse
is_finalFalse
is_primitiveFalse
nameWand
owned_properties
  1. Property "owner" (9b1f6d9c-58d6-4e5e-a0f1-822cb5440a51)
  2. Property "core" (32f70910-a1fd-4ec9-8d22-4c585ceaf7b9)
  3. Property "wood" (df884a71-e774-49e1-8aee-0a675179c647)
parentDataPkg "Data" (39e99d4a-a32c-4b70-b4b6-d03fec612e17)
progress_statusNOT_SET
properties
  1. Property "owner" (9b1f6d9c-58d6-4e5e-a0f1-822cb5440a51)
  2. Property "core" (32f70910-a1fd-4ec9-8d22-4c585ceaf7b9)
  3. Property "wood" (df884a71-e774-49e1-8aee-0a675179c647)
  4. Property "wood" (87f356eb-c79e-4155-b297-8d733685621c)
property_value_groups

(Empty list)

property_values

(Empty list)

realizations
  1. InformationRealization "" (793520e1-acbf-4f93-a219-587840aa5a3b)
realized_byBackreference to Class - omitted: can be slow to compute. Display this property directly to show.
realized_classes
  1. Class "SpecialTwist" (0fef2887-04ce-4406-b1a1-a1b35e1ce0f3)
requirements

(Empty list)

state_machines

(Empty list)

subBackreference to Class - omitted: can be slow to compute. Display this property directly to show.
summaryNone
superClass "Branch" (2b34c799-769c-42f2-8a1b-4533dba209a0)
traces

(Empty list)

uuidc710f1c2-ede6-444e-9e2b-0ff30d7fd040
visibility<VisibilityKind.UNSET: 1>
xtypeorg.polarsys.capella.core.data.information:Class
" + ], + "text/plain": [ + "\n", + ".applied_property_value_groups = []\n", + ".applied_property_values = []\n", + ".constraints = []\n", + ".description = ''\n", + ".diagrams = []\n", + ".filtering_criteria = []\n", + ".is_abstract = False\n", + ".is_final = False\n", + ".is_primitive = False\n", + ".name = 'Wand'\n", + ".owned_properties = [0] \n", + " [1] \n", + " [2] \n", + ".parent = \n", + ".progress_status = 'NOT_SET'\n", + ".properties = [0] \n", + " [1] \n", + " [2] \n", + " [3] \n", + ".property_value_groups = []\n", + ".property_values = []\n", + ".realizations = [0] \n", + ".realized_by = ... # backreference to Class - omitted: can be slow to compute\n", + ".realized_classes = [0] \n", + ".requirements = []\n", + ".state_machines = []\n", + ".sub = ... # backreference to Class - omitted: can be slow to compute\n", + ".summary = None\n", + ".super = \n", + ".traces = []\n", + ".uuid = 'c710f1c2-ede6-444e-9e2b-0ff30d7fd040'\n", + ".visibility = \n", + ".xtype = 'org.polarsys.capella.core.data.information:Class'" + ] + }, + "execution_count": 4, + "metadata": {}, + "output_type": "execute_result" + } + ], + "source": [ + "model.la.data_package.classes[0]" + ] + }, + { + "cell_type": "markdown", + "id": "b7bc5a40", + "metadata": {}, + "source": [ + "We can investigate the properties of a Class" + ] + }, + { + "cell_type": "code", + "execution_count": 5, + "id": "56bb4b44", + "metadata": {}, + "outputs": [ + { + "data": { + "text/html": [ + "

owner (org.polarsys.capella.core.data.information:Property)

applied_property_value_groups

(Empty list)

applied_property_values

(Empty list)

associationBackreference to Association - omitted: can be slow to compute. Display this property directly to show.
constraints

(Empty list)

default_valueNone
description
diagrams

(Empty list)

filtering_criteria

(Empty list)

is_abstractFalse
is_derivedFalse
is_orderedFalse
is_part_of_keyFalse
is_read_onlyFalse
is_staticFalse
is_uniqueFalse
kind<AggregationKind.UNSET: 1>
maxNone
max_cardLiteralNumericValue "": 1 (43e39098-ace4-47c4-8f1c-1df1986063e2)
minNone
min_cardLiteralNumericValue "": 1 (95d6a6e5-6442-408d-afb2-d8f7a24c5f56)
nameowner
null_valueNone
parentClass "Wand" (c710f1c2-ede6-444e-9e2b-0ff30d7fd040)
progress_statusNOT_SET
property_value_groups

(Empty list)

property_values

(Empty list)

requirements

(Empty list)

summaryNone
traces

(Empty list)

typeGenericElement "String" (b2f035e6-78c8-4dfd-99f0-bf4a40ea3e81)
uuid9b1f6d9c-58d6-4e5e-a0f1-822cb5440a51
visibility<VisibilityKind.UNSET: 1>
xtypeorg.polarsys.capella.core.data.information:Property
" + ], + "text/plain": [ + "\n", + ".applied_property_value_groups = []\n", + ".applied_property_values = []\n", + ".association = ... # backreference to Association - omitted: can be slow to compute\n", + ".constraints = []\n", + ".default_value = None\n", + ".description = ''\n", + ".diagrams = []\n", + ".filtering_criteria = []\n", + ".is_abstract = False\n", + ".is_derived = False\n", + ".is_ordered = False\n", + ".is_part_of_key = False\n", + ".is_read_only = False\n", + ".is_static = False\n", + ".is_unique = False\n", + ".kind = \n", + ".max = None\n", + ".max_card = \n", + ".min = None\n", + ".min_card = \n", + ".name = 'owner'\n", + ".null_value = None\n", + ".parent = \n", + ".progress_status = 'NOT_SET'\n", + ".property_value_groups = []\n", + ".property_values = []\n", + ".requirements = []\n", + ".summary = None\n", + ".traces = []\n", + ".type = \n", + ".uuid = '9b1f6d9c-58d6-4e5e-a0f1-822cb5440a51'\n", + ".visibility = \n", + ".xtype = 'org.polarsys.capella.core.data.information:Property'" + ] + }, + "execution_count": 5, + "metadata": {}, + "output_type": "execute_result" + } + ], + "source": [ + "model.la.data_package.classes[0].properties[0]" + ] + }, + { + "cell_type": "markdown", + "id": "36670ae2", + "metadata": {}, + "source": [ + "As you can see the `kind` attribute is `UNSET`. That means this property isn't modelled as an association. An example for a `COMPOSITION`:\n", + "\n", + "![Waypoint](../_static/img/waypoints.png)" + ] + }, + { + "cell_type": "code", + "execution_count": 6, + "id": "7ca2429b", + "metadata": {}, + "outputs": [ + { + "data": { + "text/html": [ + "

waypoints (org.polarsys.capella.core.data.information:Property)

applied_property_value_groups

(Empty list)

applied_property_values

(Empty list)

associationBackreference to Association - omitted: can be slow to compute. Display this property directly to show.
constraints

(Empty list)

default_valueNone
description
diagrams

(Empty list)

filtering_criteria

(Empty list)

is_abstractFalse
is_derivedFalse
is_orderedFalse
is_part_of_keyFalse
is_read_onlyFalse
is_staticFalse
is_uniqueFalse
kind<AggregationKind.COMPOSITION: 4>
maxNone
max_cardLiteralNumericValue "": inf (57d146cf-6e40-42f4-9413-1cd0240d1431)
minNone
min_cardLiteralNumericValue "": 1 (1df38231-5a0a-4c47-8c99-96119b8b8af8)
namewaypoints
null_valueNone
parentClass "Trajectory" (c3c96805-d6f6-4092-b9f4-df7970651cdc)
progress_statusNOT_SET
property_value_groups

(Empty list)

property_values

(Empty list)

requirements

(Empty list)

summaryNone
traces

(Empty list)

typeClass "Waypoint" (c89849fd-0643-4708-a4da-74c9ea9ca7b1)
uuid424efd65-eaa9-4220-b61b-fb3340dbc19a
visibility<VisibilityKind.UNSET: 1>
xtypeorg.polarsys.capella.core.data.information:Property
" + ], + "text/plain": [ + "\n", + ".applied_property_value_groups = []\n", + ".applied_property_values = []\n", + ".association = ... # backreference to Association - omitted: can be slow to compute\n", + ".constraints = []\n", + ".default_value = None\n", + ".description = ''\n", + ".diagrams = []\n", + ".filtering_criteria = []\n", + ".is_abstract = False\n", + ".is_derived = False\n", + ".is_ordered = False\n", + ".is_part_of_key = False\n", + ".is_read_only = False\n", + ".is_static = False\n", + ".is_unique = False\n", + ".kind = \n", + ".max = None\n", + ".max_card = \n", + ".min = None\n", + ".min_card = \n", + ".name = 'waypoints'\n", + ".null_value = None\n", + ".parent = \n", + ".progress_status = 'NOT_SET'\n", + ".property_value_groups = []\n", + ".property_values = []\n", + ".requirements = []\n", + ".summary = None\n", + ".traces = []\n", + ".type = \n", + ".uuid = '424efd65-eaa9-4220-b61b-fb3340dbc19a'\n", + ".visibility = \n", + ".xtype = 'org.polarsys.capella.core.data.information:Property'" + ] + }, + "execution_count": 6, + "metadata": {}, + "output_type": "execute_result" + } + ], + "source": [ + "prop = model.sa.all_classes.by_name(\"Trajectory\").properties[0]\n", + "prop" + ] + }, + { + "cell_type": "markdown", + "id": "7ab5f3a6", + "metadata": {}, + "source": [ + "The role name is exposed as the `name` attribute of the property. As you can see the only *navigable* role is `waypoints` and its min- and max-card are also accessible. The `Class` can be accessed via the `type` attribute and the `association` is there to receive information about the incoming role/property. Let's have a look at it:" + ] + }, + { + "cell_type": "code", + "execution_count": 7, + "id": "90c3cf8c", + "metadata": {}, + "outputs": [ + { + "data": { + "text/html": [ + "

waypoint association (org.polarsys.capella.core.data.information:Association)

applied_property_value_groups

(Empty list)

applied_property_values

(Empty list)

constraints

(Empty list)

description
diagrams

(Empty list)

filtering_criteria

(Empty list)

members
  1. Property "trajectory" (c0e5b34c-297e-4b3c-8957-58bbe4d36199)
namewaypoint association
navigable_members
  1. Property "waypoints" (424efd65-eaa9-4220-b61b-fb3340dbc19a)
parentDataPkg "Data" (814464a3-3278-48eb-b66c-e255ed11afa8)
progress_statusNOT_SET
property_value_groups

(Empty list)

property_values

(Empty list)

requirements

(Empty list)

roles
  1. Property "trajectory" (c0e5b34c-297e-4b3c-8957-58bbe4d36199)
  2. Property "waypoints" (424efd65-eaa9-4220-b61b-fb3340dbc19a)
source_roleProperty "trajectory" (c0e5b34c-297e-4b3c-8957-58bbe4d36199)
summaryFind waypoints and you will finish consistently.
traces

(Empty list)

uuid3d738685-83e8-45f9-ade2-d5bcc6de1a0c
xtypeorg.polarsys.capella.core.data.information:Association
" + ], + "text/plain": [ + "\n", + ".applied_property_value_groups = []\n", + ".applied_property_values = []\n", + ".constraints = []\n", + ".description = ''\n", + ".diagrams = []\n", + ".filtering_criteria = []\n", + ".members = [0] \n", + ".name = 'waypoint association'\n", + ".navigable_members = [0] \n", + ".parent = \n", + ".progress_status = 'NOT_SET'\n", + ".property_value_groups = []\n", + ".property_values = []\n", + ".requirements = []\n", + ".roles = [0] \n", + " [1] \n", + ".source_role = \n", + ".summary = 'Find waypoints and you will finish consistently.'\n", + ".traces = []\n", + ".uuid = '3d738685-83e8-45f9-ade2-d5bcc6de1a0c'\n", + ".xtype = 'org.polarsys.capella.core.data.information:Association'" + ] + }, + "execution_count": 7, + "metadata": {}, + "output_type": "execute_result" + } + ], + "source": [ + "prop.association" + ] + }, + { + "cell_type": "markdown", + "id": "19c3fffc", + "metadata": {}, + "source": [ + "An `Association` has `navigable_members` which can be at most 2 (the source and target roles) and a `source_role`. Whenever the `is Navigable` option is ticked in Capella the property element will appear underneath the target `Class` of the `Association`." + ] + }, + { + "cell_type": "markdown", + "id": "eb05ea54", + "metadata": {}, + "source": [ + "## Example 2: Complex Values (instances of Classes)\n", + "\n", + "Capella allows to create Complex Values which have the type of a class model element. Complex Values can contain Value Parts that instantiate the properties of the class. Let's have a look at Harry's wand:" + ] + }, + { + "cell_type": "code", + "execution_count": 8, + "id": "77a5dc04", + "metadata": {}, + "outputs": [ + { + "data": { + "text/html": [ + "

Harry's Wand (org.polarsys.capella.core.data.information.datavalue:ComplexValue)

applied_property_value_groups

(Empty list)

applied_property_values

(Empty list)

constraints

(Empty list)

description
diagrams

(Empty list)

filtering_criteria

(Empty list)

nameHarry's Wand
parentDataPkg "Wand Objects" (880af86d-6fac-4fba-a559-2fffd036fa9a)
progress_statusNOT_SET
property_value_groups

(Empty list)

property_values

(Empty list)

requirements

(Empty list)

summaryNone
traces

(Empty list)

typeClass "Wand" (c710f1c2-ede6-444e-9e2b-0ff30d7fd040)
uuid3a467d68-f53c-4d66-9d32-fe032a8cb2c5
value_parts
  1. ValuePart "": \n", + ".applied_property_value_groups = []\n", + ".applied_property_values = []\n", + ".constraints = []\n", + ".description = ''\n", + ".diagrams = []\n", + ".filtering_criteria = []\n", + ".is_abstract = False\n", + ".name = 'LiteralStringValue'\n", + ".parent = \n", + ".progress_status = 'NOT_SET'\n", + ".property_value_groups = []\n", + ".property_values = []\n", + ".requirements = []\n", + ".summary = None\n", + ".traces = []\n", + ".type = \n", + ".uuid = 'f7b00d88-cf53-4ae0-a0e6-bb2049b4bdea'\n", + ".value = 'Harry Potter'\n", + ".xtype = 'org.polarsys.capella.core.data.information.datavalue:LiteralStringValue' (c996225b-5b1f-4d53-83cb-2bc72597e8ad)
  2. ValuePart "": \n", + ".applied_property_value_groups = []\n", + ".applied_property_values = []\n", + ".constraints = []\n", + ".description = ''\n", + ".diagrams = []\n", + ".filtering_criteria = []\n", + ".name = 'EnumerationReference'\n", + ".parent = \n", + ".progress_status = 'NOT_SET'\n", + ".property_value_groups = []\n", + ".property_values = []\n", + ".requirements = []\n", + ".summary = None\n", + ".traces = []\n", + ".type = \n", + ".uuid = '3406b669-4572-44e9-b703-1e319c350e9b'\n", + ".value = \n", + ".xtype = 'org.polarsys.capella.core.data.information.datavalue:EnumerationReference' (66da894f-6261-47ac-9ad7-217db04671d2)
  3. ValuePart "": \n", + ".applied_property_value_groups = []\n", + ".applied_property_values = []\n", + ".constraints = []\n", + ".description = ''\n", + ".diagrams = []\n", + ".filtering_criteria = []\n", + ".name = 'EnumerationReference'\n", + ".parent = \n", + ".progress_status = 'NOT_SET'\n", + ".property_value_groups = []\n", + ".property_values = []\n", + ".requirements = []\n", + ".summary = None\n", + ".traces = []\n", + ".type = \n", + ".uuid = '9645687f-9485-45f0-a10b-01f8b9d24914'\n", + ".value = \n", + ".xtype = 'org.polarsys.capella.core.data.information.datavalue:EnumerationReference' (a10de770-c6de-43fc-8d9f-868efe5cd29f)
xtypeorg.polarsys.capella.core.data.information.datavalue:ComplexValue
" + ], + "text/plain": [ + "\n", + ".applied_property_value_groups = []\n", + ".applied_property_values = []\n", + ".constraints = []\n", + ".description = ''\n", + ".diagrams = []\n", + ".filtering_criteria = []\n", + ".name = \"Harry's Wand\"\n", + ".parent = \n", + ".progress_status = 'NOT_SET'\n", + ".property_value_groups = []\n", + ".property_values = []\n", + ".requirements = []\n", + ".summary = None\n", + ".traces = []\n", + ".type = \n", + ".uuid = '3a467d68-f53c-4d66-9d32-fe032a8cb2c5'\n", + ".value_parts = [0] \n", + " [1] \n", + " [2] \n", + ".xtype = 'org.polarsys.capella.core.data.information.datavalue:ComplexValue'" + ] + }, + "execution_count": 8, + "metadata": {}, + "output_type": "execute_result" + } + ], + "source": [ + "model.la.data_package.packages[0].complex_values[0]" + ] + }, + { + "cell_type": "markdown", + "id": "8e7b17c0", + "metadata": {}, + "source": [ + "and let's see what wood Harry's wand is made of:" + ] + }, + { + "cell_type": "code", + "execution_count": 9, + "id": "0557c2a5", + "metadata": {}, + "outputs": [ + { + "name": "stdout", + "output_type": "stream", + "text": [ + "The owner of Harry's Wand is Harry Potter.\n", + "The core of Harry's Wand is Pheonix Feather.\n", + "The wood of Harry's Wand is Holly.\n" + ] + } + ], + "source": [ + "from capellambse.model.crosslayer.information import datavalue\n", + "\n", + "instance = model.la.data_package.packages[0].complex_values[0]\n", + "for value_part in instance.value_parts:\n", + " value = value_part.value.value\n", + " if isinstance(value, datavalue.EnumerationLiteral):\n", + " value = value.name\n", + "\n", + " print(f\"The {value_part.referenced_property.name} of {instance.name} is {value}.\")" + ] + } + ], + "metadata": { + "kernelspec": { + "display_name": "Python 3 (ipykernel)", + "language": "python", + "name": "python3" + }, + "language_info": { + "codemirror_mode": { + "name": "ipython", + "version": 3 + }, + "file_extension": ".py", + "mimetype": "text/x-python", + "name": "python", + "nbconvert_exporter": "python", + "pygments_lexer": "ipython3", + "version": "3.11.5" + } + }, + "nbformat": 4, + "nbformat_minor": 5 +} diff --git a/_sources/examples/04 Intro to Jinja templating.ipynb.txt b/_sources/examples/04 Intro to Jinja templating.ipynb.txt new file mode 100644 index 000000000..fbbe8125e --- /dev/null +++ b/_sources/examples/04 Intro to Jinja templating.ipynb.txt @@ -0,0 +1,802 @@ +{ + "cells": [ + { + "cell_type": "markdown", + "metadata": {}, + "source": [ + "# Using capellambse with Jinja2\n", + "\n", + "Welcome to the py-capellambse jinja2 templating showcase. When using capella\n", + "for systems engineering you might want to generate documentation for your model\n", + "you can use M2DOC, one of capella's [addons] which we found too challenging or\n", + "you can use Jinja2 a richful templating language with high degree of freedom.\n", + "\n", + "This notebook will introduce you into writing jinja2 templates where you'll plant model\n", + "information and diagrams. Additonally we'll give a side-note on how to handle\n", + "unique identifiers professionally with PVMT and some hints onto how to use jinja\n", + "in a professional manner which could give you the option onto developping an\n", + "automated document generation system.\n", + "\n", + "With Jinja2 you are able to generate any text-based format(HTML, XML, CSV, LaTex,...)\n", + "but during this tutorial we will only generate .html files. The jinja2 syntax is\n", + "inspired by python. Check out their [docs]!\n", + "\n", + "[docs]: https://jinja2docs.readthedocs.io/en/stable/\n", + "[addons]: https://www.eclipse.org/capella/addons.html\n", + "\n", + "Below code loads the needed libraries and instantiates a test model:" + ] + }, + { + "cell_type": "code", + "execution_count": 1, + "metadata": {}, + "outputs": [ + { + "name": "stderr", + "output_type": "stream", + "text": [ + "Cannot load PVMT extension: ValueError: Provided model does not have a PropertyValuePkg\n", + "Property values are not available in this model\n" + ] + } + ], + "source": [ + "import jinja2\n", + "import capellambse\n", + "\n", + "from IPython.core.display import HTML\n", + "\n", + "path_to_model = \"../../../tests/data/melodymodel/5_0/Melody Model Test.aird\"\n", + "model = capellambse.MelodyModel(path_to_model, jupyter_untrusted=True)\n", + "env = jinja2.Environment()" + ] + }, + { + "cell_type": "markdown", + "metadata": {}, + "source": [ + "In the following we want to make a template to document all modelled actors from\n", + "the logical layer. Therefore we define a template string where we iterate over\n", + "all actors and plant the name, uuid and description into it.\n", + "\n", + "*Hint: Make sure that you know of capella's [metamodel] as we are implementing the\n", + "capellambse.layers as close as possible to it while being as efficient and pythonic\n", + "we can be currently. This knowledge can shorten used statements in the template immensely!*\n", + "\n", + "[metamodel]: https://dsd-dbs.github.io/py-capellambse/start/intro-to-api.html" + ] + }, + { + "cell_type": "code", + "execution_count": 2, + "metadata": {}, + "outputs": [ + { + "data": { + "text/html": [ + "\n", + "

Actor definitions

\n", + "\n", + "

Prof. A. P. W. B. Dumbledore

\n", + "

Actor definition

\n", + "

UUID: 08e02248-504d-4ed8-a295-c7682a614f66

\n", + "

Principal of Hogwarts, wearer of the elder wand and greatest mage of all time.

\n", + "

\n", + "\n", + "

Prof. S. Snape

\n", + "

Actor definition

\n", + "

UUID: 6f463eed-c77b-4568-8078-beec0536f243

\n", + "

Good guy and teacher of brewing arts.

\n", + "

\n", + "\n", + "

Harry J. Potter

\n", + "

Actor definition

\n", + "

UUID: a8c46457-a702-41c4-a971-c815c4c5a674

\n", + "

\n", + "\n", + "

R. Weasley

\n", + "

Actor definition

\n", + "

UUID: ff7b8672-84db-4b93-9fea-22a410907fb1

\n", + "

\n", + "\n", + "

Voldemort

\n", + "

Actor definition

\n", + "

UUID: 3e0ee19f-0e3f-49d4-ae99-29bd4a3260c5

\n", + "

\n", + "\n", + "

Multiport

\n", + "

Actor definition

\n", + "

UUID: b3888dad-a870-4b8b-97d4-0ddb83ef9251

\n", + "

\n" + ], + "text/plain": [ + "" + ] + }, + "execution_count": 2, + "metadata": {}, + "output_type": "execute_result" + } + ], + "source": [ + "templ = \"\"\"\n", + "

Actor definitions

\n", + "{% for actor in model.la.all_components.by_is_actor(True) %}\n", + "

{{ actor.name }}

\n", + "

Actor definition

\n", + "

UUID: {{ actor.uuid }}

\n", + "

{{ actor.description }}

\n", + "{% endfor %}\n", + "\"\"\"\n", + "\n", + "HTML(env.from_string(templ).render(model=model))" + ] + }, + { + "cell_type": "markdown", + "metadata": {}, + "source": [ + "As an extra: We don't use the UUID from capella in our documents. If you still\n", + "need an identifier, in the following it's explained how we are doing it:\n", + "\n", + "With the capella PVMT, one of the various capella addons, you can make property\n", + "value groups and then set arbitrary values with these. Capellambse is able to recognize\n", + "the pvmt extension and gives read and write access. In our workflows we are maintaining\n", + "an ID database for all model elements. If that is done you can access pvmt attributes\n", + "like:\n", + "\n", + "```html\n", + "{{ ... }}\n", + "

ID: {{ actor.pvmt[\"Group.Identification.MY MODEL ID\"] }}

\n", + "{{ ... }}\n", + "```\n", + "\n", + "For a PVMT showcase look into [TODO: pvmt-showcase notebook]." + ] + }, + { + "cell_type": "markdown", + "metadata": {}, + "source": [ + "## Filters and object manipulation\n", + "\n", + "Now to step up our templating-game, we'll bring in more complexity.\n", + "We want a template that documents functional context of all actors. For that\n", + "iff the actor has a non-empty functions attribute we make a table with columns\n", + "for function's uid, name, description and `FunctionalExchange`s.\n", + "\n", + "We can define variables in the template via the set statement. Furthermore\n", + "the builtin jinja [filters] already give much power for object manipulation during\n", + "rendering. Here we use map(.) to create `FunctionalExchange` iterators and sum these\n", + "up into one large list that stores all `FunctionalExchange`s that have an an actor\n", + "as either source or target. In the table for-loop we then filter on this lookup\n", + "container and set outgoing and incoming `FunctionalExchange`s that we need for\n", + "the last column.\n", + "\n", + "The jupyter environment is great for writing templates b/c you can investigate\n", + "possible attributes of objects right away in another cell.\n", + "\n", + "*Hint: You can define custom filter-functions and add them to the Environment.filters.*\n", + "\n", + "[filters]: https://jinja.palletsprojects.com/en/3.0.x/templates/#builtin-filters" + ] + }, + { + "cell_type": "code", + "execution_count": 3, + "metadata": {}, + "outputs": [ + { + "data": { + "text/html": [ + "\n", + "

Actor definitions

\n", + "\n", + "\n", + "

Harry J. Potter

\n", + "

Actor definition

\n", + "

\n", + "

Actor functions

\n", + " \n", + "

The table below identifies functions of Harry J. Potter.

\n", + " \n", + " \n", + " \n", + " \n", + " \n", + " \n", + " \n", + " \n", + " \n", + " \n", + " \n", + " \n", + " \n", + " \n", + " \n", + " \n", + " \n", + " \n", + " \n", + " \n", + " \n", + " \n", + "
IDFunctionDescriptionInvolved Subsystems
aa9931e3-116c-461e-8215-6b9fdbdd4a1bkill He Who Must Not Be NamedLearning
\n", + "

Functions of Harry J. Potter

\n", + " \n", + "\n", + "

Prof. A. P. W. B. Dumbledore

\n", + "

Actor definition

\n", + "

Principal of Hogwarts, wearer of the elder wand and greatest mage of all time.

\n", + "

\n", + "

Actor functions

\n", + " \n", + "

The table below identifies functions of Prof. A. P. W. B. Dumbledore.

\n", + " \n", + " \n", + " \n", + " \n", + " \n", + " \n", + " \n", + " \n", + " \n", + " \n", + " \n", + " \n", + " \n", + " \n", + " \n", + " \n", + " \n", + " \n", + " \n", + " \n", + " \n", + " \n", + " \n", + " \n", + " \n", + " \n", + " \n", + " \n", + " \n", + " \n", + " \n", + " \n", + "
IDFunctionDescriptionInvolved Subsystems
f708bc29-d69f-42a0-90cc-11fc01054cd0manage the school
beaf5ba4-8fa9-4342-911f-0266bb29be45advise Harry
\n", + "

Functions of Prof. A. P. W. B. Dumbledore

\n", + " \n", + "\n", + "

Prof. S. Snape

\n", + "

Actor definition

\n", + "

Good guy and teacher of brewing arts.

\n", + "

\n", + "

Actor functions

\n", + " \n", + "

The table below identifies functions of Prof. S. Snape.

\n", + " \n", + " \n", + " \n", + " \n", + " \n", + " \n", + " \n", + " \n", + " \n", + " \n", + " \n", + " \n", + " \n", + " \n", + " \n", + " \n", + " \n", + " \n", + " \n", + " \n", + " \n", + " \n", + " \n", + " \n", + " \n", + " \n", + " \n", + " \n", + " \n", + " \n", + " \n", + " \n", + "
IDFunctionDescriptionInvolved Subsystems
a7acb298-d14b-4707-a419-fea272434541TeachingTeacher Responsibilities
4a2a7f3c-d223-4d44-94a7-50dd2906a70cmaintain a layer of defense for the Sorcerer's Stone
\n", + "

Functions of Prof. S. Snape

\n", + " \n", + "\n", + "

Multiport

\n", + "

Actor definition

\n", + "

\n", + "

Actor functions

\n", + " \n", + "

The table below identifies functions of Multiport.

\n", + " \n", + " \n", + " \n", + " \n", + " \n", + " \n", + " \n", + " \n", + " \n", + " \n", + " \n", + " \n", + " \n", + " \n", + " \n", + " \n", + " \n", + " \n", + " \n", + " \n", + " \n", + " \n", + "
IDFunctionDescriptionInvolved Subsystems
9c1885f5-fac7-48fd-9d54-a11092508867LAF 1C 5
\n", + "

Functions of Multiport

\n", + " \n", + "\n", + "

Voldemort

\n", + "

Actor definition

\n", + "

\n", + "

Actor functions

\n", + " \n", + "

No actor functions were identified.

\n", + " \n", + "\n", + "

R. Weasley

\n", + "

Actor definition

\n", + "

\n", + "

Actor functions

\n", + " \n", + "

The table below identifies functions of R. Weasley.

\n", + " \n", + " \n", + " \n", + " \n", + " \n", + " \n", + " \n", + " \n", + " \n", + " \n", + " \n", + " \n", + " \n", + " \n", + " \n", + " \n", + " \n", + " \n", + " \n", + " \n", + " \n", + " \n", + " \n", + " \n", + " \n", + " \n", + " \n", + " \n", + " \n", + " \n", + " \n", + " \n", + "
IDFunctionDescriptionInvolved Subsystems
c1a42acc-1f53-42bb-8404-77a5c08c414bassist Harry
edbd1ad4-31c0-4d53-b856-3ffa60e0e99bbreak school rulesPunishment
\n", + "

Functions of R. Weasley

\n", + " \n" + ], + "text/plain": [ + "" + ] + }, + "execution_count": 3, + "metadata": {}, + "output_type": "execute_result" + } + ], + "source": [ + "templ1 = \"\"\"\n", + "

Actor definitions

\n", + "{% set fexs = model.la.actor_exchanges.map(\"func_exchanges\") %}\n", + "{% for actor in model.la.all_actors %}\n", + "

{{ actor.name }}

\n", + "

Actor definition

\n", + "

{{ actor.description }}

\n", + "

Actor functions

\n", + " {% if actor.functions %}\n", + "

The table below identifies functions of {{ actor.name }}.

\n", + " \n", + " \n", + " \n", + " \n", + " \n", + " \n", + " \n", + " \n", + " \n", + " \n", + " {% for fnc in actor.functions %}\n", + " {% set outs = fexs.by_source.owner(fnc) %}\n", + " {% set ins = fexs.by_target.owner(fnc) %}\n", + " {% set subs = (ins + outs) | map(attribute=\"owner.name\") | unique | sort %}\n", + " \n", + " \n", + " \n", + " \n", + " \n", + " \n", + " {% endfor %}\n", + " \n", + "
IDFunctionDescriptionInvolved Subsystems
{{ fnc.uuid }}{{ fnc.name }}{{ fnc.description }}{{ subs | join(', ') }}
\n", + "

Functions of {{ actor.name }}

\n", + " {% else %}\n", + "

No actor functions were identified.

\n", + " {% endif %}\n", + "{% endfor %}\n", + "\"\"\"\n", + "HTML(env.from_string(templ1).render(model=model))" + ] + }, + { + "cell_type": "markdown", + "metadata": {}, + "source": [ + "## Beautiful SVG diagrams\n", + "\n", + "Finally we will render a template that displays a diagram. There are many ways\n", + "to do this and with jinja2 you have full control. We like our figures inside\n", + "tables such that a caption can be displayed" + ] + }, + { + "cell_type": "code", + "execution_count": 4, + "metadata": {}, + "outputs": [ + { + "name": "stderr", + "output_type": "stream", + "text": [ + "Unknown global filter 'hide.sequencing.information.filter'\n", + "Unknown global filter 'ModelExtensionFilter'\n", + "Unknown global filter 'hide.simplified.diagram.based.component.exchanges.filter'\n", + "Unknown global filter 'hide.simplified.oriented.grouped.component.exchanges.filter'\n", + "Unknown global filter 'hide.simplified.group.of.component.exchanges.filter'\n" + ] + }, + { + "data": { + "text/html": [ + "

Hogwarts

\n", + "

This instance is the mighty Hogwarts. Do you really need a description? Then maybe read the books or watch atleast the epic movies.

\n", + "\n", + "\n", + " \n", + " \n", + "
Figure 1: [LAB] Wizzard Education
LFHogwartsproduce GreatWizardsprotectStudentsagainst theDeath EatersCampusSchooleducate WizardsWhomping Willowdefend thesurrounding areaagainst IntrudersProf. A. P. W. B.Dumbledoremanage theschooladviseHarryProf. S. SnapeTeachingmaintain alayer ofdefense fortheSorcerer'sStoneHarry J. Potterkill He WhoMust Not BeNamedR. Weasleyassist Harrybreak schoolruleswizardryHeadmasterResponsibilitiesTeacherResponsibilitiesHelp for HarryKnowledgePunishmentLearningeducate & maturefriendshipassistanceCarepunisheducate
" + ], + "text/plain": [ + "" + ] + }, + "execution_count": 4, + "metadata": {}, + "output_type": "execute_result" + } + ], + "source": [ + "templ = \"\"\"\\\n", + "

{{ component.name }}

\n", + "{{ component.description }}\n", + "\n", + " \n", + " \n", + "
Figure {{ fig_id }}: {{ fig_caption | e }}
{{ figure.as_svg | safe }}
\n", + "\"\"\"\n", + "diagram = model.diagrams.by_name(\"[LAB] Wizzard Education\")\n", + "rendered = env.from_string(templ).render(\n", + " component=model.search(\"LogicalComponent\").by_name(\"Hogwarts\"),\n", + " fig_id=1,\n", + " fig_caption=diagram.name,\n", + " figure=diagram,\n", + ")\n", + "HTML(rendered)" + ] + }, + { + "cell_type": "markdown", + "metadata": {}, + "source": [ + "*Hint: Take notice of the [jinja.Environment]. Instead of handing the figure_table-markup\n", + "over in the rendering call you could also set an insert_figure_as_table function,\n", + "which you ideally defined before, in the environment globals or you can define\n", + "[macros] right in the template. These tools can automate repetitive content placement.*\n", + "\n", + "[jinja.Environment]: https://jinja.palletsprojects.com/en/3.0.x/api/#jinja2.Environment\n", + "[macros]: https://jinja.palletsprojects.com/en/3.0.x/templates/#macros" + ] + }, + { + "cell_type": "markdown", + "metadata": {}, + "source": [ + "## Template inheritance\n", + "\n", + "The last cell will present a routine where a full .html document is generated.\n", + "Earlier rendered content were HTML-fragments to be precise. On top you can see\n", + "a showcase on jinja's [template-inheritance] functionality. The special DictLoader\n", + "gives support for finding the base template called \"template.html\". This was just\n", + "needed because we are dealing with content in memory and didn't create template.html\n", + "in our FileSystem before. Per default jinja is using the FileSystemLoader when\n", + "creating an Environment. It's not a bad idea to check the [Loaders] they have, if\n", + "you want to understand how template loading is working and/or plan on developing\n", + "a pipeline system for document distribution.\n", + "\n", + "[template-inheritance]: https://jinja.palletsprojects.com/en/3.0.x/templates/#template-inheritance\n", + "[Loaders]: https://jinja.palletsprojects.com/en/3.0.x/api/#loaders" + ] + }, + { + "cell_type": "code", + "execution_count": 5, + "metadata": {}, + "outputs": [ + { + "data": { + "text/html": [ + "\n", + "\n", + "\n", + "\n", + "\n", + " \n", + "\n", + "\n", + "\n", + "

Hogwarts

\n", + "

This instance is the mighty Hogwarts. Do you really need a description? Then maybe read the books or watch atleast the epic movies.

\n", + "\n", + "\n", + " \n", + " \n", + "
Figure 1: [LAB] Wizzard Education
LFHogwartsproduce GreatWizardsprotectStudentsagainst theDeath EatersCampusSchooleducate WizardsWhomping Willowdefend thesurrounding areaagainst IntrudersProf. A. P. W. B.Dumbledoremanage theschooladviseHarryProf. S. SnapeTeachingmaintain alayer ofdefense fortheSorcerer'sStoneHarry J. Potterkill He WhoMust Not BeNamedR. Weasleyassist Harrybreak schoolruleswizardryHeadmasterResponsibilitiesTeacherResponsibilitiesHelp for HarryKnowledgePunishmentLearningeducate & maturefriendshipassistanceCarepunisheducate
\n", + "\n", + "\n", + "" + ], + "text/plain": [ + "" + ] + }, + "execution_count": 5, + "metadata": {}, + "output_type": "execute_result" + } + ], + "source": [ + "fig_templ = \"\".join(\n", + " (\n", + " '{% extends \"template.html\" %}',\n", + " \"{% block content %}\",\n", + " templ,\n", + " \"{% endblock %}\",\n", + " )\n", + ")\n", + "final_templ = \"\"\"\n", + "\n", + "\n", + "\n", + "\n", + " \n", + "\n", + "\n", + "\n", + " {% block content %}\n", + " {% endblock %}\n", + "\n", + "\n", + "\"\"\"\n", + "\n", + "env = jinja2.Environment(loader=jinja2.DictLoader({\"template.html\": final_templ}))\n", + "rendered = env.from_string(fig_templ).render(\n", + " component=model.search(\"LogicalComponent\").by_name(\"Hogwarts\"),\n", + " fig_id=1,\n", + " fig_caption=diagram.name,\n", + " figure=diagram,\n", + ")\n", + "HTML(rendered)" + ] + } + ], + "metadata": { + "kernelspec": { + "display_name": "Python 3 (ipykernel)", + "language": "python", + "name": "python3" + }, + "language_info": { + "codemirror_mode": { + "name": "ipython", + "version": 3 + }, + "file_extension": ".py", + "mimetype": "text/x-python", + "name": "python", + "nbconvert_exporter": "python", + "pygments_lexer": "ipython3", + "version": "3.11.5" + }, + "vscode": { + "interpreter": { + "hash": "e7370f93d1d0cde622a1f8e1c04877d8463912d04d973331ad4851f04de6915a" + } + } + }, + "nbformat": 4, + "nbformat_minor": 4 +} diff --git a/_sources/examples/05 Introduction to Libraries.ipynb.txt b/_sources/examples/05 Introduction to Libraries.ipynb.txt new file mode 100644 index 000000000..f679d16b9 --- /dev/null +++ b/_sources/examples/05 Introduction to Libraries.ipynb.txt @@ -0,0 +1,254 @@ +{ + "cells": [ + { + "cell_type": "markdown", + "metadata": {}, + "source": [ + "# Introduction to Libraries\n", + "\n", + "This notebook illustrates the use of Capella Libraries. When trying to load a\n", + "model that uses a library, you may encounter an error similar to this:" + ] + }, + { + "cell_type": "code", + "execution_count": 1, + "metadata": {}, + "outputs": [ + { + "name": "stdout", + "output_type": "stream", + "text": [ + "MissingResourceLocationError: 'Library Test'\n" + ] + } + ], + "source": [ + "import capellambse\n", + "path_to_model = \"../../../tests/data/Library Project/Library Project.aird\"\n", + "\n", + "try:\n", + " capellambse.MelodyModel(path_to_model, jupyter_untrusted=True)\n", + "except Exception as err:\n", + " print(f\"{type(err).__name__}: {err}\")" + ] + }, + { + "cell_type": "markdown", + "metadata": {}, + "source": [ + "This tells us the name that was given to the library: `Library Test`. So, we need to define a resource location with that name. As there can be arbitrarily many resource locations (i.e. linked libraries) with arbitrary names given to them, these locations are handed over in a Python `dict`.\n", + "\n", + "There are three ways of defining a resource location:\n", + "\n", + "1. A simple `str` containing only a path or URL, similar to the first positional argument of `MelodyModel`.\n", + "2. A nested dictionary with a `path` key, as well as other keys needed to find and access that resource. These may include `subdir` or `username` and `password`.\n", + "3. A constructed `FileHandler` object.\n", + "\n", + "The following cell shows a concrete example of each of them." + ] + }, + { + "cell_type": "code", + "execution_count": 2, + "metadata": {}, + "outputs": [], + "source": [ + "# Simple `str`\n", + "resources = {\n", + " \"Library Test\": \"/data/models/Library Test\",\n", + "}\n", + "\n", + "# Nested `dict`\n", + "resources = {\n", + " \"Library Test\": {\n", + " \"path\": \"https://raw.githubusercontent.com/DSD-DBS/py-capellambse/master/tests/data/Library%20Test\",\n", + " # More options can be added here if necessary, e.g.:\n", + " # \"username\": \"demouser\",\n", + " # \"password\": \"super secret passphrase\",\n", + " }\n", + "}\n", + "\n", + "# `FileHandler` object\n", + "# (Be aware that constructing a ´FileHandler` may already involve network access,\n", + "# for example cloning a remote git repository into a local cache.)\n", + "lib_handler = capellambse.get_filehandler(\n", + " \"git+https://github.com/DSD-DBS/py-capellambse.git\",\n", + " subdir=\"tests/data/Library Test\",\n", + " revision=\"master\"\n", + " # More options can be added here as well, e.g.:\n", + " # username=\"demouser\",\n", + " # password=\"super secret passphrase\",\n", + ")\n", + "resources = {\"Library Test\": lib_handler}" + ] + }, + { + "cell_type": "markdown", + "metadata": {}, + "source": [ + "Equipped with this `resources` dictionary, we can now try to load the model again:" + ] + }, + { + "cell_type": "code", + "execution_count": 3, + "metadata": {}, + "outputs": [ + { + "name": "stderr", + "output_type": "stream", + "text": [ + "Cannot load PVMT extension: ValueError: Provided model does not have a PropertyValuePkg\n", + "Property values are not available in this model\n" + ] + } + ], + "source": [ + "model = capellambse.MelodyModel(path_to_model, resources=resources, jupyter_untrusted=True)" + ] + }, + { + "cell_type": "markdown", + "metadata": {}, + "source": [ + "If you receive more `MissingResourceLocationError`s, add them to the same `resources` dictionary and pass them to the `MelodyModel` as well." + ] + }, + { + "cell_type": "markdown", + "metadata": {}, + "source": [ + "Once you no longer receive errors during loading, you can use the model as normal." + ] + }, + { + "cell_type": "code", + "execution_count": 4, + "metadata": {}, + "outputs": [ + { + "data": { + "text/html": [ + "
  1. [CDB] Library Product (uuid: _q-c-0KN2EeyJNLcTD9ngpQ)
  2. [LAB] E-Commerce (uuid: _SMS4sKFFEeyn0YWM8vjd5w)
" + ], + "text/plain": [ + "[0] \n", + "[1] " + ] + }, + "execution_count": 4, + "metadata": {}, + "output_type": "execute_result" + } + ], + "source": [ + "model.diagrams" + ] + }, + { + "cell_type": "code", + "execution_count": 5, + "metadata": {}, + "outputs": [ + { + "name": "stderr", + "output_type": "stream", + "text": [ + "Unknown global filter 'hide.sequencing.information.filter'\n", + "Unknown global filter 'hide.simplified.diagram.based.component.exchanges.filter'\n", + "Unknown global filter 'hide.simplified.oriented.grouped.component.exchanges.filter'\n", + "Unknown global filter 'ModelExtensionFilter'\n", + "Unknown global filter 'hide.simplified.group.of.component.exchanges.filter'\n" + ] + }, + { + "data": { + "image/png": "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", + "text/plain": [ + "" + ] + }, + "execution_count": 5, + "metadata": {}, + "output_type": "execute_result" + } + ], + "source": [ + "model.diagrams.by_name(\"[LAB] E-Commerce\")" + ] + }, + { + "cell_type": "markdown", + "metadata": {}, + "source": [ + "## Common pitfalls\n", + "\n", + "In order for `capellambse` to be able to find your elements, ensure that you:\n", + "\n", + "1. Created a replicable element collection (REC) in the Capella Library. To do that, right-click the element in the library, and choose \"REC / RPL\" > \"Create REC\".\n", + "2. Have instantiated the REC as a so-called Replica (RPL) in your primary project. For this, right-click any element in your project and choose \"REC / RPL\" > \"Instantiate RPL from REC\".\n", + "\n", + "For more information, watch this [short video about REC / RPL](https://youtu.be/h-ax61eVlxM)." + ] + }, + { + "cell_type": "code", + "execution_count": 6, + "metadata": {}, + "outputs": [ + { + "name": "stdout", + "output_type": "stream", + "text": [ + "Logical System\n", + "E-Commerce\n", + "Truck Company\n", + "Client\n" + ] + } + ], + "source": [ + "for component in model.la.all_components:\n", + " print(component.name)" + ] + }, + { + "cell_type": "markdown", + "metadata": {}, + "source": [ + "## Saving a model\n", + "\n", + "You can save the model as usual by calling `model.save()`. However, **this will only save the primary model**. Modifications to elements that are defined only in a library will be lost.\n", + "\n", + "This is normally not a problem when following the standard REC / RPL based workflow, as the definitions of those elements will be copied to and modified in the primary model.\n", + "\n", + "In order to modify elements in one of the used libraries, you can load the library itself as a model. This allows you to apply any needed modifications and then save the library. Afterwards, the updated library can be used in the `resources` of other models." + ] + } + ], + "metadata": { + "interpreter": { + "hash": "e5607734f15d1b90b8506e690cd35fc527f703262c75faee6ce3cf6c6d2ff5b2" + }, + "kernelspec": { + "display_name": "Python 3 (ipykernel)", + "language": "python", + "name": "python3" + }, + "language_info": { + "codemirror_mode": { + "name": "ipython", + "version": 3 + }, + "file_extension": ".py", + "mimetype": "text/x-python", + "name": "python", + "nbconvert_exporter": "python", + "pygments_lexer": "ipython3", + "version": "3.11.5" + } + }, + "nbformat": 4, + "nbformat_minor": 4 +} diff --git a/_sources/examples/06 Introduction to Requirement access and management.ipynb.txt b/_sources/examples/06 Introduction to Requirement access and management.ipynb.txt new file mode 100644 index 000000000..986ff719a --- /dev/null +++ b/_sources/examples/06 Introduction to Requirement access and management.ipynb.txt @@ -0,0 +1,686 @@ +{ + "cells": [ + { + "cell_type": "markdown", + "id": "7bcee230", + "metadata": {}, + "source": [ + "# Introduction to Requirement access and management\n", + "\n", + "Welcome to the ReqIF extension Showcase notebook. This notebook will show you some basic (and not so basic) things that you can get done using this library.\n", + "\n", + "The below code loads the library and one of the test models:" + ] + }, + { + "cell_type": "code", + "execution_count": 1, + "id": "a52777dd", + "metadata": {}, + "outputs": [ + { + "data": { + "text/plain": [ + "" + ] + }, + "execution_count": 1, + "metadata": {}, + "output_type": "execute_result" + } + ], + "source": [ + "import capellambse\n", + "\n", + "path_to_model = \"../../../tests/data/melodymodel/5_0/Melody Model Test.aird\"\n", + "model = capellambse.MelodyModel(path_to_model, jupyter_untrusted=True)\n", + "model" + ] + }, + { + "cell_type": "markdown", + "id": "d8442183", + "metadata": {}, + "source": [ + "You can access a lookup for all requirements defined in a specific layer." + ] + }, + { + "cell_type": "code", + "execution_count": 2, + "id": "42a46a49", + "metadata": {}, + "outputs": [ + { + "data": { + "text/html": [ + "
  1. Requirement "TestReq1" (3c2d312c-37c9-41b5-8c32-67578fa52dc3)
  2. Requirement "TypedReq2" (0a9a68b1-ba9a-4793-b2cf-4448f0b4b8cc)
  3. Requirement "TestReq3" (79291c33-5147-4543-9398-9077d582576d)
  4. Requirement "TypedReq1" (85d41db2-9e17-438b-95cf-49342452ddf3)
  5. Requirement "TestReq" (1092f69a-5f3a-4fe6-a8fd-b2dffde90650)
" + ], + "text/plain": [ + "[0] \n", + "[1] \n", + "[2] \n", + "[3] \n", + "[4] " + ] + }, + "execution_count": 2, + "metadata": {}, + "output_type": "execute_result" + } + ], + "source": [ + "model.oa.all_requirements" + ] + }, + { + "cell_type": "markdown", + "id": "904c842b", + "metadata": {}, + "source": [ + "Have a look at all of the available attributes for a Requirement ModelElement:" + ] + }, + { + "cell_type": "code", + "execution_count": 3, + "id": "cb0a5b75", + "metadata": {}, + "outputs": [ + { + "data": { + "text/html": [ + "

TestReq1 (Requirements:Requirement)

applied_property_value_groups

(Empty list)

applied_property_values

(Empty list)

attributes
  1. BooleanValueAttribute "AttrDef" (9c692405-b8aa-4caa-b988-51d27db5cd1b)
  2. DateValueAttribute "AttrDef" (b97c09b5-948a-46e8-a656-69d764ddce7d)
  3. IntegerValueAttribute "AttrDef" (85dfd42c-7f6e-4236-a181-bdd784040431)
  4. RealValueAttribute "AttrDef" (d2231d14-854d-4625-b48b-6cf1c2554367)
  5. StringValueAttribute "AttrDef" (ee8a69ef-61b9-4db9-9a0f-628e5d4704e1)
  6. BooleanValueAttribute "None" (dcb8614e-2d1c-4cb3-aa0c-667a297e7489)
chapter_name2
constraints

(Empty list)

descriptionThis is a test requirement of kind 1.
diagrams

(Empty list)

filtering_criteria

(Empty list)

foreign_id1
identifierREQTYPE-1
long_name1
nameTestReq1
ownerFolder "Folder" (e16f5cc1-3299-43d0-b1a0-82d31a137111)
parentFolder "Folder" (e16f5cc1-3299-43d0-b1a0-82d31a137111)
prefix3
progress_statusNOT_SET
property_value_groups

(Empty list)

property_values

(Empty list)

pvmt<capellambse.extensions.pvmt.PropertyValueProxy object at 0x7f3afe4e68d0>
related
  1. Entity "Weather" (4bf0356c-89dd-45e9-b8a6-e0332c026d33)
  2. SystemFunction "Sysexfunc" (00e7b925-cf4c-4cb0-929e-5409a1cd872b)
  3. Requirement "TypedReq1" (85d41db2-9e17-438b-95cf-49342452ddf3)
  4. LogicalComponent "Hogwarts" (0d2edb8f-fa34-4e73-89ec-fb9a63001440)
relations
  1. CapellaIncomingRelation "Controlling the weather" (078b2c69-4352-4cf9-9ea5-6573b75e5eec)
  2. CapellaIncomingRelation "Test req" (24c824ef-b187-4725-a051-a68707e82d70)
  3. InternalRelation "None" (7de4c1a5-e106-4171-902a-502b816b60b0)
  4. CapellaOutgoingRelation "None" (57033242-3766-4961-8091-ce3d9326ed67)
requirements
  1. Entity "Weather" (4bf0356c-89dd-45e9-b8a6-e0332c026d33)
  2. SystemFunction "Sysexfunc" (00e7b925-cf4c-4cb0-929e-5409a1cd872b)
  3. Requirement "TypedReq1" (85d41db2-9e17-438b-95cf-49342452ddf3)
  4. LogicalComponent "Hogwarts" (0d2edb8f-fa34-4e73-89ec-fb9a63001440)
summaryNone
text

Test requirement 1 really l o n g text that is way too long to display here as that

\n", + "\n", + "

< > " '

\n", + "\n", + "
    \n", + "\t
  • This is a list
  • \n", + "\t
  • an unordered one
  • \n", + "
\n", + "\n", + "
    \n", + "\t
  1. Ordered list
  2. \n", + "\t
  3. Ok
  4. \n", + "
\n", + "
traces

(Empty list)

typeRequirementType "ReqType" (db47fca9-ddb6-4397-8d4b-e397e53d277e)
uuid3c2d312c-37c9-41b5-8c32-67578fa52dc3
xtypeRequirements:Requirement
" + ], + "text/plain": [ + "\n", + ".applied_property_value_groups = []\n", + ".applied_property_values = []\n", + ".attributes = [0] \n", + " [1] \n", + " [2] \n", + " [3] \n", + " [4] \n", + " [5] \n", + ".chapter_name = '2'\n", + ".constraints = []\n", + ".description = 'This is a test requirement of kind 1.'\n", + ".diagrams = []\n", + ".filtering_criteria = []\n", + ".foreign_id = 1\n", + ".identifier = 'REQTYPE-1'\n", + ".long_name = '1'\n", + ".name = 'TestReq1'\n", + ".owner = \n", + ".parent = \n", + ".prefix = '3'\n", + ".progress_status = 'NOT_SET'\n", + ".property_value_groups = []\n", + ".property_values = []\n", + ".pvmt = \n", + ".related = [0] \n", + " [1] \n", + " [2] \n", + " [3] \n", + ".relations = [0] to (078b2c69-4352-4cf9-9ea5-6573b75e5eec)>\n", + " [1] to (24c824ef-b187-4725-a051-a68707e82d70)>\n", + " [2] to (7de4c1a5-e106-4171-902a-502b816b60b0)>\n", + " [3] to (57033242-3766-4961-8091-ce3d9326ed67)>\n", + ".requirements = [0] \n", + " [1] \n", + " [2] \n", + " [3] \n", + ".summary = None\n", + ".text = Markup('

Test requirement 1 really l o n g text that is way too long to display here as that

\\n\\n

< > " '

\\n\\n
    \\n\\t
  • This is a list
  • \\n\\t
  • an unordered one
  • \\n
\\n\\n
    \\n\\t
  1. Ordered [...]\n", + ".traces = []\n", + ".type = \n", + ".uuid = '3c2d312c-37c9-41b5-8c32-67578fa52dc3'\n", + ".xtype = 'Requirements:Requirement'" + ] + }, + "execution_count": 3, + "metadata": {}, + "output_type": "execute_result" + } + ], + "source": [ + "model.oa.all_requirements[0]" + ] + }, + { + "cell_type": "markdown", + "id": "896e3633", + "metadata": {}, + "source": [ + "## Filtering for requirements by type\n", + "\n", + "You probably know about typing of your Requirements. You can have a view of all requirement type folders directly on a specific layer and see the stored requirement_types:" + ] + }, + { + "cell_type": "code", + "execution_count": 4, + "id": "8dad0bed", + "metadata": {}, + "outputs": [ + { + "data": { + "text/html": [ + "

    CapellaTypesFolder (CapellaRequirements:CapellaTypesFolder)

    applied_property_value_groups

    (Empty list)

    applied_property_values

    (Empty list)

    constraints

    (Empty list)

    data_type_definitions
    1. DataTypeDefinition "DataTypeDef" (3b7ec38a-e26a-4c23-9fa3-275af3f629ee)
    2. EnumerationDataTypeDefinition "EnumDataTypeDef" (637caf95-3229-4607-99a0-7d7b990bc97f)
    descriptionNone
    diagrams

    (Empty list)

    filtering_criteria

    (Empty list)

    identifierNone
    long_nameTypes
    module_types
    1. ModuleType "ModuleType" (a67e7f43-4b49-425c-a6a7-d44e1054a488)
    nameNone
    parentOperationalAnalysis "Operational Analysis" (ddbef16d-ddb9-4162-934c-f1e40e6f8bed)
    prefixNone
    progress_statusNOT_SET
    property_value_groups

    (Empty list)

    property_values

    (Empty list)

    pvmt<capellambse.extensions.pvmt.PropertyValueProxy object at 0x7f3ad72c8f50>
    relation_types
    1. RelationType "RelationType" (f1aceb81-5f70-4469-a127-94830eb9be04)
    requirement_types
    1. RequirementType "ReqType" (db47fca9-ddb6-4397-8d4b-e397e53d277e)
    requirements

    (Empty list)

    summaryNone
    traces

    (Empty list)

    typeNone
    uuid67bba9cf-953c-4f0b-9986-41991c68d241
    xtypeCapellaRequirements:CapellaTypesFolder
    " + ], + "text/plain": [ + "\n", + ".applied_property_value_groups = []\n", + ".applied_property_values = []\n", + ".constraints = []\n", + ".data_type_definitions = [0] \n", + " [1] \n", + ".description = None\n", + ".diagrams = []\n", + ".filtering_criteria = []\n", + ".identifier = None\n", + ".long_name = 'Types'\n", + ".module_types = [0] \n", + ".name = None\n", + ".parent = \n", + ".prefix = None\n", + ".progress_status = 'NOT_SET'\n", + ".property_value_groups = []\n", + ".property_values = []\n", + ".pvmt = \n", + ".relation_types = [0] \n", + ".requirement_types = [0] \n", + ".requirements = []\n", + ".summary = None\n", + ".traces = []\n", + ".type = None\n", + ".uuid = '67bba9cf-953c-4f0b-9986-41991c68d241'\n", + ".xtype = 'CapellaRequirements:CapellaTypesFolder'" + ] + }, + "execution_count": 4, + "metadata": {}, + "output_type": "execute_result" + } + ], + "source": [ + "model.oa.requirement_types_folders[0]" + ] + }, + { + "cell_type": "code", + "execution_count": 5, + "id": "28c463c1", + "metadata": {}, + "outputs": [ + { + "data": { + "text/html": [ + "

    RequirementType (Requirements:RequirementType)

    applied_property_value_groups

    (Empty list)

    applied_property_values

    (Empty list)

    attribute_definitions
    1. AttributeDefinition "AttrDef" (682bd51d-5451-4930-a97e-8bfca6c3a127)
    2. AttributeDefinitionEnumeration "AttrDefEnum" (c316ab07-c5c3-4866-a896-92e34733055c)
    3. AttributeDefinitionEnumeration "MultiEnum" (f31d4dd3-99f7-41d1-b1ce-f07b20d26eac)
    4. AttributeDefinition "version" (0bd40628-10a5-451e-86da-187c93bc3b10)
    constraints

    (Empty list)

    descriptionNone
    diagrams

    (Empty list)

    filtering_criteria

    (Empty list)

    identifierNone
    long_nameReqType
    nameNone
    ownerCapellaTypesFolder "Types" (67bba9cf-953c-4f0b-9986-41991c68d241)
    parentCapellaTypesFolder "Types" (67bba9cf-953c-4f0b-9986-41991c68d241)
    prefixNone
    progress_statusNOT_SET
    property_value_groups

    (Empty list)

    property_values

    (Empty list)

    pvmt<capellambse.extensions.pvmt.PropertyValueProxy object at 0x7f3ad72f7910>
    requirements

    (Empty list)

    summaryNone
    traces

    (Empty list)

    typeNone
    uuiddb47fca9-ddb6-4397-8d4b-e397e53d277e
    xtypeRequirements:RequirementType
    " + ], + "text/plain": [ + "\n", + ".applied_property_value_groups = []\n", + ".applied_property_values = []\n", + ".attribute_definitions = [0] \n", + " [1] \n", + " [2] \n", + " [3] \n", + ".constraints = []\n", + ".description = None\n", + ".diagrams = []\n", + ".filtering_criteria = []\n", + ".identifier = None\n", + ".long_name = 'ReqType'\n", + ".name = None\n", + ".owner = \n", + ".parent = \n", + ".prefix = None\n", + ".progress_status = 'NOT_SET'\n", + ".property_value_groups = []\n", + ".property_values = []\n", + ".pvmt = \n", + ".requirements = []\n", + ".summary = None\n", + ".traces = []\n", + ".type = None\n", + ".uuid = 'db47fca9-ddb6-4397-8d4b-e397e53d277e'\n", + ".xtype = 'Requirements:RequirementType'" + ] + }, + "execution_count": 5, + "metadata": {}, + "output_type": "execute_result" + } + ], + "source": [ + "reqtype = model.oa.requirement_types_folders[0].requirement_types[0]\n", + "reqtype" + ] + }, + { + "cell_type": "markdown", + "id": "62baecda", + "metadata": {}, + "source": [ + "Now we actually want to filter the requirements by this type with name 'ReqType'. Very original name, we know. You wouldn't really see the filtering effect since all requirements in the oa layer are of that type. So let us create one new requirement dynamically:" + ] + }, + { + "cell_type": "code", + "execution_count": 6, + "id": "2d985277", + "metadata": {}, + "outputs": [ + { + "data": { + "text/html": [ + "

    New showcase req1 (Requirements:Requirement)

    applied_property_value_groups

    (Empty list)

    applied_property_values

    (Empty list)

    attributes

    (Empty list)

    chapter_nameNone
    constraints

    (Empty list)

    descriptionNone
    diagrams

    (Empty list)

    filtering_criteria

    (Empty list)

    foreign_idNone
    identifierNone
    long_nameNone
    nameNew showcase req1
    ownerCapellaModule "Test Module" (f8e2195d-b5f5-4452-a12b-79233d943d5e)
    parentCapellaModule "Test Module" (f8e2195d-b5f5-4452-a12b-79233d943d5e)
    prefixNone
    progress_statusNOT_SET
    property_value_groups

    (Empty list)

    property_values

    (Empty list)

    pvmt<capellambse.extensions.pvmt.PropertyValueProxy object at 0x7f3afc3ddd50>
    related

    (Empty list)

    relations

    (Empty list)

    requirements

    (Empty list)

    summaryNone
    text
    traces

    (Empty list)

    typeNone
    uuid0ce7877e-68d6-4113-8aca-be69b2e25253
    xtypeRequirements:Requirement
    " + ], + "text/plain": [ + "\n", + ".applied_property_value_groups = []\n", + ".applied_property_values = []\n", + ".attributes = []\n", + ".chapter_name = None\n", + ".constraints = []\n", + ".description = None\n", + ".diagrams = []\n", + ".filtering_criteria = []\n", + ".foreign_id = None\n", + ".identifier = None\n", + ".long_name = None\n", + ".name = 'New showcase req1'\n", + ".owner = \n", + ".parent = \n", + ".prefix = None\n", + ".progress_status = 'NOT_SET'\n", + ".property_value_groups = []\n", + ".property_values = []\n", + ".pvmt = \n", + ".related = []\n", + ".relations = []\n", + ".requirements = []\n", + ".summary = None\n", + ".text = ''\n", + ".traces = []\n", + ".type = None\n", + ".uuid = '0ce7877e-68d6-4113-8aca-be69b2e25253'\n", + ".xtype = 'Requirements:Requirement'" + ] + }, + "metadata": {}, + "output_type": "display_data" + }, + { + "data": { + "text/html": [ + "
    1. Requirement "TestReq1" (3c2d312c-37c9-41b5-8c32-67578fa52dc3)
    2. Requirement "TypedReq2" (0a9a68b1-ba9a-4793-b2cf-4448f0b4b8cc)
    3. Requirement "TestReq3" (79291c33-5147-4543-9398-9077d582576d)
    4. Requirement "TypedReq1" (85d41db2-9e17-438b-95cf-49342452ddf3)
    5. Requirement "New showcase req1" (0ce7877e-68d6-4113-8aca-be69b2e25253)
    6. Requirement "TestReq" (1092f69a-5f3a-4fe6-a8fd-b2dffde90650)
    " + ], + "text/plain": [ + "[0] \n", + "[1] \n", + "[2] \n", + "[3] \n", + "[4] \n", + "[5] " + ] + }, + "execution_count": 6, + "metadata": {}, + "output_type": "execute_result" + } + ], + "source": [ + "new_req = model.oa.requirement_modules[0].requirements.create(name=\"New showcase req1\")\n", + "display(new_req)\n", + "model.oa.all_requirements" + ] + }, + { + "cell_type": "markdown", + "id": "58941b28", + "metadata": {}, + "source": [ + "Info about creation: Whenever you have an ElementList which deals with one\n", + "specified classtype, here it is Requirement, then the classtype name doesn't\n", + "need to be included." + ] + }, + { + "cell_type": "code", + "execution_count": 7, + "id": "9156aac3", + "metadata": {}, + "outputs": [ + { + "data": { + "text/html": [ + "
    1. Requirement "TestReq1" (3c2d312c-37c9-41b5-8c32-67578fa52dc3)
    2. Requirement "TypedReq2" (0a9a68b1-ba9a-4793-b2cf-4448f0b4b8cc)
    3. Requirement "TestReq3" (79291c33-5147-4543-9398-9077d582576d)
    4. Requirement "TypedReq1" (85d41db2-9e17-438b-95cf-49342452ddf3)
    5. Requirement "TestReq" (1092f69a-5f3a-4fe6-a8fd-b2dffde90650)
    " + ], + "text/plain": [ + "[0] \n", + "[1] \n", + "[2] \n", + "[3] \n", + "[4] " + ] + }, + "execution_count": 7, + "metadata": {}, + "output_type": "execute_result" + } + ], + "source": [ + "model.oa.all_requirements.by_type(\"ReqType\")" + ] + }, + { + "cell_type": "markdown", + "id": "f6687b07", + "metadata": {}, + "source": [ + "We see our Requirements with type `ReqType` are missing the new one.\n", + "Let's change that, without using capella." + ] + }, + { + "cell_type": "code", + "execution_count": 8, + "id": "19f5f917", + "metadata": {}, + "outputs": [ + { + "data": { + "text/html": [ + "
    1. Requirement "TestReq1" (3c2d312c-37c9-41b5-8c32-67578fa52dc3)
    2. Requirement "TypedReq2" (0a9a68b1-ba9a-4793-b2cf-4448f0b4b8cc)
    3. Requirement "TestReq3" (79291c33-5147-4543-9398-9077d582576d)
    4. Requirement "TypedReq1" (85d41db2-9e17-438b-95cf-49342452ddf3)
    5. Requirement "New showcase req1" (0ce7877e-68d6-4113-8aca-be69b2e25253)
    6. Requirement "TestReq" (1092f69a-5f3a-4fe6-a8fd-b2dffde90650)
    " + ], + "text/plain": [ + "[0] \n", + "[1] \n", + "[2] \n", + "[3] \n", + "[4] \n", + "[5] " + ] + }, + "execution_count": 8, + "metadata": {}, + "output_type": "execute_result" + } + ], + "source": [ + "new_req.type = reqtype\n", + "model.oa.all_requirements.by_type(\"ReqType\")" + ] + }, + { + "cell_type": "markdown", + "id": "3878d8d5", + "metadata": {}, + "source": [ + "If you are sure about attributes when creating a Requirement you can also include\n", + "it during creation. " + ] + }, + { + "cell_type": "code", + "execution_count": 9, + "id": "f46766d6", + "metadata": {}, + "outputs": [ + { + "data": { + "text/html": [ + "
    1. Requirement "TestReq1" (3c2d312c-37c9-41b5-8c32-67578fa52dc3)
    2. Requirement "TypedReq2" (0a9a68b1-ba9a-4793-b2cf-4448f0b4b8cc)
    3. Requirement "TestReq3" (79291c33-5147-4543-9398-9077d582576d)
    4. Requirement "TypedReq1" (85d41db2-9e17-438b-95cf-49342452ddf3)
    5. Requirement "New showcase req1" (0ce7877e-68d6-4113-8aca-be69b2e25253)
    6. Requirement "ReqType during Creation" (ec5f6680-501c-4a6c-a2ce-3ae8fe93be00)
    7. Requirement "TestReq" (1092f69a-5f3a-4fe6-a8fd-b2dffde90650)
    " + ], + "text/plain": [ + "[0] \n", + "[1] \n", + "[2] \n", + "[3] \n", + "[4] \n", + "[5] \n", + "[6] " + ] + }, + "execution_count": 9, + "metadata": {}, + "output_type": "execute_result" + } + ], + "source": [ + "model.oa.requirement_modules[0].requirements.create(\n", + " name=\"ReqType during Creation\",\n", + " type=reqtype,\n", + ")\n", + "model.oa.all_requirements.by_type(\"ReqType\")" + ] + }, + { + "cell_type": "code", + "execution_count": 10, + "id": "2530e4a4", + "metadata": {}, + "outputs": [ + { + "data": { + "text/html": [ + "

    TestReq1 (Requirements:Requirement)

    applied_property_value_groups

    (Empty list)

    applied_property_values

    (Empty list)

    attributes
    1. BooleanValueAttribute "AttrDef" (9c692405-b8aa-4caa-b988-51d27db5cd1b)
    2. DateValueAttribute "AttrDef" (b97c09b5-948a-46e8-a656-69d764ddce7d)
    3. IntegerValueAttribute "AttrDef" (85dfd42c-7f6e-4236-a181-bdd784040431)
    4. RealValueAttribute "AttrDef" (d2231d14-854d-4625-b48b-6cf1c2554367)
    5. StringValueAttribute "AttrDef" (ee8a69ef-61b9-4db9-9a0f-628e5d4704e1)
    6. BooleanValueAttribute "None" (dcb8614e-2d1c-4cb3-aa0c-667a297e7489)
    chapter_name2
    constraints

    (Empty list)

    descriptionThis is a test requirement of kind 1.
    diagrams

    (Empty list)

    filtering_criteria

    (Empty list)

    foreign_id1
    identifierREQTYPE-1
    long_name1
    nameTestReq1
    ownerFolder "Folder" (e16f5cc1-3299-43d0-b1a0-82d31a137111)
    parentFolder "Folder" (e16f5cc1-3299-43d0-b1a0-82d31a137111)
    prefix3
    progress_statusNOT_SET
    property_value_groups

    (Empty list)

    property_values

    (Empty list)

    pvmt<capellambse.extensions.pvmt.PropertyValueProxy object at 0x7f3ad72dfd90>
    related
    1. Entity "Weather" (4bf0356c-89dd-45e9-b8a6-e0332c026d33)
    2. SystemFunction "Sysexfunc" (00e7b925-cf4c-4cb0-929e-5409a1cd872b)
    3. Requirement "TypedReq1" (85d41db2-9e17-438b-95cf-49342452ddf3)
    4. LogicalComponent "Hogwarts" (0d2edb8f-fa34-4e73-89ec-fb9a63001440)
    relations
    1. CapellaIncomingRelation "Controlling the weather" (078b2c69-4352-4cf9-9ea5-6573b75e5eec)
    2. CapellaIncomingRelation "Test req" (24c824ef-b187-4725-a051-a68707e82d70)
    3. InternalRelation "None" (7de4c1a5-e106-4171-902a-502b816b60b0)
    4. CapellaOutgoingRelation "None" (57033242-3766-4961-8091-ce3d9326ed67)
    requirements
    1. Entity "Weather" (4bf0356c-89dd-45e9-b8a6-e0332c026d33)
    2. SystemFunction "Sysexfunc" (00e7b925-cf4c-4cb0-929e-5409a1cd872b)
    3. Requirement "TypedReq1" (85d41db2-9e17-438b-95cf-49342452ddf3)
    4. LogicalComponent "Hogwarts" (0d2edb8f-fa34-4e73-89ec-fb9a63001440)
    summaryNone
    text

    Test requirement 1 really l o n g text that is way too long to display here as that

    \n", + "\n", + "

    < > " '

    \n", + "\n", + "
      \n", + "\t
    • This is a list
    • \n", + "\t
    • an unordered one
    • \n", + "
    \n", + "\n", + "
      \n", + "\t
    1. Ordered list
    2. \n", + "\t
    3. Ok
    4. \n", + "
    \n", + "
    traces

    (Empty list)

    typeRequirementType "ReqType" (db47fca9-ddb6-4397-8d4b-e397e53d277e)
    uuid3c2d312c-37c9-41b5-8c32-67578fa52dc3
    xtypeRequirements:Requirement
    " + ], + "text/plain": [ + "\n", + ".applied_property_value_groups = []\n", + ".applied_property_values = []\n", + ".attributes = [0] \n", + " [1] \n", + " [2] \n", + " [3] \n", + " [4] \n", + " [5] \n", + ".chapter_name = '2'\n", + ".constraints = []\n", + ".description = 'This is a test requirement of kind 1.'\n", + ".diagrams = []\n", + ".filtering_criteria = []\n", + ".foreign_id = 1\n", + ".identifier = 'REQTYPE-1'\n", + ".long_name = '1'\n", + ".name = 'TestReq1'\n", + ".owner = \n", + ".parent = \n", + ".prefix = '3'\n", + ".progress_status = 'NOT_SET'\n", + ".property_value_groups = []\n", + ".property_values = []\n", + ".pvmt = \n", + ".related = [0] \n", + " [1] \n", + " [2] \n", + " [3] \n", + ".relations = [0] to (078b2c69-4352-4cf9-9ea5-6573b75e5eec)>\n", + " [1] to (24c824ef-b187-4725-a051-a68707e82d70)>\n", + " [2] to (7de4c1a5-e106-4171-902a-502b816b60b0)>\n", + " [3] to (57033242-3766-4961-8091-ce3d9326ed67)>\n", + ".requirements = [0] \n", + " [1] \n", + " [2] \n", + " [3] \n", + ".summary = None\n", + ".text = Markup('

    Test requirement 1 really l o n g text that is way too long to display here as that

    \\n\\n

    < > " '

    \\n\\n
      \\n\\t
    • This is a list
    • \\n\\t
    • an unordered one
    • \\n
    \\n\\n
      \\n\\t
    1. Ordered [...]\n", + ".traces = []\n", + ".type = \n", + ".uuid = '3c2d312c-37c9-41b5-8c32-67578fa52dc3'\n", + ".xtype = 'Requirements:Requirement'" + ] + }, + "execution_count": 10, + "metadata": {}, + "output_type": "execute_result" + } + ], + "source": [ + "model.oa.all_requirements[0]" + ] + }, + { + "cell_type": "markdown", + "id": "ec7a07c8", + "metadata": {}, + "source": [ + "## Export all requirements to excel with pandas\n", + "\n", + "With `Pandas` you are able to generate data tables and export to many file-types\n", + "like .xlsx for excel. Note that this requires the `xlsxwriter` module in\n", + "addition to `pandas` itself." + ] + }, + { + "cell_type": "code", + "execution_count": 11, + "id": "acbe3e06", + "metadata": {}, + "outputs": [], + "source": [ + "import pandas as pd\n", + "\n", + "requirements = [{\"id\": f\"Req_{i}\", \"uuid\": req.uuid, \"text\": req.text} for i, req in enumerate(model.oa.all_requirements)]\n", + "data = pd.DataFrame(requirements)\n", + "data.to_excel(excel_writer=\"Test_requirements.xlsx\")" + ] + }, + { + "cell_type": "markdown", + "id": "03bf76ca-0703-4ae7-9270-e743063d7372", + "metadata": {}, + "source": [ + "## Export a Requirements module as Requirements Interchange Format (`.reqif`) file\n", + "\n", + "`capellambse` features basic export functionality for requirements modules, which allows to write `.reqif` or compressed `.reqifz` files. These can then be imported into other ReqIF-compliant applications, like CodeBeamer or IBM DOORS.\n", + "\n", + "Let's first take a glance at the module we're going to export:" + ] + }, + { + "cell_type": "code", + "execution_count": 12, + "id": "e3f44e4c", + "metadata": {}, + "outputs": [ + { + "data": { + "text/html": [ + "

      Test Module (CapellaRequirements:CapellaModule)

      applied_property_value_groups

      (Empty list)

      applied_property_values

      (Empty list)

      attributes
      1. EnumerationValueAttribute "AttrDefEnum" (11a2e07a-41f7-4eee-a94e-f3e33738c487)
      constraints

      (Empty list)

      descriptionThis is a test requirement module.
      diagrams

      (Empty list)

      filtering_criteria

      (Empty list)

      folders
      1. Folder "Folder" (e16f5cc1-3299-43d0-b1a0-82d31a137111)
      identifier1
      long_nameModule
      nameTest Module
      parentOperationalAnalysis "Operational Analysis" (ddbef16d-ddb9-4162-934c-f1e40e6f8bed)
      prefixT
      progress_statusNOT_SET
      property_value_groups

      (Empty list)

      property_values

      (Empty list)

      pvmt<capellambse.extensions.pvmt.PropertyValueProxy object at 0x7f3ad4787a10>
      requirement_types_folders

      (Empty list)

      requirements
      1. Requirement "TypedReq1" (85d41db2-9e17-438b-95cf-49342452ddf3)
      2. Requirement "New showcase req1" (0ce7877e-68d6-4113-8aca-be69b2e25253)
      3. Requirement "ReqType during Creation" (ec5f6680-501c-4a6c-a2ce-3ae8fe93be00)
      summaryNone
      traces

      (Empty list)

      typeModuleType "ModuleType" (a67e7f43-4b49-425c-a6a7-d44e1054a488)
      uuidf8e2195d-b5f5-4452-a12b-79233d943d5e
      xtypeCapellaRequirements:CapellaModule
      " + ], + "text/plain": [ + "\n", + ".applied_property_value_groups = []\n", + ".applied_property_values = []\n", + ".attributes = [0] \n", + ".constraints = []\n", + ".description = 'This is a test requirement module.'\n", + ".diagrams = []\n", + ".filtering_criteria = []\n", + ".folders = [0] \n", + ".identifier = '1'\n", + ".long_name = 'Module'\n", + ".name = 'Test Module'\n", + ".parent = \n", + ".prefix = 'T'\n", + ".progress_status = 'NOT_SET'\n", + ".property_value_groups = []\n", + ".property_values = []\n", + ".pvmt = \n", + ".requirement_types_folders = []\n", + ".requirements = [0] \n", + " [1] \n", + " [2] \n", + ".summary = None\n", + ".traces = []\n", + ".type = \n", + ".uuid = 'f8e2195d-b5f5-4452-a12b-79233d943d5e'\n", + ".xtype = 'CapellaRequirements:CapellaModule'" + ] + }, + "execution_count": 12, + "metadata": {}, + "output_type": "execute_result" + } + ], + "source": [ + "req_module = model.by_uuid(\"f8e2195d-b5f5-4452-a12b-79233d943d5e\")\n", + "req_module" + ] + }, + { + "cell_type": "code", + "execution_count": 13, + "id": "5c555118-246d-47be-b462-6ce01822e156", + "metadata": {}, + "outputs": [], + "source": [ + "req_module.to_reqif(req_module.name + \".reqif\")" + ] + }, + { + "cell_type": "markdown", + "id": "94297761-841e-4900-945c-b5f87825627f", + "metadata": {}, + "source": [ + "And that's it! This code has created the file \"Test Module.reqif\" next to this notebook, which contains all Requirements defined in that module." + ] + } + ], + "metadata": { + "kernelspec": { + "display_name": "Python 3 (ipykernel)", + "language": "python", + "name": "python3" + }, + "language_info": { + "codemirror_mode": { + "name": "ipython", + "version": 3 + }, + "file_extension": ".py", + "mimetype": "text/x-python", + "name": "python", + "nbconvert_exporter": "python", + "pygments_lexer": "ipython3", + "version": "3.11.5" + }, + "vscode": { + "interpreter": { + "hash": "e7370f93d1d0cde622a1f8e1c04877d8463912d04d973331ad4851f04de6915a" + } + } + }, + "nbformat": 4, + "nbformat_minor": 5 +} diff --git a/_sources/examples/07 Code Generation.ipynb.txt b/_sources/examples/07 Code Generation.ipynb.txt new file mode 100644 index 000000000..852511d03 --- /dev/null +++ b/_sources/examples/07 Code Generation.ipynb.txt @@ -0,0 +1,396 @@ +{ + "cells": [ + { + "cell_type": "markdown", + "metadata": {}, + "source": [ + "# Introduction to Code Generation\n", + "\n", + "This notebook exemplifies how to generate automatically code in terms of interfaces. For this three examples are provided. The first one creates a ROS message, the second a standard python class and the last a protobuf interface." + ] + }, + { + "cell_type": "code", + "execution_count": 1, + "metadata": {}, + "outputs": [], + "source": [ + "import capellambse" + ] + }, + { + "cell_type": "code", + "execution_count": 2, + "metadata": {}, + "outputs": [], + "source": [ + "path_to_model = \"../../../tests/data/melodymodel/5_0/Melody Model Test.aird\"\n", + "model = capellambse.MelodyModel(path_to_model, jupyter_untrusted=True)" + ] + }, + { + "cell_type": "markdown", + "metadata": {}, + "source": [ + "In particular, we want to create code from our class:" + ] + }, + { + "cell_type": "code", + "execution_count": 3, + "metadata": {}, + "outputs": [ + { + "name": "stderr", + "output_type": "stream", + "text": [ + "Unknown global filter 'hide.technical.interfaces.filter'\n", + "Unknown global filter 'ModelExtensionFilter'\n" + ] + }, + { + "data": { + "image/png": "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", + "text/plain": [ + "" + ] + }, + "execution_count": 3, + "metadata": {}, + "output_type": "execute_result" + } + ], + "source": [ + "model.diagrams.by_name(\"[CDB] CodeGeneration\")" + ] + }, + { + "cell_type": "markdown", + "metadata": {}, + "source": [ + "In order to access the classes, we can simply access the `data_package` of the operational layer, and from there access the attribute `classes`." + ] + }, + { + "cell_type": "code", + "execution_count": 4, + "metadata": {}, + "outputs": [ + { + "data": { + "text/html": [ + "
      1. Class "Twist" (8164ae8b-36d5-4502-a184-5ec064db4ec3)
      2. Class "Trajectory" (c5ea0585-7657-4764-9eb2-3a6584980ce6)
      3. Class "Waypoint" (2a923851-a4ca-4fd2-a4b3-302edb8ac178)
      4. Class "Example" (a7ecc231-c55e-4ab9-ae14-9558e3ec2a34)
      " + ], + "text/plain": [ + "[0] \n", + "[1] \n", + "[2] \n", + "[3] " + ] + }, + "execution_count": 4, + "metadata": {}, + "output_type": "execute_result" + } + ], + "source": [ + "data_pkg = model.oa.data_package\n", + "data_pkg.classes" + ] + }, + { + "cell_type": "markdown", + "metadata": {}, + "source": [ + "## ROS2 IDL Message\n", + "\n", + "Let's have a brief look into the structure of ROS2 Message descriptions. They are stored in `.msg` files and comprised of a type and name, separated by whitespace, i.e.:\n", + "\n", + "```\n", + "fieldtype1 fieldname1\n", + "fieldtype2[] fieldname2\n", + "```" + ] + }, + { + "cell_type": "code", + "execution_count": 5, + "metadata": {}, + "outputs": [], + "source": [ + "def class_to_ros2_idl(cls):\n", + " filename = f\"{cls.name}.msg\"\n", + " lines = []\n", + " for prop in cls.properties:\n", + " multiplicity = (\"\", \"[]\")[prop.max_card.value > 1]\n", + " lines.append(f\"{prop.type.name}{multiplicity} {prop.name}\")\n", + " text = \"\\n\".join(lines)\n", + " with open(filename, \"w\") as file:\n", + " file.write(text)\n", + " print(f\"# file: {filename} \\n{text}\\n\")" + ] + }, + { + "cell_type": "markdown", + "metadata": {}, + "source": [ + "In our example, files would be generated with the following content:" + ] + }, + { + "cell_type": "code", + "execution_count": 6, + "metadata": {}, + "outputs": [ + { + "name": "stdout", + "output_type": "stream", + "text": [ + "# file: Twist.msg \n", + "\n", + "\n", + "# file: Trajectory.msg \n", + "Waypoint[] waypoints\n", + "\n", + "# file: Waypoint.msg \n", + "float lat\n", + "float lon\n", + "float alt\n", + "Example[] examples\n", + "\n", + "# file: Example.msg \n", + "str test\n", + "\n" + ] + } + ], + "source": [ + "data_pkg = model.oa.data_package\n", + "for cls in data_pkg.classes:\n", + " class_to_ros2_idl(cls)" + ] + }, + { + "cell_type": "markdown", + "metadata": {}, + "source": [ + "## Interface for python classes\n", + "\n", + "A python class has the following structure:\n", + "\n", + "\n", + "```\n", + "class class_name:\n", + " name1: type \n", + " name2: [type]\n", + "```" + ] + }, + { + "cell_type": "markdown", + "metadata": {}, + "source": [ + "A python interface can be generated as follows:" + ] + }, + { + "cell_type": "code", + "execution_count": 7, + "metadata": {}, + "outputs": [], + "source": [ + "def class_to_python(cls, current_classes=None):\n", + " lines = [f\"class {cls.name}:\"]\n", + " current_classes = [cls]\n", + " if not cls.properties:\n", + " lines.append(4*\" \" + \"pass\")\n", + " for prop in cls.properties:\n", + " if (\n", + " isinstance(\n", + " prop.type, capellambse.model.crosslayer.information.Class\n", + " )\n", + " and prop.type not in current_classes\n", + " ):\n", + " nested_text = class_to_python(prop.type, current_classes)\n", + " lines = [nested_text] + [\"\\n\"] + lines\n", + " multiplicity = (f\"{prop.type.name}\", f\"list[{prop.type.name}]\")[\n", + " prop.max_card.value > 1\n", + " ]\n", + " lines.append(4*\" \" + f\"{prop.name}: {multiplicity}\")\n", + " text = \"\\n\".join(lines)\n", + "\n", + " return text" + ] + }, + { + "cell_type": "code", + "execution_count": 8, + "metadata": {}, + "outputs": [ + { + "name": "stdout", + "output_type": "stream", + "text": [ + "# file: trajectory.py \n", + "class Example:\n", + " test: str\n", + "\n", + "\n", + "class Waypoint:\n", + " lat: float\n", + " lon: float\n", + " alt: float\n", + " examples: list[Example]\n", + "\n", + "\n", + "class Trajectory:\n", + " waypoints: list[Waypoint]\n", + "\n" + ] + } + ], + "source": [ + "trajectory = data_pkg.classes.by_name(\"Trajectory\")\n", + "text = class_to_python(trajectory)\n", + "filename = f\"{trajectory.name.lower()}.py\"\n", + "with open(filename, \"w\") as file:\n", + " file.write(text)\n", + "print(f\"# file: {filename} \\n{text}\\n\")" + ] + }, + { + "cell_type": "markdown", + "metadata": {}, + "source": [ + "## Interface for Protocol Buffers (Protobuf) \n", + "\n", + "Protobuf Message descriptions are stored in `.proto` files where a class definition starts with `message` and each property of the class is defined by at least three parts: the data type, name and its order number. Classes can also be nested in other classes. An example is shown in the following:\n", + "\n", + "\n", + "```\n", + "syntax = \"proto3\";\n", + "\n", + "message class1 {\n", + " datatype class1_name1 = 1;\n", + " datatype class1_name2 = 2;\n", + " message class2 {\n", + " datatype class2_name1 = 1;\n", + " }\n", + " repeated class2 class_name = 3;\n", + "}\n", + "\n", + "```\n" + ] + }, + { + "cell_type": "code", + "execution_count": 9, + "metadata": {}, + "outputs": [], + "source": [ + "def class_to_proto(cls, current_classes=None, indent=\"\"):\n", + " if current_classes is None:\n", + " current_classes = [cls]\n", + " lines = ['syntax = \"proto3\";\\n']\n", + " indent += \" \"*4\n", + " lines.append(f\"{indent[:-4]}message {cls.name} {{\")\n", + " else:\n", + " lines = [f\"{indent[:-4]}message {cls.name} {{\"]\n", + "\n", + " for counter, prop in enumerate(cls.properties, start=1):\n", + " multiplicity = (\"\", \"[]\")[prop.max_card.value > 1]\n", + " if (\n", + " isinstance(\n", + " prop.type, capellambse.model.crosslayer.information.Class\n", + " )\n", + " and prop.type not in current_classes\n", + " ):\n", + " current_classes.append(prop.type)\n", + " nested_text = class_to_proto(\n", + " prop.type, current_classes, indent + \" \"*4\n", + " )\n", + " lines.append(nested_text)\n", + " lines.append(\n", + " f\"{indent}repeated {prop.type.name}{multiplicity} {prop.name} = {counter};\"\n", + " )\n", + " else:\n", + " lines.append(\n", + " f\"{indent}{prop.type.name}{multiplicity} {prop.name} = {counter};\"\n", + " )\n", + " lines.append(f\"{indent[:-4]}}}\")\n", + " text = \"\\n\".join(lines)\n", + " return text" + ] + }, + { + "cell_type": "markdown", + "metadata": {}, + "source": [ + "The protobuf interface of class `Trajectory` would look as follows:\n" + ] + }, + { + "cell_type": "code", + "execution_count": 10, + "metadata": {}, + "outputs": [ + { + "name": "stdout", + "output_type": "stream", + "text": [ + "# file: Trajectory.proto \n", + "syntax = \"proto3\";\n", + "\n", + "message Trajectory {\n", + " message Waypoint {\n", + " float lat = 1;\n", + " float lon = 2;\n", + " float alt = 3;\n", + " message Example {\n", + " str test = 1;\n", + " }\n", + " repeated Example[] examples = 4;\n", + " }\n", + " repeated Waypoint[] waypoints = 1;\n", + "}\n", + "\n" + ] + } + ], + "source": [ + "trajectory = data_pkg.classes.by_name(\"Trajectory\")\n", + "text = class_to_proto(trajectory)\n", + "filename = f\"{trajectory.name}.proto\"\n", + "with open(filename, \"w\") as file:\n", + " file.write(text)\n", + "print(f\"# file: {filename} \\n{text}\\n\")" + ] + } + ], + "metadata": { + "kernelspec": { + "display_name": "Python 3 (ipykernel)", + "language": "python", + "name": "python3" + }, + "language_info": { + "codemirror_mode": { + "name": "ipython", + "version": 3 + }, + "file_extension": ".py", + "mimetype": "text/x-python", + "name": "python", + "nbconvert_exporter": "python", + "pygments_lexer": "ipython3", + "version": "3.11.5" + }, + "vscode": { + "interpreter": { + "hash": "c5ea7dc634d8047a259e5b898f154d237fbe6934b444b1a949475949608d751e" + } + } + }, + "nbformat": 4, + "nbformat_minor": 4 +} diff --git a/_sources/examples/08 Property Values.ipynb.txt b/_sources/examples/08 Property Values.ipynb.txt new file mode 100644 index 000000000..347f93245 --- /dev/null +++ b/_sources/examples/08 Property Values.ipynb.txt @@ -0,0 +1,343 @@ +{ + "cells": [ + { + "cell_type": "markdown", + "id": "f4599550", + "metadata": {}, + "source": [ + "# Property Values\n", + "\n", + "capellambse provides access to property values and property value groups, as well as the Property Value Management (PVMT) extension.\n", + "\n", + "This notebook will show how to access and work with basic property values and groups." + ] + }, + { + "cell_type": "code", + "execution_count": 1, + "id": "fb62b658", + "metadata": {}, + "outputs": [], + "source": [ + "import capellambse\n", + "model = capellambse.MelodyModel(\n", + " \"../../../tests/data/melodymodel/5_0/Melody Model Test.aird\",\n", + " jupyter_untrusted=True\n", + ")" + ] + }, + { + "cell_type": "markdown", + "id": "f4effeec", + "metadata": {}, + "source": [ + "Model objects can own property values and PV groups. To access those, use the `property_values` and `property_value_groups` attributes respectively:" + ] + }, + { + "cell_type": "code", + "execution_count": 2, + "id": "3491dc40", + "metadata": {}, + "outputs": [], + "source": [ + "obj = model.search(\"LogicalComponent\").by_name(\"Whomping Willow\")" + ] + }, + { + "cell_type": "code", + "execution_count": 3, + "id": "6f1304b7", + "metadata": {}, + "outputs": [ + { + "data": { + "text/html": [ + "
      1. IntegerPropertyValue "cars_defeated": 1 (a928fa22-cef7-4357-9b87-675a432f6591)
      " + ], + "text/plain": [ + "[0] " + ] + }, + "execution_count": 3, + "metadata": {}, + "output_type": "execute_result" + } + ], + "source": [ + "obj.property_values" + ] + }, + { + "cell_type": "code", + "execution_count": 4, + "id": "573ce32d", + "metadata": {}, + "outputs": [ + { + "data": { + "text/html": [ + "

      cars_defeated (org.polarsys.capella.core.data.capellacore:IntegerPropertyValue)

      applied_property_value_groups

      (Empty list)

      applied_property_values

      (Empty list)

      constraints

      (Empty list)

      description
      diagrams

      (Empty list)

      enumerations

      (Empty list)

      filtering_criteria

      (Empty list)

      namecars_defeated
      parentLogicalComponent "Whomping Willow" (3bdd4fa2-5646-44a1-9fa6-80c68433ddb7)
      progress_statusNOT_SET
      property_value_groups

      (Empty list)

      property_values

      (Empty list)

      pvmt<capellambse.extensions.pvmt.PropertyValueProxy object at 0x7f7a6eb58b10>
      requirements

      (Empty list)

      summaryNone
      traces

      (Empty list)

      uuida928fa22-cef7-4357-9b87-675a432f6591
      value1
      xtypeorg.polarsys.capella.core.data.capellacore:IntegerPropertyValue
      " + ], + "text/plain": [ + "\n", + ".applied_property_value_groups = []\n", + ".applied_property_values = []\n", + ".constraints = []\n", + ".description = ''\n", + ".diagrams = []\n", + ".enumerations = []\n", + ".filtering_criteria = []\n", + ".name = 'cars_defeated'\n", + ".parent = \n", + ".progress_status = 'NOT_SET'\n", + ".property_value_groups = []\n", + ".property_values = []\n", + ".pvmt = \n", + ".requirements = []\n", + ".summary = None\n", + ".traces = []\n", + ".uuid = 'a928fa22-cef7-4357-9b87-675a432f6591'\n", + ".value = 1\n", + ".xtype = 'org.polarsys.capella.core.data.capellacore:IntegerPropertyValue'" + ] + }, + "execution_count": 4, + "metadata": {}, + "output_type": "execute_result" + } + ], + "source": [ + "obj.property_values[0]" + ] + }, + { + "cell_type": "code", + "execution_count": 5, + "id": "3c105268", + "metadata": {}, + "outputs": [ + { + "data": { + "text/html": [ + "
      1. PropertyValueGroup "Stats" (81bcc3d3-24d2-411e-a296-9943029acfd9)
      " + ], + "text/plain": [ + "[0] " + ] + }, + "execution_count": 5, + "metadata": {}, + "output_type": "execute_result" + } + ], + "source": [ + "obj.property_value_groups" + ] + }, + { + "cell_type": "markdown", + "id": "06426fae", + "metadata": {}, + "source": [ + "## dict-like access to property values\n", + "\n", + "In addition to standard attribute-based access, these property value-related attributes can also behave like dicts in some cases. Here the dict key equates to the `name` of a property value or group, and the dict value equates to either the `value` of a property value object, or a list of PV objects in the group (which again can behave dict-like)." + ] + }, + { + "cell_type": "markdown", + "id": "574022f7", + "metadata": {}, + "source": [ + "This significantly shortens the way to a specific value. For comparison, this would be the \"usual\" attribute-based way of accessing it:" + ] + }, + { + "cell_type": "code", + "execution_count": 6, + "id": "a10cf035", + "metadata": {}, + "outputs": [ + { + "data": { + "text/plain": [ + "150" + ] + }, + "execution_count": 6, + "metadata": {}, + "output_type": "execute_result" + } + ], + "source": [ + "obj.property_value_groups.by_name(\"Stats\").property_values.by_name(\"WIS\").value" + ] + }, + { + "cell_type": "markdown", + "id": "12800ba6", + "metadata": {}, + "source": [ + "And here is the same again, leveraging the dict-like behavior:" + ] + }, + { + "cell_type": "code", + "execution_count": 7, + "id": "0e03011b", + "metadata": {}, + "outputs": [ + { + "data": { + "text/plain": [ + "150" + ] + }, + "execution_count": 7, + "metadata": {}, + "output_type": "execute_result" + } + ], + "source": [ + "obj.property_value_groups[\"Stats\"][\"WIS\"]" + ] + }, + { + "cell_type": "markdown", + "id": "3e8f3666", + "metadata": {}, + "source": [ + "This of course works for all property value related attributes." + ] + }, + { + "cell_type": "code", + "execution_count": 8, + "id": "c56a533f", + "metadata": {}, + "outputs": [ + { + "data": { + "text/plain": [ + "1" + ] + }, + "execution_count": 8, + "metadata": {}, + "output_type": "execute_result" + } + ], + "source": [ + "obj.property_values[\"cars_defeated\"]" + ] + }, + { + "cell_type": "code", + "execution_count": 9, + "id": "2cc26810", + "metadata": {}, + "outputs": [ + { + "data": { + "text/html": [ + "
      1. IntegerPropertyValue "HP": 8000 (0e95950d-272f-4b50-b68d-ee8fed002c94)
      2. IntegerPropertyValue "STR": 42 (32be6f26-4b33-4861-b501-3bfce3df94cb)
      3. IntegerPropertyValue "AGI": 0 (addb1e4a-9ecc-411c-b7ef-b76388e5840d)
      4. IntegerPropertyValue "WIS": 150 (7c8c9c21-86b9-4dba-9c7c-cd9db2f09c11)
      5. IntegerPropertyValue "INT": 12 (647a5565-a1de-4e15-ab21-eb628eea413c)
      " + ], + "text/plain": [ + "[0] \n", + "[1] \n", + "[2] \n", + "[3] \n", + "[4] " + ] + }, + "execution_count": 9, + "metadata": {}, + "output_type": "execute_result" + } + ], + "source": [ + "obj.property_value_groups[\"Stats\"]" + ] + }, + { + "cell_type": "code", + "execution_count": 10, + "id": "b7943324", + "metadata": {}, + "outputs": [ + { + "data": { + "text/plain": [ + "150" + ] + }, + "execution_count": 10, + "metadata": {}, + "output_type": "execute_result" + } + ], + "source": [ + "obj.property_value_groups[\"Stats\"][\"WIS\"]" + ] + }, + { + "cell_type": "markdown", + "id": "65b98527", + "metadata": {}, + "source": [ + "These property values can also be written to:" + ] + }, + { + "cell_type": "code", + "execution_count": 11, + "id": "b34fdf10", + "metadata": {}, + "outputs": [ + { + "data": { + "text/html": [ + "
      1. IntegerPropertyValue "HP": 8000 (0e95950d-272f-4b50-b68d-ee8fed002c94)
      2. IntegerPropertyValue "STR": 42 (32be6f26-4b33-4861-b501-3bfce3df94cb)
      3. IntegerPropertyValue "AGI": 0 (addb1e4a-9ecc-411c-b7ef-b76388e5840d)
      4. IntegerPropertyValue "WIS": 150 (7c8c9c21-86b9-4dba-9c7c-cd9db2f09c11)
      5. IntegerPropertyValue "INT": 18 (647a5565-a1de-4e15-ab21-eb628eea413c)
      " + ], + "text/plain": [ + "[0] \n", + "[1] \n", + "[2] \n", + "[3] \n", + "[4] " + ] + }, + "execution_count": 11, + "metadata": {}, + "output_type": "execute_result" + } + ], + "source": [ + "obj.property_value_groups[\"Stats\"][\"INT\"] = 18\n", + "obj.property_value_groups[\"Stats\"]" + ] + } + ], + "metadata": { + "kernelspec": { + "display_name": "Python 3 (ipykernel)", + "language": "python", + "name": "python3" + }, + "language_info": { + "codemirror_mode": { + "name": "ipython", + "version": 3 + }, + "file_extension": ".py", + "mimetype": "text/x-python", + "name": "python", + "nbconvert_exporter": "python", + "pygments_lexer": "ipython3", + "version": "3.11.5" + } + }, + "nbformat": 4, + "nbformat_minor": 5 +} diff --git a/_sources/examples/09 Context Diagrams.ipynb.txt b/_sources/examples/09 Context Diagrams.ipynb.txt new file mode 100644 index 000000000..705369808 --- /dev/null +++ b/_sources/examples/09 Context Diagrams.ipynb.txt @@ -0,0 +1,710 @@ +{ + "cells": [ + { + "cell_type": "markdown", + "id": "2cdf5dd5", + "metadata": {}, + "source": [ + "# Context Diagrams Extension\n", + "\n", + "We have an extension that visualizes contexts of Capella objects. The viewpoint\n", + "definiton depends on the object type. The extension is external to\n", + "`capellambse` library and needs to be installed separately. You may use the\n", + "below command to install it or find more guidance in the [package\n", + "documentation](https://dsd-dbs.github.io/capellambse-context-diagrams/)." + ] + }, + { + "cell_type": "code", + "execution_count": null, + "id": "e49dac7a", + "metadata": {}, + "outputs": [], + "source": [ + "!pip install capellambse_context_diagrams" + ] + }, + { + "cell_type": "markdown", + "id": "c93ced3d", + "metadata": {}, + "source": [ + "Now that the lib is installed we can load a test model" + ] + }, + { + "cell_type": "code", + "execution_count": 1, + "id": "d64ff435", + "metadata": {}, + "outputs": [ + { + "name": "stderr", + "output_type": "stream", + "text": [ + "Cannot load PVMT extension: ValueError: Provided model does not have a PropertyValuePkg\n", + "Property values are not available in this model\n" + ] + } + ], + "source": [ + "from IPython.display import display, HTML # we'll need that later\n", + "import capellambse\n", + "\n", + "model = capellambse.MelodyModel(\n", + " \"../../../tests/data/melodymodel/5_2/Melody Model Test.aird\",\n", + " jupyter_untrusted=True\n", + ")" + ] + }, + { + "cell_type": "markdown", + "id": "e88f088c", + "metadata": {}, + "source": [ + "## Logical Component Context" + ] + }, + { + "cell_type": "markdown", + "id": "71d8da10", + "metadata": {}, + "source": [ + "and now as the model is loaded lets have a look at a context diagram for the `School` component" + ] + }, + { + "cell_type": "code", + "execution_count": 2, + "id": "09c4d2ee", + "metadata": {}, + "outputs": [ + { + "data": { + "image/png": "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", + "text/plain": [ + "" + ] + }, + "execution_count": 2, + "metadata": {}, + "output_type": "execute_result" + } + ], + "source": [ + "cmp = model.la.all_components.by_name(\"Whomping Willow\")\n", + "cmp.context_diagram" + ] + }, + { + "cell_type": "markdown", + "id": "e721cfa4", + "metadata": {}, + "source": [ + "## Component Exchange Context\n", + "\n", + "We also found it useful to spell-out the `ComponentExchange`s - this gives us a nice overview of functional interactions between the components" + ] + }, + { + "cell_type": "code", + "execution_count": 3, + "id": "040bfc14", + "metadata": {}, + "outputs": [ + { + "data": { + "image/png": "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", + "text/plain": [ + "" + ] + }, + "execution_count": 3, + "metadata": {}, + "output_type": "execute_result" + } + ], + "source": [ + "cmp.related_exchanges.by_name(\"Punishment\").context_diagram" + ] + }, + { + "cell_type": "markdown", + "id": "8cff65de", + "metadata": {}, + "source": [ + "it also works for more complex arrangements but we dont have one in the current test model ;-)" + ] + }, + { + "cell_type": "markdown", + "id": "e9ecb867", + "metadata": {}, + "source": [ + "## Functional (dataflow) Context\n", + "\n", + "same applies for functions - we saw it useful to see what functions are dependent on an output of a function of interest and what inputs it needs to do what it should. We frequently use that view in documentation" + ] + }, + { + "cell_type": "code", + "execution_count": 4, + "id": "ba760fda", + "metadata": {}, + "outputs": [ + { + "data": { + "image/png": "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", + "text/plain": [ + "" + ] + }, + "execution_count": 4, + "metadata": {}, + "output_type": "execute_result" + } + ], + "source": [ + "fnc = model.la.all_functions.by_name(\"educate Wizards\")\n", + "fnc.context_diagram" + ] + }, + { + "cell_type": "markdown", + "id": "b073bfc3", + "metadata": {}, + "source": [ + "## Layers, other than LA\n", + "\n", + "Other layers are supported too - for complete list see https://dsd-dbs.github.io/capellambse-context-diagrams/ (Features section)" + ] + }, + { + "cell_type": "code", + "execution_count": 5, + "id": "99743283", + "metadata": {}, + "outputs": [ + { + "data": { + "text/html": [ + "

      Context of Root Operational Activity

      " + ], + "text/plain": [ + "" + ] + }, + "metadata": {}, + "output_type": "display_data" + }, + { + "data": { + "image/png": "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", + "text/plain": [ + "" + ] + }, + "metadata": {}, + "output_type": "display_data" + }, + { + "data": { + "text/html": [ + "

      Context of Sleep

      " + ], + "text/plain": [ + "" + ] + }, + "metadata": {}, + "output_type": "display_data" + }, + { + "data": { + "image/png": "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", + "text/plain": [ + "" + ] + }, + "metadata": {}, + "output_type": "display_data" + }, + { + "data": { + "text/html": [ + "

      Context of Eat

      " + ], + "text/plain": [ + "" + ] + }, + "metadata": {}, + "output_type": "display_data" + }, + { + "data": { + "image/png": "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", + "text/plain": [ + "" + ] + }, + "metadata": {}, + "output_type": "display_data" + }, + { + "data": { + "text/html": [ + "

      Context of Spawn wild animal

      " + ], + "text/plain": [ + "" + ] + }, + "metadata": {}, + "output_type": "display_data" + }, + { + "data": { + "image/png": "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", + "text/plain": [ + "" + ] + }, + "metadata": {}, + "output_type": "display_data" + }, + { + "data": { + "text/html": [ + "

      Context of Make Food

      " + ], + "text/plain": [ + "" + ] + }, + "metadata": {}, + "output_type": "display_data" + }, + { + "data": { + "image/png": "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", + "text/plain": [ + "" + ] + }, + "metadata": {}, + "output_type": "display_data" + }, + { + "data": { + "text/html": [ + "

      Context of Hunt wild animal

      " + ], + "text/plain": [ + "" + ] + }, + "metadata": {}, + "output_type": "display_data" + }, + { + "data": { + "image/png": "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", + "text/plain": [ + "" + ] + }, + "metadata": {}, + "output_type": "display_data" + }, + { + "data": { + "text/html": [ + "

      Context of Search for animals

      " + ], + "text/plain": [ + "" + ] + }, + "metadata": {}, + "output_type": "display_data" + }, + { + "data": { + "image/png": "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", + "text/plain": [ + "" + ] + }, + "metadata": {}, + "output_type": "display_data" + }, + { + "data": { + "text/html": [ + "

      Context of Play games

      " + ], + "text/plain": [ + "" + ] + }, + "metadata": {}, + "output_type": "display_data" + }, + { + "data": { + "image/png": "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", + "text/plain": [ + "" + ] + }, + "metadata": {}, + "output_type": "display_data" + }, + { + "data": { + "text/html": [ + "

      Context of Spawn tree

      " + ], + "text/plain": [ + "" + ] + }, + "metadata": {}, + "output_type": "display_data" + }, + { + "data": { + "image/png": "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", + "text/plain": [ + "" + ] + }, + "metadata": {}, + "output_type": "display_data" + }, + { + "data": { + "text/html": [ + "

      Context of Create landslide

      " + ], + "text/plain": [ + "" + ] + }, + "metadata": {}, + "output_type": "display_data" + }, + { + "data": { + "image/png": "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", + "text/plain": [ + "" + ] + }, + "metadata": {}, + "output_type": "display_data" + }, + { + "data": { + "text/html": [ + "

      Context of Spawn water

      " + ], + "text/plain": [ + "" + ] + }, + "metadata": {}, + "output_type": "display_data" + }, + { + "data": { + "image/png": "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", + "text/plain": [ + "" + ] + }, + "metadata": {}, + "output_type": "display_data" + }, + { + "data": { + "text/html": [ + "

      Context of Build house

      " + ], + "text/plain": [ + "" + ] + }, + "metadata": {}, + "output_type": "display_data" + }, + { + "data": { + "image/png": "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", + "text/plain": [ + "" + ] + }, + "metadata": {}, + "output_type": "display_data" + }, + { + "data": { + "text/html": [ + "

      Context of Run away

      " + ], + "text/plain": [ + "" + ] + }, + "metadata": {}, + "output_type": "display_data" + }, + { + "data": { + "image/png": "iVBORw0KGgoAAAANSUhEUgAAAYEAAAA7CAIAAAASHaHUAAAABmJLR0QA/wD/AP+gvaeTAAAVeElEQVR4nO3deVgUR9oA8Ld6DubgHIbT4b4RBEQRDwQUURQvNBp1VXSNumuMiWTjtRo1Jpq4RhPj9a3JatTPG0XFA49IwCiIAopGRUAIN4RrgLm79g8QAYXBARxc6/f044PT1e/U9FS/U1XT040wxkAQBKEllLYrQBDEO43kIIIgtInkIIIgtInkIIIgtInkIIIgtInkIIIgtInkIIIgtInkIIIgtInkIIIgtImp7QoQxNsnO7s8MTH7blp+eVltdZWEpt/1HxswGJShgGdpaTB4oF1goKO+Pqfj26LX/a0G2fs9WWeaAtERhYXVe3+69TSr3CfQ0b2fjbGZvq4hl6KQtuulZbSKrq6oL86rSEt4+ntKXkSEV8RELyazQ8Os18hBZO/3fJ1pCoRat2/nbt12PWii95BwDyaLoe3q9FCVpeLoPYlysXT92jA9PfWfgh3NQWTvv3VetykQ7bt9O3fb9/GzV4yycTbVdl16Oozh4qHkh7dytmwer7btdSgHdeHer6mq4vJ4LDa7k3GIjnitpkC0o7CwOuofp+esCiMJqONi9yeV55RtWD+m/dGS+hzUuPdXar73FXL56T3flj1O4SEln8eR0SwJ4qk4huPm/d3S2kazmETHxf7coaZAtGP9hktmLuaBE7w6E0QqkSSci3565wYDMGawdY3N/UdPcHB166pK9jQ0jf/9+bmgQbbjx3m2U0x9Durk3i/+I3f/slkfD9ZxcbBGPCHiGSOuEPGE9cD/188XzL38h02YpFnktwBdX1uF+AKudg/9DjYFoi3Z2eVrv7j02Q/vd2YW4vy+HXm/npjSz8Db3Ympa4J4wko553Ti4zs55bOWrTE2NevCCvccxXkV//f52b3/nsbjtTn0Yaxdu7adENnZ5dGn709bMoyiNJzX3LUoYkuo3MzYoOJZ8X9u1erYWhmXPtqXZeznYjZsaMCv128w9AUCoYmaKMrCRz99HXvgVNqFc3evZXF9/YSdOK5x4dGD3zxzDHJ+aa/Iis9sPrsv9vEfujZeos4PF+msQ4e2ZVkP9+B1uLJt101zCCGRg8nPP8SHhbmzyFze6zt7NkPfSuDsY6VxhJPbv3TKP7IwyNjSzJhi8xGLh1g8nq5hX2/P4MH+m77a7DEwgM1pa7CMxWkxyVeSnt3PKMqrpIQiPe5b8yWDrgH3j8xSSkU7OAjbKqPmxSQmZnsHODJYDAygwfLkQcZgwwIdJgWysh2JzMmD6QPHsxN/ybgcn5RPAwB8OG/2tVMn1MXB4is/JEgnTv12++xtOyO/XOhghDSrT9MCr3xceT/1Em/gF5smRfrzUKfiNyzIMXLO5veNqa6oW+cWM2uBnbtFYmJ2V7Wtd8rdtHxXXxuN9760XlKTEj3KlQ1YEn3w1pq9Vzck1hSc2jXg63QJAJ/H/XLF0pj/7G07Al1792y21L6Xs72eLOnip19klOGubyLdtvQZ5Jhwo72Gp+Ycxbtp+aGzBzUeGa9PVi/RZakAAIBWAMVmM3SgOCalXkd2+1zx9EWOwGazaZVKXXwsraikdPUoAMCAmGwmAND1xbE74n8rVUgZ5lOihvub4JyY09+dq8KgpJwH/WNpbzh5eEeeMX6azwubuHI0Stgddy5bToNgzIowV6DzL55fmyCvKMee8yfOG8BHAFCfffDHR89KC1YX9wr5aOQwwZ9Xd1299Exar9ALXjQ6wl2n6FizgOMFFNCtn/HEgXVJuhYsyfOwvKKjh3bzJ64L5xUdPbAuSc+CVffnn8hztKMq/VlumVhmP2jZUlfJ2ZZBGl4xYPG9377flyNWyetNfFeu8jLv7HiuzyDHhOuPQkNdOxnnHVRaWmtsrq/pQQAFuc9c9WsBOFBfkc5xXjONt+loZizHYQ7rQZJsWDAPTE2E0lpxO/ExIJ6dr30/HvTz0y2ek5RW6x7CKfv1itxnlEgP0QW/pZc4evcVViYnSswMq+4/lup7uQ1x5jV2MLC0IDXr7lMxbSwaEmgpS07JsuwXYEvhqvwrCQrvcDsThMtS0rPMXaxKs5uKSZPu5tr0HSSiAEsfXc5iBPR24mr28kVOJud/vtlOATU5qLS01thMX+NLTrt6+Wz/1jgcAHTMPhxYfeg6770Qi4rQsGBh2S/1NAAcO3XGJyBYXXxKEDLNbM3qfc+GegYHu/u56+kAnX38Srr3hA0jePW3LvzzSJ7vYuteI8K+HsdhQ/21tYfPPHAdixX5CuutO0KNGLg45vglQdCGxaZsmlYhKAHgeg9eMc+CkX9z+dfp+f0HiigAnt30SKfk5N7rFtswgc4/HnfVfMSGj4WqJwkr/nXTY2eg7ouAAIAxoNbPCKDjOWjlHDOqMaw/BQ0lMQbQ6TNoVaQpzvz14/WV8/ZMWcgVX1hx7Nwjl1kvBQHAGEtuHsu0XTxzul1DQ+r8Vb9FjmqaAtEWcbWEr8fV+JPYQmR9Q8wHAOAautZmfn2Ez+5l8ttNlQ2vPDdVETwMKior2Tx+e/Gbuscgl0nZuoY6CMtKrl+otR0p0kX0H4mp6XpePoKKhK1x4okBobaqXzaeLFk7Y7INBQC4puhWSg3f2VB+5/Lqx8NXW+cceWg9cJ5ZXfLNHd9LZvraRljU3jrxBC3QL2hW7HO7nCNxIr+55oy6nNPnq6aEaPzyDYz4lX/WtVNATQ4SV0v4+lyN2z/FZA5fvHn1ns9WhNFmro4f9m2Yk2YjntcIHvPY2bj7JZLpkyLVxxcODfu+T+md+IdXt+7bbzfsq2WijLSCzPSrW+9SIP2zQl4pxVa6bMWT+IzUrKrfs6s41SoAysZTZMjAgOvTk6R9FgjZgIFCDMAAyNhcjwUYmZtaybOraCx6/pHR+C+uv3dH5fOhgAWY5dR7KO/8gzJ6QFPA51itnxEZmvCZzcIKXvRJkaGQxwCMrIUibqm+DgbEs7GGh1Wql4JgAAyIbeeM/3/LeRjjHhRoa9nxCaU26QvUNAWiLTSNEYU0Pgo4fD7LfXRi1qUAT+NpkQMavpNB7wkRV4h4LJlcvurLXX9Zub7dFKQqTziQ8CQ7/WyGzvClk33YGOTQbMTT8DcgoeOk6R4eTNwr7/d9j2TYhoMAkIFtxHxbAFrmUntrY4Fygr3uv3LLaKOcNGrcZGZ6Su3EYXkZKutZtnaW8+2aisnfczLakF1Im5ncy6r07m9NafzyEYNSqeh2CqjJQTSNEUIap0AA8Bgw1ER0fN3W1Tr1+V62taaCSglV/KBAUliD+46cMH3KyI4GZxia+o039Rvlsn9x3JU8GwFHEDB7zBzX5xNaWJa07WSc3bB5Uzw8lM/O0s3fH1qlwk3vFbR666iWb2PjKqxSqZSNg0QGxWIA1WKrhmdMbvMZX4RttQAC9OJv1Ha1KYdZM77xfZpw9daaU7mfbA9y19H8PQAAAIpS0xSI9nSuHzo1asOpnbpXL1yc4c92cWycnZVI5Wd/+fVqWs60pf80EgjbfQqKZ+1l7eNcn56rGDTAALVoq81qiBonStkcllKhalwlL4vb/muyQteMX10i76UydfKm4zMqTTLrROHT8e69z/60KBB7eJsqyi41LyawH8A/f7esv9Xtmt6jhVRn90A71P9mtfOjAFNLqwWb96mUypzMx/mlJVwef/gkV119g44HV1YUSXjmehwEWFpbqeS7G3A8+nNjYrMjXBz0EcYYIajLz2e5z+hlpid/kFtNuzZExhgwIK6tvfynG2UTbUxY0JSM8PN5PcDPq9H0OEZcNw/V9qvF4+aaMwufpCit/2rSfKsGbT9ji7C4+R+N+/T5gxjqWwUBBLRKhQHLZVjQ22WCuyleeimzmnYzJef2aFHnj4IJf1teW/O389GHfjybzqCqMZXH4Bn2HzHuk1l+auNjjHg2faw8eRYLH+zfd6a873Qhg4EYCrkM41bN7HnTwk3tTZl6O4bh+22UDbs2o2RZDU0Z+npJz57Jljh5WlhSfepuxNyU2gQL6dRLLYohvu8Aas+tpyVFlkPtEe6+FPQGfzfPYDId3Xo7uvV+/U1x3aOYy9FpdTSTAsTznDMsWEAxw0dO2Xl5/dI7+hxkNjJsfpDhwHD9TasO3DHX44p5rBbf91EuU0O8vrmwbClbBxuErhjpov4pKdvJI4duubJiCYPDMQpePNyagsJWRVA7z9hhyKBlECTobSXbdu6gxSin2xdPZiIOk6ac/ZaYkAT0P0BX3yA88u+dicDsPcWXF3UzaWz4IL65C+fm8VMWw41KL6bJLMLa3AYJ9VgPHly7pWTdTX2gtEdA2fiZZnycOej7IIqC/r3Ll5yxWLOAQvmtiiHjgXbyqOuZAWMju/d8DjXnKI4Zs3vTsYXdWgPijVk+ZXdsLHk31asTi/l6ek3/1fZRgGvTz+UbhLnaMgFA+eyXO3k2vkPtmXR18Y2rf4iNLNyM6qotnb2FVUnnq+zH2JkgXPf7gxSmS6ATCwAAlCWpv998ojDxEvHL5HZDRAaq8vijZfZT3ayYQBc+OZmmN3a0BeflYlAZveRkxYK583p38nyk9hteB/pB3dcJI4ieh6bp1QsXGgoEk+fOdffx0XZ1AADpeo1tOqWCaRs8wBYAACgD84AI82bljAaEGzVswHfzCHzxONPMx3NCwwtpCMMUBs5onJSiLJ3fs2yjmKyyUGEX6NzdJ0R2YD6om2tAED3K/du3SwsLSwsLv1q61KNfv4jZs6Er5oPePrK0x5nOrh+wuvu1k37QO4SpKo3et0/btXgLxKWmetraWhgZZaSkZKSk6FGCgodDerl5aLtebxSuZVlEjBexuj0DkH7QO4RFl0bv/0XbtXgLlFVXX0tPd7e29rG3BwAmXXFo2eLAyAV+Ee9ru2pvkKBvnwCA7s8Ab+K7eaKHUFCm02f303Yt3gJN/aCG/yopwayv1vVy8yAfyN2BjMXeIUqGaURkpLZr0dOlJyWFPp+KbpgP+mRZoqWbBzkOugkZixGaq6urk8vlDX9zuVzOS1efkMlkTQUacDgcFov1huqnEc/+/U0tLVt+L5ZIDoPuQ/pBhOZmz5x+4eIlNosJAGy2zqHDR0NCQpoXWL58+a5dO1nMxmYmk8k3btoUFRWlhbp2GEVRX+ze3fz8ICAzEt2JzAcRmsMqxbdzPMcOcWVwDFKya2ZMf//R40yj59MoDUW+XLn4w5ljVPWVKknVii0HXwohfnLl9OX0Irm+nW/IqCH2ehqdjVL729aVqQHfLOrXNdfMbpWAiG5F7nFIdApisCkWl2Jx/L1MxwZ4jAkL9fR48R128u1kc4He/MmBlFJKK6SYVrXYGJedmR+2hT3rbyNcGBVPE5PzBtl35JRcOn/H6EjlnrglNs8L65i6+DgJWa9a1TXIJ3H3If0gohMQQkw2xeIgFodicaaM6r/o8z1u1llN691cIDo9d/Oug/+YPRIrpUArW2wujT8c7/d5xkfDXly5VpUZe/pPR8vca8mVFkFTxnkJKABQFqfExCQWcvuMeW+YPafq1oHo9HuyDV/ozFgwP8icAgAKKxWYQvgVq4ieTs27xGBQtLLZVSnI8tYuWIW7/L4aCABRLMTkUEwOxeR4uTsVVUoj/fnNl3Wj+afOX6cVUlohad0PYjm66cdt3Xolpw4/f0j15MTC95bESswsag7PHLM+RQa45MQHkzc+0rcxeLQpfMLOTJpr27+PpaV36MjBjvrU861idl/MVaFXrOoaGJNF86V9avpBhka8mqp6A2PdLnszCS0RV9cbGPK6OipCTBbF4lAsLmJxdZh8uVLVqoStMTOvuBwrpVghxa36QUzvFad3f792Y5jzB4LAuSs3fBZuj4CymrTy88ihLByM7wYduLVc9+Z3ZbOO/zTNHE3tXxc25sc7879ydTQRKv38PEWt84yOeZurOkndgURoTE0OsrI2Ksgu0xeQHPTWqyitEZp0/ftIMdiIxW0Yi+UUVdmY8FsVeFik6G1vSisktFIKdOsMxRKFRO0NiZL+cf2HRXMjvrJMWQUACAEAID1HB8apYmnes2pRXwECAMrM3QUnl6jAsstfRrsYDEqloikGGdppAtNqOuBqclDAYPvryU/dfO26tFaEFjxJy/PtK+rioAgBg90wEENMzs20VCaT+uTUiyvGKpXKlFzFrEn+WNEwJ618dRyOVdBHi4bv3JejAA5dUlisAmBBfW4eQzRGx9SMnfmoiA61oXB1br6ujXXD5bzb7OK3s0pDhkY8cSUZDWiotkpNB1xdDgpw2P9zcmlBpUkvo/ZLEj2ZUqF6kJT5/tq2r3SlKYrBapiTBobOj8fjQsbP9PR8cSfFY8eO+vaVzJ/gR0ur8cvzQbVXv/zoPHuwn5MxnRO7/U7Iio0cSALZ5S2f/cgJkcdsq5q9dwDX0WIBN3z+GuPFPkWH/s398Ignk5LaWRXvOHbZb1xff1eTlperE7S5SnNW1kYFOWX6JAdppKJMTQdcTQ7i89lTp/a9dChxRlQ4IncKfmslX77n5mJqZ2fc1YFRvQJXS1QMrPrx0BkOX/+7775rfjvM3x9m8GqflJZVqKTVKmm1RCprsbVu4N8/Q5fi059ksiym7IsPcdZDcqBEE+YEKTNzuO//dDTEkQFg98Hxc47Hz6WVWC05ssS/FwXAC/vmYN3By7ef2vdzNWEDAMN5/EKmDQNesaoLkNFAZ6jtgKu/1zNN41X/jDUQmQRPHtCldSPekPyskhPbL279dqK5uX7XRo6cPfPMmTMACACcnBxPRp8WiVq0tjVrVu/etbPxQscAALB+w8aFC9u5JqE89q/+VyKTtgb0oN9z1NXJ531w+C+fjSOjgdelVKh2rjj8xdqwdj7/1OcgABCLpVGfxjh42wVO9EOkM/RWyc8qObnzUtQnwb6+mt+q+A3qiTkIAE7H3L+W+IyMBl7XbxdS6wrKVq0MbaeMmvvNN9DRYQYFOZ2NTn14J7uXgzmH19n7zBBvgFKhSopLv3z4xqdLh70lCQgAACGGsWs/R4Oedag7O5smxGcWFVTZunX1vP7/rvysksuHb6xYPkJXt72M0aF+UAOlkj516t6Jk2kOHtZufo6mvYz0DPnkC8seBdO4rqa+orTm6b28h8mZbq6mf53r3+VDsHcTGQ28lo53wF8jBzUQi6Xx8U8Tb+QUFtVUVdSR2+b1KBSF9A14Jia6vr6iIYPtu2ES+p0mFsvWrL2AdNih0wMMheR3ra+mVKhuX72XHHfv06gOdcBfOwcRxLuMjAZeqTMdcJKDCOK1kdFAK53pgJMcRBCENr3rfUiCILSL5CCCILSJ5CCCILSJ5CCCILSJ5CCCILSJ5CCCILSJ5CCCILSJ5CCCILTpvxn6IjUg1/3kAAAAAElFTkSuQmCC", + "text/plain": [ + "" + ] + }, + "metadata": {}, + "output_type": "display_data" + }, + { + "data": { + "text/html": [ + "

      Context of Repeat

      " + ], + "text/plain": [ + "" + ] + }, + "metadata": {}, + "output_type": "display_data" + }, + { + "data": { + "image/png": "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", + "text/plain": [ + "" + ] + }, + "metadata": {}, + "output_type": "display_data" + }, + { + "data": { + "text/html": [ + "

      Context of Kill

      " + ], + "text/plain": [ + "" + ] + }, + "metadata": {}, + "output_type": "display_data" + }, + { + "data": { + "image/png": "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", + "text/plain": [ + "" + ] + }, + "metadata": {}, + "output_type": "display_data" + } + ], + "source": [ + "for act in model.oa.all_activities:\n", + " display(HTML(f\"

      Context of {act.name}

      \"))\n", + " display(act.context_diagram)" + ] + }, + { + "cell_type": "markdown", + "id": "b9b28af3", + "metadata": {}, + "source": [ + "## Customizing context views\n", + "\n", + "For almost every context view you can do some tuning. Just like in Capella itself you can apply view filters, like hide or show exchange items instead or next to exchange names. You can see more of the tuning options here: https://dsd-dbs.github.io/capellambse-context-diagrams/extras/filters/#capella-filters" + ] + }, + { + "cell_type": "code", + "execution_count": 6, + "id": "2c88ed82", + "metadata": {}, + "outputs": [ + { + "data": { + "text/html": [ + "

      Context of Stay alive

      " + ], + "text/plain": [ + "" + ] + }, + "metadata": {}, + "output_type": "display_data" + }, + { + "data": { + "image/png": "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", + "text/plain": [ + "" + ] + }, + "metadata": {}, + "output_type": "display_data" + }, + { + "data": { + "text/html": [ + "

      Context of Escape predators

      " + ], + "text/plain": [ + "" + ] + }, + "metadata": {}, + "output_type": "display_data" + }, + { + "data": { + "image/png": "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", + "text/plain": [ + "" + ] + }, + "metadata": {}, + "output_type": "display_data" + }, + { + "data": { + "text/html": [ + "

      Context of Eat food

      " + ], + "text/plain": [ + "" + ] + }, + "metadata": {}, + "output_type": "display_data" + }, + { + "data": { + "image/png": "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", + "text/plain": [ + "" + ] + }, + "metadata": {}, + "output_type": "display_data" + }, + { + "data": { + "text/html": [ + "

      Context of Provide environment to live in

      " + ], + "text/plain": [ + "" + ] + }, + "metadata": {}, + "output_type": "display_data" + }, + { + "data": { + "image/png": "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", + "text/plain": [ + "" + ] + }, + "metadata": {}, + "output_type": "display_data" + }, + { + "data": { + "text/html": [ + "

      Context of Provide clean water

      " + ], + "text/plain": [ + "" + ] + }, + "metadata": {}, + "output_type": "display_data" + }, + { + "data": { + "image/png": "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", + "text/plain": [ + "" + ] + }, + "metadata": {}, + "output_type": "display_data" + }, + { + "data": { + "text/html": [ + "

      Context of Provide food sources

      " + ], + "text/plain": [ + "" + ] + }, + "metadata": {}, + "output_type": "display_data" + }, + { + "data": { + "image/png": "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", + "text/plain": [ + "" + ] + }, + "metadata": {}, + "output_type": "display_data" + } + ], + "source": [ + "for cap in model.oa.all_capabilities:\n", + " display(HTML(f\"

      Context of {cap.name}

      \"))\n", + " diag = cap.context_diagram\n", + " diag.display_symbols_as_boxes = True\n", + " diag.render(None, no_edgelabels=True)\n", + " display(diag)" + ] + } + ], + "metadata": { + "kernelspec": { + "display_name": "Python 3 (ipykernel)", + "language": "python", + "name": "python3" + }, + "language_info": { + "codemirror_mode": { + "name": "ipython", + "version": 3 + }, + "file_extension": ".py", + "mimetype": "text/x-python", + "name": "python", + "nbconvert_exporter": "python", + "pygments_lexer": "ipython3", + "version": "3.11.5" + }, + "vscode": { + "interpreter": { + "hash": "e7370f93d1d0cde622a1f8e1c04877d8463912d04d973331ad4851f04de6915a" + } + } + }, + "nbformat": 4, + "nbformat_minor": 5 +} diff --git a/_sources/examples/10 Declarative Modeling.ipynb.txt b/_sources/examples/10 Declarative Modeling.ipynb.txt new file mode 100644 index 000000000..cce8f731c --- /dev/null +++ b/_sources/examples/10 Declarative Modeling.ipynb.txt @@ -0,0 +1,272 @@ +{ + "cells": [ + { + "cell_type": "markdown", + "metadata": {}, + "source": [ + "# Declarative Modeling Example\n", + "\n", + "Declarative approach to modeling means that one could define or update a model using a fragment of structured text. A number of fragments could be \"played\" against a model in a sequence to build it up.\n", + "\n", + "Enabling declarative modeling for Capella models enables a range of complex automations around modeling process that are explainable / transparent to human auditors.\n", + "\n", + "This notebook will demonstrate a basic application of this approach to modeling on a coffee machine example. Please note that we will not model any specific modeling process but rather a \"free-form\" demo." + ] + }, + { + "cell_type": "markdown", + "metadata": {}, + "source": [ + "## System Analysis of a Coffee Machine\n", + "\n", + "Lets do a quick system analysis of a coffee machine. Lets assume that our meta-solution is an automated coffee machine for a household use. We may look into variant management scenario in a separate example.\n", + "\n", + "### 0. Initialize\n", + "\n", + "But before we can model something lets first initialize the model. We will use an empty Capella 5.2 model as a starting point." + ] + }, + { + "cell_type": "code", + "execution_count": 1, + "metadata": {}, + "outputs": [ + { + "name": "stderr", + "output_type": "stream", + "text": [ + "Cannot load PVMT extension: ValueError: Provided model does not have a PropertyValuePkg\n", + "Property values are not available in this model\n" + ] + } + ], + "source": [ + "import capellambse\n", + "import io\n", + "from capellambse import decl\n", + "\n", + "model = capellambse.MelodyModel(\n", + " \"../../../tests/data/decl/empty_project_52/empty_project_52.aird\",\n", + " jupyter_untrusted=True\n", + ")" + ] + }, + { + "cell_type": "markdown", + "metadata": {}, + "source": [ + "to visualize the modeling results we'll use context-diagrams extension, you may get one by uncommenting and running the command below" + ] + }, + { + "cell_type": "code", + "execution_count": null, + "metadata": {}, + "outputs": [], + "source": [ + "!pip install capellambse_context_diagrams" + ] + }, + { + "cell_type": "markdown", + "metadata": {}, + "source": [ + "lets verify that the model is empty at SA layer:" + ] + }, + { + "cell_type": "code", + "execution_count": 2, + "metadata": {}, + "outputs": [ + { + "name": "stdout", + "output_type": "stream", + "text": [ + "At SA layer the model has 0, out of which 0 are allocated to Root Component\n" + ] + } + ], + "source": [ + "functions_allocated = model.sa.root_component.allocated_functions\n", + "functions_available = model.sa.root_function.functions\n", + "print(f\"At SA layer the model has {len(functions_available)}, out of which {len(functions_allocated)} are allocated to Root Component\")" + ] + }, + { + "cell_type": "markdown", + "metadata": {}, + "source": [ + "Also for this to work we'll need \"coordinates\" of some key elements in the model:" + ] + }, + { + "cell_type": "code", + "execution_count": 3, + "metadata": {}, + "outputs": [], + "source": [ + "root_function = model.sa.root_function\n", + "root_component = model.sa.root_component\n", + "structure = model.sa.component_package" + ] + }, + { + "cell_type": "markdown", + "metadata": {}, + "source": [ + "### 1. Context\n", + "\n", + "Lets start by renaming the root component from **System** to **Coffee Machine**, creating a human actor **User** and a component exchange between those two.\n", + "\n", + "We can achieve this by applying the following YAML patch to an empty Capella model:" + ] + }, + { + "cell_type": "code", + "execution_count": 4, + "metadata": { + "scrolled": true + }, + "outputs": [ + { + "data": { + "text/plain": [ + "{Promise(identifier='usr-port-promise'): \n", + " .applied_property_value_groups = []\n", + " .applied_property_values = []\n", + " .constraints = []\n", + " .description = ''\n", + " .direction = \n", + " .exchanges = ... # backreference to ComponentExchange - omitted: can be slow to compute\n", + " .filtering_criteria = []\n", + " .name = 'usr'\n", + " .owner = \n", + " .parent = \n", + " .progress_status = 'NOT_SET'\n", + " .property_value_groups = []\n", + " .property_values = []\n", + " .requirements = []\n", + " .summary = None\n", + " .traces = []\n", + " .uuid = '3a2ff9e0-adcc-4d2d-ae4f-7001a0c25475'\n", + " .xtype = 'org.polarsys.capella.core.data.fa:ComponentPort',\n", + " Promise(identifier='cm-port-promise'): \n", + " .applied_property_value_groups = []\n", + " .applied_property_values = []\n", + " .constraints = []\n", + " .description = ''\n", + " .direction = \n", + " .exchanges = ... # backreference to ComponentExchange - omitted: can be slow to compute\n", + " .filtering_criteria = []\n", + " .name = 'cm'\n", + " .owner = \n", + " .parent = \n", + " .progress_status = 'NOT_SET'\n", + " .property_value_groups = []\n", + " .property_values = []\n", + " .requirements = []\n", + " .summary = None\n", + " .traces = []\n", + " .uuid = '99f8db47-4771-4e4e-993a-b252398d8806'\n", + " .xtype = 'org.polarsys.capella.core.data.fa:ComponentPort'}" + ] + }, + "execution_count": 4, + "metadata": {}, + "output_type": "execute_result" + } + ], + "source": [ + "model_update = f\"\"\"\n", + "- parent: !uuid {root_component.uuid}\n", + " modify:\n", + " name: Coffee Machine\n", + "- parent: !uuid {root_component.uuid}\n", + " extend:\n", + " ports:\n", + " - name: usr\n", + " direction: INOUT\n", + " promise_id: usr-port-promise\n", + " exchanges:\n", + " - name: user interactions\n", + " source: !promise usr-port-promise\n", + " target: !promise cm-port-promise\n", + "- parent: !uuid {structure.uuid}\n", + " extend:\n", + " components:\n", + " - name: User\n", + " is_actor: true\n", + " is_human: true\n", + " ports:\n", + " - name: cm\n", + " direction: INOUT\n", + " promise_id: cm-port-promise\n", + "\"\"\"\n", + "# the below line applies the model_update to the model\n", + "decl.apply(model, io.StringIO(model_update))" + ] + }, + { + "cell_type": "markdown", + "metadata": {}, + "source": [ + "and now we can verify the changes by visualizing the context of our system under analysis:" + ] + }, + { + "cell_type": "code", + "execution_count": 5, + "metadata": {}, + "outputs": [ + { + "data": { + "image/png": "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", + "text/plain": [ + "" + ] + }, + "execution_count": 5, + "metadata": {}, + "output_type": "execute_result" + } + ], + "source": [ + "root_component.context_diagram" + ] + }, + { + "cell_type": "markdown", + "metadata": {}, + "source": [ + "Please note: the changes we made are not yet stored - if you like those to be saved you may use `model.save()` method. This will save the model back to where it was loaded from, for example by writing back into local files, or by creating a Git commit and pushing it back to the remote.\n" + ] + } + ], + "metadata": { + "kernelspec": { + "display_name": "Python 3 (ipykernel)", + "language": "python", + "name": "python3" + }, + "language_info": { + "codemirror_mode": { + "name": "ipython", + "version": 3 + }, + "file_extension": ".py", + "mimetype": "text/x-python", + "name": "python", + "nbconvert_exporter": "python", + "pygments_lexer": "ipython3", + "version": "3.11.5" + }, + "vscode": { + "interpreter": { + "hash": "c5ea7dc634d8047a259e5b898f154d237fbe6934b444b1a949475949608d751e" + } + } + }, + "nbformat": 4, + "nbformat_minor": 4 +} diff --git a/_sources/examples/11 Complexity Assessment.ipynb.txt b/_sources/examples/11 Complexity Assessment.ipynb.txt new file mode 100644 index 000000000..f76d88176 --- /dev/null +++ b/_sources/examples/11 Complexity Assessment.ipynb.txt @@ -0,0 +1,119 @@ +{ + "cells": [ + { + "cell_type": "markdown", + "metadata": {}, + "source": [ + "# Complexity Assessment\n", + "\n", + "This notebook demonstrates how to use / view the model complexity badge for a Capella model." + ] + }, + { + "cell_type": "code", + "execution_count": 1, + "metadata": {}, + "outputs": [], + "source": [ + "from IPython.display import SVG, display\n", + "import capellambse" + ] + }, + { + "cell_type": "code", + "execution_count": 2, + "metadata": {}, + "outputs": [ + { + "name": "stderr", + "output_type": "stream", + "text": [ + "Cannot load PVMT extension: ValueError: Provided model does not have a PropertyValuePkg\n", + "Property values are not available in this model\n" + ] + }, + { + "data": { + "image/svg+xml": [ + "\n", + "\n", + "\n", + "180\n", + "objects\n", + "\n", + "\n", + "31%\n", + "\n", + "16%\n", + "\n", + "27%\n", + "\n", + "27%\n", + "\n", + "18\n", + "diagrams\n", + "\n", + "\n", + "39%\n", + "\n", + "22%\n", + "\n", + "28%\n", + "\n", + "11%\n", + "\n", + "\n", + "\n", + "Operational Analysis\n", + "\n", + "System Analysis\n", + "\n", + "Logical Architecture\n", + "\n", + "Physical Architecture\n", + "\n", + "\n", + "" + ], + "text/plain": [ + "" + ] + }, + "execution_count": 2, + "metadata": {}, + "output_type": "execute_result" + } + ], + "source": [ + "model = capellambse.MelodyModel(\"../../../tests/data/melodymodel/5_2/Melody Model Test.aird\")\n", + "SVG(model.description_badge)" + ] + } + ], + "metadata": { + "kernelspec": { + "display_name": "Python 3 (ipykernel)", + "language": "python", + "name": "python3" + }, + "language_info": { + "codemirror_mode": { + "name": "ipython", + "version": 3 + }, + "file_extension": ".py", + "mimetype": "text/x-python", + "name": "python", + "nbconvert_exporter": "python", + "pygments_lexer": "ipython3", + "version": "3.11.5" + }, + "vscode": { + "interpreter": { + "hash": "f5c9192943a88c565de58e6fd7c534c9de794a35889e45ad6809ab92c821a96c" + } + } + }, + "nbformat": 4, + "nbformat_minor": 4 +} diff --git a/_sources/howtos/howtos.rst.txt b/_sources/howtos/howtos.rst.txt new file mode 100644 index 000000000..fe0051595 --- /dev/null +++ b/_sources/howtos/howtos.rst.txt @@ -0,0 +1,21 @@ +.. + SPDX-FileCopyrightText: Copyright DB Netz AG and the capellambse contributors + SPDX-License-Identifier: Apache-2.0 + +.. _howtos: + +******* +How Tos +******* + +In this section you can view dedicated tutorial-notebooks of key +features. + + +.. toctree:: + :maxdepth: 4 + :caption: How tos: + :numbered: + :glob: + + ../examples/* diff --git a/_sources/index.rst.txt b/_sources/index.rst.txt new file mode 100644 index 000000000..d7af388a1 --- /dev/null +++ b/_sources/index.rst.txt @@ -0,0 +1,76 @@ +.. + SPDX-FileCopyrightText: Copyright DB Netz AG and the capellambse contributors + SPDX-License-Identifier: Apache-2.0 + +***************************** +Welcome to the documentation! +***************************** + +Python Capella MBSE Tools +========================= + +.. image:: https://img.shields.io/badge/code%20style-black-000000.svg + :target: https://github.com/psf/black + :alt: Black + +**Date**: |today| **Version**: |Version| + +Description +----------- + +This library was designed to enable and support Model Based System Engineering +using Polarsys' Capella_ with Python. Common usage for this API: + +* parsing .aird files +* easy access to model elements and objects +* property-value access and manipulation +* diagram access and export as SVG + +Additionally and as a core idea it provides an interface for the underlying +database of the Capella model. + +Since v0.5, it also supports a simple, but powerful :ref:`declarative modelling +language `, which is based on the API for the semantic +model. + +If you want a quickstart at how to use this package, head right into the +:ref:`how-tos section `. + +.. toctree:: + :caption: Start + :maxdepth: 1 + :titlesonly: + + start/installation + start/intro-to-api + start/declarative + start/audit-events + +.. toctree:: + :caption: Tutorials + :titlesonly: + + howtos/howtos + +.. toctree:: + :caption: API reference + :maxdepth: 4 + + code/modules + +.. toctree:: + :caption: Integration with other tools + :maxdepth: 2 + + tools/sphinx-extension.rst + +.. toctree:: + :caption: Development + :maxdepth: 2 + + development/low-level-api + development/how-to-explore-capella-mm + development/developing-docs + development/repl + +.. _Capella: https://www.eclipse.org/capella/ diff --git a/_sources/start/audit-events.rst.txt b/_sources/start/audit-events.rst.txt new file mode 100644 index 000000000..b5baf6848 --- /dev/null +++ b/_sources/start/audit-events.rst.txt @@ -0,0 +1,152 @@ +.. + SPDX-FileCopyrightText: Copyright DB Netz AG and the capellambse contributors + SPDX-License-Identifier: Apache-2.0 + +.. _audit-events: + +************ +Audit events +************ + +|project| fires :external:py:func:`sys.audit` events for certain calls in the +high-level API. These events can be inspected by a callable registered with +:external:py:func:`sys.addaudithook`. + +.. warning:: + + System audit hooks cannot be removed once they were registered. When storing + a reference to a model in an audit hook, make sure that the reference is + destroyed properly to avoid unnecessary memory consumption. + +Also refer to the :py:class:`capellambse.auditing.AttributeAuditor` for an +example class that records read access to all attributes in a model using the +``capellambse.getattr`` audit event. + +For programmatic use, a set of all events that signify changes to the model is +available as the ``events`` class variable on the +:py:class:`~capellambse.auditing.WriteProtector`. + +List of audit events fired by |project| +======================================= + +The following table shows all audit events that are fired by ``capellambse``. + +.. note:: + + For some calls, multiple events may be fired. For example, a + ``capellambse.create`` event is always followed by another event (such as + ``capellambse.insert``) when adding the newly created object to the model + tree. + ++--------------------------------+--------------------------------------------+ +| Event | Description | ++================================+============================================+ +| ``capellambse.getattr`` | An attribute is accessed for reading. | +| | | +| .. versionadded:: 0.5.11 | **Arguments:** | +| | | +| | 1. ``obj``: The object that was accessed. | +| | 2. ``attr``: The attribute on that object. | +| | 3. ``value``: The value that is going to | +| | be returned. Use this to avoid | +| | recursive loops when inspecting the | +| | object. | +| | | +| | .. note:: | +| | | +| | This event will also be fired for | +| | internal read accesses, for example | +| | when searching the model for references | +| | to another object. Currently there is | +| | no reliable way to distinguish between | +| | explicit (user) access and these | +| | internal calls. | ++--------------------------------+--------------------------------------------+ +| ``capellambse.read_attribute`` | Deprecated alias of | +| | ``capellambse.getattr``. | +| .. versionadded:: pre-0.5 | | +| | | +| .. deprecated:: 0.5.11 | | +| Use the ``getattr`` event | | +| instead. | | ++--------------------------------+--------------------------------------------+ +| ``capellambse.setattr`` | An attribute is about to be changed. | +| | | +| .. versionadded:: 0.5.11 | **Arguments:** | +| | | +| | 1. ``obj``: The object being changed. | +| | 2. ``attr``: The name of the attribute. | +| | 3. ``value``: The new value. | ++--------------------------------+--------------------------------------------+ +| ``capellambse.delete`` | An object or a list of objects is about to | +| | be deleted from the model. | +| .. versionadded:: 0.5.11 | | +| | This is also fired when purging left-over | +| | references while deleting another object. | +| | | +| | **Arguments:** | +| | | +| | 1. ``parent``: The current parent object. | +| | 2. ``attr``: The attribute that contains | +| | the object to be deleted. | +| | 3. ``index``: If a single object from a | +| | list is being deleted, contains the | +| | index of that object into the list. If | +| | the entire attribute is deleted (in the | +| | case of lists: the list is emptied), | +| | contains ``None``. | ++--------------------------------+--------------------------------------------+ +| ``capellambse.insert`` | An item is about to be inserted into a | +| | coupled ``ElementList``. | +| .. versionadded:: 0.5.11 | | +| | **Arguments:** | +| | | +| | 1. ``parent``: The object being changed. | +| | 2. ``attr``: The attribute that contains | +| | this list. | +| | 3. ``index``: The index into the list to | +| | insert into. May be ``len(the_list)`` | +| | (or greater) to signify appending to | +| | the end. | +| | 4. ``value``: The value being inserted. | ++--------------------------------+--------------------------------------------+ +| ``capellambse.create`` | A new object was just created, but is not | +| | yet part of the model. | +| .. versionadded:: 0.5.11 | | +| | **Arguments:** | +| | | +| | 1. ``parent``: The new parent object. | +| | 2. ``attr``: The attribute that contains | +| | this list. | +| | 3. ``index``: The index into the list to | +| | insert into. May be ``len(the_list)`` | +| | (or greater) to signify appending to | +| | the end. | +| | 4. ``value``: The newly created object. | ++--------------------------------+--------------------------------------------+ + +Implementation notes +==================== + +Audit events are generally fired from these locations: + +1. Read access events (i.e. ``capellambse.getattr``) are fired by each Accessor + subclass, just before returning the final value from ``__get__()``. + +2. Events that signify modifications to a list are fired by the overridden + methods in ``CoupledElementListMixin`` (include ``create``), as well as by + ``__setattr__()`` of ``GenericElement``, before passing the values on to the + actual accessor implementation. + +3. The ``capellambse.delete`` event for deleting an entire attribute (i.e. the + case where the ``index`` argument is ``None``) is fired by the relevant + Accessor's ``__delete__()`` method. + + Note that for lists, Accessors may instead fire individual ``delete`` events + for each list item. + +In order to prevent audit events from being fired for elements that are still +under construction, ``GenericElement`` keeps track of the construction state in +the ``_constructed`` attribute. It becomes True when construction is finished +and audit events may be fired. Accessors must not fire any audit events if the +object they're acting on has not been fully constructed. diff --git a/_sources/start/declarative.rst.txt b/_sources/start/declarative.rst.txt new file mode 100644 index 000000000..d037ed79c --- /dev/null +++ b/_sources/start/declarative.rst.txt @@ -0,0 +1,281 @@ +.. + SPDX-FileCopyrightText: Copyright DB Netz AG and the capellambse contributors + SPDX-License-Identifier: Apache-2.0 + +.. _declarative-modelling: + +********************* +Declarative modelling +********************* + +.. versionadded:: 0.5.0 + Introduced the declarative modelling module. + +|project| supports declarative modelling with the :py:mod:`capellambse.decl` +module. This requires the optional dependency ``capellambse[decl]`` to be +installed. + +The YAML-based declarative modelling engine combines a few simple concepts into +a powerful, but easy to use file format. These files can then be applied to any +model supported by |project|. + +Example +======= + +Here is an example YAML file that declares a simple coffee machine with a +couple of functions, and functional exchanges between them: + +.. literalinclude:: ../../../tests/data/decl/coffee-machine.yml + :language: yaml + :lines: 4- + :lineno-start: 1 + :linenos: + +CLI usage +--------- + +If the additional optional dependency ``capellambse[decl,cli]`` is installed, +this file can be applied from the command line. Assuming it is saved as +``coffee-machine.yml``, it can then be applied to a locally stored Capella +model like this: + +.. code-block:: sh + + python -m capellambse.decl --model path/to/model.aird coffee-machine.yml + +Refer to the :py:func:`capellambse.cli_helpers.loadcli` documentation to find +out the supported argument format for ``--model``. + +API usage +--------- + +Declarative YAML can also be applied programmatically, by calling the +:py:func:`capellambse.decl.apply` function. It takes a (loaded) |project| +model, and either a path to a file or a file-like object. To pass in a string +containing YAML, wrap it in :external:class:`io.StringIO`: + +.. code-block:: python + :emphasize-lines: 5 + + import io, capellambse.decl + my_model = capellambse.MelodyModel(...) + my_yaml = "..." + + capellambse.decl.apply(my_model, io.StringIO(my_yaml)) + + my_model.save() + +Format description +================== + +The expected YAML follows a simple format, where a parent object (i.e. an +object that already exists in the model) is selected, and one or more of three +different operations is applied to it: + +- ``extend``-ing the object on list attributes, +- ``modify``-ing the object itself, or +- ``delete``-ing one or more children. + +Parents can be selected by their universally unique ID (UUID), using the +``!uuid`` YAML tag. The following query selects the root logical function in +our test model: + +.. code-block:: yaml + + - parent: !uuid f28ec0f8-f3b3-43a0-8af7-79f194b29a2d + +Extending objects +----------------- + +The following subsections show how to create completely new objects, or +reference and move already existing ones, using examples of declarative YAML +files on +:py:class:`~capellambse.model.common.accessors.ElementListCouplingMixin`-ish +attributes. The extension of one-to-one attributes works in the same way, +adhering to the YAML syntax. + +Creating new objects +^^^^^^^^^^^^^^^^^^^^ + +:py:class:`~capellambse.model.layers.la.LogicalFunction` objects have several +different attributes which can be modified from a declarative YAML file. For +example, it is possible to create new +sub-:py:attr:`~capellambse.model.layers.la.LogicalFunction.functions`. This +snippet creates a function with the name "brew coffee" directly below the root +function: + +.. code-block:: yaml + :emphasize-lines: 2 + + - parent: !uuid f28ec0f8-f3b3-43a0-8af7-79f194b29a2d + extend: + functions: + - name: brew coffee + +Functions can be nested arbitrarily deeply, and can also receive any other +supported attributes at the same time. The "brew coffee" function for example +could further receive nested child functions, each providing an output port: + +.. code-block:: yaml + :emphasize-lines: 4-5 + + - parent: !uuid f28ec0f8-f3b3-43a0-8af7-79f194b29a2d + extend: + functions: + - name: brew coffee + functions: + - name: grind beans + outputs: + - name: Ground Beans port + - name: heat water + outputs: + - name: Hot Water port + +While objects that already exist in the base model can be referenced with +``!uuid``, this is not possible for objects declared by the YAML file, as they +will have a random UUID assigned to ensure uniqueness. For this reason, a +promise mechanic exists, which allows to "tag" any declared object with a +``promise_id``, and later reference that object with the ``!promise`` YAML tag. +These promise IDs are user defined strings. The only requirement is that two +objects cannot receive the same ID, however they can be referenced any number +of times. This example snippet demonstrates how to declare two logical +functions, which communicate through a functional exchange: + +.. code-block:: yaml + :emphasize-lines: 7,11,14-15 + + - parent: !uuid f28ec0f8-f3b3-43a0-8af7-79f194b29a2d + extend: + functions: + - name: brew coffee + inputs: + - name: Steam port + promise_id: steam-input + - name: produce steam + outputs: + - name: Steam port + promise_id: steam-output + exchanges: + - name: Steam + source: !promise steam-output + target: !promise steam-input + +The ``!promise`` tag (and the ``!uuid`` tag as well) can be used anywhere where +a model object is expected. + +Creating new references +^^^^^^^^^^^^^^^^^^^^^^^ + +It is important to understand when new model objects are created and when only +references are added. The following example would create a reference in the +``.allocated_functions`` attribute of the +:py:class:`~capellambse.model.layers.la.LogicalComponent` which is also the +logical ``root_component`` (parent) to the logical ``root_function``: + +.. code-block:: yaml + :emphasize-lines: 2 + + - parent: !uuid 0d2edb8f-fa34-4e73-89ec-fb9a63001440 + extend: + allocated_functions: + - !uuid f28ec0f8-f3b3-43a0-8af7-79f194b29a2d + +This is caused by the type of relationship (non- +:py:class:`~capellambse.model.common.accessors.DirectProxyAccessor`) between +the parent and its ``allocated_functions``. + +It is also possible to create references to promised objects, but extra caution +for declaring ``promise_id``\ s for resolving these promises successfully: + +.. code-block:: yaml + :emphasize-lines: 5,9 + + - parent: !uuid f28ec0f8-f3b3-43a0-8af7-79f194b29a2d + extend: + functions: + - name: The promised one + promise_id: promised-fnc + - parent: !uuid 0d2edb8f-fa34-4e73-89ec-fb9a63001440 + extend: + functions: + - !promise promised-fnc + +The ``promise_id`` declaration can also happen after referencing it. + +Moving objects +^^^^^^^^^^^^^^ + +The following example would move a logical function from underneath a +:py:class:`~capellambse.model.layers.la.LogicalFunctionPkg` (accessible via +``functions``) into ``functions`` of the logical ``root_function`` (parent) +since the ``functions`` attribute has a parent/children relationship (i.e. the +:py:class:`~capellambse.model.common.accessors.DirectProxyAccessor` is used). + +.. code-block:: yaml + + - parent: !uuid f28ec0f8-f3b3-43a0-8af7-79f194b29a2d + extend: + functions: + - !uuid 8833d2dc-b862-4a50-b26c-6f7e0f17faef + +Modifying objects +----------------- + +After selecting a parent, it is also possible to directly change its properties +without introducing new objects into the model. This happens by specifying the +attributes in the ``modify:`` key. + +The following example would change the ``name`` of the root +:py:class:`~capellambse.model.layers.la.LogicalComponent` to "Coffee Machine" +(notice how we use a different UUID than before): + +.. code-block:: yaml + :emphasize-lines: 2 + + - parent: !uuid 0d2edb8f-fa34-4e73-89ec-fb9a63001440 + modify: + name: Coffee Machine + +This is not limited to string attributes; it is just as well possible to change +e.g. numeric properties. This example changes the ``min_card`` property of an +:py:class:`~capellambse.model.crosslayer.information.ExchangeItemElement` to +``0`` and the ``max_card`` to infinity, effectively removing both limitations: + +.. code-block:: yaml + :emphasize-lines: 3- + + - parent: !uuid 81b87fcc-03cf-434b-ad5b-ef18266c5a3e + modify: + min_card: 0 + max_card: .inf + +Deleting objects +---------------- + +Finally, with declarative modelling files, it is possible to delete objects +from the model. Depending on where the delete operation occurs, either the +target object is deleted entirely, or only the link to it is destroyed. + +Currently, objects to be deleted can only be selected by their UUID. + +For example, this snippet deletes the logical function named "produce Great +Wizards" from the model: + +.. code-block:: yaml + :emphasize-lines: 3 + + - parent: !uuid f28ec0f8-f3b3-43a0-8af7-79f194b29a2d + delete: + functions: + - !uuid 0e71a0d3-0a18-4671-bba0-71b5f88f95dd + +In contrast, this snippet only removes its allocation to the "Hogwarts" root +component, but the function still exists afterwards: + +.. code-block:: yaml + :emphasize-lines: 3 + + - parent: !uuid 0d2edb8f-fa34-4e73-89ec-fb9a63001440 + delete: + allocated_functions: + - !uuid 0e71a0d3-0a18-4671-bba0-71b5f88f95dd diff --git a/_sources/start/installation.rst.txt b/_sources/start/installation.rst.txt new file mode 100644 index 000000000..3feeed8fc --- /dev/null +++ b/_sources/start/installation.rst.txt @@ -0,0 +1,65 @@ +.. + SPDX-FileCopyrightText: Copyright DB Netz AG and the capellambse contributors + SPDX-License-Identifier: Apache-2.0 + +************ +Installation +************ + +.. image:: https://img.shields.io/pypi/pyversions/capellambse + :target: https://pypi.org/project/capellambse/ + :alt: PyPI - Python Version + +This guide helps you to get |project| installed. There are a few ways to get it +done: + +Install from PyPI +================= + +Installing |project| from Python Package Index via pip__ is the quickest way to +get started. + +__ http://www.pip-installer.org/ + +.. code:: bash + + pip install capellambse + +Windows +======= + +If you intend to use |project|'s PNG export functionality, you need a working +installation of cairosvg__. Unfortunately, the Windows wheels on PyPI do not +ship with all necessary libraries. However, they can be manually installed with +the `GTK for Windows Runtime Environment Installer`__. + +__ https://pypi.org/project/CairoSVG/ +__ https://github.com/tschoonj/GTK-for-Windows-Runtime-Environment-Installer/releases + +This should give you a fully functioning |project|. + +Install as a package from Github +================================ + +If you want to have a comfortable playground with examples / Jupyter notebooks +/ export to excel demo and test models you may clone the repository directly +from Github, create a virtual environment and install all the extras: + +.. code-block:: bash + + git clone https://github.com/DSD-DBS/py-capellambse.git + cd py-capellambse + python3 -m venv .venv + source .venv/bin/activate + pip install . + pip install jupyter + cd examples + jupyter-notebook + +Install for development +======================= + +In case you'd like to contribute to the development or improve documentation, +sample models or examples collection please follow the `contribution guide`__. + +__ https://github.com/DSD-DBS/py-capellambse/blob/master/CONTRIBUTING.md diff --git a/_sources/start/intro-to-api.rst.txt b/_sources/start/intro-to-api.rst.txt new file mode 100644 index 000000000..4e95cd818 --- /dev/null +++ b/_sources/start/intro-to-api.rst.txt @@ -0,0 +1,63 @@ +.. + SPDX-FileCopyrightText: Copyright DB Netz AG and the capellambse contributors + SPDX-License-Identifier: Apache-2.0 + +********************************** +Introduction to py-capellambse API +********************************** + +|project| provides access to model elements using a meta-model similar to the +one of Capella. However in this meta-model we make a few simplifications. A +collection of automated tests and design reviews help us to ensure that those +simplifications don't break compatibility with original Capella models (however +coverage isn't complete yet). + +As you may know the meta-model behind Capella is layered. There are many +packages involved and there is a long inheritance chain behind almost every +model element. We are simplifying that by "flattening" the lower layers. + +You may see an example of how that works in the figure below: + +.. image:: ../_static/img/crosslayer_intro.jpg + +In the example above we see that `LogicalFunction` is a subtype of +`AbstractFunction`, just like `SystemFunction` or `OperationalActivity`. +Because of that, all of those subtypes can be `.available_in_states` or have a +layer-specific structural `owner`, like `LogicalComponent` for +`LogicalFunction`. Any layer-specific class that inherits from `Component` may +also have `state_machines`. + +The API reference part of this documentation provides you with the complete (as +it is generated from the code base) list of available methods and attributes. + +Layer-specific packages +======================= + +The following packages enable working with model layers: + +* :mod:`capellambse.model.layers.oa` - covers Operational Analysis layer. +* :mod:`capellambse.model.layers.ctx` - covers System Analysis layer. +* :mod:`capellambse.model.layers.la` - covers Logical Architecture layer. +* :mod:`capellambse.model.layers.pa` - covers Physical Architecture layer. + +Cross-layer packages +==================== + +The following packages enable all (almost) of the layer packages: + +* :mod:`capellambse.model.crosslayer.fa` - covers Functional Analysis concerns, + defines things like AbstractFunction or FunctionalExchange +* :mod:`capellambse.model.crosslayer.cs` - covers Composite Structure concerns, + defines things like Component +* :mod:`capellambse.model.crosslayer.capellacommon` - covers common concerns, + defines things like StateMachine, State +* :mod:`capellambse.model.crosslayer.information` - covers Information + concerns, defines things like Class, DataPkg, ExchangeItem + +Extension packages +================== + +* :mod:`capellambse.extensions.reqif` - provides means for working with ReqIF + Requirements within Capella model. +* :mod:`capellambse.extensions.pvmt` - provides means for working with object + attributes created with PVMT package. diff --git a/_sources/tools/sphinx-extension.rst.txt b/_sources/tools/sphinx-extension.rst.txt new file mode 100644 index 000000000..a9224801b --- /dev/null +++ b/_sources/tools/sphinx-extension.rst.txt @@ -0,0 +1,8 @@ +.. + SPDX-FileCopyrightText: Copyright DB Netz AG and the capellambse contributors + SPDX-License-Identifier: Apache-2.0 + +The |project| Sphinx extension +============================== + +.. automodule:: capellambse.sphinx diff --git a/_static/basic.css b/_static/basic.css new file mode 100644 index 000000000..30fee9d0f --- /dev/null +++ b/_static/basic.css @@ -0,0 +1,925 @@ +/* + * 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: 230px; + 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; +} + +a:visited { + color: #551A8B; +} + +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; +} + +.sig dd { + margin-top: 0px; + margin-bottom: 0px; +} + +.sig dl { + margin-top: 0px; + margin-bottom: 0px; +} + +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; +} + +.translated { + background-color: rgba(207, 255, 207, 0.2) +} + +.untranslated { + background-color: rgba(255, 207, 207, 0.2) +} + +/* -- 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/debug.css b/_static/debug.css new file mode 100644 index 000000000..74d4aec33 --- /dev/null +++ b/_static/debug.css @@ -0,0 +1,69 @@ +/* + This CSS file should be overridden by the theme authors. It's + meant for debugging and developing the skeleton that this theme provides. +*/ +body { + font-family: -apple-system, "Segoe UI", Roboto, Helvetica, Arial, sans-serif, + "Apple Color Emoji", "Segoe UI Emoji"; + background: lavender; +} +.sb-announcement { + background: rgb(131, 131, 131); +} +.sb-announcement__inner { + background: black; + color: white; +} +.sb-header { + background: lightskyblue; +} +.sb-header__inner { + background: royalblue; + color: white; +} +.sb-header-secondary { + background: lightcyan; +} +.sb-header-secondary__inner { + background: cornflowerblue; + color: white; +} +.sb-sidebar-primary { + background: lightgreen; +} +.sb-main { + background: blanchedalmond; +} +.sb-main__inner { + background: antiquewhite; +} +.sb-header-article { + background: lightsteelblue; +} +.sb-article-container { + background: snow; +} +.sb-article-main { + background: white; +} +.sb-footer-article { + background: lightpink; +} +.sb-sidebar-secondary { + background: lightgoldenrodyellow; +} +.sb-footer-content { + background: plum; +} +.sb-footer-content__inner { + background: palevioletred; +} +.sb-footer { + background: pink; +} +.sb-footer__inner { + background: salmon; +} +.sb-article { + background: white; +} diff --git a/_static/doctools.js b/_static/doctools.js new file mode 100644 index 000000000..d06a71d75 --- /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 000000000..7e4c114f2 --- /dev/null +++ b/_static/documentation_options.js @@ -0,0 +1,13 @@ +const DOCUMENTATION_OPTIONS = { + VERSION: '', + LANGUAGE: 'en', + COLLAPSE_INDEX: false, + BUILDER: 'html', + FILE_SUFFIX: '.html', + LINK_SUFFIX: '.html', + HAS_SOURCE: true, + SOURCELINK_SUFFIX: '.txt', + 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 000000000..a858a410e Binary files /dev/null and b/_static/file.png differ diff --git a/_static/img/2021-05-09_10-32.jpg b/_static/img/2021-05-09_10-32.jpg new file mode 100644 index 000000000..d16a02951 Binary files /dev/null and b/_static/img/2021-05-09_10-32.jpg differ diff --git a/_static/img/2021-05-09_10-32.jpg.license b/_static/img/2021-05-09_10-32.jpg.license new file mode 100644 index 000000000..6605644e7 --- /dev/null +++ b/_static/img/2021-05-09_10-32.jpg.license @@ -0,0 +1,2 @@ +SPDX-FileCopyrightText: Copyright DB Netz AG and the capellambse contributors +SPDX-License-Identifier: Apache-2.0 diff --git a/_static/img/2021-05-09_10-36.jpg b/_static/img/2021-05-09_10-36.jpg new file mode 100644 index 000000000..db2876cb3 Binary files /dev/null and b/_static/img/2021-05-09_10-36.jpg differ diff --git a/_static/img/2021-05-09_10-36.jpg.license b/_static/img/2021-05-09_10-36.jpg.license new file mode 100644 index 000000000..6605644e7 --- /dev/null +++ b/_static/img/2021-05-09_10-36.jpg.license @@ -0,0 +1,2 @@ +SPDX-FileCopyrightText: Copyright DB Netz AG and the capellambse contributors +SPDX-License-Identifier: Apache-2.0 diff --git a/_static/img/2021-05-09_10-50.jpg b/_static/img/2021-05-09_10-50.jpg new file mode 100644 index 000000000..cd12ca5d9 Binary files /dev/null and b/_static/img/2021-05-09_10-50.jpg differ diff --git a/_static/img/2021-05-09_10-50.jpg.license b/_static/img/2021-05-09_10-50.jpg.license new file mode 100644 index 000000000..6605644e7 --- /dev/null +++ b/_static/img/2021-05-09_10-50.jpg.license @@ -0,0 +1,2 @@ +SPDX-FileCopyrightText: Copyright DB Netz AG and the capellambse contributors +SPDX-License-Identifier: Apache-2.0 diff --git a/_static/img/2021-05-09_17-36.jpg b/_static/img/2021-05-09_17-36.jpg new file mode 100644 index 000000000..b5581088b Binary files /dev/null and b/_static/img/2021-05-09_17-36.jpg differ diff --git a/_static/img/2021-05-09_17-36.jpg.license b/_static/img/2021-05-09_17-36.jpg.license new file mode 100644 index 000000000..6605644e7 --- /dev/null +++ b/_static/img/2021-05-09_17-36.jpg.license @@ -0,0 +1,2 @@ +SPDX-FileCopyrightText: Copyright DB Netz AG and the capellambse contributors +SPDX-License-Identifier: Apache-2.0 diff --git a/_static/img/2021-05-09_17-38.jpg b/_static/img/2021-05-09_17-38.jpg new file mode 100644 index 000000000..60c056e85 Binary files /dev/null and b/_static/img/2021-05-09_17-38.jpg differ diff --git a/_static/img/2021-05-09_17-38.jpg.license b/_static/img/2021-05-09_17-38.jpg.license new file mode 100644 index 000000000..6605644e7 --- /dev/null +++ b/_static/img/2021-05-09_17-38.jpg.license @@ -0,0 +1,2 @@ +SPDX-FileCopyrightText: Copyright DB Netz AG and the capellambse contributors +SPDX-License-Identifier: Apache-2.0 diff --git a/_static/img/2021-05-09_17-44.jpg b/_static/img/2021-05-09_17-44.jpg new file mode 100644 index 000000000..83b414d0f Binary files /dev/null and b/_static/img/2021-05-09_17-44.jpg differ diff --git a/_static/img/2021-05-09_17-44.jpg.license b/_static/img/2021-05-09_17-44.jpg.license new file mode 100644 index 000000000..6605644e7 --- /dev/null +++ b/_static/img/2021-05-09_17-44.jpg.license @@ -0,0 +1,2 @@ +SPDX-FileCopyrightText: Copyright DB Netz AG and the capellambse contributors +SPDX-License-Identifier: Apache-2.0 diff --git a/_static/img/2021-05-09_17-57.jpg b/_static/img/2021-05-09_17-57.jpg new file mode 100644 index 000000000..b9dd0947e Binary files /dev/null and b/_static/img/2021-05-09_17-57.jpg differ diff --git a/_static/img/2021-05-09_17-57.jpg.license b/_static/img/2021-05-09_17-57.jpg.license new file mode 100644 index 000000000..6605644e7 --- /dev/null +++ b/_static/img/2021-05-09_17-57.jpg.license @@ -0,0 +1,2 @@ +SPDX-FileCopyrightText: Copyright DB Netz AG and the capellambse contributors +SPDX-License-Identifier: Apache-2.0 diff --git a/_static/img/2021-05-09_19-50.jpg b/_static/img/2021-05-09_19-50.jpg new file mode 100644 index 000000000..7decd28c0 Binary files /dev/null and b/_static/img/2021-05-09_19-50.jpg differ diff --git a/_static/img/2021-05-09_19-50.jpg.license b/_static/img/2021-05-09_19-50.jpg.license new file mode 100644 index 000000000..6605644e7 --- /dev/null +++ b/_static/img/2021-05-09_19-50.jpg.license @@ -0,0 +1,2 @@ +SPDX-FileCopyrightText: Copyright DB Netz AG and the capellambse contributors +SPDX-License-Identifier: Apache-2.0 diff --git a/_static/img/2021-05-09_21-16.jpg b/_static/img/2021-05-09_21-16.jpg new file mode 100644 index 000000000..165f0ba53 Binary files /dev/null and b/_static/img/2021-05-09_21-16.jpg differ diff --git a/_static/img/2021-05-09_21-16.jpg.license b/_static/img/2021-05-09_21-16.jpg.license new file mode 100644 index 000000000..6605644e7 --- /dev/null +++ b/_static/img/2021-05-09_21-16.jpg.license @@ -0,0 +1,2 @@ +SPDX-FileCopyrightText: Copyright DB Netz AG and the capellambse contributors +SPDX-License-Identifier: Apache-2.0 diff --git a/_static/img/2021-05-09_21-21.jpg b/_static/img/2021-05-09_21-21.jpg new file mode 100644 index 000000000..1b27de5f0 Binary files /dev/null and b/_static/img/2021-05-09_21-21.jpg differ diff --git a/_static/img/2021-05-09_21-21.jpg.license b/_static/img/2021-05-09_21-21.jpg.license new file mode 100644 index 000000000..6605644e7 --- /dev/null +++ b/_static/img/2021-05-09_21-21.jpg.license @@ -0,0 +1,2 @@ +SPDX-FileCopyrightText: Copyright DB Netz AG and the capellambse contributors +SPDX-License-Identifier: Apache-2.0 diff --git a/_static/img/2021-05-09_21-46.jpg b/_static/img/2021-05-09_21-46.jpg new file mode 100644 index 000000000..903bd8e92 Binary files /dev/null and b/_static/img/2021-05-09_21-46.jpg differ diff --git a/_static/img/2021-05-09_21-46.jpg.license b/_static/img/2021-05-09_21-46.jpg.license new file mode 100644 index 000000000..6605644e7 --- /dev/null +++ b/_static/img/2021-05-09_21-46.jpg.license @@ -0,0 +1,2 @@ +SPDX-FileCopyrightText: Copyright DB Netz AG and the capellambse contributors +SPDX-License-Identifier: Apache-2.0 diff --git a/_static/img/2021-05-09_22-09.jpg b/_static/img/2021-05-09_22-09.jpg new file mode 100644 index 000000000..5b3a1fcd2 Binary files /dev/null and b/_static/img/2021-05-09_22-09.jpg differ diff --git a/_static/img/2021-05-09_22-09.jpg.license b/_static/img/2021-05-09_22-09.jpg.license new file mode 100644 index 000000000..6605644e7 --- /dev/null +++ b/_static/img/2021-05-09_22-09.jpg.license @@ -0,0 +1,2 @@ +SPDX-FileCopyrightText: Copyright DB Netz AG and the capellambse contributors +SPDX-License-Identifier: Apache-2.0 diff --git a/_static/img/2021-05-09_22-39.jpg b/_static/img/2021-05-09_22-39.jpg new file mode 100644 index 000000000..d02b6668f Binary files /dev/null and b/_static/img/2021-05-09_22-39.jpg differ diff --git a/_static/img/2021-05-09_22-39.jpg.license b/_static/img/2021-05-09_22-39.jpg.license new file mode 100644 index 000000000..6605644e7 --- /dev/null +++ b/_static/img/2021-05-09_22-39.jpg.license @@ -0,0 +1,2 @@ +SPDX-FileCopyrightText: Copyright DB Netz AG and the capellambse contributors +SPDX-License-Identifier: Apache-2.0 diff --git a/_static/img/2021-05-10_22-35.jpg b/_static/img/2021-05-10_22-35.jpg new file mode 100644 index 000000000..9c3650585 Binary files /dev/null and b/_static/img/2021-05-10_22-35.jpg differ diff --git a/_static/img/2021-05-10_22-35.jpg.license b/_static/img/2021-05-10_22-35.jpg.license new file mode 100644 index 000000000..6605644e7 --- /dev/null +++ b/_static/img/2021-05-10_22-35.jpg.license @@ -0,0 +1,2 @@ +SPDX-FileCopyrightText: Copyright DB Netz AG and the capellambse contributors +SPDX-License-Identifier: Apache-2.0 diff --git a/_static/img/2021-05-10_22-59.jpg b/_static/img/2021-05-10_22-59.jpg new file mode 100644 index 000000000..ca184bc6c Binary files /dev/null and b/_static/img/2021-05-10_22-59.jpg differ diff --git a/_static/img/2021-05-10_22-59.jpg.license b/_static/img/2021-05-10_22-59.jpg.license new file mode 100644 index 000000000..6605644e7 --- /dev/null +++ b/_static/img/2021-05-10_22-59.jpg.license @@ -0,0 +1,2 @@ +SPDX-FileCopyrightText: Copyright DB Netz AG and the capellambse contributors +SPDX-License-Identifier: Apache-2.0 diff --git a/_static/img/2021-05-12_18-18.jpg b/_static/img/2021-05-12_18-18.jpg new file mode 100644 index 000000000..5ee9dac5a Binary files /dev/null and b/_static/img/2021-05-12_18-18.jpg differ diff --git a/_static/img/2021-05-12_18-18.jpg.license b/_static/img/2021-05-12_18-18.jpg.license new file mode 100644 index 000000000..6605644e7 --- /dev/null +++ b/_static/img/2021-05-12_18-18.jpg.license @@ -0,0 +1,2 @@ +SPDX-FileCopyrightText: Copyright DB Netz AG and the capellambse contributors +SPDX-License-Identifier: Apache-2.0 diff --git a/_static/img/2021-05-12_18-51.jpg b/_static/img/2021-05-12_18-51.jpg new file mode 100644 index 000000000..afb248858 Binary files /dev/null and b/_static/img/2021-05-12_18-51.jpg differ diff --git a/_static/img/2021-05-12_18-51.jpg.license b/_static/img/2021-05-12_18-51.jpg.license new file mode 100644 index 000000000..6605644e7 --- /dev/null +++ b/_static/img/2021-05-12_18-51.jpg.license @@ -0,0 +1,2 @@ +SPDX-FileCopyrightText: Copyright DB Netz AG and the capellambse contributors +SPDX-License-Identifier: Apache-2.0 diff --git a/_static/img/2021-05-12_19-12.jpg b/_static/img/2021-05-12_19-12.jpg new file mode 100644 index 000000000..cda80cf9d Binary files /dev/null and b/_static/img/2021-05-12_19-12.jpg differ diff --git a/_static/img/2021-05-12_19-12.jpg.license b/_static/img/2021-05-12_19-12.jpg.license new file mode 100644 index 000000000..6605644e7 --- /dev/null +++ b/_static/img/2021-05-12_19-12.jpg.license @@ -0,0 +1,2 @@ +SPDX-FileCopyrightText: Copyright DB Netz AG and the capellambse contributors +SPDX-License-Identifier: Apache-2.0 diff --git a/_static/img/2021-05-12_20-06.jpg b/_static/img/2021-05-12_20-06.jpg new file mode 100644 index 000000000..b34855edf Binary files /dev/null and b/_static/img/2021-05-12_20-06.jpg differ diff --git a/_static/img/2021-05-12_20-06.jpg.license b/_static/img/2021-05-12_20-06.jpg.license new file mode 100644 index 000000000..6605644e7 --- /dev/null +++ b/_static/img/2021-05-12_20-06.jpg.license @@ -0,0 +1,2 @@ +SPDX-FileCopyrightText: Copyright DB Netz AG and the capellambse contributors +SPDX-License-Identifier: Apache-2.0 diff --git a/_static/img/2021-05-12_20-10.jpg b/_static/img/2021-05-12_20-10.jpg new file mode 100644 index 000000000..04d6402f1 Binary files /dev/null and b/_static/img/2021-05-12_20-10.jpg differ diff --git a/_static/img/2021-05-12_20-10.jpg.license b/_static/img/2021-05-12_20-10.jpg.license new file mode 100644 index 000000000..6605644e7 --- /dev/null +++ b/_static/img/2021-05-12_20-10.jpg.license @@ -0,0 +1,2 @@ +SPDX-FileCopyrightText: Copyright DB Netz AG and the capellambse contributors +SPDX-License-Identifier: Apache-2.0 diff --git a/_static/img/2021-05-12_21-03.jpg b/_static/img/2021-05-12_21-03.jpg new file mode 100644 index 000000000..384e2ba0d Binary files /dev/null and b/_static/img/2021-05-12_21-03.jpg differ diff --git a/_static/img/2021-05-12_21-03.jpg.license b/_static/img/2021-05-12_21-03.jpg.license new file mode 100644 index 000000000..6605644e7 --- /dev/null +++ b/_static/img/2021-05-12_21-03.jpg.license @@ -0,0 +1,2 @@ +SPDX-FileCopyrightText: Copyright DB Netz AG and the capellambse contributors +SPDX-License-Identifier: Apache-2.0 diff --git a/_static/img/2021-05-13_20-46.jpg b/_static/img/2021-05-13_20-46.jpg new file mode 100644 index 000000000..d3448b877 Binary files /dev/null and b/_static/img/2021-05-13_20-46.jpg differ diff --git a/_static/img/2021-05-13_20-46.jpg.license b/_static/img/2021-05-13_20-46.jpg.license new file mode 100644 index 000000000..6605644e7 --- /dev/null +++ b/_static/img/2021-05-13_20-46.jpg.license @@ -0,0 +1,2 @@ +SPDX-FileCopyrightText: Copyright DB Netz AG and the capellambse contributors +SPDX-License-Identifier: Apache-2.0 diff --git a/_static/img/2021-05-17_11-50.jpg b/_static/img/2021-05-17_11-50.jpg new file mode 100644 index 000000000..3fb35b9c0 Binary files /dev/null and b/_static/img/2021-05-17_11-50.jpg differ diff --git a/_static/img/2021-05-17_11-50.jpg.license b/_static/img/2021-05-17_11-50.jpg.license new file mode 100644 index 000000000..6605644e7 --- /dev/null +++ b/_static/img/2021-05-17_11-50.jpg.license @@ -0,0 +1,2 @@ +SPDX-FileCopyrightText: Copyright DB Netz AG and the capellambse contributors +SPDX-License-Identifier: Apache-2.0 diff --git a/_static/img/core-pkg-deps-raw.svg b/_static/img/core-pkg-deps-raw.svg new file mode 100644 index 000000000..04321b962 --- /dev/null +++ b/_static/img/core-pkg-deps-raw.svg @@ -0,0 +1,1146 @@ + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + http://www.polarsys.org/capella/common/activity/5.0.0 + + + + + + + + + + + http://www.polarsys.org/capella/common/behavior/5.0.0 + + + + + + + + + + + http://www.polarsys.org/capella/core/common/5.0.0 + + + + + + + + + + + http://www.polarsys.org/capella/core/core/5.0.0 + + + + + + + + + + + http://www.polarsys.org/capella/core/cs/5.0.0 + + + + + + + + + + + http://www.polarsys.org/capella/core/ctx/5.0.0 + + + + + + + + + + + http://www.polarsys.org/kitalpha/emde/1.0.0 + + + + + + + + + + + http://www.polarsys.org/capella/core/epbs/5.0.0 + + + + + + + + + + + http://www.polarsys.org/capella/core/fa/5.0.0 + + + + + + + + + + + http://www.polarsys.org/capella/core/information/5.0.0 + + + + + + + + + + + http://www.polarsys.org/capella/core/interaction/5.0.0 + + + + + + + + + + + http://www.polarsys.org/capella/core/la/5.0.0 + + + + + + + + + + + http://www.polarsys.org/capella/common/core/5.0.0 + + + + + + + + + + + http://www.polarsys.org/capella/core/oa/5.0.0 + + + + + + + + + + + http://www.polarsys.org/capella/core/pa/5.0.0 + + + + + + + + + + + http://www.polarsys.org/capella/core/requirement/5.0.0 + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + diff --git a/_static/img/core-pkg-deps-raw.svg.license b/_static/img/core-pkg-deps-raw.svg.license new file mode 100644 index 000000000..6605644e7 --- /dev/null +++ b/_static/img/core-pkg-deps-raw.svg.license @@ -0,0 +1,2 @@ +SPDX-FileCopyrightText: Copyright DB Netz AG and the capellambse contributors +SPDX-License-Identifier: Apache-2.0 diff --git a/_static/img/crosslayer_intro.jpg b/_static/img/crosslayer_intro.jpg new file mode 100644 index 000000000..41349a91a Binary files /dev/null and b/_static/img/crosslayer_intro.jpg differ diff --git a/_static/img/crosslayer_intro.jpg.license b/_static/img/crosslayer_intro.jpg.license new file mode 100644 index 000000000..6605644e7 --- /dev/null +++ b/_static/img/crosslayer_intro.jpg.license @@ -0,0 +1,2 @@ +SPDX-FileCopyrightText: Copyright DB Netz AG and the capellambse contributors +SPDX-License-Identifier: Apache-2.0 diff --git a/_static/img/dep-graph-oa.jpg b/_static/img/dep-graph-oa.jpg new file mode 100644 index 000000000..1e771de2f Binary files /dev/null and b/_static/img/dep-graph-oa.jpg differ diff --git a/_static/img/dep-graph-oa.jpg.license b/_static/img/dep-graph-oa.jpg.license new file mode 100644 index 000000000..6605644e7 --- /dev/null +++ b/_static/img/dep-graph-oa.jpg.license @@ -0,0 +1,2 @@ +SPDX-FileCopyrightText: Copyright DB Netz AG and the capellambse contributors +SPDX-License-Identifier: Apache-2.0 diff --git a/_static/img/github-logo.svg b/_static/img/github-logo.svg new file mode 100644 index 000000000..a407b96cd --- /dev/null +++ b/_static/img/github-logo.svg @@ -0,0 +1,9 @@ + + + + + diff --git a/_static/img/harrys_wand.png b/_static/img/harrys_wand.png new file mode 100644 index 000000000..effece64b Binary files /dev/null and b/_static/img/harrys_wand.png differ diff --git a/_static/img/harrys_wand.png.license b/_static/img/harrys_wand.png.license new file mode 100644 index 000000000..6605644e7 --- /dev/null +++ b/_static/img/harrys_wand.png.license @@ -0,0 +1,2 @@ +SPDX-FileCopyrightText: Copyright DB Netz AG and the capellambse contributors +SPDX-License-Identifier: Apache-2.0 diff --git a/_static/img/waypoints.png b/_static/img/waypoints.png new file mode 100644 index 000000000..fd021c986 Binary files /dev/null and b/_static/img/waypoints.png differ diff --git a/_static/img/waypoints.png.license b/_static/img/waypoints.png.license new file mode 100644 index 000000000..6605644e7 --- /dev/null +++ b/_static/img/waypoints.png.license @@ -0,0 +1,2 @@ +SPDX-FileCopyrightText: Copyright DB Netz AG and the capellambse contributors +SPDX-License-Identifier: Apache-2.0 diff --git a/_static/language_data.js b/_static/language_data.js new file mode 100644 index 000000000..250f5665f --- /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/minus.png b/_static/minus.png new file mode 100644 index 000000000..d96755fda Binary files /dev/null and b/_static/minus.png differ diff --git a/_static/nbsphinx-broken-thumbnail.svg b/_static/nbsphinx-broken-thumbnail.svg new file mode 100644 index 000000000..4919ca882 --- /dev/null +++ b/_static/nbsphinx-broken-thumbnail.svg @@ -0,0 +1,9 @@ + + + + diff --git a/_static/nbsphinx-code-cells.css b/_static/nbsphinx-code-cells.css new file mode 100644 index 000000000..a3fb27c30 --- /dev/null +++ b/_static/nbsphinx-code-cells.css @@ -0,0 +1,259 @@ +/* remove conflicting styling from Sphinx themes */ +div.nbinput.container div.prompt *, +div.nboutput.container div.prompt *, +div.nbinput.container div.input_area pre, +div.nboutput.container div.output_area pre, +div.nbinput.container div.input_area .highlight, +div.nboutput.container div.output_area .highlight { + border: none; + padding: 0; + margin: 0; + box-shadow: none; +} + +div.nbinput.container > div[class*=highlight], +div.nboutput.container > div[class*=highlight] { + margin: 0; +} + +div.nbinput.container div.prompt *, +div.nboutput.container div.prompt * { + background: none; +} + +div.nboutput.container div.output_area .highlight, +div.nboutput.container div.output_area pre { + background: unset; +} + +div.nboutput.container div.output_area div.highlight { + color: unset; /* override Pygments text color */ +} + +/* avoid gaps between output lines */ +div.nboutput.container div[class*=highlight] pre { + line-height: normal; +} + +/* input/output containers */ +div.nbinput.container, +div.nboutput.container { + display: -webkit-flex; + display: flex; + align-items: flex-start; + margin: 0; + width: 100%; +} +@media (max-width: 540px) { + div.nbinput.container, + div.nboutput.container { + flex-direction: column; + } +} + +/* input container */ +div.nbinput.container { + padding-top: 5px; +} + +/* last container */ +div.nblast.container { + padding-bottom: 5px; +} + +/* input prompt */ +div.nbinput.container div.prompt pre, +/* for sphinx_immaterial theme: */ +div.nbinput.container div.prompt pre > code { + color: #307FC1; +} + +/* output prompt */ +div.nboutput.container div.prompt pre, +/* for sphinx_immaterial theme: */ +div.nboutput.container div.prompt pre > code { + color: #BF5B3D; +} + +/* all prompts */ +div.nbinput.container div.prompt, +div.nboutput.container div.prompt { + width: 4.5ex; + padding-top: 5px; + position: relative; + user-select: none; +} + +div.nbinput.container div.prompt > div, +div.nboutput.container div.prompt > div { + position: absolute; + right: 0; + margin-right: 0.3ex; +} + +@media (max-width: 540px) { + div.nbinput.container div.prompt, + div.nboutput.container div.prompt { + width: unset; + text-align: left; + padding: 0.4em; + } + div.nboutput.container div.prompt.empty { + padding: 0; + } + + div.nbinput.container div.prompt > div, + div.nboutput.container div.prompt > div { + position: unset; + } +} + +/* disable scrollbars and line breaks on prompts */ +div.nbinput.container div.prompt pre, +div.nboutput.container div.prompt pre { + overflow: hidden; + white-space: pre; +} + +/* input/output area */ +div.nbinput.container div.input_area, +div.nboutput.container div.output_area { + -webkit-flex: 1; + flex: 1; + overflow: auto; +} +@media (max-width: 540px) { + div.nbinput.container div.input_area, + div.nboutput.container div.output_area { + width: 100%; + } +} + +/* input area */ +div.nbinput.container div.input_area { + border: 1px solid #e0e0e0; + border-radius: 2px; + /*background: #f5f5f5;*/ +} + +/* override MathJax center alignment in output cells */ +div.nboutput.container div[class*=MathJax] { + text-align: left !important; +} + +/* override sphinx.ext.imgmath center alignment in output cells */ +div.nboutput.container div.math p { + text-align: left; +} + +/* standard error */ +div.nboutput.container div.output_area.stderr { + background: #fdd; +} + +/* ANSI colors */ +.ansi-black-fg { color: #3E424D; } +.ansi-black-bg { background-color: #3E424D; } +.ansi-black-intense-fg { color: #282C36; } +.ansi-black-intense-bg { background-color: #282C36; } +.ansi-red-fg { color: #E75C58; } +.ansi-red-bg { background-color: #E75C58; } +.ansi-red-intense-fg { color: #B22B31; } +.ansi-red-intense-bg { background-color: #B22B31; } +.ansi-green-fg { color: #00A250; } +.ansi-green-bg { background-color: #00A250; } +.ansi-green-intense-fg { color: #007427; } +.ansi-green-intense-bg { background-color: #007427; } +.ansi-yellow-fg { color: #DDB62B; } +.ansi-yellow-bg { background-color: #DDB62B; } +.ansi-yellow-intense-fg { color: #B27D12; } +.ansi-yellow-intense-bg { background-color: #B27D12; } +.ansi-blue-fg { color: #208FFB; } +.ansi-blue-bg { background-color: #208FFB; } +.ansi-blue-intense-fg { color: #0065CA; } +.ansi-blue-intense-bg { background-color: #0065CA; } +.ansi-magenta-fg { color: #D160C4; } +.ansi-magenta-bg { background-color: #D160C4; } +.ansi-magenta-intense-fg { color: #A03196; } +.ansi-magenta-intense-bg { background-color: #A03196; } +.ansi-cyan-fg { color: #60C6C8; } +.ansi-cyan-bg { background-color: #60C6C8; } +.ansi-cyan-intense-fg { color: #258F8F; } +.ansi-cyan-intense-bg { background-color: #258F8F; } +.ansi-white-fg { color: #C5C1B4; } +.ansi-white-bg { background-color: #C5C1B4; } +.ansi-white-intense-fg { color: #A1A6B2; } +.ansi-white-intense-bg { background-color: #A1A6B2; } + +.ansi-default-inverse-fg { color: #FFFFFF; } +.ansi-default-inverse-bg { background-color: #000000; } + +.ansi-bold { font-weight: bold; } +.ansi-underline { text-decoration: underline; } + + +div.nbinput.container div.input_area div[class*=highlight] > pre, +div.nboutput.container div.output_area div[class*=highlight] > pre, +div.nboutput.container div.output_area div[class*=highlight].math, +div.nboutput.container div.output_area.rendered_html, +div.nboutput.container div.output_area > div.output_javascript, +div.nboutput.container div.output_area:not(.rendered_html) > img{ + padding: 5px; + margin: 0; +} + +/* fix copybtn overflow problem in chromium (needed for 'sphinx_copybutton') */ +div.nbinput.container div.input_area > div[class^='highlight'], +div.nboutput.container div.output_area > div[class^='highlight']{ + overflow-y: hidden; +} + +/* hide copy button on prompts for 'sphinx_copybutton' extension ... */ +.prompt .copybtn, +/* ... and 'sphinx_immaterial' theme */ +.prompt .md-clipboard.md-icon { + display: none; +} + +/* Some additional styling taken form the Jupyter notebook CSS */ +.jp-RenderedHTMLCommon table, +div.rendered_html table { + border: none; + border-collapse: collapse; + border-spacing: 0; + color: black; + font-size: 12px; + table-layout: fixed; +} +.jp-RenderedHTMLCommon thead, +div.rendered_html thead { + border-bottom: 1px solid black; + vertical-align: bottom; +} +.jp-RenderedHTMLCommon tr, +.jp-RenderedHTMLCommon th, +.jp-RenderedHTMLCommon td, +div.rendered_html tr, +div.rendered_html th, +div.rendered_html td { + text-align: right; + vertical-align: middle; + padding: 0.5em 0.5em; + line-height: normal; + white-space: normal; + max-width: none; + border: none; +} +.jp-RenderedHTMLCommon th, +div.rendered_html th { + font-weight: bold; +} +.jp-RenderedHTMLCommon tbody tr:nth-child(odd), +div.rendered_html tbody tr:nth-child(odd) { + background: #f5f5f5; +} +.jp-RenderedHTMLCommon tbody tr:hover, +div.rendered_html tbody tr:hover { + background: rgba(66, 165, 245, 0.2); +} + diff --git a/_static/nbsphinx-gallery.css b/_static/nbsphinx-gallery.css new file mode 100644 index 000000000..365c27a96 --- /dev/null +++ b/_static/nbsphinx-gallery.css @@ -0,0 +1,31 @@ +.nbsphinx-gallery { + display: grid; + grid-template-columns: repeat(auto-fill, minmax(160px, 1fr)); + gap: 5px; + margin-top: 1em; + margin-bottom: 1em; +} + +.nbsphinx-gallery > a { + padding: 5px; + border: 1px dotted currentColor; + border-radius: 2px; + text-align: center; +} + +.nbsphinx-gallery > a:hover { + border-style: solid; +} + +.nbsphinx-gallery img { + max-width: 100%; + max-height: 100%; +} + +.nbsphinx-gallery > a > div:first-child { + display: flex; + align-items: start; + justify-content: center; + height: 120px; + margin-bottom: 5px; +} diff --git a/_static/nbsphinx-no-thumbnail.svg b/_static/nbsphinx-no-thumbnail.svg new file mode 100644 index 000000000..9dca7588f --- /dev/null +++ b/_static/nbsphinx-no-thumbnail.svg @@ -0,0 +1,9 @@ + + + + diff --git a/_static/plus.png b/_static/plus.png new file mode 100644 index 000000000..7107cec93 Binary files /dev/null and b/_static/plus.png differ diff --git a/_static/pygments.css b/_static/pygments.css new file mode 100644 index 000000000..5c8cad8b4 --- /dev/null +++ b/_static/pygments.css @@ -0,0 +1,258 @@ +.highlight pre { line-height: 125%; } +.highlight td.linenos .normal { color: inherit; background-color: transparent; padding-left: 5px; padding-right: 5px; } +.highlight span.linenos { color: inherit; background-color: transparent; padding-left: 5px; padding-right: 5px; } +.highlight td.linenos .special { color: #000000; background-color: #ffffc0; padding-left: 5px; padding-right: 5px; } +.highlight span.linenos.special { color: #000000; background-color: #ffffc0; padding-left: 5px; padding-right: 5px; } +.highlight .hll { background-color: #ffffcc } +.highlight { background: #f8f8f8; } +.highlight .c { color: #8f5902; font-style: italic } /* Comment */ +.highlight .err { color: #a40000; border: 1px solid #ef2929 } /* Error */ +.highlight .g { color: #000000 } /* Generic */ +.highlight .k { color: #204a87; font-weight: bold } /* Keyword */ +.highlight .l { color: #000000 } /* Literal */ +.highlight .n { color: #000000 } /* Name */ +.highlight .o { color: #ce5c00; font-weight: bold } /* Operator */ +.highlight .x { color: #000000 } /* Other */ +.highlight .p { color: #000000; font-weight: bold } /* Punctuation */ +.highlight .ch { color: #8f5902; font-style: italic } /* Comment.Hashbang */ +.highlight .cm { color: #8f5902; font-style: italic } /* Comment.Multiline */ +.highlight .cp { color: #8f5902; font-style: italic } /* Comment.Preproc */ +.highlight .cpf { color: #8f5902; font-style: italic } /* Comment.PreprocFile */ +.highlight .c1 { color: #8f5902; font-style: italic } /* Comment.Single */ +.highlight .cs { color: #8f5902; font-style: italic } /* Comment.Special */ +.highlight .gd { color: #a40000 } /* Generic.Deleted */ +.highlight .ge { color: #000000; font-style: italic } /* Generic.Emph */ +.highlight .ges { color: #000000; font-weight: bold; font-style: italic } /* Generic.EmphStrong */ +.highlight .gr { color: #ef2929 } /* Generic.Error */ +.highlight .gh { color: #000080; font-weight: bold } /* Generic.Heading */ +.highlight .gi { color: #00A000 } /* Generic.Inserted */ +.highlight .go { color: #000000; font-style: italic } /* Generic.Output */ +.highlight .gp { color: #8f5902 } /* Generic.Prompt */ +.highlight .gs { color: #000000; font-weight: bold } /* Generic.Strong */ +.highlight .gu { color: #800080; font-weight: bold } /* Generic.Subheading */ +.highlight .gt { color: #a40000; font-weight: bold } /* Generic.Traceback */ +.highlight .kc { color: #204a87; font-weight: bold } /* Keyword.Constant */ +.highlight .kd { color: #204a87; font-weight: bold } /* Keyword.Declaration */ +.highlight .kn { color: #204a87; font-weight: bold } /* Keyword.Namespace */ +.highlight .kp { color: #204a87; font-weight: bold } /* Keyword.Pseudo */ +.highlight .kr { color: #204a87; font-weight: bold } /* Keyword.Reserved */ +.highlight .kt { color: #204a87; font-weight: bold } /* Keyword.Type */ +.highlight .ld { color: #000000 } /* Literal.Date */ +.highlight .m { color: #0000cf; font-weight: bold } /* Literal.Number */ +.highlight .s { color: #4e9a06 } /* Literal.String */ +.highlight .na { color: #c4a000 } /* Name.Attribute */ +.highlight .nb { color: #204a87 } /* Name.Builtin */ +.highlight .nc { color: #000000 } /* Name.Class */ +.highlight .no { color: #000000 } /* Name.Constant */ +.highlight .nd { color: #5c35cc; font-weight: bold } /* Name.Decorator */ +.highlight .ni { color: #ce5c00 } /* Name.Entity */ +.highlight .ne { color: #cc0000; font-weight: bold } /* Name.Exception */ +.highlight .nf { color: #000000 } /* Name.Function */ +.highlight .nl { color: #f57900 } /* Name.Label */ +.highlight .nn { color: #000000 } /* Name.Namespace */ +.highlight .nx { color: #000000 } /* Name.Other */ +.highlight .py { color: #000000 } /* Name.Property */ +.highlight .nt { color: #204a87; font-weight: bold } /* Name.Tag */ +.highlight .nv { color: #000000 } /* Name.Variable */ +.highlight .ow { color: #204a87; font-weight: bold } /* Operator.Word */ +.highlight .pm { color: #000000; font-weight: bold } /* Punctuation.Marker */ +.highlight .w { color: #f8f8f8 } /* Text.Whitespace */ +.highlight .mb { color: #0000cf; font-weight: bold } /* Literal.Number.Bin */ +.highlight .mf { color: #0000cf; font-weight: bold } /* Literal.Number.Float */ +.highlight .mh { color: #0000cf; font-weight: bold } /* Literal.Number.Hex */ +.highlight .mi { color: #0000cf; font-weight: bold } /* Literal.Number.Integer */ +.highlight .mo { color: #0000cf; font-weight: bold } /* Literal.Number.Oct */ +.highlight .sa { color: #4e9a06 } /* Literal.String.Affix */ +.highlight .sb { color: #4e9a06 } /* Literal.String.Backtick */ +.highlight .sc { color: #4e9a06 } /* Literal.String.Char */ +.highlight .dl { color: #4e9a06 } /* Literal.String.Delimiter */ +.highlight .sd { color: #8f5902; font-style: italic } /* Literal.String.Doc */ +.highlight .s2 { color: #4e9a06 } /* Literal.String.Double */ +.highlight .se { color: #4e9a06 } /* Literal.String.Escape */ +.highlight .sh { color: #4e9a06 } /* Literal.String.Heredoc */ +.highlight .si { color: #4e9a06 } /* Literal.String.Interpol */ +.highlight .sx { color: #4e9a06 } /* Literal.String.Other */ +.highlight .sr { color: #4e9a06 } /* Literal.String.Regex */ +.highlight .s1 { color: #4e9a06 } /* Literal.String.Single */ +.highlight .ss { color: #4e9a06 } /* Literal.String.Symbol */ +.highlight .bp { color: #3465a4 } /* Name.Builtin.Pseudo */ +.highlight .fm { color: #000000 } /* Name.Function.Magic */ +.highlight .vc { color: #000000 } /* Name.Variable.Class */ +.highlight .vg { color: #000000 } /* Name.Variable.Global */ +.highlight .vi { color: #000000 } /* Name.Variable.Instance */ +.highlight .vm { color: #000000 } /* Name.Variable.Magic */ +.highlight .il { color: #0000cf; font-weight: bold } /* Literal.Number.Integer.Long */ +@media not print { +body[data-theme="dark"] .highlight pre { line-height: 125%; } +body[data-theme="dark"] .highlight td.linenos .normal { color: inherit; background-color: transparent; padding-left: 5px; padding-right: 5px; } +body[data-theme="dark"] .highlight span.linenos { color: inherit; background-color: transparent; padding-left: 5px; padding-right: 5px; } +body[data-theme="dark"] .highlight td.linenos .special { color: #000000; background-color: #ffffc0; padding-left: 5px; padding-right: 5px; } +body[data-theme="dark"] .highlight span.linenos.special { color: #000000; background-color: #ffffc0; padding-left: 5px; padding-right: 5px; } +body[data-theme="dark"] .highlight .hll { background-color: #49483e } +body[data-theme="dark"] .highlight { background: #272822; color: #f8f8f2 } +body[data-theme="dark"] .highlight .c { color: #959077 } /* Comment */ +body[data-theme="dark"] .highlight .err { color: #ed007e; background-color: #1e0010 } /* Error */ +body[data-theme="dark"] .highlight .esc { color: #f8f8f2 } /* Escape */ +body[data-theme="dark"] .highlight .g { color: #f8f8f2 } /* Generic */ +body[data-theme="dark"] .highlight .k { color: #66d9ef } /* Keyword */ +body[data-theme="dark"] .highlight .l { color: #ae81ff } /* Literal */ +body[data-theme="dark"] .highlight .n { color: #f8f8f2 } /* Name */ +body[data-theme="dark"] .highlight .o { color: #ff4689 } /* Operator */ +body[data-theme="dark"] .highlight .x { color: #f8f8f2 } /* Other */ +body[data-theme="dark"] .highlight .p { color: #f8f8f2 } /* Punctuation */ +body[data-theme="dark"] .highlight .ch { color: #959077 } /* Comment.Hashbang */ +body[data-theme="dark"] .highlight .cm { color: #959077 } /* Comment.Multiline */ +body[data-theme="dark"] .highlight .cp { color: #959077 } /* Comment.Preproc */ +body[data-theme="dark"] .highlight .cpf { color: #959077 } /* Comment.PreprocFile */ +body[data-theme="dark"] .highlight .c1 { color: #959077 } /* Comment.Single */ +body[data-theme="dark"] .highlight .cs { color: #959077 } /* Comment.Special */ +body[data-theme="dark"] .highlight .gd { color: #ff4689 } /* Generic.Deleted */ +body[data-theme="dark"] .highlight .ge { color: #f8f8f2; font-style: italic } /* Generic.Emph */ +body[data-theme="dark"] .highlight .ges { color: #f8f8f2; font-weight: bold; font-style: italic } /* Generic.EmphStrong */ +body[data-theme="dark"] .highlight .gr { color: #f8f8f2 } /* Generic.Error */ +body[data-theme="dark"] .highlight .gh { color: #f8f8f2 } /* Generic.Heading */ +body[data-theme="dark"] .highlight .gi { color: #a6e22e } /* Generic.Inserted */ +body[data-theme="dark"] .highlight .go { color: #66d9ef } /* Generic.Output */ +body[data-theme="dark"] .highlight .gp { color: #ff4689; font-weight: bold } /* Generic.Prompt */ +body[data-theme="dark"] .highlight .gs { color: #f8f8f2; font-weight: bold } /* Generic.Strong */ +body[data-theme="dark"] .highlight .gu { color: #959077 } /* Generic.Subheading */ +body[data-theme="dark"] .highlight .gt { color: #f8f8f2 } /* Generic.Traceback */ +body[data-theme="dark"] .highlight .kc { color: #66d9ef } /* Keyword.Constant */ +body[data-theme="dark"] .highlight .kd { color: #66d9ef } /* Keyword.Declaration */ +body[data-theme="dark"] .highlight .kn { color: #ff4689 } /* Keyword.Namespace */ +body[data-theme="dark"] .highlight .kp { color: #66d9ef } /* Keyword.Pseudo */ +body[data-theme="dark"] .highlight .kr { color: #66d9ef } /* Keyword.Reserved */ +body[data-theme="dark"] .highlight .kt { color: #66d9ef } /* Keyword.Type */ +body[data-theme="dark"] .highlight .ld { color: #e6db74 } /* Literal.Date */ +body[data-theme="dark"] .highlight .m { color: #ae81ff } /* Literal.Number */ +body[data-theme="dark"] .highlight .s { color: #e6db74 } /* Literal.String */ +body[data-theme="dark"] .highlight .na { color: #a6e22e } /* Name.Attribute */ +body[data-theme="dark"] .highlight .nb { color: #f8f8f2 } /* Name.Builtin */ +body[data-theme="dark"] .highlight .nc { color: #a6e22e } /* Name.Class */ +body[data-theme="dark"] .highlight .no { color: #66d9ef } /* Name.Constant */ +body[data-theme="dark"] .highlight .nd { color: #a6e22e } /* Name.Decorator */ +body[data-theme="dark"] .highlight .ni { color: #f8f8f2 } /* Name.Entity */ +body[data-theme="dark"] .highlight .ne { color: #a6e22e } /* Name.Exception */ +body[data-theme="dark"] .highlight .nf { color: #a6e22e } /* Name.Function */ +body[data-theme="dark"] .highlight .nl { color: #f8f8f2 } /* Name.Label */ +body[data-theme="dark"] .highlight .nn { color: #f8f8f2 } /* Name.Namespace */ +body[data-theme="dark"] .highlight .nx { color: #a6e22e } /* Name.Other */ +body[data-theme="dark"] .highlight .py { color: #f8f8f2 } /* Name.Property */ +body[data-theme="dark"] .highlight .nt { color: #ff4689 } /* Name.Tag */ +body[data-theme="dark"] .highlight .nv { color: #f8f8f2 } /* Name.Variable */ +body[data-theme="dark"] .highlight .ow { color: #ff4689 } /* Operator.Word */ +body[data-theme="dark"] .highlight .pm { color: #f8f8f2 } /* Punctuation.Marker */ +body[data-theme="dark"] .highlight .w { color: #f8f8f2 } /* Text.Whitespace */ +body[data-theme="dark"] .highlight .mb { color: #ae81ff } /* Literal.Number.Bin */ +body[data-theme="dark"] .highlight .mf { color: #ae81ff } /* Literal.Number.Float */ +body[data-theme="dark"] .highlight .mh { color: #ae81ff } /* Literal.Number.Hex */ +body[data-theme="dark"] .highlight .mi { color: #ae81ff } /* Literal.Number.Integer */ +body[data-theme="dark"] .highlight .mo { color: #ae81ff } /* Literal.Number.Oct */ +body[data-theme="dark"] .highlight .sa { color: #e6db74 } /* Literal.String.Affix */ +body[data-theme="dark"] .highlight .sb { color: #e6db74 } /* Literal.String.Backtick */ +body[data-theme="dark"] .highlight .sc { color: #e6db74 } /* Literal.String.Char */ +body[data-theme="dark"] .highlight .dl { color: #e6db74 } /* Literal.String.Delimiter */ +body[data-theme="dark"] .highlight .sd { color: #e6db74 } /* Literal.String.Doc */ +body[data-theme="dark"] .highlight .s2 { color: #e6db74 } /* Literal.String.Double */ +body[data-theme="dark"] .highlight .se { color: #ae81ff } /* Literal.String.Escape */ +body[data-theme="dark"] .highlight .sh { color: #e6db74 } /* Literal.String.Heredoc */ +body[data-theme="dark"] .highlight .si { color: #e6db74 } /* Literal.String.Interpol */ +body[data-theme="dark"] .highlight .sx { color: #e6db74 } /* Literal.String.Other */ +body[data-theme="dark"] .highlight .sr { color: #e6db74 } /* Literal.String.Regex */ +body[data-theme="dark"] .highlight .s1 { color: #e6db74 } /* Literal.String.Single */ +body[data-theme="dark"] .highlight .ss { color: #e6db74 } /* Literal.String.Symbol */ +body[data-theme="dark"] .highlight .bp { color: #f8f8f2 } /* Name.Builtin.Pseudo */ +body[data-theme="dark"] .highlight .fm { color: #a6e22e } /* Name.Function.Magic */ +body[data-theme="dark"] .highlight .vc { color: #f8f8f2 } /* Name.Variable.Class */ +body[data-theme="dark"] .highlight .vg { color: #f8f8f2 } /* Name.Variable.Global */ +body[data-theme="dark"] .highlight .vi { color: #f8f8f2 } /* Name.Variable.Instance */ +body[data-theme="dark"] .highlight .vm { color: #f8f8f2 } /* Name.Variable.Magic */ +body[data-theme="dark"] .highlight .il { color: #ae81ff } /* Literal.Number.Integer.Long */ +@media (prefers-color-scheme: dark) { +body:not([data-theme="light"]) .highlight pre { line-height: 125%; } +body:not([data-theme="light"]) .highlight td.linenos .normal { color: inherit; background-color: transparent; padding-left: 5px; padding-right: 5px; } +body:not([data-theme="light"]) .highlight span.linenos { color: inherit; background-color: transparent; padding-left: 5px; padding-right: 5px; } +body:not([data-theme="light"]) .highlight td.linenos .special { color: #000000; background-color: #ffffc0; padding-left: 5px; padding-right: 5px; } +body:not([data-theme="light"]) .highlight span.linenos.special { color: #000000; background-color: #ffffc0; padding-left: 5px; padding-right: 5px; } +body:not([data-theme="light"]) .highlight .hll { background-color: #49483e } +body:not([data-theme="light"]) .highlight { background: #272822; color: #f8f8f2 } +body:not([data-theme="light"]) .highlight .c { color: #959077 } /* Comment */ +body:not([data-theme="light"]) .highlight .err { color: #ed007e; background-color: #1e0010 } /* Error */ +body:not([data-theme="light"]) .highlight .esc { color: #f8f8f2 } /* Escape */ +body:not([data-theme="light"]) .highlight .g { color: #f8f8f2 } /* Generic */ +body:not([data-theme="light"]) .highlight .k { color: #66d9ef } /* Keyword */ +body:not([data-theme="light"]) .highlight .l { color: #ae81ff } /* Literal */ +body:not([data-theme="light"]) .highlight .n { color: #f8f8f2 } /* Name */ +body:not([data-theme="light"]) .highlight .o { color: #ff4689 } /* Operator */ +body:not([data-theme="light"]) .highlight .x { color: #f8f8f2 } /* Other */ +body:not([data-theme="light"]) .highlight .p { color: #f8f8f2 } /* Punctuation */ +body:not([data-theme="light"]) .highlight .ch { color: #959077 } /* Comment.Hashbang */ +body:not([data-theme="light"]) .highlight .cm { color: #959077 } /* Comment.Multiline */ +body:not([data-theme="light"]) .highlight .cp { color: #959077 } /* Comment.Preproc */ +body:not([data-theme="light"]) .highlight .cpf { color: #959077 } /* Comment.PreprocFile */ +body:not([data-theme="light"]) .highlight .c1 { color: #959077 } /* Comment.Single */ +body:not([data-theme="light"]) .highlight .cs { color: #959077 } /* Comment.Special */ +body:not([data-theme="light"]) .highlight .gd { color: #ff4689 } /* Generic.Deleted */ +body:not([data-theme="light"]) .highlight .ge { color: #f8f8f2; font-style: italic } /* Generic.Emph */ +body:not([data-theme="light"]) .highlight .ges { color: #f8f8f2; font-weight: bold; font-style: italic } /* Generic.EmphStrong */ +body:not([data-theme="light"]) .highlight .gr { color: #f8f8f2 } /* Generic.Error */ +body:not([data-theme="light"]) .highlight .gh { color: #f8f8f2 } /* Generic.Heading */ +body:not([data-theme="light"]) .highlight .gi { color: #a6e22e } /* Generic.Inserted */ +body:not([data-theme="light"]) .highlight .go { color: #66d9ef } /* Generic.Output */ +body:not([data-theme="light"]) .highlight .gp { color: #ff4689; font-weight: bold } /* Generic.Prompt */ +body:not([data-theme="light"]) .highlight .gs { color: #f8f8f2; font-weight: bold } /* Generic.Strong */ +body:not([data-theme="light"]) .highlight .gu { color: #959077 } /* Generic.Subheading */ +body:not([data-theme="light"]) .highlight .gt { color: #f8f8f2 } /* Generic.Traceback */ +body:not([data-theme="light"]) .highlight .kc { color: #66d9ef } /* Keyword.Constant */ +body:not([data-theme="light"]) .highlight .kd { color: #66d9ef } /* Keyword.Declaration */ +body:not([data-theme="light"]) .highlight .kn { color: #ff4689 } /* Keyword.Namespace */ +body:not([data-theme="light"]) .highlight .kp { color: #66d9ef } /* Keyword.Pseudo */ +body:not([data-theme="light"]) .highlight .kr { color: #66d9ef } /* Keyword.Reserved */ +body:not([data-theme="light"]) .highlight .kt { color: #66d9ef } /* Keyword.Type */ +body:not([data-theme="light"]) .highlight .ld { color: #e6db74 } /* Literal.Date */ +body:not([data-theme="light"]) .highlight .m { color: #ae81ff } /* Literal.Number */ +body:not([data-theme="light"]) .highlight .s { color: #e6db74 } /* Literal.String */ +body:not([data-theme="light"]) .highlight .na { color: #a6e22e } /* Name.Attribute */ +body:not([data-theme="light"]) .highlight .nb { color: #f8f8f2 } /* Name.Builtin */ +body:not([data-theme="light"]) .highlight .nc { color: #a6e22e } /* Name.Class */ +body:not([data-theme="light"]) .highlight .no { color: #66d9ef } /* Name.Constant */ +body:not([data-theme="light"]) .highlight .nd { color: #a6e22e } /* Name.Decorator */ +body:not([data-theme="light"]) .highlight .ni { color: #f8f8f2 } /* Name.Entity */ +body:not([data-theme="light"]) .highlight .ne { color: #a6e22e } /* Name.Exception */ +body:not([data-theme="light"]) .highlight .nf { color: #a6e22e } /* Name.Function */ +body:not([data-theme="light"]) .highlight .nl { color: #f8f8f2 } /* Name.Label */ +body:not([data-theme="light"]) .highlight .nn { color: #f8f8f2 } /* Name.Namespace */ +body:not([data-theme="light"]) .highlight .nx { color: #a6e22e } /* Name.Other */ +body:not([data-theme="light"]) .highlight .py { color: #f8f8f2 } /* Name.Property */ +body:not([data-theme="light"]) .highlight .nt { color: #ff4689 } /* Name.Tag */ +body:not([data-theme="light"]) .highlight .nv { color: #f8f8f2 } /* Name.Variable */ +body:not([data-theme="light"]) .highlight .ow { color: #ff4689 } /* Operator.Word */ +body:not([data-theme="light"]) .highlight .pm { color: #f8f8f2 } /* Punctuation.Marker */ +body:not([data-theme="light"]) .highlight .w { color: #f8f8f2 } /* Text.Whitespace */ +body:not([data-theme="light"]) .highlight .mb { color: #ae81ff } /* Literal.Number.Bin */ +body:not([data-theme="light"]) .highlight .mf { color: #ae81ff } /* Literal.Number.Float */ +body:not([data-theme="light"]) .highlight .mh { color: #ae81ff } /* Literal.Number.Hex */ +body:not([data-theme="light"]) .highlight .mi { color: #ae81ff } /* Literal.Number.Integer */ +body:not([data-theme="light"]) .highlight .mo { color: #ae81ff } /* Literal.Number.Oct */ +body:not([data-theme="light"]) .highlight .sa { color: #e6db74 } /* Literal.String.Affix */ +body:not([data-theme="light"]) .highlight .sb { color: #e6db74 } /* Literal.String.Backtick */ +body:not([data-theme="light"]) .highlight .sc { color: #e6db74 } /* Literal.String.Char */ +body:not([data-theme="light"]) .highlight .dl { color: #e6db74 } /* Literal.String.Delimiter */ +body:not([data-theme="light"]) .highlight .sd { color: #e6db74 } /* Literal.String.Doc */ +body:not([data-theme="light"]) .highlight .s2 { color: #e6db74 } /* Literal.String.Double */ +body:not([data-theme="light"]) .highlight .se { color: #ae81ff } /* Literal.String.Escape */ +body:not([data-theme="light"]) .highlight .sh { color: #e6db74 } /* Literal.String.Heredoc */ +body:not([data-theme="light"]) .highlight .si { color: #e6db74 } /* Literal.String.Interpol */ +body:not([data-theme="light"]) .highlight .sx { color: #e6db74 } /* Literal.String.Other */ +body:not([data-theme="light"]) .highlight .sr { color: #e6db74 } /* Literal.String.Regex */ +body:not([data-theme="light"]) .highlight .s1 { color: #e6db74 } /* Literal.String.Single */ +body:not([data-theme="light"]) .highlight .ss { color: #e6db74 } /* Literal.String.Symbol */ +body:not([data-theme="light"]) .highlight .bp { color: #f8f8f2 } /* Name.Builtin.Pseudo */ +body:not([data-theme="light"]) .highlight .fm { color: #a6e22e } /* Name.Function.Magic */ +body:not([data-theme="light"]) .highlight .vc { color: #f8f8f2 } /* Name.Variable.Class */ +body:not([data-theme="light"]) .highlight .vg { color: #f8f8f2 } /* Name.Variable.Global */ +body:not([data-theme="light"]) .highlight .vi { color: #f8f8f2 } /* Name.Variable.Instance */ +body:not([data-theme="light"]) .highlight .vm { color: #f8f8f2 } /* Name.Variable.Magic */ +body:not([data-theme="light"]) .highlight .il { color: #ae81ff } /* Literal.Number.Integer.Long */ +} +} \ No newline at end of file diff --git a/_static/scripts/furo-extensions.js b/_static/scripts/furo-extensions.js new file mode 100644 index 000000000..e69de29bb diff --git a/_static/scripts/furo.js b/_static/scripts/furo.js new file mode 100644 index 000000000..32e7c05be --- /dev/null +++ b/_static/scripts/furo.js @@ -0,0 +1,3 @@ +/*! For license information please see furo.js.LICENSE.txt */ +(()=>{var t={212:function(t,e,n){var o,r;r=void 0!==n.g?n.g:"undefined"!=typeof window?window:this,o=function(){return function(t){"use strict";var e={navClass:"active",contentClass:"active",nested:!1,nestedClass:"active",offset:0,reflow:!1,events:!0},n=function(t,e,n){if(n.settings.events){var o=new CustomEvent(t,{bubbles:!0,cancelable:!0,detail:n});e.dispatchEvent(o)}},o=function(t){var e=0;if(t.offsetParent)for(;t;)e+=t.offsetTop,t=t.offsetParent;return e>=0?e:0},r=function(t){t&&t.sort((function(t,e){return o(t.content)=Math.max(document.body.scrollHeight,document.documentElement.scrollHeight,document.body.offsetHeight,document.documentElement.offsetHeight,document.body.clientHeight,document.documentElement.clientHeight)},l=function(t,e){var n=t[t.length-1];if(function(t,e){return!(!s()||!c(t.content,e,!0))}(n,e))return n;for(var o=t.length-1;o>=0;o--)if(c(t[o].content,e))return t[o]},a=function(t,e){if(e.nested&&t.parentNode){var n=t.parentNode.closest("li");n&&(n.classList.remove(e.nestedClass),a(n,e))}},i=function(t,e){if(t){var o=t.nav.closest("li");o&&(o.classList.remove(e.navClass),t.content.classList.remove(e.contentClass),a(o,e),n("gumshoeDeactivate",o,{link:t.nav,content:t.content,settings:e}))}},u=function(t,e){if(e.nested){var n=t.parentNode.closest("li");n&&(n.classList.add(e.nestedClass),u(n,e))}};return function(o,c){var s,a,d,f,m,v={setup:function(){s=document.querySelectorAll(o),a=[],Array.prototype.forEach.call(s,(function(t){var e=document.getElementById(decodeURIComponent(t.hash.substr(1)));e&&a.push({nav:t,content:e})})),r(a)},detect:function(){var t=l(a,m);t?d&&t.content===d.content||(i(d,m),function(t,e){if(t){var o=t.nav.closest("li");o&&(o.classList.add(e.navClass),t.content.classList.add(e.contentClass),u(o,e),n("gumshoeActivate",o,{link:t.nav,content:t.content,settings:e}))}}(t,m),d=t):d&&(i(d,m),d=null)}},h=function(e){f&&t.cancelAnimationFrame(f),f=t.requestAnimationFrame(v.detect)},g=function(e){f&&t.cancelAnimationFrame(f),f=t.requestAnimationFrame((function(){r(a),v.detect()}))};return v.destroy=function(){d&&i(d,m),t.removeEventListener("scroll",h,!1),m.reflow&&t.removeEventListener("resize",g,!1),a=null,s=null,d=null,f=null,m=null},m=function(){var t={};return Array.prototype.forEach.call(arguments,(function(e){for(var n in e){if(!e.hasOwnProperty(n))return;t[n]=e[n]}})),t}(e,c||{}),v.setup(),v.detect(),t.addEventListener("scroll",h,!1),m.reflow&&t.addEventListener("resize",g,!1),v}}(r)}.apply(e,[]),void 0===o||(t.exports=o)}},e={};function n(o){var r=e[o];if(void 0!==r)return r.exports;var c=e[o]={exports:{}};return t[o].call(c.exports,c,c.exports,n),c.exports}n.n=t=>{var e=t&&t.__esModule?()=>t.default:()=>t;return n.d(e,{a:e}),e},n.d=(t,e)=>{for(var o in e)n.o(e,o)&&!n.o(t,o)&&Object.defineProperty(t,o,{enumerable:!0,get:e[o]})},n.g=function(){if("object"==typeof globalThis)return globalThis;try{return this||new Function("return this")()}catch(t){if("object"==typeof window)return window}}(),n.o=(t,e)=>Object.prototype.hasOwnProperty.call(t,e),(()=>{"use strict";var t=n(212),e=n.n(t),o=null,r=null,c=window.pageYOffset||document.documentElement.scrollTop;const s=64;function l(){const t=localStorage.getItem("theme")||"auto";var e;"light"!==(e=window.matchMedia("(prefers-color-scheme: dark)").matches?"auto"===t?"light":"light"==t?"dark":"auto":"auto"===t?"dark":"dark"==t?"light":"auto")&&"dark"!==e&&"auto"!==e&&(console.error(`Got invalid theme mode: ${e}. Resetting to auto.`),e="auto"),document.body.dataset.theme=e,localStorage.setItem("theme",e),console.log(`Changed to ${e} mode.`)}function a(){!function(){const t=document.getElementsByClassName("theme-toggle");Array.from(t).forEach((t=>{t.addEventListener("click",l)}))}(),function(){let t=0,e=!1;window.addEventListener("scroll",(function(n){t=window.scrollY,e||(window.requestAnimationFrame((function(){var n;n=t,0==Math.floor(r.getBoundingClientRect().top)?r.classList.add("scrolled"):r.classList.remove("scrolled"),function(t){tc&&document.documentElement.classList.remove("show-back-to-top"),c=t}(n),function(t){null!==o&&(0==t?o.scrollTo(0,0):Math.ceil(t)>=Math.floor(document.documentElement.scrollHeight-window.innerHeight)?o.scrollTo(0,o.scrollHeight):document.querySelector(".scroll-current"))}(n),e=!1})),e=!0)})),window.scroll()}(),null!==o&&new(e())(".toc-tree a",{reflow:!0,recursive:!0,navClass:"scroll-current",offset:()=>{let t=parseFloat(getComputedStyle(document.documentElement).fontSize);return r.getBoundingClientRect().height+.5*t+1}})}document.addEventListener("DOMContentLoaded",(function(){document.body.parentNode.classList.remove("no-js"),r=document.querySelector("header"),o=document.querySelector(".toc-scroll"),a()}))})()})(); +//# sourceMappingURL=furo.js.map \ No newline at end of file diff --git a/_static/scripts/furo.js.LICENSE.txt b/_static/scripts/furo.js.LICENSE.txt new file mode 100644 index 000000000..1632189c7 --- /dev/null +++ b/_static/scripts/furo.js.LICENSE.txt @@ -0,0 +1,7 @@ +/*! + * gumshoejs v5.1.2 (patched by @pradyunsg) + * A simple, framework-agnostic scrollspy script. + * (c) 2019 Chris Ferdinandi + * MIT License + * http://github.com/cferdinandi/gumshoe + */ diff --git a/_static/scripts/furo.js.map b/_static/scripts/furo.js.map new file mode 100644 index 000000000..7b7ddb113 --- /dev/null +++ b/_static/scripts/furo.js.map @@ -0,0 +1 @@ +{"version":3,"file":"scripts/furo.js","mappings":"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","sources":["webpack:///./src/furo/assets/scripts/gumshoe-patched.js","webpack:///webpack/bootstrap","webpack:///webpack/runtime/compat get default export","webpack:///webpack/runtime/define property getters","webpack:///webpack/runtime/global","webpack:///webpack/runtime/hasOwnProperty shorthand","webpack:///./src/furo/assets/scripts/furo.js"],"sourcesContent":["/*!\n * gumshoejs v5.1.2 (patched by @pradyunsg)\n * A simple, framework-agnostic scrollspy script.\n * (c) 2019 Chris Ferdinandi\n * MIT License\n * http://github.com/cferdinandi/gumshoe\n */\n\n(function (root, factory) {\n if (typeof define === \"function\" && define.amd) {\n define([], function () {\n return factory(root);\n });\n } else if (typeof exports === \"object\") {\n module.exports = factory(root);\n } else {\n root.Gumshoe = factory(root);\n }\n})(\n typeof global !== \"undefined\"\n ? global\n : typeof window !== \"undefined\"\n ? window\n : this,\n function (window) {\n \"use strict\";\n\n //\n // Defaults\n //\n\n var defaults = {\n // Active classes\n navClass: \"active\",\n contentClass: \"active\",\n\n // Nested navigation\n nested: false,\n nestedClass: \"active\",\n\n // Offset & reflow\n offset: 0,\n reflow: false,\n\n // Event support\n events: true,\n };\n\n //\n // Methods\n //\n\n /**\n * Merge two or more objects together.\n * @param {Object} objects The objects to merge together\n * @returns {Object} Merged values of defaults and options\n */\n var extend = function () {\n var merged = {};\n Array.prototype.forEach.call(arguments, function (obj) {\n for (var key in obj) {\n if (!obj.hasOwnProperty(key)) return;\n merged[key] = obj[key];\n }\n });\n return merged;\n };\n\n /**\n * Emit a custom event\n * @param {String} type The event type\n * @param {Node} elem The element to attach the event to\n * @param {Object} detail Any details to pass along with the event\n */\n var emitEvent = function (type, elem, detail) {\n // Make sure events are enabled\n if (!detail.settings.events) return;\n\n // Create a new event\n var event = new CustomEvent(type, {\n bubbles: true,\n cancelable: true,\n detail: detail,\n });\n\n // Dispatch the event\n elem.dispatchEvent(event);\n };\n\n /**\n * Get an element's distance from the top of the Document.\n * @param {Node} elem The element\n * @return {Number} Distance from the top in pixels\n */\n var getOffsetTop = function (elem) {\n var location = 0;\n if (elem.offsetParent) {\n while (elem) {\n location += elem.offsetTop;\n elem = elem.offsetParent;\n }\n }\n return location >= 0 ? location : 0;\n };\n\n /**\n * Sort content from first to last in the DOM\n * @param {Array} contents The content areas\n */\n var sortContents = function (contents) {\n if (contents) {\n contents.sort(function (item1, item2) {\n var offset1 = getOffsetTop(item1.content);\n var offset2 = getOffsetTop(item2.content);\n if (offset1 < offset2) return -1;\n return 1;\n });\n }\n };\n\n /**\n * Get the offset to use for calculating position\n * @param {Object} settings The settings for this instantiation\n * @return {Float} The number of pixels to offset the calculations\n */\n var getOffset = function (settings) {\n // if the offset is a function run it\n if (typeof settings.offset === \"function\") {\n return parseFloat(settings.offset());\n }\n\n // Otherwise, return it as-is\n return parseFloat(settings.offset);\n };\n\n /**\n * Get the document element's height\n * @private\n * @returns {Number}\n */\n var getDocumentHeight = function () {\n return Math.max(\n document.body.scrollHeight,\n document.documentElement.scrollHeight,\n document.body.offsetHeight,\n document.documentElement.offsetHeight,\n document.body.clientHeight,\n document.documentElement.clientHeight,\n );\n };\n\n /**\n * Determine if an element is in view\n * @param {Node} elem The element\n * @param {Object} settings The settings for this instantiation\n * @param {Boolean} bottom If true, check if element is above bottom of viewport instead\n * @return {Boolean} Returns true if element is in the viewport\n */\n var isInView = function (elem, settings, bottom) {\n var bounds = elem.getBoundingClientRect();\n var offset = getOffset(settings);\n if (bottom) {\n return (\n parseInt(bounds.bottom, 10) <\n (window.innerHeight || document.documentElement.clientHeight)\n );\n }\n return parseInt(bounds.top, 10) <= offset;\n };\n\n /**\n * Check if at the bottom of the viewport\n * @return {Boolean} If true, page is at the bottom of the viewport\n */\n var isAtBottom = function () {\n if (\n Math.ceil(window.innerHeight + window.pageYOffset) >=\n getDocumentHeight()\n )\n return true;\n return false;\n };\n\n /**\n * Check if the last item should be used (even if not at the top of the page)\n * @param {Object} item The last item\n * @param {Object} settings The settings for this instantiation\n * @return {Boolean} If true, use the last item\n */\n var useLastItem = function (item, settings) {\n if (isAtBottom() && isInView(item.content, settings, true)) return true;\n return false;\n };\n\n /**\n * Get the active content\n * @param {Array} contents The content areas\n * @param {Object} settings The settings for this instantiation\n * @return {Object} The content area and matching navigation link\n */\n var getActive = function (contents, settings) {\n var last = contents[contents.length - 1];\n if (useLastItem(last, settings)) return last;\n for (var i = contents.length - 1; i >= 0; i--) {\n if (isInView(contents[i].content, settings)) return contents[i];\n }\n };\n\n /**\n * Deactivate parent navs in a nested navigation\n * @param {Node} nav The starting navigation element\n * @param {Object} settings The settings for this instantiation\n */\n var deactivateNested = function (nav, settings) {\n // If nesting isn't activated, bail\n if (!settings.nested || !nav.parentNode) return;\n\n // Get the parent navigation\n var li = nav.parentNode.closest(\"li\");\n if (!li) return;\n\n // Remove the active class\n li.classList.remove(settings.nestedClass);\n\n // Apply recursively to any parent navigation elements\n deactivateNested(li, settings);\n };\n\n /**\n * Deactivate a nav and content area\n * @param {Object} items The nav item and content to deactivate\n * @param {Object} settings The settings for this instantiation\n */\n var deactivate = function (items, settings) {\n // Make sure there are items to deactivate\n if (!items) return;\n\n // Get the parent list item\n var li = items.nav.closest(\"li\");\n if (!li) return;\n\n // Remove the active class from the nav and content\n li.classList.remove(settings.navClass);\n items.content.classList.remove(settings.contentClass);\n\n // Deactivate any parent navs in a nested navigation\n deactivateNested(li, settings);\n\n // Emit a custom event\n emitEvent(\"gumshoeDeactivate\", li, {\n link: items.nav,\n content: items.content,\n settings: settings,\n });\n };\n\n /**\n * Activate parent navs in a nested navigation\n * @param {Node} nav The starting navigation element\n * @param {Object} settings The settings for this instantiation\n */\n var activateNested = function (nav, settings) {\n // If nesting isn't activated, bail\n if (!settings.nested) return;\n\n // Get the parent navigation\n var li = nav.parentNode.closest(\"li\");\n if (!li) return;\n\n // Add the active class\n li.classList.add(settings.nestedClass);\n\n // Apply recursively to any parent navigation elements\n activateNested(li, settings);\n };\n\n /**\n * Activate a nav and content area\n * @param {Object} items The nav item and content to activate\n * @param {Object} settings The settings for this instantiation\n */\n var activate = function (items, settings) {\n // Make sure there are items to activate\n if (!items) return;\n\n // Get the parent list item\n var li = items.nav.closest(\"li\");\n if (!li) return;\n\n // Add the active class to the nav and content\n li.classList.add(settings.navClass);\n items.content.classList.add(settings.contentClass);\n\n // Activate any parent navs in a nested navigation\n activateNested(li, settings);\n\n // Emit a custom event\n emitEvent(\"gumshoeActivate\", li, {\n link: items.nav,\n content: items.content,\n settings: settings,\n });\n };\n\n /**\n * Create the Constructor object\n * @param {String} selector The selector to use for navigation items\n * @param {Object} options User options and settings\n */\n var Constructor = function (selector, options) {\n //\n // Variables\n //\n\n var publicAPIs = {};\n var navItems, contents, current, timeout, settings;\n\n //\n // Methods\n //\n\n /**\n * Set variables from DOM elements\n */\n publicAPIs.setup = function () {\n // Get all nav items\n navItems = document.querySelectorAll(selector);\n\n // Create contents array\n contents = [];\n\n // Loop through each item, get it's matching content, and push to the array\n Array.prototype.forEach.call(navItems, function (item) {\n // Get the content for the nav item\n var content = document.getElementById(\n decodeURIComponent(item.hash.substr(1)),\n );\n if (!content) return;\n\n // Push to the contents array\n contents.push({\n nav: item,\n content: content,\n });\n });\n\n // Sort contents by the order they appear in the DOM\n sortContents(contents);\n };\n\n /**\n * Detect which content is currently active\n */\n publicAPIs.detect = function () {\n // Get the active content\n var active = getActive(contents, settings);\n\n // if there's no active content, deactivate and bail\n if (!active) {\n if (current) {\n deactivate(current, settings);\n current = null;\n }\n return;\n }\n\n // If the active content is the one currently active, do nothing\n if (current && active.content === current.content) return;\n\n // Deactivate the current content and activate the new content\n deactivate(current, settings);\n activate(active, settings);\n\n // Update the currently active content\n current = active;\n };\n\n /**\n * Detect the active content on scroll\n * Debounced for performance\n */\n var scrollHandler = function (event) {\n // If there's a timer, cancel it\n if (timeout) {\n window.cancelAnimationFrame(timeout);\n }\n\n // Setup debounce callback\n timeout = window.requestAnimationFrame(publicAPIs.detect);\n };\n\n /**\n * Update content sorting on resize\n * Debounced for performance\n */\n var resizeHandler = function (event) {\n // If there's a timer, cancel it\n if (timeout) {\n window.cancelAnimationFrame(timeout);\n }\n\n // Setup debounce callback\n timeout = window.requestAnimationFrame(function () {\n sortContents(contents);\n publicAPIs.detect();\n });\n };\n\n /**\n * Destroy the current instantiation\n */\n publicAPIs.destroy = function () {\n // Undo DOM changes\n if (current) {\n deactivate(current, settings);\n }\n\n // Remove event listeners\n window.removeEventListener(\"scroll\", scrollHandler, false);\n if (settings.reflow) {\n window.removeEventListener(\"resize\", resizeHandler, false);\n }\n\n // Reset variables\n contents = null;\n navItems = null;\n current = null;\n timeout = null;\n settings = null;\n };\n\n /**\n * Initialize the current instantiation\n */\n var init = function () {\n // Merge user options into defaults\n settings = extend(defaults, options || {});\n\n // Setup variables based on the current DOM\n publicAPIs.setup();\n\n // Find the currently active content\n publicAPIs.detect();\n\n // Setup event listeners\n window.addEventListener(\"scroll\", scrollHandler, false);\n if (settings.reflow) {\n window.addEventListener(\"resize\", resizeHandler, false);\n }\n };\n\n //\n // Initialize and return the public APIs\n //\n\n init();\n return publicAPIs;\n };\n\n //\n // Return the Constructor\n //\n\n return Constructor;\n },\n);\n","// The module cache\nvar __webpack_module_cache__ = {};\n\n// The require function\nfunction __webpack_require__(moduleId) {\n\t// Check if module is in cache\n\tvar cachedModule = __webpack_module_cache__[moduleId];\n\tif (cachedModule !== undefined) {\n\t\treturn cachedModule.exports;\n\t}\n\t// Create a new module (and put it into the cache)\n\tvar module = __webpack_module_cache__[moduleId] = {\n\t\t// no module.id needed\n\t\t// no module.loaded needed\n\t\texports: {}\n\t};\n\n\t// Execute the module function\n\t__webpack_modules__[moduleId].call(module.exports, module, module.exports, __webpack_require__);\n\n\t// Return the exports of the module\n\treturn module.exports;\n}\n\n","// getDefaultExport function for compatibility with non-harmony modules\n__webpack_require__.n = (module) => {\n\tvar getter = module && module.__esModule ?\n\t\t() => (module['default']) :\n\t\t() => (module);\n\t__webpack_require__.d(getter, { a: getter });\n\treturn getter;\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__.g = (function() {\n\tif (typeof globalThis === 'object') return globalThis;\n\ttry {\n\t\treturn this || new Function('return this')();\n\t} catch (e) {\n\t\tif (typeof window === 'object') return window;\n\t}\n})();","__webpack_require__.o = (obj, prop) => (Object.prototype.hasOwnProperty.call(obj, prop))","import Gumshoe from \"./gumshoe-patched.js\";\n\n////////////////////////////////////////////////////////////////////////////////\n// Scroll Handling\n////////////////////////////////////////////////////////////////////////////////\nvar tocScroll = null;\nvar header = null;\nvar lastScrollTop = window.pageYOffset || document.documentElement.scrollTop;\nconst GO_TO_TOP_OFFSET = 64;\n\nfunction scrollHandlerForHeader() {\n if (Math.floor(header.getBoundingClientRect().top) == 0) {\n header.classList.add(\"scrolled\");\n } else {\n header.classList.remove(\"scrolled\");\n }\n}\n\nfunction scrollHandlerForBackToTop(positionY) {\n if (positionY < GO_TO_TOP_OFFSET) {\n document.documentElement.classList.remove(\"show-back-to-top\");\n } else {\n if (positionY < lastScrollTop) {\n document.documentElement.classList.add(\"show-back-to-top\");\n } else if (positionY > lastScrollTop) {\n document.documentElement.classList.remove(\"show-back-to-top\");\n }\n }\n lastScrollTop = positionY;\n}\n\nfunction scrollHandlerForTOC(positionY) {\n if (tocScroll === null) {\n return;\n }\n\n // top of page.\n if (positionY == 0) {\n tocScroll.scrollTo(0, 0);\n } else if (\n // bottom of page.\n Math.ceil(positionY) >=\n Math.floor(document.documentElement.scrollHeight - window.innerHeight)\n ) {\n tocScroll.scrollTo(0, tocScroll.scrollHeight);\n } else {\n // somewhere in the middle.\n const current = document.querySelector(\".scroll-current\");\n if (current == null) {\n return;\n }\n\n // https://github.com/pypa/pip/issues/9159 This breaks scroll behaviours.\n // // scroll the currently \"active\" heading in toc, into view.\n // const rect = current.getBoundingClientRect();\n // if (0 > rect.top) {\n // current.scrollIntoView(true); // the argument is \"alignTop\"\n // } else if (rect.bottom > window.innerHeight) {\n // current.scrollIntoView(false);\n // }\n }\n}\n\nfunction scrollHandler(positionY) {\n scrollHandlerForHeader();\n scrollHandlerForBackToTop(positionY);\n scrollHandlerForTOC(positionY);\n}\n\n////////////////////////////////////////////////////////////////////////////////\n// Theme Toggle\n////////////////////////////////////////////////////////////////////////////////\nfunction setTheme(mode) {\n if (mode !== \"light\" && mode !== \"dark\" && mode !== \"auto\") {\n console.error(`Got invalid theme mode: ${mode}. Resetting to auto.`);\n mode = \"auto\";\n }\n\n document.body.dataset.theme = mode;\n localStorage.setItem(\"theme\", mode);\n console.log(`Changed to ${mode} mode.`);\n}\n\nfunction cycleThemeOnce() {\n const currentTheme = localStorage.getItem(\"theme\") || \"auto\";\n const prefersDark = window.matchMedia(\"(prefers-color-scheme: dark)\").matches;\n\n if (prefersDark) {\n // Auto (dark) -> Light -> Dark\n if (currentTheme === \"auto\") {\n setTheme(\"light\");\n } else if (currentTheme == \"light\") {\n setTheme(\"dark\");\n } else {\n setTheme(\"auto\");\n }\n } else {\n // Auto (light) -> Dark -> Light\n if (currentTheme === \"auto\") {\n setTheme(\"dark\");\n } else if (currentTheme == \"dark\") {\n setTheme(\"light\");\n } else {\n setTheme(\"auto\");\n }\n }\n}\n\n////////////////////////////////////////////////////////////////////////////////\n// Setup\n////////////////////////////////////////////////////////////////////////////////\nfunction setupScrollHandler() {\n // Taken from https://developer.mozilla.org/en-US/docs/Web/API/Document/scroll_event\n let last_known_scroll_position = 0;\n let ticking = false;\n\n window.addEventListener(\"scroll\", function (e) {\n last_known_scroll_position = window.scrollY;\n\n if (!ticking) {\n window.requestAnimationFrame(function () {\n scrollHandler(last_known_scroll_position);\n ticking = false;\n });\n\n ticking = true;\n }\n });\n window.scroll();\n}\n\nfunction setupScrollSpy() {\n if (tocScroll === null) {\n return;\n }\n\n // Scrollspy -- highlight table on contents, based on scroll\n new Gumshoe(\".toc-tree a\", {\n reflow: true,\n recursive: true,\n navClass: \"scroll-current\",\n offset: () => {\n let rem = parseFloat(getComputedStyle(document.documentElement).fontSize);\n return header.getBoundingClientRect().height + 0.5 * rem + 1;\n },\n });\n}\n\nfunction setupTheme() {\n // Attach event handlers for toggling themes\n const buttons = document.getElementsByClassName(\"theme-toggle\");\n Array.from(buttons).forEach((btn) => {\n btn.addEventListener(\"click\", cycleThemeOnce);\n });\n}\n\nfunction setup() {\n setupTheme();\n setupScrollHandler();\n setupScrollSpy();\n}\n\n////////////////////////////////////////////////////////////////////////////////\n// Main entrypoint\n////////////////////////////////////////////////////////////////////////////////\nfunction main() {\n document.body.parentNode.classList.remove(\"no-js\");\n\n header = document.querySelector(\"header\");\n tocScroll = document.querySelector(\".toc-scroll\");\n\n setup();\n}\n\ndocument.addEventListener(\"DOMContentLoaded\", main);\n"],"names":["root","g","window","this","defaults","navClass","contentClass","nested","nestedClass","offset","reflow","events","emitEvent","type","elem","detail","settings","event","CustomEvent","bubbles","cancelable","dispatchEvent","getOffsetTop","location","offsetParent","offsetTop","sortContents","contents","sort","item1","item2","content","isInView","bottom","bounds","getBoundingClientRect","parseFloat","getOffset","parseInt","innerHeight","document","documentElement","clientHeight","top","isAtBottom","Math","ceil","pageYOffset","max","body","scrollHeight","offsetHeight","getActive","last","length","item","useLastItem","i","deactivateNested","nav","parentNode","li","closest","classList","remove","deactivate","items","link","activateNested","add","selector","options","navItems","current","timeout","publicAPIs","querySelectorAll","Array","prototype","forEach","call","getElementById","decodeURIComponent","hash","substr","push","active","activate","scrollHandler","cancelAnimationFrame","requestAnimationFrame","detect","resizeHandler","destroy","removeEventListener","merged","arguments","obj","key","hasOwnProperty","extend","setup","addEventListener","factory","__webpack_module_cache__","__webpack_require__","moduleId","cachedModule","undefined","exports","module","__webpack_modules__","n","getter","__esModule","d","a","definition","o","Object","defineProperty","enumerable","get","globalThis","Function","e","prop","tocScroll","header","lastScrollTop","scrollTop","GO_TO_TOP_OFFSET","cycleThemeOnce","currentTheme","localStorage","getItem","mode","matchMedia","matches","console","error","dataset","theme","setItem","log","buttons","getElementsByClassName","from","btn","setupTheme","last_known_scroll_position","ticking","scrollY","positionY","floor","scrollHandlerForBackToTop","scrollTo","querySelector","scrollHandlerForTOC","scroll","setupScrollHandler","recursive","rem","getComputedStyle","fontSize","height"],"sourceRoot":""} \ No newline at end of file diff --git a/_static/searchtools.js b/_static/searchtools.js new file mode 100644 index 000000000..7918c3fab --- /dev/null +++ b/_static/searchtools.js @@ -0,0 +1,574 @@ +/* + * searchtools.js + * ~~~~~~~~~~~~~~~~ + * + * Sphinx JavaScript utilities for the full-text search. + * + * :copyright: Copyright 2007-2023 by the Sphinx team, see AUTHORS. + * :license: BSD, see LICENSE for details. + * + */ +"use strict"; + +/** + * Simple result scoring code. + */ +if (typeof Scorer === "undefined") { + var Scorer = { + // Implement the following function to further tweak the score for each result + // The function takes a result array [docname, title, anchor, descr, score, filename] + // and returns the new score. + /* + score: result => { + const [docname, title, anchor, descr, score, filename] = result + return score + }, + */ + + // query matches the full name of an object + objNameMatch: 11, + // or matches in the last dotted part of the object name + objPartialMatch: 6, + // Additive scores depending on the priority of the object + objPrio: { + 0: 15, // used to be importantResults + 1: 5, // used to be objectResults + 2: -5, // used to be unimportantResults + }, + // Used when the priority is not in the mapping. + objPrioDefault: 0, + + // query found in title + title: 15, + partialTitle: 7, + // query found in terms + term: 5, + partialTerm: 2, + }; +} + +const _removeChildren = (element) => { + while (element && element.lastChild) element.removeChild(element.lastChild); +}; + +/** + * See https://developer.mozilla.org/en-US/docs/Web/JavaScript/Guide/Regular_Expressions#escaping + */ +const _escapeRegExp = (string) => + string.replace(/[.*+\-?^${}()|[\]\\]/g, "\\$&"); // $& means the whole matched string + +const _displayItem = (item, searchTerms, highlightTerms) => { + const docBuilder = DOCUMENTATION_OPTIONS.BUILDER; + const docFileSuffix = DOCUMENTATION_OPTIONS.FILE_SUFFIX; + const docLinkSuffix = DOCUMENTATION_OPTIONS.LINK_SUFFIX; + const showSearchSummary = DOCUMENTATION_OPTIONS.SHOW_SEARCH_SUMMARY; + const contentRoot = document.documentElement.dataset.content_root; + + const [docName, title, anchor, descr, score, _filename] = item; + + let listItem = document.createElement("li"); + let requestUrl; + let linkUrl; + if (docBuilder === "dirhtml") { + // dirhtml builder + let dirname = docName + "/"; + if (dirname.match(/\/index\/$/)) + dirname = dirname.substring(0, dirname.length - 6); + else if (dirname === "index/") dirname = ""; + requestUrl = contentRoot + dirname; + linkUrl = requestUrl; + } else { + // normal html builders + requestUrl = contentRoot + docName + docFileSuffix; + linkUrl = docName + docLinkSuffix; + } + let linkEl = listItem.appendChild(document.createElement("a")); + linkEl.href = linkUrl + anchor; + linkEl.dataset.score = score; + linkEl.innerHTML = title; + if (descr) { + listItem.appendChild(document.createElement("span")).innerHTML = + " (" + descr + ")"; + // highlight search terms in the description + if (SPHINX_HIGHLIGHT_ENABLED) // set in sphinx_highlight.js + highlightTerms.forEach((term) => _highlightText(listItem, term, "highlighted")); + } + else if (showSearchSummary) + fetch(requestUrl) + .then((responseData) => responseData.text()) + .then((data) => { + if (data) + listItem.appendChild( + Search.makeSearchSummary(data, searchTerms) + ); + // highlight search terms in the summary + if (SPHINX_HIGHLIGHT_ENABLED) // set in sphinx_highlight.js + highlightTerms.forEach((term) => _highlightText(listItem, term, "highlighted")); + }); + Search.output.appendChild(listItem); +}; +const _finishSearch = (resultCount) => { + Search.stopPulse(); + Search.title.innerText = _("Search Results"); + if (!resultCount) + Search.status.innerText = Documentation.gettext( + "Your search did not match any documents. Please make sure that all words are spelled correctly and that you've selected enough categories." + ); + else + Search.status.innerText = _( + `Search finished, found ${resultCount} page(s) matching the search query.` + ); +}; +const _displayNextItem = ( + results, + resultCount, + searchTerms, + highlightTerms, +) => { + // results left, load the summary and display it + // this is intended to be dynamic (don't sub resultsCount) + if (results.length) { + _displayItem(results.pop(), searchTerms, highlightTerms); + setTimeout( + () => _displayNextItem(results, resultCount, searchTerms, highlightTerms), + 5 + ); + } + // search finished, update title and status message + else _finishSearch(resultCount); +}; + +/** + * Default splitQuery function. Can be overridden in ``sphinx.search`` with a + * custom function per language. + * + * The regular expression works by splitting the string on consecutive characters + * that are not Unicode letters, numbers, underscores, or emoji characters. + * This is the same as ``\W+`` in Python, preserving the surrogate pair area. + */ +if (typeof splitQuery === "undefined") { + var splitQuery = (query) => query + .split(/[^\p{Letter}\p{Number}_\p{Emoji_Presentation}]+/gu) + .filter(term => term) // remove remaining empty strings +} + +/** + * Search Module + */ +const Search = { + _index: null, + _queued_query: null, + _pulse_status: -1, + + htmlToText: (htmlString) => { + const htmlElement = new DOMParser().parseFromString(htmlString, 'text/html'); + htmlElement.querySelectorAll(".headerlink").forEach((el) => { el.remove() }); + const docContent = htmlElement.querySelector('[role="main"]'); + if (docContent !== undefined) return docContent.textContent; + console.warn( + "Content block not found. Sphinx search tries to obtain it via '[role=main]'. Could you check your theme or template." + ); + return ""; + }, + + init: () => { + const query = new URLSearchParams(window.location.search).get("q"); + document + .querySelectorAll('input[name="q"]') + .forEach((el) => (el.value = query)); + if (query) Search.performSearch(query); + }, + + loadIndex: (url) => + (document.body.appendChild(document.createElement("script")).src = url), + + setIndex: (index) => { + Search._index = index; + if (Search._queued_query !== null) { + const query = Search._queued_query; + Search._queued_query = null; + Search.query(query); + } + }, + + hasIndex: () => Search._index !== null, + + deferQuery: (query) => (Search._queued_query = query), + + stopPulse: () => (Search._pulse_status = -1), + + startPulse: () => { + if (Search._pulse_status >= 0) return; + + const pulse = () => { + Search._pulse_status = (Search._pulse_status + 1) % 4; + Search.dots.innerText = ".".repeat(Search._pulse_status); + if (Search._pulse_status >= 0) window.setTimeout(pulse, 500); + }; + pulse(); + }, + + /** + * perform a search for something (or wait until index is loaded) + */ + performSearch: (query) => { + // create the required interface elements + const searchText = document.createElement("h2"); + searchText.textContent = _("Searching"); + const searchSummary = document.createElement("p"); + searchSummary.classList.add("search-summary"); + searchSummary.innerText = ""; + const searchList = document.createElement("ul"); + searchList.classList.add("search"); + + const out = document.getElementById("search-results"); + Search.title = out.appendChild(searchText); + Search.dots = Search.title.appendChild(document.createElement("span")); + Search.status = out.appendChild(searchSummary); + Search.output = out.appendChild(searchList); + + const searchProgress = document.getElementById("search-progress"); + // Some themes don't use the search progress node + if (searchProgress) { + searchProgress.innerText = _("Preparing search..."); + } + Search.startPulse(); + + // index already loaded, the browser was quick! + if (Search.hasIndex()) Search.query(query); + else Search.deferQuery(query); + }, + + /** + * execute search (requires search index to be loaded) + */ + query: (query) => { + const filenames = Search._index.filenames; + const docNames = Search._index.docnames; + const titles = Search._index.titles; + const allTitles = Search._index.alltitles; + const indexEntries = Search._index.indexentries; + + // stem the search terms and add them to the correct list + const stemmer = new Stemmer(); + const searchTerms = new Set(); + const excludedTerms = new Set(); + const highlightTerms = new Set(); + const objectTerms = new Set(splitQuery(query.toLowerCase().trim())); + splitQuery(query.trim()).forEach((queryTerm) => { + const queryTermLower = queryTerm.toLowerCase(); + + // maybe skip this "word" + // stopwords array is from language_data.js + if ( + stopwords.indexOf(queryTermLower) !== -1 || + queryTerm.match(/^\d+$/) + ) + return; + + // stem the word + let word = stemmer.stemWord(queryTermLower); + // select the correct list + if (word[0] === "-") excludedTerms.add(word.substr(1)); + else { + searchTerms.add(word); + highlightTerms.add(queryTermLower); + } + }); + + if (SPHINX_HIGHLIGHT_ENABLED) { // set in sphinx_highlight.js + localStorage.setItem("sphinx_highlight_terms", [...highlightTerms].join(" ")) + } + + // console.debug("SEARCH: searching for:"); + // console.info("required: ", [...searchTerms]); + // console.info("excluded: ", [...excludedTerms]); + + // array of [docname, title, anchor, descr, score, filename] + let results = []; + _removeChildren(document.getElementById("search-progress")); + + const queryLower = query.toLowerCase(); + for (const [title, foundTitles] of Object.entries(allTitles)) { + if (title.toLowerCase().includes(queryLower) && (queryLower.length >= title.length/2)) { + for (const [file, id] of foundTitles) { + let score = Math.round(100 * queryLower.length / title.length) + results.push([ + docNames[file], + titles[file] !== title ? `${titles[file]} > ${title}` : title, + id !== null ? "#" + id : "", + null, + score, + filenames[file], + ]); + } + } + } + + // search for explicit entries in index directives + for (const [entry, foundEntries] of Object.entries(indexEntries)) { + if (entry.includes(queryLower) && (queryLower.length >= entry.length/2)) { + for (const [file, id] of foundEntries) { + let score = Math.round(100 * queryLower.length / entry.length) + results.push([ + docNames[file], + titles[file], + id ? "#" + id : "", + null, + score, + filenames[file], + ]); + } + } + } + + // lookup as object + objectTerms.forEach((term) => + results.push(...Search.performObjectSearch(term, objectTerms)) + ); + + // lookup as search terms in fulltext + results.push(...Search.performTermsSearch(searchTerms, excludedTerms)); + + // let the scorer override scores with a custom scoring function + if (Scorer.score) results.forEach((item) => (item[4] = Scorer.score(item))); + + // now sort the results by score (in opposite order of appearance, since the + // display function below uses pop() to retrieve items) and then + // alphabetically + results.sort((a, b) => { + const leftScore = a[4]; + const rightScore = b[4]; + if (leftScore === rightScore) { + // same score: sort alphabetically + const leftTitle = a[1].toLowerCase(); + const rightTitle = b[1].toLowerCase(); + if (leftTitle === rightTitle) return 0; + return leftTitle > rightTitle ? -1 : 1; // inverted is intentional + } + return leftScore > rightScore ? 1 : -1; + }); + + // remove duplicate search results + // note the reversing of results, so that in the case of duplicates, the highest-scoring entry is kept + let seen = new Set(); + results = results.reverse().reduce((acc, result) => { + let resultStr = result.slice(0, 4).concat([result[5]]).map(v => String(v)).join(','); + if (!seen.has(resultStr)) { + acc.push(result); + seen.add(resultStr); + } + return acc; + }, []); + + results = results.reverse(); + + // for debugging + //Search.lastresults = results.slice(); // a copy + // console.info("search results:", Search.lastresults); + + // print the results + _displayNextItem(results, results.length, searchTerms, highlightTerms); + }, + + /** + * search for object names + */ + performObjectSearch: (object, objectTerms) => { + const filenames = Search._index.filenames; + const docNames = Search._index.docnames; + const objects = Search._index.objects; + const objNames = Search._index.objnames; + const titles = Search._index.titles; + + const results = []; + + const objectSearchCallback = (prefix, match) => { + const name = match[4] + const fullname = (prefix ? prefix + "." : "") + name; + const fullnameLower = fullname.toLowerCase(); + if (fullnameLower.indexOf(object) < 0) return; + + let score = 0; + const parts = fullnameLower.split("."); + + // check for different match types: exact matches of full name or + // "last name" (i.e. last dotted part) + if (fullnameLower === object || parts.slice(-1)[0] === object) + score += Scorer.objNameMatch; + else if (parts.slice(-1)[0].indexOf(object) > -1) + score += Scorer.objPartialMatch; // matches in last name + + const objName = objNames[match[1]][2]; + const title = titles[match[0]]; + + // If more than one term searched for, we require other words to be + // found in the name/title/description + const otherTerms = new Set(objectTerms); + otherTerms.delete(object); + if (otherTerms.size > 0) { + const haystack = `${prefix} ${name} ${objName} ${title}`.toLowerCase(); + if ( + [...otherTerms].some((otherTerm) => haystack.indexOf(otherTerm) < 0) + ) + return; + } + + let anchor = match[3]; + if (anchor === "") anchor = fullname; + else if (anchor === "-") anchor = objNames[match[1]][1] + "-" + fullname; + + const descr = objName + _(", in ") + title; + + // add custom score for some objects according to scorer + if (Scorer.objPrio.hasOwnProperty(match[2])) + score += Scorer.objPrio[match[2]]; + else score += Scorer.objPrioDefault; + + results.push([ + docNames[match[0]], + fullname, + "#" + anchor, + descr, + score, + filenames[match[0]], + ]); + }; + Object.keys(objects).forEach((prefix) => + objects[prefix].forEach((array) => + objectSearchCallback(prefix, array) + ) + ); + return results; + }, + + /** + * search for full-text terms in the index + */ + performTermsSearch: (searchTerms, excludedTerms) => { + // prepare search + const terms = Search._index.terms; + const titleTerms = Search._index.titleterms; + const filenames = Search._index.filenames; + const docNames = Search._index.docnames; + const titles = Search._index.titles; + + const scoreMap = new Map(); + const fileMap = new Map(); + + // perform the search on the required terms + searchTerms.forEach((word) => { + const files = []; + const arr = [ + { files: terms[word], score: Scorer.term }, + { files: titleTerms[word], score: Scorer.title }, + ]; + // add support for partial matches + if (word.length > 2) { + const escapedWord = _escapeRegExp(word); + Object.keys(terms).forEach((term) => { + if (term.match(escapedWord) && !terms[word]) + arr.push({ files: terms[term], score: Scorer.partialTerm }); + }); + Object.keys(titleTerms).forEach((term) => { + if (term.match(escapedWord) && !titleTerms[word]) + arr.push({ files: titleTerms[word], score: Scorer.partialTitle }); + }); + } + + // no match but word was a required one + if (arr.every((record) => record.files === undefined)) return; + + // found search word in contents + arr.forEach((record) => { + if (record.files === undefined) return; + + let recordFiles = record.files; + if (recordFiles.length === undefined) recordFiles = [recordFiles]; + files.push(...recordFiles); + + // set score for the word in each file + recordFiles.forEach((file) => { + if (!scoreMap.has(file)) scoreMap.set(file, {}); + scoreMap.get(file)[word] = record.score; + }); + }); + + // create the mapping + files.forEach((file) => { + if (fileMap.has(file) && fileMap.get(file).indexOf(word) === -1) + fileMap.get(file).push(word); + else fileMap.set(file, [word]); + }); + }); + + // now check if the files don't contain excluded terms + const results = []; + for (const [file, wordList] of fileMap) { + // check if all requirements are matched + + // as search terms with length < 3 are discarded + const filteredTermCount = [...searchTerms].filter( + (term) => term.length > 2 + ).length; + if ( + wordList.length !== searchTerms.size && + wordList.length !== filteredTermCount + ) + continue; + + // ensure that none of the excluded terms is in the search result + if ( + [...excludedTerms].some( + (term) => + terms[term] === file || + titleTerms[term] === file || + (terms[term] || []).includes(file) || + (titleTerms[term] || []).includes(file) + ) + ) + break; + + // select one (max) score for the file. + const score = Math.max(...wordList.map((w) => scoreMap.get(file)[w])); + // add result to the result list + results.push([ + docNames[file], + titles[file], + "", + null, + score, + filenames[file], + ]); + } + return results; + }, + + /** + * helper function to return a node containing the + * search summary for a given text. keywords is a list + * of stemmed words. + */ + makeSearchSummary: (htmlText, keywords) => { + const text = Search.htmlToText(htmlText); + if (text === "") return null; + + const textLower = text.toLowerCase(); + const actualStartPosition = [...keywords] + .map((k) => textLower.indexOf(k.toLowerCase())) + .filter((i) => i > -1) + .slice(-1)[0]; + const startWithContext = Math.max(actualStartPosition - 120, 0); + + const top = startWithContext === 0 ? "" : "..."; + const tail = startWithContext + 240 < text.length ? "..." : ""; + + let summary = document.createElement("p"); + summary.classList.add("context"); + summary.textContent = top + text.substr(startWithContext, 240).trim() + tail; + + return summary; + }, +}; + +_ready(Search.init); diff --git a/_static/skeleton.css b/_static/skeleton.css new file mode 100644 index 000000000..467c878c6 --- /dev/null +++ b/_static/skeleton.css @@ -0,0 +1,296 @@ +/* Some sane resets. */ +html { + height: 100%; +} + +body { + margin: 0; + min-height: 100%; +} + +/* All the flexbox magic! */ +body, +.sb-announcement, +.sb-content, +.sb-main, +.sb-container, +.sb-container__inner, +.sb-article-container, +.sb-footer-content, +.sb-header, +.sb-header-secondary, +.sb-footer { + display: flex; +} + +/* These order things vertically */ +body, +.sb-main, +.sb-article-container { + flex-direction: column; +} + +/* Put elements in the center */ +.sb-header, +.sb-header-secondary, +.sb-container, +.sb-content, +.sb-footer, +.sb-footer-content { + justify-content: center; +} +/* Put elements at the ends */ +.sb-article-container { + justify-content: space-between; +} + +/* These elements grow. */ +.sb-main, +.sb-content, +.sb-container, +article { + flex-grow: 1; +} + +/* Because padding making this wider is not fun */ +article { + box-sizing: border-box; +} + +/* The announcements element should never be wider than the page. */ +.sb-announcement { + max-width: 100%; +} + +.sb-sidebar-primary, +.sb-sidebar-secondary { + flex-shrink: 0; + width: 17rem; +} + +.sb-announcement__inner { + justify-content: center; + + box-sizing: border-box; + height: 3rem; + + overflow-x: auto; + white-space: nowrap; +} + +/* Sidebars, with checkbox-based toggle */ +.sb-sidebar-primary, +.sb-sidebar-secondary { + position: fixed; + height: 100%; + top: 0; +} + +.sb-sidebar-primary { + left: -17rem; + transition: left 250ms ease-in-out; +} +.sb-sidebar-secondary { + right: -17rem; + transition: right 250ms ease-in-out; +} + +.sb-sidebar-toggle { + display: none; +} +.sb-sidebar-overlay { + position: fixed; + top: 0; + width: 0; + height: 0; + + transition: width 0ms ease 250ms, height 0ms ease 250ms, opacity 250ms ease; + + opacity: 0; + background-color: rgba(0, 0, 0, 0.54); +} + +#sb-sidebar-toggle--primary:checked + ~ .sb-sidebar-overlay[for="sb-sidebar-toggle--primary"], +#sb-sidebar-toggle--secondary:checked + ~ .sb-sidebar-overlay[for="sb-sidebar-toggle--secondary"] { + width: 100%; + height: 100%; + opacity: 1; + transition: width 0ms ease, height 0ms ease, opacity 250ms ease; +} + +#sb-sidebar-toggle--primary:checked ~ .sb-container .sb-sidebar-primary { + left: 0; +} +#sb-sidebar-toggle--secondary:checked ~ .sb-container .sb-sidebar-secondary { + right: 0; +} + +/* Full-width mode */ +.drop-secondary-sidebar-for-full-width-content + .hide-when-secondary-sidebar-shown { + display: none !important; +} +.drop-secondary-sidebar-for-full-width-content .sb-sidebar-secondary { + display: none !important; +} + +/* Mobile views */ +.sb-page-width { + width: 100%; +} + +.sb-article-container, +.sb-footer-content__inner, +.drop-secondary-sidebar-for-full-width-content .sb-article, +.drop-secondary-sidebar-for-full-width-content .match-content-width { + width: 100vw; +} + +.sb-article, +.match-content-width { + padding: 0 1rem; + box-sizing: border-box; +} + +@media (min-width: 32rem) { + .sb-article, + .match-content-width { + padding: 0 2rem; + } +} + +/* Tablet views */ +@media (min-width: 42rem) { + .sb-article-container { + width: auto; + } + .sb-footer-content__inner, + .drop-secondary-sidebar-for-full-width-content .sb-article, + .drop-secondary-sidebar-for-full-width-content .match-content-width { + width: 42rem; + } + .sb-article, + .match-content-width { + width: 42rem; + } +} +@media (min-width: 46rem) { + .sb-footer-content__inner, + .drop-secondary-sidebar-for-full-width-content .sb-article, + .drop-secondary-sidebar-for-full-width-content .match-content-width { + width: 46rem; + } + .sb-article, + .match-content-width { + width: 46rem; + } +} +@media (min-width: 50rem) { + .sb-footer-content__inner, + .drop-secondary-sidebar-for-full-width-content .sb-article, + .drop-secondary-sidebar-for-full-width-content .match-content-width { + width: 50rem; + } + .sb-article, + .match-content-width { + width: 50rem; + } +} + +/* Tablet views */ +@media (min-width: 59rem) { + .sb-sidebar-secondary { + position: static; + } + .hide-when-secondary-sidebar-shown { + display: none !important; + } + .sb-footer-content__inner, + .drop-secondary-sidebar-for-full-width-content .sb-article, + .drop-secondary-sidebar-for-full-width-content .match-content-width { + width: 59rem; + } + .sb-article, + .match-content-width { + width: 42rem; + } +} +@media (min-width: 63rem) { + .sb-footer-content__inner, + .drop-secondary-sidebar-for-full-width-content .sb-article, + .drop-secondary-sidebar-for-full-width-content .match-content-width { + width: 63rem; + } + .sb-article, + .match-content-width { + width: 46rem; + } +} +@media (min-width: 67rem) { + .sb-footer-content__inner, + .drop-secondary-sidebar-for-full-width-content .sb-article, + .drop-secondary-sidebar-for-full-width-content .match-content-width { + width: 67rem; + } + .sb-article, + .match-content-width { + width: 50rem; + } +} + +/* Desktop views */ +@media (min-width: 76rem) { + .sb-sidebar-primary { + position: static; + } + .hide-when-primary-sidebar-shown { + display: none !important; + } + .sb-footer-content__inner, + .drop-secondary-sidebar-for-full-width-content .sb-article, + .drop-secondary-sidebar-for-full-width-content .match-content-width { + width: 59rem; + } + .sb-article, + .match-content-width { + width: 42rem; + } +} + +/* Full desktop views */ +@media (min-width: 80rem) { + .sb-article, + .match-content-width { + width: 46rem; + } + .sb-footer-content__inner, + .drop-secondary-sidebar-for-full-width-content .sb-article, + .drop-secondary-sidebar-for-full-width-content .match-content-width { + width: 63rem; + } +} + +@media (min-width: 84rem) { + .sb-article, + .match-content-width { + width: 50rem; + } + .sb-footer-content__inner, + .drop-secondary-sidebar-for-full-width-content .sb-article, + .drop-secondary-sidebar-for-full-width-content .match-content-width { + width: 67rem; + } +} + +@media (min-width: 88rem) { + .sb-footer-content__inner, + .drop-secondary-sidebar-for-full-width-content .sb-article, + .drop-secondary-sidebar-for-full-width-content .match-content-width { + width: 67rem; + } + .sb-page-width { + width: 88rem; + } +} diff --git a/_static/sphinx_highlight.js b/_static/sphinx_highlight.js new file mode 100644 index 000000000..8a96c69a1 --- /dev/null +++ b/_static/sphinx_highlight.js @@ -0,0 +1,154 @@ +/* Highlighting utilities for Sphinx HTML documentation. */ +"use strict"; + +const SPHINX_HIGHLIGHT_ENABLED = true + +/** + * highlight a given string on a node by wrapping it in + * span elements with the given class name. + */ +const _highlight = (node, addItems, text, className) => { + if (node.nodeType === Node.TEXT_NODE) { + const val = node.nodeValue; + const parent = node.parentNode; + const pos = val.toLowerCase().indexOf(text); + if ( + pos >= 0 && + !parent.classList.contains(className) && + !parent.classList.contains("nohighlight") + ) { + let span; + + const closestNode = parent.closest("body, svg, foreignObject"); + const isInSVG = closestNode && closestNode.matches("svg"); + if (isInSVG) { + span = document.createElementNS("http://www.w3.org/2000/svg", "tspan"); + } else { + span = document.createElement("span"); + span.classList.add(className); + } + + span.appendChild(document.createTextNode(val.substr(pos, text.length))); + const rest = document.createTextNode(val.substr(pos + text.length)); + parent.insertBefore( + span, + parent.insertBefore( + rest, + node.nextSibling + ) + ); + node.nodeValue = val.substr(0, pos); + /* There may be more occurrences of search term in this node. So call this + * function recursively on the remaining fragment. + */ + _highlight(rest, addItems, text, className); + + if (isInSVG) { + const rect = document.createElementNS( + "http://www.w3.org/2000/svg", + "rect" + ); + const bbox = parent.getBBox(); + rect.x.baseVal.value = bbox.x; + rect.y.baseVal.value = bbox.y; + rect.width.baseVal.value = bbox.width; + rect.height.baseVal.value = bbox.height; + rect.setAttribute("class", className); + addItems.push({ parent: parent, target: rect }); + } + } + } else if (node.matches && !node.matches("button, select, textarea")) { + node.childNodes.forEach((el) => _highlight(el, addItems, text, className)); + } +}; +const _highlightText = (thisNode, text, className) => { + let addItems = []; + _highlight(thisNode, addItems, text, className); + addItems.forEach((obj) => + obj.parent.insertAdjacentElement("beforebegin", obj.target) + ); +}; + +/** + * Small JavaScript module for the documentation. + */ +const SphinxHighlight = { + + /** + * highlight the search words provided in localstorage in the text + */ + highlightSearchWords: () => { + if (!SPHINX_HIGHLIGHT_ENABLED) return; // bail if no highlight + + // get and clear terms from localstorage + const url = new URL(window.location); + const highlight = + localStorage.getItem("sphinx_highlight_terms") + || url.searchParams.get("highlight") + || ""; + localStorage.removeItem("sphinx_highlight_terms") + url.searchParams.delete("highlight"); + window.history.replaceState({}, "", url); + + // get individual terms from highlight string + const terms = highlight.toLowerCase().split(/\s+/).filter(x => x); + if (terms.length === 0) return; // nothing to do + + // There should never be more than one element matching "div.body" + const divBody = document.querySelectorAll("div.body"); + const body = divBody.length ? divBody[0] : document.querySelector("body"); + window.setTimeout(() => { + terms.forEach((term) => _highlightText(body, term, "highlighted")); + }, 10); + + const searchBox = document.getElementById("searchbox"); + if (searchBox === null) return; + searchBox.appendChild( + document + .createRange() + .createContextualFragment( + '" + ) + ); + }, + + /** + * helper function to hide the search marks again + */ + hideSearchWords: () => { + document + .querySelectorAll("#searchbox .highlight-link") + .forEach((el) => el.remove()); + document + .querySelectorAll("span.highlighted") + .forEach((el) => el.classList.remove("highlighted")); + localStorage.removeItem("sphinx_highlight_terms") + }, + + initEscapeListener: () => { + // only install a listener if it is really needed + if (!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.shiftKey || event.altKey || event.ctrlKey || event.metaKey) return; + if (DOCUMENTATION_OPTIONS.ENABLE_SEARCH_SHORTCUTS && (event.key === "Escape")) { + SphinxHighlight.hideSearchWords(); + event.preventDefault(); + } + }); + }, +}; + +_ready(() => { + /* Do not call highlightSearchWords() when we are on the search page. + * It will highlight words from the *previous* search query. + */ + if (typeof Search === "undefined") SphinxHighlight.highlightSearchWords(); + SphinxHighlight.initEscapeListener(); +}); diff --git a/_static/styles/furo-extensions.css b/_static/styles/furo-extensions.css new file mode 100644 index 000000000..bc447f228 --- /dev/null +++ b/_static/styles/furo-extensions.css @@ -0,0 +1,2 @@ +#furo-sidebar-ad-placement{padding:var(--sidebar-item-spacing-vertical) var(--sidebar-item-spacing-horizontal)}#furo-sidebar-ad-placement .ethical-sidebar{background:var(--color-background-secondary);border:none;box-shadow:none}#furo-sidebar-ad-placement .ethical-sidebar:hover{background:var(--color-background-hover)}#furo-sidebar-ad-placement .ethical-sidebar a{color:var(--color-foreground-primary)}#furo-sidebar-ad-placement .ethical-callout a{color:var(--color-foreground-secondary)!important}#furo-readthedocs-versions{background:transparent;display:block;position:static;width:100%}#furo-readthedocs-versions .rst-versions{background:#1a1c1e}#furo-readthedocs-versions .rst-current-version{background:var(--color-sidebar-item-background);cursor:unset}#furo-readthedocs-versions .rst-current-version:hover{background:var(--color-sidebar-item-background)}#furo-readthedocs-versions .rst-current-version .fa-book{color:var(--color-foreground-primary)}#furo-readthedocs-versions>.rst-other-versions{padding:0}#furo-readthedocs-versions>.rst-other-versions small{opacity:1}#furo-readthedocs-versions .injected .rst-versions{position:unset}#furo-readthedocs-versions:focus-within,#furo-readthedocs-versions:hover{box-shadow:0 0 0 1px var(--color-sidebar-background-border)}#furo-readthedocs-versions:focus-within .rst-current-version,#furo-readthedocs-versions:hover .rst-current-version{background:#1a1c1e;font-size:inherit;height:auto;line-height:inherit;padding:12px;text-align:right}#furo-readthedocs-versions:focus-within .rst-current-version .fa-book,#furo-readthedocs-versions:hover .rst-current-version .fa-book{color:#fff;float:left}#furo-readthedocs-versions:focus-within .fa-caret-down,#furo-readthedocs-versions:hover .fa-caret-down{display:none}#furo-readthedocs-versions:focus-within .injected,#furo-readthedocs-versions:focus-within .rst-current-version,#furo-readthedocs-versions:focus-within .rst-other-versions,#furo-readthedocs-versions:hover .injected,#furo-readthedocs-versions:hover .rst-current-version,#furo-readthedocs-versions:hover .rst-other-versions{display:block}#furo-readthedocs-versions:focus-within>.rst-current-version,#furo-readthedocs-versions:hover>.rst-current-version{display:none}.highlight:hover button.copybtn{color:var(--color-code-foreground)}.highlight button.copybtn{align-items:center;background-color:var(--color-code-background);border:none;color:var(--color-background-item);cursor:pointer;height:1.25em;opacity:1;right:.5rem;top:.625rem;transition:color .3s,opacity .3s;width:1.25em}.highlight button.copybtn:hover{background-color:var(--color-code-background);color:var(--color-brand-content)}.highlight button.copybtn:after{background-color:transparent;color:var(--color-code-foreground);display:none}.highlight button.copybtn.success{color:#22863a;transition:color 0ms}.highlight button.copybtn.success:after{display:block}.highlight button.copybtn svg{padding:0}body{--sd-color-primary:var(--color-brand-primary);--sd-color-primary-highlight:var(--color-brand-content);--sd-color-primary-text:var(--color-background-primary);--sd-color-shadow:rgba(0,0,0,.05);--sd-color-card-border:var(--color-card-border);--sd-color-card-border-hover:var(--color-brand-content);--sd-color-card-background:var(--color-card-background);--sd-color-card-text:var(--color-foreground-primary);--sd-color-card-header:var(--color-card-marginals-background);--sd-color-card-footer:var(--color-card-marginals-background);--sd-color-tabs-label-active:var(--color-brand-content);--sd-color-tabs-label-hover:var(--color-foreground-muted);--sd-color-tabs-label-inactive:var(--color-foreground-muted);--sd-color-tabs-underline-active:var(--color-brand-content);--sd-color-tabs-underline-hover:var(--color-foreground-border);--sd-color-tabs-underline-inactive:var(--color-background-border);--sd-color-tabs-overline:var(--color-background-border);--sd-color-tabs-underline:var(--color-background-border)}.sd-tab-content{box-shadow:0 -2px var(--sd-color-tabs-overline),0 1px var(--sd-color-tabs-underline)}.sd-card{box-shadow:0 .1rem .25rem var(--sd-color-shadow),0 0 .0625rem rgba(0,0,0,.1)}.sd-shadow-sm{box-shadow:0 .1rem .25rem var(--sd-color-shadow),0 0 .0625rem rgba(0,0,0,.1)!important}.sd-shadow-md{box-shadow:0 .3rem .75rem var(--sd-color-shadow),0 0 .0625rem rgba(0,0,0,.1)!important}.sd-shadow-lg{box-shadow:0 .6rem 1.5rem var(--sd-color-shadow),0 0 .0625rem rgba(0,0,0,.1)!important}.sd-card-hover:hover{transform:none}.sd-cards-carousel{gap:.25rem;padding:.25rem}body{--tabs--label-text:var(--color-foreground-muted);--tabs--label-text--hover:var(--color-foreground-muted);--tabs--label-text--active:var(--color-brand-content);--tabs--label-text--active--hover:var(--color-brand-content);--tabs--label-background:transparent;--tabs--label-background--hover:transparent;--tabs--label-background--active:transparent;--tabs--label-background--active--hover:transparent;--tabs--padding-x:0.25em;--tabs--margin-x:1em;--tabs--border:var(--color-background-border);--tabs--label-border:transparent;--tabs--label-border--hover:var(--color-foreground-muted);--tabs--label-border--active:var(--color-brand-content);--tabs--label-border--active--hover:var(--color-brand-content)}[role=main] .container{max-width:none;padding-left:0;padding-right:0}.shadow.docutils{border:none;box-shadow:0 .2rem .5rem rgba(0,0,0,.05),0 0 .0625rem rgba(0,0,0,.1)!important}.sphinx-bs .card{background-color:var(--color-background-secondary);color:var(--color-foreground)} +/*# sourceMappingURL=furo-extensions.css.map*/ \ No newline at end of file diff --git a/_static/styles/furo-extensions.css.map b/_static/styles/furo-extensions.css.map new file mode 100644 index 000000000..9ba5637f9 --- /dev/null +++ b/_static/styles/furo-extensions.css.map @@ -0,0 +1 @@ +{"version":3,"file":"styles/furo-extensions.css","mappings":"AAGA,2BACE,oFACA,4CAKE,6CAHA,YACA,eAEA,CACA,kDACE,yCAEF,8CACE,sCAEJ,8CACE,kDAEJ,2BAGE,uBACA,cAHA,gBACA,UAEA,CAGA,yCACE,mBAEF,gDAEE,gDADA,YACA,CACA,sDACE,gDACF,yDACE,sCAEJ,+CACE,UACA,qDACE,UAGF,mDACE,eAEJ,yEAEE,4DAEA,mHASE,mBAPA,kBAEA,YADA,oBAGA,aADA,gBAIA,CAEA,qIAEE,WADA,UACA,CAEJ,uGACE,aAEF,iUAGE,cAEF,mHACE,aC1EJ,gCACE,mCAEF,0BAKE,mBAUA,8CACA,YAFA,mCAKA,eAZA,cALA,UASA,YADA,YAYA,iCAdA,YAcA,CAEA,gCAEE,8CADA,gCACA,CAEF,gCAGE,6BADA,mCADA,YAEA,CAEF,kCAEE,cADA,oBACA,CACA,wCACE,cAEJ,8BACE,UC5CN,KAEE,6CAA8C,CAC9C,uDAAwD,CACxD,uDAAwD,CAGxD,iCAAsC,CAGtC,+CAAgD,CAChD,uDAAwD,CACxD,uDAAwD,CACxD,oDAAqD,CACrD,6DAA8D,CAC9D,6DAA8D,CAG9D,uDAAwD,CACxD,yDAA0D,CAC1D,4DAA6D,CAC7D,2DAA4D,CAC5D,8DAA+D,CAC/D,iEAAkE,CAClE,uDAAwD,CACxD,wDAAyD,CAG3D,gBACE,qFAGF,SACE,6EAEF,cACE,uFAEF,cACE,uFAEF,cACE,uFAGF,qBACE,eAEF,mBACE,WACA,eChDF,KACE,gDAAiD,CACjD,uDAAwD,CACxD,qDAAsD,CACtD,4DAA6D,CAC7D,oCAAqC,CACrC,2CAA4C,CAC5C,4CAA6C,CAC7C,mDAAoD,CACpD,wBAAyB,CACzB,oBAAqB,CACrB,6CAA8C,CAC9C,gCAAiC,CACjC,yDAA0D,CAC1D,uDAAwD,CACxD,8DAA+D,CCbjE,uBACE,eACA,eACA,gBAGF,iBACE,YACA,+EAGF,iBACE,mDACA","sources":["webpack:///./src/furo/assets/styles/extensions/_readthedocs.sass","webpack:///./src/furo/assets/styles/extensions/_copybutton.sass","webpack:///./src/furo/assets/styles/extensions/_sphinx-design.sass","webpack:///./src/furo/assets/styles/extensions/_sphinx-inline-tabs.sass","webpack:///./src/furo/assets/styles/extensions/_sphinx-panels.sass"],"sourcesContent":["// This file contains the styles used for tweaking how ReadTheDoc's embedded\n// contents would show up inside the theme.\n\n#furo-sidebar-ad-placement\n padding: var(--sidebar-item-spacing-vertical) var(--sidebar-item-spacing-horizontal)\n .ethical-sidebar\n // Remove the border and box-shadow.\n border: none\n box-shadow: none\n // Manage the background colors.\n background: var(--color-background-secondary)\n &:hover\n background: var(--color-background-hover)\n // Ensure the text is legible.\n a\n color: var(--color-foreground-primary)\n\n .ethical-callout a\n color: var(--color-foreground-secondary) !important\n\n#furo-readthedocs-versions\n position: static\n width: 100%\n background: transparent\n display: block\n\n // Make the background color fit with the theme's aesthetic.\n .rst-versions\n background: rgb(26, 28, 30)\n\n .rst-current-version\n cursor: unset\n background: var(--color-sidebar-item-background)\n &:hover\n background: var(--color-sidebar-item-background)\n .fa-book\n color: var(--color-foreground-primary)\n\n > .rst-other-versions\n padding: 0\n small\n opacity: 1\n\n .injected\n .rst-versions\n position: unset\n\n &:hover,\n &:focus-within\n box-shadow: 0 0 0 1px var(--color-sidebar-background-border)\n\n .rst-current-version\n // Undo the tweaks done in RTD's CSS\n font-size: inherit\n line-height: inherit\n height: auto\n text-align: right\n padding: 12px\n\n // Match the rest of the body\n background: #1a1c1e\n\n .fa-book\n float: left\n color: white\n\n .fa-caret-down\n display: none\n\n .rst-current-version,\n .rst-other-versions,\n .injected\n display: block\n\n > .rst-current-version\n display: none\n",".highlight\n &:hover button.copybtn\n color: var(--color-code-foreground)\n\n button.copybtn\n // Make it visible\n opacity: 1\n\n // Align things correctly\n align-items: center\n\n height: 1.25em\n width: 1.25em\n\n top: 0.625rem // $code-spacing-vertical\n right: 0.5rem\n\n // Make it look better\n color: var(--color-background-item)\n background-color: var(--color-code-background)\n border: none\n\n // Change to cursor to make it obvious that you can click on it\n cursor: pointer\n\n // Transition smoothly, for aesthetics\n transition: color 300ms, opacity 300ms\n\n &:hover\n color: var(--color-brand-content)\n background-color: var(--color-code-background)\n\n &::after\n display: none\n color: var(--color-code-foreground)\n background-color: transparent\n\n &.success\n transition: color 0ms\n color: #22863a\n &::after\n display: block\n\n svg\n padding: 0\n","body\n // Colors\n --sd-color-primary: var(--color-brand-primary)\n --sd-color-primary-highlight: var(--color-brand-content)\n --sd-color-primary-text: var(--color-background-primary)\n\n // Shadows\n --sd-color-shadow: rgba(0, 0, 0, 0.05)\n\n // Cards\n --sd-color-card-border: var(--color-card-border)\n --sd-color-card-border-hover: var(--color-brand-content)\n --sd-color-card-background: var(--color-card-background)\n --sd-color-card-text: var(--color-foreground-primary)\n --sd-color-card-header: var(--color-card-marginals-background)\n --sd-color-card-footer: var(--color-card-marginals-background)\n\n // Tabs\n --sd-color-tabs-label-active: var(--color-brand-content)\n --sd-color-tabs-label-hover: var(--color-foreground-muted)\n --sd-color-tabs-label-inactive: var(--color-foreground-muted)\n --sd-color-tabs-underline-active: var(--color-brand-content)\n --sd-color-tabs-underline-hover: var(--color-foreground-border)\n --sd-color-tabs-underline-inactive: var(--color-background-border)\n --sd-color-tabs-overline: var(--color-background-border)\n --sd-color-tabs-underline: var(--color-background-border)\n\n// Tabs\n.sd-tab-content\n box-shadow: 0 -2px var(--sd-color-tabs-overline), 0 1px var(--sd-color-tabs-underline)\n\n// Shadows\n.sd-card // Have a shadow by default\n box-shadow: 0 0.1rem 0.25rem var(--sd-color-shadow), 0 0 0.0625rem rgba(0, 0, 0, 0.1)\n\n.sd-shadow-sm\n box-shadow: 0 0.1rem 0.25rem var(--sd-color-shadow), 0 0 0.0625rem rgba(0, 0, 0, 0.1) !important\n\n.sd-shadow-md\n box-shadow: 0 0.3rem 0.75rem var(--sd-color-shadow), 0 0 0.0625rem rgba(0, 0, 0, 0.1) !important\n\n.sd-shadow-lg\n box-shadow: 0 0.6rem 1.5rem var(--sd-color-shadow), 0 0 0.0625rem rgba(0, 0, 0, 0.1) !important\n\n// Cards\n.sd-card-hover:hover // Don't change scale on hover\n transform: none\n\n.sd-cards-carousel // Have a bit of gap in the carousel by default\n gap: 0.25rem\n padding: 0.25rem\n","// This file contains styles to tweak sphinx-inline-tabs to work well with Furo.\n\nbody\n --tabs--label-text: var(--color-foreground-muted)\n --tabs--label-text--hover: var(--color-foreground-muted)\n --tabs--label-text--active: var(--color-brand-content)\n --tabs--label-text--active--hover: var(--color-brand-content)\n --tabs--label-background: transparent\n --tabs--label-background--hover: transparent\n --tabs--label-background--active: transparent\n --tabs--label-background--active--hover: transparent\n --tabs--padding-x: 0.25em\n --tabs--margin-x: 1em\n --tabs--border: var(--color-background-border)\n --tabs--label-border: transparent\n --tabs--label-border--hover: var(--color-foreground-muted)\n --tabs--label-border--active: var(--color-brand-content)\n --tabs--label-border--active--hover: var(--color-brand-content)\n","// This file contains styles to tweak sphinx-panels to work well with Furo.\n\n// sphinx-panels includes Bootstrap 4, which uses .container which can conflict\n// with docutils' `.. container::` directive.\n[role=\"main\"] .container\n max-width: initial\n padding-left: initial\n padding-right: initial\n\n// Make the panels look nicer!\n.shadow.docutils\n border: none\n box-shadow: 0 0.2rem 0.5rem rgba(0, 0, 0, 0.05), 0 0 0.0625rem rgba(0, 0, 0, 0.1) !important\n\n// Make panel colors respond to dark mode\n.sphinx-bs .card\n background-color: var(--color-background-secondary)\n color: var(--color-foreground)\n"],"names":[],"sourceRoot":""} \ No newline at end of file diff --git a/_static/styles/furo.css b/_static/styles/furo.css new file mode 100644 index 000000000..3d29a218f --- /dev/null +++ b/_static/styles/furo.css @@ -0,0 +1,2 @@ +/*! normalize.css v8.0.1 | MIT License | github.com/necolas/normalize.css */html{-webkit-text-size-adjust:100%;line-height:1.15}body{margin:0}main{display:block}h1{font-size:2em;margin:.67em 0}hr{box-sizing:content-box;height:0;overflow:visible}pre{font-family:monospace,monospace;font-size:1em}a{background-color:transparent}abbr[title]{border-bottom:none;text-decoration:underline;text-decoration:underline dotted}b,strong{font-weight:bolder}code,kbd,samp{font-family:monospace,monospace;font-size:1em}sub,sup{font-size:75%;line-height:0;position:relative;vertical-align:baseline}sub{bottom:-.25em}sup{top:-.5em}img{border-style:none}button,input,optgroup,select,textarea{font-family:inherit;font-size:100%;line-height:1.15;margin:0}button,input{overflow:visible}button,select{text-transform:none}[type=button],[type=reset],[type=submit],button{-webkit-appearance:button}[type=button]::-moz-focus-inner,[type=reset]::-moz-focus-inner,[type=submit]::-moz-focus-inner,button::-moz-focus-inner{border-style:none;padding:0}[type=button]:-moz-focusring,[type=reset]:-moz-focusring,[type=submit]:-moz-focusring,button:-moz-focusring{outline:1px dotted ButtonText}fieldset{padding:.35em .75em .625em}legend{box-sizing:border-box;color:inherit;display:table;max-width:100%;padding:0;white-space:normal}progress{vertical-align:baseline}textarea{overflow:auto}[type=checkbox],[type=radio]{box-sizing:border-box;padding:0}[type=number]::-webkit-inner-spin-button,[type=number]::-webkit-outer-spin-button{height:auto}[type=search]{-webkit-appearance:textfield;outline-offset:-2px}[type=search]::-webkit-search-decoration{-webkit-appearance:none}::-webkit-file-upload-button{-webkit-appearance:button;font:inherit}details{display:block}summary{display:list-item}[hidden],template{display:none}@media print{.content-icon-container,.headerlink,.mobile-header,.related-pages{display:none!important}.highlight{border:.1pt solid var(--color-foreground-border)}a,blockquote,dl,ol,pre,table,ul{page-break-inside:avoid}caption,figure,h1,h2,h3,h4,h5,h6,img{page-break-after:avoid;page-break-inside:avoid}dl,ol,ul{page-break-before:avoid}}.visually-hidden{clip:rect(0,0,0,0)!important;border:0!important;height:1px!important;margin:-1px!important;overflow:hidden!important;padding:0!important;position:absolute!important;white-space:nowrap!important;width:1px!important}:-moz-focusring{outline:auto}body{--font-stack:-apple-system,BlinkMacSystemFont,Segoe UI,Helvetica,Arial,sans-serif,Apple Color Emoji,Segoe UI Emoji;--font-stack--monospace:"SFMono-Regular",Menlo,Consolas,Monaco,Liberation Mono,Lucida Console,monospace;--font-size--normal:100%;--font-size--small:87.5%;--font-size--small--2:81.25%;--font-size--small--3:75%;--font-size--small--4:62.5%;--sidebar-caption-font-size:var(--font-size--small--2);--sidebar-item-font-size:var(--font-size--small);--sidebar-search-input-font-size:var(--font-size--small);--toc-font-size:var(--font-size--small--3);--toc-font-size--mobile:var(--font-size--normal);--toc-title-font-size:var(--font-size--small--4);--admonition-font-size:0.8125rem;--admonition-title-font-size:0.8125rem;--code-font-size:var(--font-size--small--2);--api-font-size:var(--font-size--small);--header-height:calc(var(--sidebar-item-line-height) + var(--sidebar-item-spacing-vertical)*4);--header-padding:0.5rem;--sidebar-tree-space-above:1.5rem;--sidebar-caption-space-above:1rem;--sidebar-item-line-height:1rem;--sidebar-item-spacing-vertical:0.5rem;--sidebar-item-spacing-horizontal:1rem;--sidebar-item-height:calc(var(--sidebar-item-line-height) + var(--sidebar-item-spacing-vertical)*2);--sidebar-expander-width:var(--sidebar-item-height);--sidebar-search-space-above:0.5rem;--sidebar-search-input-spacing-vertical:0.5rem;--sidebar-search-input-spacing-horizontal:0.5rem;--sidebar-search-input-height:1rem;--sidebar-search-icon-size:var(--sidebar-search-input-height);--toc-title-padding:0.25rem 0;--toc-spacing-vertical:1.5rem;--toc-spacing-horizontal:1.5rem;--toc-item-spacing-vertical:0.4rem;--toc-item-spacing-horizontal:1rem;--icon-search:url('data:image/svg+xml;charset=utf-8,');--icon-pencil:url('data:image/svg+xml;charset=utf-8,');--icon-abstract:url('data:image/svg+xml;charset=utf-8,');--icon-info:url('data:image/svg+xml;charset=utf-8,');--icon-flame:url('data:image/svg+xml;charset=utf-8,');--icon-question:url('data:image/svg+xml;charset=utf-8,');--icon-warning:url('data:image/svg+xml;charset=utf-8,');--icon-failure:url('data:image/svg+xml;charset=utf-8,');--icon-spark:url('data:image/svg+xml;charset=utf-8,');--color-admonition-title--caution:#ff9100;--color-admonition-title-background--caution:rgba(255,145,0,.2);--color-admonition-title--warning:#ff9100;--color-admonition-title-background--warning:rgba(255,145,0,.2);--color-admonition-title--danger:#ff5252;--color-admonition-title-background--danger:rgba(255,82,82,.2);--color-admonition-title--attention:#ff5252;--color-admonition-title-background--attention:rgba(255,82,82,.2);--color-admonition-title--error:#ff5252;--color-admonition-title-background--error:rgba(255,82,82,.2);--color-admonition-title--hint:#00c852;--color-admonition-title-background--hint:rgba(0,200,82,.2);--color-admonition-title--tip:#00c852;--color-admonition-title-background--tip:rgba(0,200,82,.2);--color-admonition-title--important:#00bfa5;--color-admonition-title-background--important:rgba(0,191,165,.2);--color-admonition-title--note:#00b0ff;--color-admonition-title-background--note:rgba(0,176,255,.2);--color-admonition-title--seealso:#448aff;--color-admonition-title-background--seealso:rgba(68,138,255,.2);--color-admonition-title--admonition-todo:grey;--color-admonition-title-background--admonition-todo:hsla(0,0%,50%,.2);--color-admonition-title:#651fff;--color-admonition-title-background:rgba(101,31,255,.2);--icon-admonition-default:var(--icon-abstract);--color-topic-title:#14b8a6;--color-topic-title-background:rgba(20,184,166,.2);--icon-topic-default:var(--icon-pencil);--color-problematic:#b30000;--color-foreground-primary:#000;--color-foreground-secondary:#5a5c63;--color-foreground-muted:#646776;--color-foreground-border:#878787;--color-background-primary:#fff;--color-background-secondary:#f8f9fb;--color-background-hover:#efeff4;--color-background-hover--transparent:#efeff400;--color-background-border:#eeebee;--color-background-item:#ccc;--color-announcement-background:#000000dd;--color-announcement-text:#eeebee;--color-brand-primary:#2962ff;--color-brand-content:#2a5adf;--color-api-background:var(--color-background-hover--transparent);--color-api-background-hover:var(--color-background-hover);--color-api-overall:var(--color-foreground-secondary);--color-api-name:var(--color-problematic);--color-api-pre-name:var(--color-problematic);--color-api-paren:var(--color-foreground-secondary);--color-api-keyword:var(--color-foreground-primary);--color-highlight-on-target:#ffc;--color-inline-code-background:var(--color-background-secondary);--color-highlighted-background:#def;--color-highlighted-text:var(--color-foreground-primary);--color-guilabel-background:#ddeeff80;--color-guilabel-border:#bedaf580;--color-guilabel-text:var(--color-foreground-primary);--color-admonition-background:transparent;--color-table-header-background:var(--color-background-secondary);--color-table-border:var(--color-background-border);--color-card-border:var(--color-background-secondary);--color-card-background:transparent;--color-card-marginals-background:var(--color-background-secondary);--color-header-background:var(--color-background-primary);--color-header-border:var(--color-background-border);--color-header-text:var(--color-foreground-primary);--color-sidebar-background:var(--color-background-secondary);--color-sidebar-background-border:var(--color-background-border);--color-sidebar-brand-text:var(--color-foreground-primary);--color-sidebar-caption-text:var(--color-foreground-muted);--color-sidebar-link-text:var(--color-foreground-secondary);--color-sidebar-link-text--top-level:var(--color-brand-primary);--color-sidebar-item-background:var(--color-sidebar-background);--color-sidebar-item-background--current:var( --color-sidebar-item-background );--color-sidebar-item-background--hover:linear-gradient(90deg,var(--color-background-hover--transparent) 0%,var(--color-background-hover) var(--sidebar-item-spacing-horizontal),var(--color-background-hover) 100%);--color-sidebar-item-expander-background:transparent;--color-sidebar-item-expander-background--hover:var( --color-background-hover );--color-sidebar-search-text:var(--color-foreground-primary);--color-sidebar-search-background:var(--color-background-secondary);--color-sidebar-search-background--focus:var(--color-background-primary);--color-sidebar-search-border:var(--color-background-border);--color-sidebar-search-icon:var(--color-foreground-muted);--color-toc-background:var(--color-background-primary);--color-toc-title-text:var(--color-foreground-muted);--color-toc-item-text:var(--color-foreground-secondary);--color-toc-item-text--hover:var(--color-foreground-primary);--color-toc-item-text--active:var(--color-brand-primary);--color-content-foreground:var(--color-foreground-primary);--color-content-background:transparent;--color-link:var(--color-brand-content);--color-link--hover:var(--color-brand-content);--color-link-underline:var(--color-background-border);--color-link-underline--hover:var(--color-foreground-border)}.only-light{display:block!important}html body .only-dark{display:none!important}@media not print{body[data-theme=dark]{--color-problematic:#ee5151;--color-foreground-primary:#ffffffcc;--color-foreground-secondary:#9ca0a5;--color-foreground-muted:#81868d;--color-foreground-border:#666;--color-background-primary:#131416;--color-background-secondary:#1a1c1e;--color-background-hover:#1e2124;--color-background-hover--transparent:#1e212400;--color-background-border:#303335;--color-background-item:#444;--color-announcement-background:#000000dd;--color-announcement-text:#eeebee;--color-brand-primary:#2b8cee;--color-brand-content:#368ce2;--color-highlighted-background:#083563;--color-guilabel-background:#08356380;--color-guilabel-border:#13395f80;--color-api-keyword:var(--color-foreground-secondary);--color-highlight-on-target:#330;--color-admonition-background:#18181a;--color-card-border:var(--color-background-secondary);--color-card-background:#18181a;--color-card-marginals-background:var(--color-background-hover)}html body[data-theme=dark] .only-light{display:none!important}body[data-theme=dark] .only-dark{display:block!important}@media(prefers-color-scheme:dark){body:not([data-theme=light]){--color-problematic:#ee5151;--color-foreground-primary:#ffffffcc;--color-foreground-secondary:#9ca0a5;--color-foreground-muted:#81868d;--color-foreground-border:#666;--color-background-primary:#131416;--color-background-secondary:#1a1c1e;--color-background-hover:#1e2124;--color-background-hover--transparent:#1e212400;--color-background-border:#303335;--color-background-item:#444;--color-announcement-background:#000000dd;--color-announcement-text:#eeebee;--color-brand-primary:#2b8cee;--color-brand-content:#368ce2;--color-highlighted-background:#083563;--color-guilabel-background:#08356380;--color-guilabel-border:#13395f80;--color-api-keyword:var(--color-foreground-secondary);--color-highlight-on-target:#330;--color-admonition-background:#18181a;--color-card-border:var(--color-background-secondary);--color-card-background:#18181a;--color-card-marginals-background:var(--color-background-hover)}html body:not([data-theme=light]) .only-light{display:none!important}body:not([data-theme=light]) .only-dark{display:block!important}}}body[data-theme=auto] .theme-toggle svg.theme-icon-when-auto,body[data-theme=dark] .theme-toggle svg.theme-icon-when-dark,body[data-theme=light] .theme-toggle svg.theme-icon-when-light{display:block}body{font-family:var(--font-stack)}code,kbd,pre,samp{font-family:var(--font-stack--monospace)}body{-webkit-font-smoothing:antialiased;-moz-osx-font-smoothing:grayscale}article{line-height:1.5}h1,h2,h3,h4,h5,h6{border-radius:.5rem;font-weight:700;line-height:1.25;margin:.5rem -.5rem;padding-left:.5rem;padding-right:.5rem}h1+p,h2+p,h3+p,h4+p,h5+p,h6+p{margin-top:0}h1{font-size:2.5em;margin-bottom:1rem}h1,h2{margin-top:1.75rem}h2{font-size:2em}h3{font-size:1.5em}h4{font-size:1.25em}h5{font-size:1.125em}h6{font-size:1em}small{font-size:80%;opacity:75%}p{margin-bottom:.75rem;margin-top:.5rem}hr.docutils{background-color:var(--color-background-border);border:0;height:1px;margin:2rem 0;padding:0}.centered{text-align:center}a{color:var(--color-link);text-decoration:underline;text-decoration-color:var(--color-link-underline)}a:hover{color:var(--color-link--hover);text-decoration-color:var(--color-link-underline--hover)}a.muted-link{color:inherit}a.muted-link:hover{color:var(--color-link);text-decoration-color:var(--color-link-underline--hover)}html{overflow-x:hidden;overflow-y:scroll;scroll-behavior:smooth}.sidebar-scroll,.toc-scroll,article[role=main] *{scrollbar-color:var(--color-foreground-border) transparent;scrollbar-width:thin}.sidebar-scroll::-webkit-scrollbar,.toc-scroll::-webkit-scrollbar,article[role=main] ::-webkit-scrollbar{height:.25rem;width:.25rem}.sidebar-scroll::-webkit-scrollbar-thumb,.toc-scroll::-webkit-scrollbar-thumb,article[role=main] ::-webkit-scrollbar-thumb{background-color:var(--color-foreground-border);border-radius:.125rem}body,html{background:var(--color-background-primary);color:var(--color-foreground-primary);height:100%}article{background:var(--color-content-background);color:var(--color-content-foreground);overflow-wrap:break-word}.page{display:flex;min-height:100%}.mobile-header{background-color:var(--color-header-background);border-bottom:1px solid var(--color-header-border);color:var(--color-header-text);display:none;height:var(--header-height);width:100%;z-index:10}.mobile-header.scrolled{border-bottom:none;box-shadow:0 0 .2rem rgba(0,0,0,.1),0 .2rem .4rem rgba(0,0,0,.2)}.mobile-header .header-center a{color:var(--color-header-text);text-decoration:none}.main{display:flex;flex:1}.sidebar-drawer{background:var(--color-sidebar-background);border-right:1px solid var(--color-sidebar-background-border);box-sizing:border-box;display:flex;justify-content:flex-end;min-width:15em;width:calc(50% - 26em)}.sidebar-container,.toc-drawer{box-sizing:border-box;width:15em}.toc-drawer{background:var(--color-toc-background);padding-right:1rem}.sidebar-sticky,.toc-sticky{display:flex;flex-direction:column;height:min(100%,100vh);height:100vh;position:sticky;top:0}.sidebar-scroll,.toc-scroll{flex-grow:1;flex-shrink:1;overflow:auto;scroll-behavior:smooth}.content{display:flex;flex-direction:column;justify-content:space-between;padding:0 3em;width:46em}.icon{display:inline-block;height:1rem;width:1rem}.icon svg{height:100%;width:100%}.announcement{align-items:center;background-color:var(--color-announcement-background);color:var(--color-announcement-text);display:flex;height:var(--header-height);overflow-x:auto}.announcement+.page{min-height:calc(100% - var(--header-height))}.announcement-content{box-sizing:border-box;min-width:100%;padding:.5rem;text-align:center;white-space:nowrap}.announcement-content a{color:var(--color-announcement-text);text-decoration-color:var(--color-announcement-text)}.announcement-content a:hover{color:var(--color-announcement-text);text-decoration-color:var(--color-link--hover)}.no-js .theme-toggle-container{display:none}.theme-toggle-container{vertical-align:middle}.theme-toggle{background:transparent;border:none;cursor:pointer;padding:0}.theme-toggle svg{color:var(--color-foreground-primary);display:none;height:1rem;vertical-align:middle;width:1rem}.theme-toggle-header{float:left;padding:1rem .5rem}.nav-overlay-icon,.toc-overlay-icon{cursor:pointer;display:none}.nav-overlay-icon .icon,.toc-overlay-icon .icon{color:var(--color-foreground-secondary);height:1rem;width:1rem}.nav-overlay-icon,.toc-header-icon{align-items:center;justify-content:center}.toc-content-icon{height:1.5rem;width:1.5rem}.content-icon-container{display:flex;float:right;gap:.5rem;margin-bottom:1rem;margin-left:1rem;margin-top:1.5rem}.content-icon-container .edit-this-page svg{color:inherit;height:1rem;width:1rem}.sidebar-toggle{display:none;position:absolute}.sidebar-toggle[name=__toc]{left:20px}.sidebar-toggle:checked{left:40px}.overlay{background-color:rgba(0,0,0,.54);height:0;opacity:0;position:fixed;top:0;transition:width 0ms,height 0ms,opacity .25s ease-out;width:0}.sidebar-overlay{z-index:20}.toc-overlay{z-index:40}.sidebar-drawer{transition:left .25s ease-in-out;z-index:30}.toc-drawer{transition:right .25s ease-in-out;z-index:50}#__navigation:checked~.sidebar-overlay{height:100%;opacity:1;width:100%}#__navigation:checked~.page .sidebar-drawer{left:0;top:0}#__toc:checked~.toc-overlay{height:100%;opacity:1;width:100%}#__toc:checked~.page .toc-drawer{right:0;top:0}.back-to-top{background:var(--color-background-primary);border-radius:1rem;box-shadow:0 .2rem .5rem rgba(0,0,0,.05),0 0 1px 0 hsla(220,9%,46%,.502);display:none;font-size:.8125rem;left:0;margin-left:50%;padding:.5rem .75rem .5rem .5rem;position:fixed;text-decoration:none;top:1rem;transform:translateX(-50%);z-index:10}.back-to-top svg{fill:currentColor;display:inline-block;height:1rem;width:1rem}.back-to-top span{margin-left:.25rem}.show-back-to-top .back-to-top{align-items:center;display:flex}@media(min-width:97em){html{font-size:110%}}@media(max-width:82em){.toc-content-icon{display:flex}.toc-drawer{border-left:1px solid var(--color-background-muted);height:100vh;position:fixed;right:-15em;top:0}.toc-tree{border-left:none;font-size:var(--toc-font-size--mobile)}.sidebar-drawer{width:calc(50% - 18.5em)}}@media(max-width:67em){.nav-overlay-icon{display:flex}.sidebar-drawer{height:100vh;left:-15em;position:fixed;top:0;width:15em}.toc-header-icon{display:flex}.theme-toggle-content,.toc-content-icon{display:none}.theme-toggle-header{display:block}.mobile-header{align-items:center;display:flex;justify-content:space-between;position:sticky;top:0}.mobile-header .header-left,.mobile-header .header-right{display:flex;height:var(--header-height);padding:0 var(--header-padding)}.mobile-header .header-left label,.mobile-header .header-right label{height:100%;-webkit-user-select:none;-moz-user-select:none;user-select:none;width:100%}.nav-overlay-icon .icon,.theme-toggle svg{height:1.25rem;width:1.25rem}:target{scroll-margin-top:var(--header-height)}.back-to-top{top:calc(var(--header-height) + .5rem)}.page{flex-direction:column;justify-content:center}.content{margin-left:auto;margin-right:auto}}@media(max-width:52em){.content{overflow-x:auto;width:100%}}@media(max-width:46em){.content{padding:0 1em}article aside.sidebar{float:none;margin:1rem 0;width:100%}}.admonition,.topic{background:var(--color-admonition-background);border-radius:.2rem;box-shadow:0 .2rem .5rem rgba(0,0,0,.05),0 0 .0625rem rgba(0,0,0,.1);font-size:var(--admonition-font-size);margin:1rem auto;overflow:hidden;padding:0 .5rem .5rem;page-break-inside:avoid}.admonition>:nth-child(2),.topic>:nth-child(2){margin-top:0}.admonition>:last-child,.topic>:last-child{margin-bottom:0}.admonition p.admonition-title,p.topic-title{font-size:var(--admonition-title-font-size);font-weight:500;line-height:1.3;margin:0 -.5rem .5rem;padding:.4rem .5rem .4rem 2rem;position:relative}.admonition p.admonition-title:before,p.topic-title:before{content:"";height:1rem;left:.5rem;position:absolute;width:1rem}p.admonition-title{background-color:var(--color-admonition-title-background)}p.admonition-title:before{background-color:var(--color-admonition-title);-webkit-mask-image:var(--icon-admonition-default);mask-image:var(--icon-admonition-default);-webkit-mask-repeat:no-repeat;mask-repeat:no-repeat}p.topic-title{background-color:var(--color-topic-title-background)}p.topic-title:before{background-color:var(--color-topic-title);-webkit-mask-image:var(--icon-topic-default);mask-image:var(--icon-topic-default);-webkit-mask-repeat:no-repeat;mask-repeat:no-repeat}.admonition{border-left:.2rem solid var(--color-admonition-title)}.admonition.caution{border-left-color:var(--color-admonition-title--caution)}.admonition.caution>.admonition-title{background-color:var(--color-admonition-title-background--caution)}.admonition.caution>.admonition-title:before{background-color:var(--color-admonition-title--caution);-webkit-mask-image:var(--icon-spark);mask-image:var(--icon-spark)}.admonition.warning{border-left-color:var(--color-admonition-title--warning)}.admonition.warning>.admonition-title{background-color:var(--color-admonition-title-background--warning)}.admonition.warning>.admonition-title:before{background-color:var(--color-admonition-title--warning);-webkit-mask-image:var(--icon-warning);mask-image:var(--icon-warning)}.admonition.danger{border-left-color:var(--color-admonition-title--danger)}.admonition.danger>.admonition-title{background-color:var(--color-admonition-title-background--danger)}.admonition.danger>.admonition-title:before{background-color:var(--color-admonition-title--danger);-webkit-mask-image:var(--icon-spark);mask-image:var(--icon-spark)}.admonition.attention{border-left-color:var(--color-admonition-title--attention)}.admonition.attention>.admonition-title{background-color:var(--color-admonition-title-background--attention)}.admonition.attention>.admonition-title:before{background-color:var(--color-admonition-title--attention);-webkit-mask-image:var(--icon-warning);mask-image:var(--icon-warning)}.admonition.error{border-left-color:var(--color-admonition-title--error)}.admonition.error>.admonition-title{background-color:var(--color-admonition-title-background--error)}.admonition.error>.admonition-title:before{background-color:var(--color-admonition-title--error);-webkit-mask-image:var(--icon-failure);mask-image:var(--icon-failure)}.admonition.hint{border-left-color:var(--color-admonition-title--hint)}.admonition.hint>.admonition-title{background-color:var(--color-admonition-title-background--hint)}.admonition.hint>.admonition-title:before{background-color:var(--color-admonition-title--hint);-webkit-mask-image:var(--icon-question);mask-image:var(--icon-question)}.admonition.tip{border-left-color:var(--color-admonition-title--tip)}.admonition.tip>.admonition-title{background-color:var(--color-admonition-title-background--tip)}.admonition.tip>.admonition-title:before{background-color:var(--color-admonition-title--tip);-webkit-mask-image:var(--icon-info);mask-image:var(--icon-info)}.admonition.important{border-left-color:var(--color-admonition-title--important)}.admonition.important>.admonition-title{background-color:var(--color-admonition-title-background--important)}.admonition.important>.admonition-title:before{background-color:var(--color-admonition-title--important);-webkit-mask-image:var(--icon-flame);mask-image:var(--icon-flame)}.admonition.note{border-left-color:var(--color-admonition-title--note)}.admonition.note>.admonition-title{background-color:var(--color-admonition-title-background--note)}.admonition.note>.admonition-title:before{background-color:var(--color-admonition-title--note);-webkit-mask-image:var(--icon-pencil);mask-image:var(--icon-pencil)}.admonition.seealso{border-left-color:var(--color-admonition-title--seealso)}.admonition.seealso>.admonition-title{background-color:var(--color-admonition-title-background--seealso)}.admonition.seealso>.admonition-title:before{background-color:var(--color-admonition-title--seealso);-webkit-mask-image:var(--icon-info);mask-image:var(--icon-info)}.admonition.admonition-todo{border-left-color:var(--color-admonition-title--admonition-todo)}.admonition.admonition-todo>.admonition-title{background-color:var(--color-admonition-title-background--admonition-todo)}.admonition.admonition-todo>.admonition-title:before{background-color:var(--color-admonition-title--admonition-todo);-webkit-mask-image:var(--icon-pencil);mask-image:var(--icon-pencil)}.admonition-todo>.admonition-title{text-transform:uppercase}dl[class]:not(.option-list):not(.field-list):not(.footnote):not(.glossary):not(.simple) dd{margin-left:2rem}dl[class]:not(.option-list):not(.field-list):not(.footnote):not(.glossary):not(.simple) dd>:first-child{margin-top:.125rem}dl[class]:not(.option-list):not(.field-list):not(.footnote):not(.glossary):not(.simple) .field-list,dl[class]:not(.option-list):not(.field-list):not(.footnote):not(.glossary):not(.simple) dd>:last-child{margin-bottom:.75rem}dl[class]:not(.option-list):not(.field-list):not(.footnote):not(.glossary):not(.simple) .field-list>dt{font-size:var(--font-size--small);text-transform:uppercase}dl[class]:not(.option-list):not(.field-list):not(.footnote):not(.glossary):not(.simple) .field-list dd:empty{margin-bottom:.5rem}dl[class]:not(.option-list):not(.field-list):not(.footnote):not(.glossary):not(.simple) .field-list dd>ul{margin-left:-1.2rem}dl[class]:not(.option-list):not(.field-list):not(.footnote):not(.glossary):not(.simple) .field-list dd>ul>li>p:nth-child(2){margin-top:0}dl[class]:not(.option-list):not(.field-list):not(.footnote):not(.glossary):not(.simple) .field-list dd>ul>li>p+p:last-child:empty{margin-bottom:0;margin-top:0}dl[class]:not(.option-list):not(.field-list):not(.footnote):not(.glossary):not(.simple)>dt{color:var(--color-api-overall)}.sig:not(.sig-inline){background:var(--color-api-background);border-radius:.25rem;font-family:var(--font-stack--monospace);font-size:var(--api-font-size);font-weight:700;margin-left:-.25rem;margin-right:-.25rem;padding:.25rem .5rem .25rem 3em;text-indent:-2.5em;transition:background .1s ease-out}.sig:not(.sig-inline):hover{background:var(--color-api-background-hover)}.sig:not(.sig-inline) a.reference .viewcode-link{font-weight:400;width:3.5rem}em.property{font-style:normal}em.property:first-child{color:var(--color-api-keyword)}.sig-name{color:var(--color-api-name)}.sig-prename{color:var(--color-api-pre-name);font-weight:400}.sig-paren{color:var(--color-api-paren)}.sig-param{font-style:normal}.versionmodified{font-style:italic}div.deprecated p,div.versionadded p,div.versionchanged p{margin-bottom:.125rem;margin-top:.125rem}.viewcode-back,.viewcode-link{float:right;text-align:right}.line-block{margin-bottom:.75rem;margin-top:.5rem}.line-block .line-block{margin-bottom:0;margin-top:0;padding-left:1rem}.code-block-caption,article p.caption,table>caption{font-size:var(--font-size--small);text-align:center}.toctree-wrapper.compound .caption,.toctree-wrapper.compound :not(.caption)>.caption-text{font-size:var(--font-size--small);margin-bottom:0;text-align:initial;text-transform:uppercase}.toctree-wrapper.compound>ul{margin-bottom:0;margin-top:0}.sig-inline,code.literal{background:var(--color-inline-code-background);border-radius:.2em;font-size:var(--font-size--small--2);padding:.1em .2em}pre.literal-block .sig-inline,pre.literal-block code.literal{font-size:inherit;padding:0}p .sig-inline,p code.literal{border:1px solid var(--color-background-border)}.sig-inline{font-family:var(--font-stack--monospace)}div[class*=" highlight-"],div[class^=highlight-]{display:flex;margin:1em 0}div[class*=" highlight-"] .table-wrapper,div[class^=highlight-] .table-wrapper,pre{margin:0;padding:0}pre{overflow:auto}article[role=main] .highlight pre{line-height:1.5}.highlight pre,pre.literal-block{font-size:var(--code-font-size);padding:.625rem .875rem}pre.literal-block{background-color:var(--color-code-background);border-radius:.2rem;color:var(--color-code-foreground);margin-bottom:1rem;margin-top:1rem}.highlight{border-radius:.2rem;width:100%}.highlight .gp,.highlight span.linenos{pointer-events:none;-webkit-user-select:none;-moz-user-select:none;user-select:none}.highlight .hll{display:block;margin-left:-.875rem;margin-right:-.875rem;padding-left:.875rem;padding-right:.875rem}.code-block-caption{background-color:var(--color-code-background);border-bottom:1px solid;border-radius:.25rem;border-bottom-left-radius:0;border-bottom-right-radius:0;border-color:var(--color-background-border);color:var(--color-code-foreground);display:flex;font-weight:300;padding:.625rem .875rem}.code-block-caption+div[class]{margin-top:0}.code-block-caption+div[class] pre{border-top-left-radius:0;border-top-right-radius:0}.highlighttable{display:block;width:100%}.highlighttable tbody{display:block}.highlighttable tr{display:flex}.highlighttable td.linenos{background-color:var(--color-code-background);border-bottom-left-radius:.2rem;border-top-left-radius:.2rem;color:var(--color-code-foreground);padding:.625rem 0 .625rem .875rem}.highlighttable .linenodiv{box-shadow:-.0625rem 0 var(--color-foreground-border) inset;font-size:var(--code-font-size);padding-right:.875rem}.highlighttable td.code{display:block;flex:1;overflow:hidden;padding:0}.highlighttable td.code .highlight{border-bottom-left-radius:0;border-top-left-radius:0}.highlight span.linenos{box-shadow:-.0625rem 0 var(--color-foreground-border) inset;display:inline-block;margin-right:.875rem;padding-left:0;padding-right:.875rem}.footnote-reference{font-size:var(--font-size--small--4);vertical-align:super}dl.footnote.brackets{color:var(--color-foreground-secondary);display:grid;font-size:var(--font-size--small);grid-template-columns:max-content auto}dl.footnote.brackets dt{margin:0}dl.footnote.brackets dt>.fn-backref{margin-left:.25rem}dl.footnote.brackets dt:after{content:":"}dl.footnote.brackets dt .brackets:before{content:"["}dl.footnote.brackets dt .brackets:after{content:"]"}dl.footnote.brackets dd{margin:0;padding:0 1rem}aside.footnote{color:var(--color-foreground-secondary);font-size:var(--font-size--small)}aside.footnote>span,div.citation>span{float:left;font-weight:500;padding-right:.25rem}aside.footnote>p,div.citation>p{margin-left:2rem}img{box-sizing:border-box;height:auto;max-width:100%}article .figure,article figure{border-radius:.2rem;margin:0}article .figure :last-child,article figure :last-child{margin-bottom:0}article .align-left{clear:left;float:left;margin:0 1rem 1rem}article .align-right{clear:right;float:right;margin:0 1rem 1rem}article .align-center,article .align-default{display:block;margin-left:auto;margin-right:auto;text-align:center}article table.align-default{display:table;text-align:initial}.domainindex-jumpbox,.genindex-jumpbox{border-bottom:1px solid var(--color-background-border);border-top:1px solid var(--color-background-border);padding:.25rem}.domainindex-section h2,.genindex-section h2{margin-bottom:.5rem;margin-top:.75rem}.domainindex-section ul,.genindex-section ul{margin-bottom:0;margin-top:0}ol,ul{margin-bottom:1rem;margin-top:1rem;padding-left:1.2rem}ol li>p:first-child,ul li>p:first-child{margin-bottom:.25rem;margin-top:.25rem}ol li>p:last-child,ul li>p:last-child{margin-top:.25rem}ol li>ol,ol li>ul,ul li>ol,ul li>ul{margin-bottom:.5rem;margin-top:.5rem}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}.simple li>ol,.simple li>ul,.toctree-wrapper li>ol,.toctree-wrapper li>ul{margin-bottom:0;margin-top:0}.field-list dt,.option-list dt,dl.footnote dt,dl.glossary dt,dl.simple dt,dl:not([class]) dt{font-weight:500;margin-top:.25rem}.field-list dt+dt,.option-list dt+dt,dl.footnote dt+dt,dl.glossary dt+dt,dl.simple dt+dt,dl:not([class]) dt+dt{margin-top:0}.field-list dt .classifier:before,.option-list dt .classifier:before,dl.footnote dt .classifier:before,dl.glossary dt .classifier:before,dl.simple dt .classifier:before,dl:not([class]) dt .classifier:before{content:":";margin-left:.2rem;margin-right:.2rem}.field-list dd ul,.field-list dd>p:first-child,.option-list dd ul,.option-list dd>p:first-child,dl.footnote dd ul,dl.footnote dd>p:first-child,dl.glossary dd ul,dl.glossary dd>p:first-child,dl.simple dd ul,dl.simple dd>p:first-child,dl:not([class]) dd ul,dl:not([class]) dd>p:first-child{margin-top:.125rem}.field-list dd ul,.option-list dd ul,dl.footnote dd ul,dl.glossary dd ul,dl.simple dd ul,dl:not([class]) dd ul{margin-bottom:.125rem}.math-wrapper{overflow-x:auto;width:100%}div.math{position:relative;text-align:center}div.math .headerlink,div.math:focus .headerlink{display:none}div.math:hover .headerlink{display:inline-block}div.math span.eqno{position:absolute;right:.5rem;top:50%;transform:translateY(-50%);z-index:1}abbr[title]{cursor:help}.problematic{color:var(--color-problematic)}kbd:not(.compound){background-color:var(--color-background-secondary);border:1px solid var(--color-foreground-border);border-radius:.2rem;box-shadow:0 .0625rem 0 rgba(0,0,0,.2),inset 0 0 0 .125rem var(--color-background-primary);color:var(--color-foreground-primary);display:inline-block;font-size:var(--font-size--small--3);margin:0 .2rem;padding:0 .2rem;vertical-align:text-bottom}blockquote{background:var(--color-background-secondary);border-left:4px solid var(--color-background-border);margin-left:0;margin-right:0;padding:.5rem 1rem}blockquote .attribution{font-weight:600;text-align:right}blockquote.highlights,blockquote.pull-quote{font-size:1.25em}blockquote.epigraph,blockquote.pull-quote{border-left-width:0;border-radius:.5rem}blockquote.highlights{background:transparent;border-left-width:0}p .reference img{vertical-align:middle}p.rubric{font-size:1.125em;font-weight:700;line-height:1.25}dd p.rubric{font-size:var(--font-size--small);font-weight:inherit;line-height:inherit;text-transform:uppercase}article .sidebar{background-color:var(--color-background-secondary);border:1px solid var(--color-background-border);border-radius:.2rem;clear:right;float:right;margin-left:1rem;margin-right:0;width:30%}article .sidebar>*{padding-left:1rem;padding-right:1rem}article .sidebar>ol,article .sidebar>ul{padding-left:2.2rem}article .sidebar .sidebar-title{border-bottom:1px solid var(--color-background-border);font-weight:500;margin:0;padding:.5rem 1rem}.table-wrapper{margin-bottom:.5rem;margin-top:1rem;overflow-x:auto;padding:.2rem .2rem .75rem;width:100%}table.docutils{border-collapse:collapse;border-radius:.2rem;border-spacing:0;box-shadow:0 .2rem .5rem rgba(0,0,0,.05),0 0 .0625rem rgba(0,0,0,.1)}table.docutils th{background:var(--color-table-header-background)}table.docutils td,table.docutils th{border-bottom:1px solid var(--color-table-border);border-left:1px solid var(--color-table-border);border-right:1px solid var(--color-table-border);padding:0 .25rem}table.docutils td p,table.docutils th p{margin:.25rem}table.docutils td:first-child,table.docutils th:first-child{border-left:none}table.docutils td:last-child,table.docutils th:last-child{border-right:none}table.docutils td.text-left,table.docutils th.text-left{text-align:left}table.docutils td.text-right,table.docutils th.text-right{text-align:right}table.docutils td.text-center,table.docutils th.text-center{text-align:center}:target{scroll-margin-top:.5rem}@media(max-width:67em){:target{scroll-margin-top:calc(.5rem + var(--header-height))}section>span:target{scroll-margin-top:calc(.8rem + var(--header-height))}}.headerlink{font-weight:100;-webkit-user-select:none;-moz-user-select:none;user-select:none}.code-block-caption>.headerlink,dl dt>.headerlink,figcaption p>.headerlink,h1>.headerlink,h2>.headerlink,h3>.headerlink,h4>.headerlink,h5>.headerlink,h6>.headerlink,p.caption>.headerlink,table>caption>.headerlink{margin-left:.5rem;visibility:hidden}.code-block-caption:hover>.headerlink,dl dt:hover>.headerlink,figcaption p:hover>.headerlink,h1:hover>.headerlink,h2:hover>.headerlink,h3:hover>.headerlink,h4:hover>.headerlink,h5:hover>.headerlink,h6:hover>.headerlink,p.caption:hover>.headerlink,table>caption:hover>.headerlink{visibility:visible}.code-block-caption>.toc-backref,dl dt>.toc-backref,figcaption p>.toc-backref,h1>.toc-backref,h2>.toc-backref,h3>.toc-backref,h4>.toc-backref,h5>.toc-backref,h6>.toc-backref,p.caption>.toc-backref,table>caption>.toc-backref{color:inherit;text-decoration-line:none}figure:hover>figcaption>p>.headerlink,table:hover>caption>.headerlink{visibility:visible}:target>h1:first-of-type,:target>h2:first-of-type,:target>h3:first-of-type,:target>h4:first-of-type,:target>h5:first-of-type,:target>h6:first-of-type,span:target~h1:first-of-type,span:target~h2:first-of-type,span:target~h3:first-of-type,span:target~h4:first-of-type,span:target~h5:first-of-type,span:target~h6:first-of-type{background-color:var(--color-highlight-on-target)}:target>h1:first-of-type code.literal,:target>h2:first-of-type code.literal,:target>h3:first-of-type code.literal,:target>h4:first-of-type code.literal,:target>h5:first-of-type code.literal,:target>h6:first-of-type code.literal,span:target~h1:first-of-type code.literal,span:target~h2:first-of-type code.literal,span:target~h3:first-of-type code.literal,span:target~h4:first-of-type code.literal,span:target~h5:first-of-type code.literal,span:target~h6:first-of-type code.literal{background-color:transparent}.literal-block-wrapper:target .code-block-caption,.this-will-duplicate-information-and-it-is-still-useful-here li :target,figure:target,table:target>caption{background-color:var(--color-highlight-on-target)}dt:target{background-color:var(--color-highlight-on-target)!important}.footnote-reference:target,.footnote>dt:target+dd{background-color:var(--color-highlight-on-target)}.guilabel{background-color:var(--color-guilabel-background);border:1px solid var(--color-guilabel-border);border-radius:.5em;color:var(--color-guilabel-text);font-size:.9em;padding:0 .3em}footer{display:flex;flex-direction:column;font-size:var(--font-size--small);margin-top:2rem}.bottom-of-page{align-items:center;border-top:1px solid var(--color-background-border);color:var(--color-foreground-secondary);display:flex;justify-content:space-between;line-height:1.5;margin-top:1rem;padding-bottom:1rem;padding-top:1rem}@media(max-width:46em){.bottom-of-page{flex-direction:column-reverse;gap:.25rem;text-align:center}}.bottom-of-page .left-details{font-size:var(--font-size--small)}.bottom-of-page .right-details{display:flex;flex-direction:column;gap:.25rem;text-align:right}.bottom-of-page .icons{display:flex;font-size:1rem;gap:.25rem;justify-content:flex-end}.bottom-of-page .icons a{text-decoration:none}.bottom-of-page .icons img,.bottom-of-page .icons svg{font-size:1.125rem;height:1em;width:1em}.related-pages a{align-items:center;display:flex;text-decoration:none}.related-pages a:hover .page-info .title{color:var(--color-link);text-decoration:underline;text-decoration-color:var(--color-link-underline)}.related-pages a svg.furo-related-icon,.related-pages a svg.furo-related-icon>use{color:var(--color-foreground-border);flex-shrink:0;height:.75rem;margin:0 .5rem;width:.75rem}.related-pages a.next-page{clear:right;float:right;max-width:50%;text-align:right}.related-pages a.prev-page{clear:left;float:left;max-width:50%}.related-pages a.prev-page svg{transform:rotate(180deg)}.page-info{display:flex;flex-direction:column;overflow-wrap:anywhere}.next-page .page-info{align-items:flex-end}.page-info .context{align-items:center;color:var(--color-foreground-muted);display:flex;font-size:var(--font-size--small);padding-bottom:.1rem;text-decoration:none}ul.search{list-style:none;padding-left:0}ul.search li{border-bottom:1px solid var(--color-background-border);padding:1rem 0}[role=main] .highlighted{background-color:var(--color-highlighted-background);color:var(--color-highlighted-text)}.sidebar-brand{display:flex;flex-direction:column;flex-shrink:0;padding:var(--sidebar-item-spacing-vertical) var(--sidebar-item-spacing-horizontal);text-decoration:none}.sidebar-brand-text{color:var(--color-sidebar-brand-text);font-size:1.5rem;overflow-wrap:break-word}.sidebar-brand-text,.sidebar-logo-container{margin:var(--sidebar-item-spacing-vertical) 0}.sidebar-logo{display:block;margin:0 auto;max-width:100%}.sidebar-search-container{align-items:center;background:var(--color-sidebar-search-background);display:flex;margin-top:var(--sidebar-search-space-above);position:relative}.sidebar-search-container:focus-within,.sidebar-search-container:hover{background:var(--color-sidebar-search-background--focus)}.sidebar-search-container:before{background-color:var(--color-sidebar-search-icon);content:"";height:var(--sidebar-search-icon-size);left:var(--sidebar-item-spacing-horizontal);-webkit-mask-image:var(--icon-search);mask-image:var(--icon-search);position:absolute;width:var(--sidebar-search-icon-size)}.sidebar-search{background:transparent;border:none;border-bottom:1px solid var(--color-sidebar-search-border);border-top:1px solid var(--color-sidebar-search-border);box-sizing:border-box;color:var(--color-sidebar-search-foreground);padding:var(--sidebar-search-input-spacing-vertical) var(--sidebar-search-input-spacing-horizontal) var(--sidebar-search-input-spacing-vertical) calc(var(--sidebar-item-spacing-horizontal) + var(--sidebar-search-input-spacing-horizontal) + var(--sidebar-search-icon-size));width:100%;z-index:10}.sidebar-search:focus{outline:none}.sidebar-search::-moz-placeholder{font-size:var(--sidebar-search-input-font-size)}.sidebar-search::placeholder{font-size:var(--sidebar-search-input-font-size)}#searchbox .highlight-link{margin:0;padding:var(--sidebar-item-spacing-vertical) var(--sidebar-item-spacing-horizontal) 0;text-align:center}#searchbox .highlight-link a{color:var(--color-sidebar-search-icon);font-size:var(--font-size--small--2)}.sidebar-tree{font-size:var(--sidebar-item-font-size);margin-bottom:var(--sidebar-item-spacing-vertical);margin-top:var(--sidebar-tree-space-above)}.sidebar-tree ul{display:flex;flex-direction:column;list-style:none;margin-bottom:0;margin-top:0;padding:0}.sidebar-tree li{margin:0;position:relative}.sidebar-tree li>ul{margin-left:var(--sidebar-item-spacing-horizontal)}.sidebar-tree .icon,.sidebar-tree .reference{color:var(--color-sidebar-link-text)}.sidebar-tree .reference{box-sizing:border-box;display:inline-block;height:100%;line-height:var(--sidebar-item-line-height);overflow-wrap:anywhere;padding:var(--sidebar-item-spacing-vertical) var(--sidebar-item-spacing-horizontal);text-decoration:none;width:100%}.sidebar-tree .reference:hover{background:var(--color-sidebar-item-background--hover)}.sidebar-tree .reference.external:after{color:var(--color-sidebar-link-text);content:url("data:image/svg+xml;charset=utf-8,%3Csvg width='12' height='12' xmlns='http://www.w3.org/2000/svg' viewBox='0 0 24 24' stroke-width='1.5' stroke='%23607D8B' fill='none' stroke-linecap='round' stroke-linejoin='round'%3E%3Cpath d='M0 0h24v24H0z' stroke='none'/%3E%3Cpath d='M11 7H6a2 2 0 0 0-2 2v9a2 2 0 0 0 2 2h9a2 2 0 0 0 2-2v-5M10 14 20 4M15 4h5v5'/%3E%3C/svg%3E");margin:0 .25rem;vertical-align:middle}.sidebar-tree .current-page>.reference{font-weight:700}.sidebar-tree label{align-items:center;cursor:pointer;display:flex;height:var(--sidebar-item-height);justify-content:center;position:absolute;right:0;top:0;-webkit-user-select:none;-moz-user-select:none;user-select:none;width:var(--sidebar-expander-width)}.sidebar-tree .caption,.sidebar-tree :not(.caption)>.caption-text{color:var(--color-sidebar-caption-text);font-size:var(--sidebar-caption-font-size);font-weight:700;margin:var(--sidebar-caption-space-above) 0 0 0;padding:var(--sidebar-item-spacing-vertical) var(--sidebar-item-spacing-horizontal);text-transform:uppercase}.sidebar-tree li.has-children>.reference{padding-right:var(--sidebar-expander-width)}.sidebar-tree .toctree-l1>.reference,.sidebar-tree .toctree-l1>label .icon{color:var(--color-sidebar-link-text--top-level)}.sidebar-tree label{background:var(--color-sidebar-item-expander-background)}.sidebar-tree label:hover{background:var(--color-sidebar-item-expander-background--hover)}.sidebar-tree .current>.reference{background:var(--color-sidebar-item-background--current)}.sidebar-tree .current>.reference:hover{background:var(--color-sidebar-item-background--hover)}.toctree-checkbox{display:none;position:absolute}.toctree-checkbox~ul{display:none}.toctree-checkbox~label .icon svg{transform:rotate(90deg)}.toctree-checkbox:checked~ul{display:block}.toctree-checkbox:checked~label .icon svg{transform:rotate(-90deg)}.toc-title-container{padding:var(--toc-title-padding);padding-top:var(--toc-spacing-vertical)}.toc-title{color:var(--color-toc-title-text);font-size:var(--toc-title-font-size);padding-left:var(--toc-spacing-horizontal);text-transform:uppercase}.no-toc{display:none}.toc-tree-container{padding-bottom:var(--toc-spacing-vertical)}.toc-tree{border-left:1px solid var(--color-background-border);font-size:var(--toc-font-size);line-height:1.3;padding-left:calc(var(--toc-spacing-horizontal) - var(--toc-item-spacing-horizontal))}.toc-tree>ul>li:first-child{padding-top:0}.toc-tree>ul>li:first-child>ul{padding-left:0}.toc-tree>ul>li:first-child>a{display:none}.toc-tree ul{list-style-type:none;margin-bottom:0;margin-top:0;padding-left:var(--toc-item-spacing-horizontal)}.toc-tree li{padding-top:var(--toc-item-spacing-vertical)}.toc-tree li.scroll-current>.reference{color:var(--color-toc-item-text--active);font-weight:700}.toc-tree .reference{color:var(--color-toc-item-text);overflow-wrap:anywhere;text-decoration:none}.toc-scroll{max-height:100vh;overflow-y:scroll}.contents:not(.this-will-duplicate-information-and-it-is-still-useful-here){background:rgba(255,0,0,.25);color:var(--color-problematic)}.contents:not(.this-will-duplicate-information-and-it-is-still-useful-here):before{content:"ERROR: Adding a table of contents in Furo-based documentation is unnecessary, and does not work well with existing styling.Add a 'this-will-duplicate-information-and-it-is-still-useful-here' class, if you want an escape hatch."}.text-align\:left>p{text-align:left}.text-align\:center>p{text-align:center}.text-align\:right>p{text-align:right} +/*# sourceMappingURL=furo.css.map*/ \ No newline at end of file diff --git a/_static/styles/furo.css.map b/_static/styles/furo.css.map new file mode 100644 index 000000000..d1dfb109d --- /dev/null +++ b/_static/styles/furo.css.map @@ -0,0 +1 @@ +{"version":3,"file":"styles/furo.css","mappings":"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0CACE,CAFF,kBACA,CACE,wEACA,CARA,YACA,CAKF,mBAFF,OAII,eACA,CAJF,iCAJE,cAGJ,CANI,oBAEA,CAKF,SAIE,2BADA,UACA,kBAGF,sCACA,CAFF,WACE,WACA,qCACE,gCACA,2EACA,sDAKJ,aACE,mDAII,CAJJ,6CAII,kEACA,iBACE,iDACA,+CACE,aACA,WADA,+BACA,uEANN,YACE,mDAEE,mBADF,0CACE,CADF,qBACE,0DACA,YACE,4DACA,sEANN,YACE,8CACA,kBADA,UACA,2CACE,2EACA,cACE,kEACA,mEANN,yBACE,4DACA,sBACE,+EAEE,iEACA,qEANN,sCACE,CAGE,iBAHF,gBAGE,qBACE,CAJJ,uBACA,gDACE,wDACA,6DAHF,2CACA,CADA,gBACA,eACE,CAGE,sBANN,8BACE,CAII,iBAFF,4DACA,WACE,YADF,uCACE,6EACA,2BANN,8CACE,kDACA,0CACE,8BACA,yFACE,sBACA,sFALJ,mEACA,sBACE,kEACA,6EACE,uCACA,kEALJ,qGAEE,kEACA,6EACE,uCACA,kEALJ,8CACA,uDACE,sEACA,2EACE,sCACA,iEALJ,mGACA,qCACE,oDACA,0DACE,6GACA,gDAGR,yDCrEA,sEACE,CACA,6GACE,gEACF,iGAIF,wFACE,qDAGA,mGAEE,2CAEF,4FACE,gCACF,wGACE,8DAEE,6FAIA,iJAKN,6GACE,gDAKF,yDACA,qCAGA,6BACA,kBACA,qDAKA,oCAEA,+DAGA,2CAGE,oDAIA,oEAEE,qBAGJ,wDAEE,uCAEF,kEAGA,8CAEA,uDAKA,oCAEA,yDAEE,gEAKF,+CC5FA,0EAGE,CACA,qDCLJ,+DAIE,sCAIA,kEACE,yBACA,2FAMA,gBACA,yGCbF,mBAOA,2MAIA,4HAYA,0DACE,8GAYF,8HAQE,mBAEA,6HAOF,YAGA,mIAME,eACA,CAFF,YAEE,4FAMJ,8BAEE,uBAYA,sCAEE,CAJF,oBAEA,CARA,wCAEA,CAHA,8BACA,CAFA,eACA,CAGA,wCAEA,CAEA,mDAIE,kCACE,6BACA,4CAKJ,kDAIA,eACE,aAGF,8BACE,uDACA,sCACA,cAEA,+BACA,CAFA,eAEA,wCAEF,YACE,iBACA,mCACA,0DAGF,qBAEE,CAFF,kBAEE,+BAIA,yCAEE,qBADA,gBACA,yBAKF,eACA,CAFF,YACE,CACA,iBACA,qDAEA,mDCvIJ,2FAOE,iCACA,CAEA,eACA,CAHA,kBAEA,CAFA,wBAGA,8BACA,eACE,CAFF,YAEE,0BACA,8CAGA,oBACE,oCAGA,kBACE,8DAEA,iBAEN,UACE,8BAIJ,+CAEE,qDAEF,kDAIE,YAEF,CAFE,YAEF,CCjCE,mFAJA,QACA,UAIE,CADF,iBACE,mCAGA,iDACE,+BAGF,wBAEA,mBAKA,6CAEF,CAHE,mBACA,CAEF,kCAIE,CARA,kBACA,CAFF,eASE,YACA,mBAGF,CAJE,UAIF,wCCjCA,oBDmCE,wBCpCJ,uCACE,8BACA,4CACA,oBAGA,2CCAA,6CAGE,CAPF,uBAIA,CDGA,gDACE,6BCVJ,CAWM,2CAEF,CAJA,kCAEE,CDJF,aCLF,gBDKE,uBCMA,gCAGA,gDAGE,wBAGJ,0BAEA,iBACE,aACF,CADE,UACF,uBACE,aACF,oBACE,YACF,4BACE,6CAMA,CAYF,6DAZE,mCAGE,iCASJ,4BAGE,4DADA,+BACA,CAFA,qBAEA,yBACE,aAEF,wBAHA,SAGA,iHACE,2DAKF,CANA,yCACE,CADF,oCAMA,uSAIA,sGACE,oDChEJ,WAEF,yBACE,QACA,eAEA,gBAEE,uCAGA,CALF,iCAKE,uCAGA,0BACA,CACA,oBACA,iCClBJ,gBACE,KAGF,qBACE,YAGF,CAHE,cAGF,gCAEE,mBACA,iEAEA,oCACA,wCAEA,sBACA,WAEA,CAFA,YAEA,8EAEA,mCAFA,iBAEA,6BAIA,wEAKA,sDAIE,CARF,mDAIA,CAIE,cAEF,8CAIA,oBAFE,iBAEF,8CAGE,eAEF,CAFE,YAEF,OAEE,kBAGJ,CAJI,eACA,CAFF,mBAKF,yCCjDE,oBACA,CAFA,iBAEA,uCAKE,iBACA,qCAGA,mBCZJ,CDWI,gBCXJ,6BAEE,eACA,sBAGA,eAEA,sBACA,oDACA,iGAMA,gBAFE,YAEF,8FAME,iJClBF,YACA,gNAUE,6BAEF,oTAcI,kBACF,gHAIA,qBACE,eACF,qDACE,kBACF,6DACE,4BCxCJ,oBAEF,qCAEI,+CAGF,uBACE,uDAGJ,oBAkBE,mDAhBA,+CAaA,CAbA,oBAaA,0FAEE,CAFF,gGAbA,+BAaA,0BAGA,mQAIA,oNAEE,iBAGJ,CAHI,gBADA,gBAIJ,8CAYI,CAZJ,wCAYI,sVACE,iCAGA,uEAHA,QAGA,qXAKJ,iDAGF,CARM,+CACE,iDAIN,CALI,gBAQN,mHACE,gBAGF,2DACE,0EAOA,0EAKA,6EC/EA,iDACA,gCACA,oDAGA,qBACA,oDCFA,cACA,eAEA,yBAGF,sBAEE,iBACA,sNAWA,iBACE,kBACA,wRAgBA,kBAEA,iOAgBA,uCACE,uEAEA,kBAEF,qUAuBE,iDAIJ,CACA,geCxFF,4BAEE,CAQA,6JACA,iDAIA,sEAGA,mDAOF,iDAGE,4DAIA,8CACA,qDAEE,eAFF,cAEE,oBAEF,uBAFE,kCAGA,eACA,iBACA,mBAIA,mDACA,CAHA,uCAEA,CAJA,0CACA,CAIA,gBAJA,gBACA,oBADA,gBAIA,wBAEJ,gBAGE,6BACA,YAHA,iBAGA,gCACA,iEAEA,6CACA,sDACA,0BADA,wBACA,0BACA,oIAIA,mBAFA,YAEA,qBACA,0CAIE,uBAEF,CAHA,yBACE,CAEF,iDACE,mFAKJ,oCACE,CANE,aAKJ,CACE,qEAIA,YAFA,WAEA,CAHA,aACA,CAEA,gBACE,4BACA,sBADA,aACA,gCAMF,oCACA,yDACA,2CAEA,qBAGE,kBAEA,CACA,mCAIF,CARE,YACA,CAOF,iCAEE,CAPA,oBACA,CAQA,oBACE,uDAEJ,sDAGA,CAHA,cAGA,0BACE,oDAIA,oCACA,4BACA,sBAGA,cAEA,oFAGA,sBAEA,yDACE,CAIA,iBAJA,wBAIA,6CAJA,6CAOA,4BAGJ,CAHI,cAGJ,yCAGA,kBACE,CAIA,iDAEA,CATA,YAEF,CACE,4CAGA,kBAIA,wEAEA,wDAIF,kCAOE,iDACA,CARF,WAIE,sCAGA,CANA,2CACA,CAMA,oEARF,iBACE,CACA,qCAMA,iBAuBE,uBAlBF,YAKA,2DALA,uDAKA,CALA,sBAiBA,4CACE,CALA,gRAIF,YACE,UAEN,uBACE,YACA,mCAOE,+CAGA,8BAGF,+CAGA,4BCjNA,SDiNA,qFCjNA,gDAGA,sCACA,qCACA,sDAIF,CAIE,kDAGA,CAPF,0CAOE,kBAEA,kDAEA,CAHA,eACA,CAFA,YACA,CADA,SAIA,mHAIE,CAGA,6CAFA,oCAeE,CAbF,yBACE,qBAEJ,CAGE,oBACA,CAEA,YAFA,2CACF,CACE,uBAEA,mFAEE,CALJ,oBACE,CAEA,UAEE,gCAGF,sDAEA,yCC7CJ,oCAGA,CD6CE,yXAQE,sCCrDJ,wCAGA,oCACE","sources":["webpack:///./node_modules/normalize.css/normalize.css","webpack:///./src/furo/assets/styles/base/_print.sass","webpack:///./src/furo/assets/styles/base/_screen-readers.sass","webpack:///./src/furo/assets/styles/base/_theme.sass","webpack:///./src/furo/assets/styles/variables/_fonts.scss","webpack:///./src/furo/assets/styles/variables/_spacing.scss","webpack:///./src/furo/assets/styles/variables/_icons.scss","webpack:///./src/furo/assets/styles/variables/_admonitions.scss","webpack:///./src/furo/assets/styles/variables/_colors.scss","webpack:///./src/furo/assets/styles/base/_typography.sass","webpack:///./src/furo/assets/styles/_scaffold.sass","webpack:///./src/furo/assets/styles/content/_admonitions.sass","webpack:///./src/furo/assets/styles/content/_api.sass","webpack:///./src/furo/assets/styles/content/_blocks.sass","webpack:///./src/furo/assets/styles/content/_captions.sass","webpack:///./src/furo/assets/styles/content/_code.sass","webpack:///./src/furo/assets/styles/content/_footnotes.sass","webpack:///./src/furo/assets/styles/content/_images.sass","webpack:///./src/furo/assets/styles/content/_indexes.sass","webpack:///./src/furo/assets/styles/content/_lists.sass","webpack:///./src/furo/assets/styles/content/_math.sass","webpack:///./src/furo/assets/styles/content/_misc.sass","webpack:///./src/furo/assets/styles/content/_rubrics.sass","webpack:///./src/furo/assets/styles/content/_sidebar.sass","webpack:///./src/furo/assets/styles/content/_tables.sass","webpack:///./src/furo/assets/styles/content/_target.sass","webpack:///./src/furo/assets/styles/content/_gui-labels.sass","webpack:///./src/furo/assets/styles/components/_footer.sass","webpack:///./src/furo/assets/styles/components/_sidebar.sass","webpack:///./src/furo/assets/styles/components/_table_of_contents.sass","webpack:///./src/furo/assets/styles/_shame.sass"],"sourcesContent":["/*! normalize.css v8.0.1 | MIT License | github.com/necolas/normalize.css */\n\n/* Document\n ========================================================================== */\n\n/**\n * 1. Correct the line height in all browsers.\n * 2. Prevent adjustments of font size after orientation changes in iOS.\n */\n\nhtml {\n line-height: 1.15; /* 1 */\n -webkit-text-size-adjust: 100%; /* 2 */\n}\n\n/* Sections\n ========================================================================== */\n\n/**\n * Remove the margin in all browsers.\n */\n\nbody {\n margin: 0;\n}\n\n/**\n * Render the `main` element consistently in IE.\n */\n\nmain {\n display: block;\n}\n\n/**\n * Correct the font size and margin on `h1` elements within `section` and\n * `article` contexts in Chrome, Firefox, and Safari.\n */\n\nh1 {\n font-size: 2em;\n margin: 0.67em 0;\n}\n\n/* Grouping content\n ========================================================================== */\n\n/**\n * 1. Add the correct box sizing in Firefox.\n * 2. Show the overflow in Edge and IE.\n */\n\nhr {\n box-sizing: content-box; /* 1 */\n height: 0; /* 1 */\n overflow: visible; /* 2 */\n}\n\n/**\n * 1. Correct the inheritance and scaling of font size in all browsers.\n * 2. Correct the odd `em` font sizing in all browsers.\n */\n\npre {\n font-family: monospace, monospace; /* 1 */\n font-size: 1em; /* 2 */\n}\n\n/* Text-level semantics\n ========================================================================== */\n\n/**\n * Remove the gray background on active links in IE 10.\n */\n\na {\n background-color: transparent;\n}\n\n/**\n * 1. Remove the bottom border in Chrome 57-\n * 2. Add the correct text decoration in Chrome, Edge, IE, Opera, and Safari.\n */\n\nabbr[title] {\n border-bottom: none; /* 1 */\n text-decoration: underline; /* 2 */\n text-decoration: underline dotted; /* 2 */\n}\n\n/**\n * Add the correct font weight in Chrome, Edge, and Safari.\n */\n\nb,\nstrong {\n font-weight: bolder;\n}\n\n/**\n * 1. Correct the inheritance and scaling of font size in all browsers.\n * 2. Correct the odd `em` font sizing in all browsers.\n */\n\ncode,\nkbd,\nsamp {\n font-family: monospace, monospace; /* 1 */\n font-size: 1em; /* 2 */\n}\n\n/**\n * Add the correct font size in all browsers.\n */\n\nsmall {\n font-size: 80%;\n}\n\n/**\n * Prevent `sub` and `sup` elements from affecting the line height in\n * all browsers.\n */\n\nsub,\nsup {\n font-size: 75%;\n line-height: 0;\n position: relative;\n vertical-align: baseline;\n}\n\nsub {\n bottom: -0.25em;\n}\n\nsup {\n top: -0.5em;\n}\n\n/* Embedded content\n ========================================================================== */\n\n/**\n * Remove the border on images inside links in IE 10.\n */\n\nimg {\n border-style: none;\n}\n\n/* Forms\n ========================================================================== */\n\n/**\n * 1. Change the font styles in all browsers.\n * 2. Remove the margin in Firefox and Safari.\n */\n\nbutton,\ninput,\noptgroup,\nselect,\ntextarea {\n font-family: inherit; /* 1 */\n font-size: 100%; /* 1 */\n line-height: 1.15; /* 1 */\n margin: 0; /* 2 */\n}\n\n/**\n * Show the overflow in IE.\n * 1. Show the overflow in Edge.\n */\n\nbutton,\ninput { /* 1 */\n overflow: visible;\n}\n\n/**\n * Remove the inheritance of text transform in Edge, Firefox, and IE.\n * 1. Remove the inheritance of text transform in Firefox.\n */\n\nbutton,\nselect { /* 1 */\n text-transform: none;\n}\n\n/**\n * Correct the inability to style clickable types in iOS and Safari.\n */\n\nbutton,\n[type=\"button\"],\n[type=\"reset\"],\n[type=\"submit\"] {\n -webkit-appearance: button;\n}\n\n/**\n * Remove the inner border and padding in Firefox.\n */\n\nbutton::-moz-focus-inner,\n[type=\"button\"]::-moz-focus-inner,\n[type=\"reset\"]::-moz-focus-inner,\n[type=\"submit\"]::-moz-focus-inner {\n border-style: none;\n padding: 0;\n}\n\n/**\n * Restore the focus styles unset by the previous rule.\n */\n\nbutton:-moz-focusring,\n[type=\"button\"]:-moz-focusring,\n[type=\"reset\"]:-moz-focusring,\n[type=\"submit\"]:-moz-focusring {\n outline: 1px dotted ButtonText;\n}\n\n/**\n * Correct the padding in Firefox.\n */\n\nfieldset {\n padding: 0.35em 0.75em 0.625em;\n}\n\n/**\n * 1. Correct the text wrapping in Edge and IE.\n * 2. Correct the color inheritance from `fieldset` elements in IE.\n * 3. Remove the padding so developers are not caught out when they zero out\n * `fieldset` elements in all browsers.\n */\n\nlegend {\n box-sizing: border-box; /* 1 */\n color: inherit; /* 2 */\n display: table; /* 1 */\n max-width: 100%; /* 1 */\n padding: 0; /* 3 */\n white-space: normal; /* 1 */\n}\n\n/**\n * Add the correct vertical alignment in Chrome, Firefox, and Opera.\n */\n\nprogress {\n vertical-align: baseline;\n}\n\n/**\n * Remove the default vertical scrollbar in IE 10+.\n */\n\ntextarea {\n overflow: auto;\n}\n\n/**\n * 1. Add the correct box sizing in IE 10.\n * 2. Remove the padding in IE 10.\n */\n\n[type=\"checkbox\"],\n[type=\"radio\"] {\n box-sizing: border-box; /* 1 */\n padding: 0; /* 2 */\n}\n\n/**\n * Correct the cursor style of increment and decrement buttons in Chrome.\n */\n\n[type=\"number\"]::-webkit-inner-spin-button,\n[type=\"number\"]::-webkit-outer-spin-button {\n height: auto;\n}\n\n/**\n * 1. Correct the odd appearance in Chrome and Safari.\n * 2. Correct the outline style in Safari.\n */\n\n[type=\"search\"] {\n -webkit-appearance: textfield; /* 1 */\n outline-offset: -2px; /* 2 */\n}\n\n/**\n * Remove the inner padding in Chrome and Safari on macOS.\n */\n\n[type=\"search\"]::-webkit-search-decoration {\n -webkit-appearance: none;\n}\n\n/**\n * 1. Correct the inability to style clickable types in iOS and Safari.\n * 2. Change font properties to `inherit` in Safari.\n */\n\n::-webkit-file-upload-button {\n -webkit-appearance: button; /* 1 */\n font: inherit; /* 2 */\n}\n\n/* Interactive\n ========================================================================== */\n\n/*\n * Add the correct display in Edge, IE 10+, and Firefox.\n */\n\ndetails {\n display: block;\n}\n\n/*\n * Add the correct display in all browsers.\n */\n\nsummary {\n display: list-item;\n}\n\n/* Misc\n ========================================================================== */\n\n/**\n * Add the correct display in IE 10+.\n */\n\ntemplate {\n display: none;\n}\n\n/**\n * Add the correct display in IE 10.\n */\n\n[hidden] {\n display: none;\n}\n","// This file contains styles for managing print media.\n\n////////////////////////////////////////////////////////////////////////////////\n// Hide elements not relevant to print media.\n////////////////////////////////////////////////////////////////////////////////\n@media print\n // Hide icon container.\n .content-icon-container\n display: none !important\n\n // Hide showing header links if hovering over when printing.\n .headerlink\n display: none !important\n\n // Hide mobile header.\n .mobile-header\n display: none !important\n\n // Hide navigation links.\n .related-pages\n display: none !important\n\n////////////////////////////////////////////////////////////////////////////////\n// Tweaks related to decolorization.\n////////////////////////////////////////////////////////////////////////////////\n@media print\n // Apply a border around code which no longer have a color background.\n .highlight\n border: 0.1pt solid var(--color-foreground-border)\n\n////////////////////////////////////////////////////////////////////////////////\n// Avoid page break in some relevant cases.\n////////////////////////////////////////////////////////////////////////////////\n@media print\n ul, ol, dl, a, table, pre, blockquote\n page-break-inside: avoid\n\n h1, h2, h3, h4, h5, h6, img, figure, caption\n page-break-inside: avoid\n page-break-after: avoid\n\n ul, ol, dl\n page-break-before: avoid\n",".visually-hidden\n position: absolute !important\n width: 1px !important\n height: 1px !important\n padding: 0 !important\n margin: -1px !important\n overflow: hidden !important\n clip: rect(0,0,0,0) !important\n white-space: nowrap !important\n border: 0 !important\n\n:-moz-focusring\n outline: auto\n","// This file serves as the \"skeleton\" of the theming logic.\n//\n// This contains the bulk of the logic for handling dark mode, color scheme\n// toggling and the handling of color-scheme-specific hiding of elements.\n\nbody\n @include fonts\n @include spacing\n @include icons\n @include admonitions\n @include default-admonition(#651fff, \"abstract\")\n @include default-topic(#14B8A6, \"pencil\")\n\n @include colors\n\n.only-light\n display: block !important\nhtml body .only-dark\n display: none !important\n\n// Ignore dark-mode hints if print media.\n@media not print\n // Enable dark-mode, if requested.\n body[data-theme=\"dark\"]\n @include colors-dark\n\n html & .only-light\n display: none !important\n .only-dark\n display: block !important\n\n // Enable dark mode, unless explicitly told to avoid.\n @media (prefers-color-scheme: dark)\n body:not([data-theme=\"light\"])\n @include colors-dark\n\n html & .only-light\n display: none !important\n .only-dark\n display: block !important\n\n//\n// Theme toggle presentation\n//\nbody[data-theme=\"auto\"]\n .theme-toggle svg.theme-icon-when-auto\n display: block\n\nbody[data-theme=\"dark\"]\n .theme-toggle svg.theme-icon-when-dark\n display: block\n\nbody[data-theme=\"light\"]\n .theme-toggle svg.theme-icon-when-light\n display: block\n","// Fonts used by this theme.\n//\n// There are basically two things here -- using the system font stack and\n// defining sizes for various elements in %ages. We could have also used `em`\n// but %age is easier to reason about for me.\n\n@mixin fonts {\n // These are adapted from https://systemfontstack.com/\n --font-stack: -apple-system, BlinkMacSystemFont, Segoe UI, Helvetica, Arial,\n sans-serif, Apple Color Emoji, Segoe UI Emoji;\n --font-stack--monospace: \"SFMono-Regular\", Menlo, Consolas, Monaco,\n Liberation Mono, Lucida Console, monospace;\n\n --font-size--normal: 100%;\n --font-size--small: 87.5%;\n --font-size--small--2: 81.25%;\n --font-size--small--3: 75%;\n --font-size--small--4: 62.5%;\n\n // Sidebar\n --sidebar-caption-font-size: var(--font-size--small--2);\n --sidebar-item-font-size: var(--font-size--small);\n --sidebar-search-input-font-size: var(--font-size--small);\n\n // Table of Contents\n --toc-font-size: var(--font-size--small--3);\n --toc-font-size--mobile: var(--font-size--normal);\n --toc-title-font-size: var(--font-size--small--4);\n\n // Admonitions\n //\n // These aren't defined in terms of %ages, since nesting these is permitted.\n --admonition-font-size: 0.8125rem;\n --admonition-title-font-size: 0.8125rem;\n\n // Code\n --code-font-size: var(--font-size--small--2);\n\n // API\n --api-font-size: var(--font-size--small);\n}\n","// Spacing for various elements on the page\n//\n// If the user wants to tweak things in a certain way, they are permitted to.\n// They also have to deal with the consequences though!\n\n@mixin spacing {\n // Header!\n --header-height: calc(\n var(--sidebar-item-line-height) + 4 * #{var(--sidebar-item-spacing-vertical)}\n );\n --header-padding: 0.5rem;\n\n // Sidebar\n --sidebar-tree-space-above: 1.5rem;\n --sidebar-caption-space-above: 1rem;\n\n --sidebar-item-line-height: 1rem;\n --sidebar-item-spacing-vertical: 0.5rem;\n --sidebar-item-spacing-horizontal: 1rem;\n --sidebar-item-height: calc(\n var(--sidebar-item-line-height) + 2 *#{var(--sidebar-item-spacing-vertical)}\n );\n\n --sidebar-expander-width: var(--sidebar-item-height); // be square\n\n --sidebar-search-space-above: 0.5rem;\n --sidebar-search-input-spacing-vertical: 0.5rem;\n --sidebar-search-input-spacing-horizontal: 0.5rem;\n --sidebar-search-input-height: 1rem;\n --sidebar-search-icon-size: var(--sidebar-search-input-height);\n\n // Table of Contents\n --toc-title-padding: 0.25rem 0;\n --toc-spacing-vertical: 1.5rem;\n --toc-spacing-horizontal: 1.5rem;\n --toc-item-spacing-vertical: 0.4rem;\n --toc-item-spacing-horizontal: 1rem;\n}\n","// Expose theme icons as CSS variables.\n\n$icons: (\n // Adapted from tabler-icons\n // url: https://tablericons.com/\n \"search\":\n url('data:image/svg+xml;charset=utf-8,'),\n // Factored out from mkdocs-material on 24-Aug-2020.\n // url: https://squidfunk.github.io/mkdocs-material/reference/admonitions/\n \"pencil\":\n url('data:image/svg+xml;charset=utf-8,'),\n \"abstract\":\n url('data:image/svg+xml;charset=utf-8,'),\n \"info\":\n url('data:image/svg+xml;charset=utf-8,'),\n \"flame\":\n url('data:image/svg+xml;charset=utf-8,'),\n \"question\":\n url('data:image/svg+xml;charset=utf-8,'),\n \"warning\":\n url('data:image/svg+xml;charset=utf-8,'),\n \"failure\":\n url('data:image/svg+xml;charset=utf-8,'),\n \"spark\":\n url('data:image/svg+xml;charset=utf-8,')\n);\n\n@mixin icons {\n @each $name, $glyph in $icons {\n --icon-#{$name}: #{$glyph};\n }\n}\n","// Admonitions\n\n// Structure of these is:\n// admonition-class: color \"icon-name\";\n//\n// The colors are translated into CSS variables below. The icons are\n// used directly in the main declarations to set the `mask-image` in\n// the title.\n\n// prettier-ignore\n$admonitions: (\n // Each of these has an reST directives for it.\n \"caution\": #ff9100 \"spark\",\n \"warning\": #ff9100 \"warning\",\n \"danger\": #ff5252 \"spark\",\n \"attention\": #ff5252 \"warning\",\n \"error\": #ff5252 \"failure\",\n \"hint\": #00c852 \"question\",\n \"tip\": #00c852 \"info\",\n \"important\": #00bfa5 \"flame\",\n \"note\": #00b0ff \"pencil\",\n \"seealso\": #448aff \"info\",\n \"admonition-todo\": #808080 \"pencil\"\n);\n\n@mixin default-admonition($color, $icon-name) {\n --color-admonition-title: #{$color};\n --color-admonition-title-background: #{rgba($color, 0.2)};\n\n --icon-admonition-default: var(--icon-#{$icon-name});\n}\n\n@mixin default-topic($color, $icon-name) {\n --color-topic-title: #{$color};\n --color-topic-title-background: #{rgba($color, 0.2)};\n\n --icon-topic-default: var(--icon-#{$icon-name});\n}\n\n@mixin admonitions {\n @each $name, $values in $admonitions {\n --color-admonition-title--#{$name}: #{nth($values, 1)};\n --color-admonition-title-background--#{$name}: #{rgba(\n nth($values, 1),\n 0.2\n )};\n }\n}\n","// Colors used throughout this theme.\n//\n// The aim is to give the user more control. Thus, instead of hard-coding colors\n// in various parts of the stylesheet, the approach taken is to define all\n// colors as CSS variables and reusing them in all the places.\n//\n// `colors-dark` depends on `colors` being included at a lower specificity.\n\n@mixin colors {\n --color-problematic: #b30000;\n\n // Base Colors\n --color-foreground-primary: black; // for main text and headings\n --color-foreground-secondary: #5a5c63; // for secondary text\n --color-foreground-muted: #646776; // for muted text\n --color-foreground-border: #878787; // for content borders\n\n --color-background-primary: white; // for content\n --color-background-secondary: #f8f9fb; // for navigation + ToC\n --color-background-hover: #efeff4ff; // for navigation-item hover\n --color-background-hover--transparent: #efeff400;\n --color-background-border: #eeebee; // for UI borders\n --color-background-item: #ccc; // for \"background\" items (eg: copybutton)\n\n // Announcements\n --color-announcement-background: #000000dd;\n --color-announcement-text: #eeebee;\n\n // Brand colors\n --color-brand-primary: #2962ff;\n --color-brand-content: #2a5adf;\n\n // API documentation\n --color-api-background: var(--color-background-hover--transparent);\n --color-api-background-hover: var(--color-background-hover);\n --color-api-overall: var(--color-foreground-secondary);\n --color-api-name: var(--color-problematic);\n --color-api-pre-name: var(--color-problematic);\n --color-api-paren: var(--color-foreground-secondary);\n --color-api-keyword: var(--color-foreground-primary);\n --color-highlight-on-target: #ffffcc;\n\n // Inline code background\n --color-inline-code-background: var(--color-background-secondary);\n\n // Highlighted text (search)\n --color-highlighted-background: #ddeeff;\n --color-highlighted-text: var(--color-foreground-primary);\n\n // GUI Labels\n --color-guilabel-background: #ddeeff80;\n --color-guilabel-border: #bedaf580;\n --color-guilabel-text: var(--color-foreground-primary);\n\n // Admonitions!\n --color-admonition-background: transparent;\n\n //////////////////////////////////////////////////////////////////////////////\n // Everything below this should be one of:\n // - var(...)\n // - *-gradient(...)\n // - special literal values (eg: transparent, none)\n //////////////////////////////////////////////////////////////////////////////\n\n // Tables\n --color-table-header-background: var(--color-background-secondary);\n --color-table-border: var(--color-background-border);\n\n // Cards\n --color-card-border: var(--color-background-secondary);\n --color-card-background: transparent;\n --color-card-marginals-background: var(--color-background-secondary);\n\n // Header\n --color-header-background: var(--color-background-primary);\n --color-header-border: var(--color-background-border);\n --color-header-text: var(--color-foreground-primary);\n\n // Sidebar (left)\n --color-sidebar-background: var(--color-background-secondary);\n --color-sidebar-background-border: var(--color-background-border);\n\n --color-sidebar-brand-text: var(--color-foreground-primary);\n --color-sidebar-caption-text: var(--color-foreground-muted);\n --color-sidebar-link-text: var(--color-foreground-secondary);\n --color-sidebar-link-text--top-level: var(--color-brand-primary);\n\n --color-sidebar-item-background: var(--color-sidebar-background);\n --color-sidebar-item-background--current: var(\n --color-sidebar-item-background\n );\n --color-sidebar-item-background--hover: linear-gradient(\n 90deg,\n var(--color-background-hover--transparent) 0%,\n var(--color-background-hover) var(--sidebar-item-spacing-horizontal),\n var(--color-background-hover) 100%\n );\n\n --color-sidebar-item-expander-background: transparent;\n --color-sidebar-item-expander-background--hover: var(\n --color-background-hover\n );\n\n --color-sidebar-search-text: var(--color-foreground-primary);\n --color-sidebar-search-background: var(--color-background-secondary);\n --color-sidebar-search-background--focus: var(--color-background-primary);\n --color-sidebar-search-border: var(--color-background-border);\n --color-sidebar-search-icon: var(--color-foreground-muted);\n\n // Table of Contents (right)\n --color-toc-background: var(--color-background-primary);\n --color-toc-title-text: var(--color-foreground-muted);\n --color-toc-item-text: var(--color-foreground-secondary);\n --color-toc-item-text--hover: var(--color-foreground-primary);\n --color-toc-item-text--active: var(--color-brand-primary);\n\n // Actual page contents\n --color-content-foreground: var(--color-foreground-primary);\n --color-content-background: transparent;\n\n // Links\n --color-link: var(--color-brand-content);\n --color-link--hover: var(--color-brand-content);\n --color-link-underline: var(--color-background-border);\n --color-link-underline--hover: var(--color-foreground-border);\n}\n\n@mixin colors-dark {\n --color-problematic: #ee5151;\n\n // Base Colors\n --color-foreground-primary: #ffffffcc; // for main text and headings\n --color-foreground-secondary: #9ca0a5; // for secondary text\n --color-foreground-muted: #81868d; // for muted text\n --color-foreground-border: #666666; // for content borders\n\n --color-background-primary: #131416; // for content\n --color-background-secondary: #1a1c1e; // for navigation + ToC\n --color-background-hover: #1e2124ff; // for navigation-item hover\n --color-background-hover--transparent: #1e212400;\n --color-background-border: #303335; // for UI borders\n --color-background-item: #444; // for \"background\" items (eg: copybutton)\n\n // Announcements\n --color-announcement-background: #000000dd;\n --color-announcement-text: #eeebee;\n\n // Brand colors\n --color-brand-primary: #2b8cee;\n --color-brand-content: #368ce2;\n\n // Highlighted text (search)\n --color-highlighted-background: #083563;\n\n // GUI Labels\n --color-guilabel-background: #08356380;\n --color-guilabel-border: #13395f80;\n\n // API documentation\n --color-api-keyword: var(--color-foreground-secondary);\n --color-highlight-on-target: #333300;\n\n // Admonitions\n --color-admonition-background: #18181a;\n\n // Cards\n --color-card-border: var(--color-background-secondary);\n --color-card-background: #18181a;\n --color-card-marginals-background: var(--color-background-hover);\n}\n","// This file contains the styling for making the content throughout the page,\n// including fonts, paragraphs, headings and spacing among these elements.\n\nbody\n font-family: var(--font-stack)\npre,\ncode,\nkbd,\nsamp\n font-family: var(--font-stack--monospace)\n\n// Make fonts look slightly nicer.\nbody\n -webkit-font-smoothing: antialiased\n -moz-osx-font-smoothing: grayscale\n\n// Line height from Bootstrap 4.1\narticle\n line-height: 1.5\n\n//\n// Headings\n//\nh1,\nh2,\nh3,\nh4,\nh5,\nh6\n line-height: 1.25\n font-weight: bold\n\n border-radius: 0.5rem\n margin-top: 0.5rem\n margin-bottom: 0.5rem\n margin-left: -0.5rem\n margin-right: -0.5rem\n padding-left: 0.5rem\n padding-right: 0.5rem\n\n + p\n margin-top: 0\n\nh1\n font-size: 2.5em\n margin-top: 1.75rem\n margin-bottom: 1rem\nh2\n font-size: 2em\n margin-top: 1.75rem\nh3\n font-size: 1.5em\nh4\n font-size: 1.25em\nh5\n font-size: 1.125em\nh6\n font-size: 1em\n\nsmall\n opacity: 75%\n font-size: 80%\n\n// Paragraph\np\n margin-top: 0.5rem\n margin-bottom: 0.75rem\n\n// Horizontal rules\nhr.docutils\n height: 1px\n padding: 0\n margin: 2rem 0\n background-color: var(--color-background-border)\n border: 0\n\n.centered\n text-align: center\n\n// Links\na\n text-decoration: underline\n\n color: var(--color-link)\n text-decoration-color: var(--color-link-underline)\n\n &:hover\n color: var(--color-link--hover)\n text-decoration-color: var(--color-link-underline--hover)\n &.muted-link\n color: inherit\n &:hover\n color: var(--color-link)\n text-decoration-color: var(--color-link-underline--hover)\n","// This file contains the styles for the overall layouting of the documentation\n// skeleton, including the responsive changes as well as sidebar toggles.\n//\n// This is implemented as a mobile-last design, which isn't ideal, but it is\n// reasonably good-enough and I got pretty tired by the time I'd finished this\n// to move the rules around to fix this. Shouldn't take more than 3-4 hours,\n// if you know what you're doing tho.\n\n// HACK: Not all browsers account for the scrollbar width in media queries.\n// This results in horizontal scrollbars in the breakpoint where we go\n// from displaying everything to hiding the ToC. We accomodate for this by\n// adding a bit of padding to the TOC drawer, disabling the horizontal\n// scrollbar and allowing the scrollbars to cover the padding.\n// https://www.456bereastreet.com/archive/201301/media_query_width_and_vertical_scrollbars/\n\n// HACK: Always having the scrollbar visible, prevents certain browsers from\n// causing the content to stutter horizontally between taller-than-viewport and\n// not-taller-than-viewport pages.\n\nhtml\n overflow-x: hidden\n overflow-y: scroll\n scroll-behavior: smooth\n\n.sidebar-scroll, .toc-scroll, article[role=main] *\n // Override Firefox scrollbar style\n scrollbar-width: thin\n scrollbar-color: var(--color-foreground-border) transparent\n\n // Override Chrome scrollbar styles\n &::-webkit-scrollbar\n width: 0.25rem\n height: 0.25rem\n &::-webkit-scrollbar-thumb\n background-color: var(--color-foreground-border)\n border-radius: 0.125rem\n\n//\n// Overalls\n//\nhtml,\nbody\n height: 100%\n color: var(--color-foreground-primary)\n background: var(--color-background-primary)\n\narticle\n color: var(--color-content-foreground)\n background: var(--color-content-background)\n overflow-wrap: break-word\n\n.page\n display: flex\n // fill the viewport for pages with little content.\n min-height: 100%\n\n.mobile-header\n width: 100%\n height: var(--header-height)\n background-color: var(--color-header-background)\n color: var(--color-header-text)\n border-bottom: 1px solid var(--color-header-border)\n\n // Looks like sub-script/super-script have this, and we need this to\n // be \"on top\" of those.\n z-index: 10\n\n // We don't show the header on large screens.\n display: none\n\n // Add shadow when scrolled\n &.scrolled\n border-bottom: none\n box-shadow: 0 0 0.2rem rgba(0, 0, 0, 0.1), 0 0.2rem 0.4rem rgba(0, 0, 0, 0.2)\n\n .header-center\n a\n color: var(--color-header-text)\n text-decoration: none\n\n.main\n display: flex\n flex: 1\n\n// Sidebar (left) also covers the entire left portion of screen.\n.sidebar-drawer\n box-sizing: border-box\n\n border-right: 1px solid var(--color-sidebar-background-border)\n background: var(--color-sidebar-background)\n\n display: flex\n justify-content: flex-end\n // These next two lines took me two days to figure out.\n width: calc((100% - #{$full-width}) / 2 + #{$sidebar-width})\n min-width: $sidebar-width\n\n// Scroll-along sidebars\n.sidebar-container,\n.toc-drawer\n box-sizing: border-box\n width: $sidebar-width\n\n.toc-drawer\n background: var(--color-toc-background)\n // See HACK described on top of this document\n padding-right: 1rem\n\n.sidebar-sticky,\n.toc-sticky\n position: sticky\n top: 0\n height: min(100%, 100vh)\n height: 100vh\n\n display: flex\n flex-direction: column\n\n.sidebar-scroll,\n.toc-scroll\n flex-grow: 1\n flex-shrink: 1\n\n overflow: auto\n scroll-behavior: smooth\n\n// Central items.\n.content\n padding: 0 $content-padding\n width: $content-width\n\n display: flex\n flex-direction: column\n justify-content: space-between\n\n.icon\n display: inline-block\n height: 1rem\n width: 1rem\n svg\n width: 100%\n height: 100%\n\n//\n// Accommodate announcement banner\n//\n.announcement\n background-color: var(--color-announcement-background)\n color: var(--color-announcement-text)\n\n height: var(--header-height)\n display: flex\n align-items: center\n overflow-x: auto\n & + .page\n min-height: calc(100% - var(--header-height))\n\n.announcement-content\n box-sizing: border-box\n padding: 0.5rem\n min-width: 100%\n white-space: nowrap\n text-align: center\n\n a\n color: var(--color-announcement-text)\n text-decoration-color: var(--color-announcement-text)\n\n &:hover\n color: var(--color-announcement-text)\n text-decoration-color: var(--color-link--hover)\n\n////////////////////////////////////////////////////////////////////////////////\n// Toggles for theme\n////////////////////////////////////////////////////////////////////////////////\n.no-js .theme-toggle-container // don't show theme toggle if there's no JS\n display: none\n\n.theme-toggle-container\n vertical-align: middle\n\n.theme-toggle\n cursor: pointer\n border: none\n padding: 0\n background: transparent\n\n.theme-toggle svg\n vertical-align: middle\n height: 1rem\n width: 1rem\n color: var(--color-foreground-primary)\n display: none\n\n.theme-toggle-header\n float: left\n padding: 1rem 0.5rem\n\n////////////////////////////////////////////////////////////////////////////////\n// Toggles for elements\n////////////////////////////////////////////////////////////////////////////////\n.toc-overlay-icon, .nav-overlay-icon\n display: none\n cursor: pointer\n\n .icon\n color: var(--color-foreground-secondary)\n height: 1rem\n width: 1rem\n\n.toc-header-icon, .nav-overlay-icon\n // for when we set display: flex\n justify-content: center\n align-items: center\n\n.toc-content-icon\n height: 1.5rem\n width: 1.5rem\n\n.content-icon-container\n float: right\n display: flex\n margin-top: 1.5rem\n margin-left: 1rem\n margin-bottom: 1rem\n gap: 0.5rem\n\n .edit-this-page svg\n color: inherit\n height: 1rem\n width: 1rem\n\n.sidebar-toggle\n position: absolute\n display: none\n// \n.sidebar-toggle[name=\"__toc\"]\n left: 20px\n.sidebar-toggle:checked\n left: 40px\n// \n\n.overlay\n position: fixed\n top: 0\n width: 0\n height: 0\n\n transition: width 0ms, height 0ms, opacity 250ms ease-out\n\n opacity: 0\n background-color: rgba(0, 0, 0, 0.54)\n.sidebar-overlay\n z-index: 20\n.toc-overlay\n z-index: 40\n\n// Keep things on top and smooth.\n.sidebar-drawer\n z-index: 30\n transition: left 250ms ease-in-out\n.toc-drawer\n z-index: 50\n transition: right 250ms ease-in-out\n\n// Show the Sidebar\n#__navigation:checked\n & ~ .sidebar-overlay\n width: 100%\n height: 100%\n opacity: 1\n & ~ .page\n .sidebar-drawer\n top: 0\n left: 0\n // Show the toc sidebar\n#__toc:checked\n & ~ .toc-overlay\n width: 100%\n height: 100%\n opacity: 1\n & ~ .page\n .toc-drawer\n top: 0\n right: 0\n\n////////////////////////////////////////////////////////////////////////////////\n// Back to top\n////////////////////////////////////////////////////////////////////////////////\n.back-to-top\n text-decoration: none\n\n display: none\n position: fixed\n left: 0\n top: 1rem\n padding: 0.5rem\n padding-right: 0.75rem\n border-radius: 1rem\n font-size: 0.8125rem\n\n background: var(--color-background-primary)\n box-shadow: 0 0.2rem 0.5rem rgba(0, 0, 0, 0.05), #6b728080 0px 0px 1px 0px\n\n z-index: 10\n\n margin-left: 50%\n transform: translateX(-50%)\n svg\n height: 1rem\n width: 1rem\n fill: currentColor\n display: inline-block\n\n span\n margin-left: 0.25rem\n\n .show-back-to-top &\n display: flex\n align-items: center\n\n////////////////////////////////////////////////////////////////////////////////\n// Responsive layouting\n////////////////////////////////////////////////////////////////////////////////\n// Make things a bit bigger on bigger screens.\n@media (min-width: $full-width + $sidebar-width)\n html\n font-size: 110%\n\n@media (max-width: $full-width)\n // Collapse \"toc\" into the icon.\n .toc-content-icon\n display: flex\n .toc-drawer\n position: fixed\n height: 100vh\n top: 0\n right: -$sidebar-width\n border-left: 1px solid var(--color-background-muted)\n .toc-tree\n border-left: none\n font-size: var(--toc-font-size--mobile)\n\n // Accomodate for a changed content width.\n .sidebar-drawer\n width: calc((100% - #{$full-width - $sidebar-width}) / 2 + #{$sidebar-width})\n\n@media (max-width: $full-width - $sidebar-width)\n // Collapse \"navigation\".\n .nav-overlay-icon\n display: flex\n .sidebar-drawer\n position: fixed\n height: 100vh\n width: $sidebar-width\n\n top: 0\n left: -$sidebar-width\n\n // Swap which icon is visible.\n .toc-header-icon\n display: flex\n .toc-content-icon, .theme-toggle-content\n display: none\n .theme-toggle-header\n display: block\n\n // Show the header.\n .mobile-header\n position: sticky\n top: 0\n display: flex\n justify-content: space-between\n align-items: center\n\n .header-left,\n .header-right\n display: flex\n height: var(--header-height)\n padding: 0 var(--header-padding)\n label\n height: 100%\n width: 100%\n user-select: none\n\n .nav-overlay-icon .icon,\n .theme-toggle svg\n height: 1.25rem\n width: 1.25rem\n\n // Add a scroll margin for the content\n :target\n scroll-margin-top: var(--header-height)\n\n // Show back-to-top below the header\n .back-to-top\n top: calc(var(--header-height) + 0.5rem)\n\n // Center the page, and accommodate for the header.\n .page\n flex-direction: column\n justify-content: center\n .content\n margin-left: auto\n margin-right: auto\n\n@media (max-width: $content-width + 2* $content-padding)\n // Content should respect window limits.\n .content\n width: 100%\n overflow-x: auto\n\n@media (max-width: $content-width)\n .content\n padding: 0 $content-padding--small\n // Don't float sidebars to the right.\n article aside.sidebar\n float: none\n width: 100%\n margin: 1rem 0\n","//\n// The design here is strongly inspired by mkdocs-material.\n.admonition, .topic\n margin: 1rem auto\n padding: 0 0.5rem 0.5rem 0.5rem\n\n background: var(--color-admonition-background)\n\n border-radius: 0.2rem\n box-shadow: 0 0.2rem 0.5rem rgba(0, 0, 0, 0.05), 0 0 0.0625rem rgba(0, 0, 0, 0.1)\n\n font-size: var(--admonition-font-size)\n\n overflow: hidden\n page-break-inside: avoid\n\n // First element should have no margin, since the title has it.\n > :nth-child(2)\n margin-top: 0\n\n // Last item should have no margin, since we'll control that w/ padding\n > :last-child\n margin-bottom: 0\n\n.admonition p.admonition-title,\np.topic-title\n position: relative\n margin: 0 -0.5rem 0.5rem\n padding-left: 2rem\n padding-right: .5rem\n padding-top: .4rem\n padding-bottom: .4rem\n\n font-weight: 500\n font-size: var(--admonition-title-font-size)\n line-height: 1.3\n\n // Our fancy icon\n &::before\n content: \"\"\n position: absolute\n left: 0.5rem\n width: 1rem\n height: 1rem\n\n// Default styles\np.admonition-title\n background-color: var(--color-admonition-title-background)\n &::before\n background-color: var(--color-admonition-title)\n mask-image: var(--icon-admonition-default)\n mask-repeat: no-repeat\n\np.topic-title\n background-color: var(--color-topic-title-background)\n &::before\n background-color: var(--color-topic-title)\n mask-image: var(--icon-topic-default)\n mask-repeat: no-repeat\n\n//\n// Variants\n//\n.admonition\n border-left: 0.2rem solid var(--color-admonition-title)\n\n @each $type, $value in $admonitions\n &.#{$type}\n border-left-color: var(--color-admonition-title--#{$type})\n > .admonition-title\n background-color: var(--color-admonition-title-background--#{$type})\n &::before\n background-color: var(--color-admonition-title--#{$type})\n mask-image: var(--icon-#{nth($value, 2)})\n\n.admonition-todo > .admonition-title\n text-transform: uppercase\n","// This file stylizes the API documentation (stuff generated by autodoc). It's\n// deeply nested due to how autodoc structures the HTML without enough classes\n// to select the relevant items.\n\n// API docs!\ndl[class]:not(.option-list):not(.field-list):not(.footnote):not(.glossary):not(.simple)\n // Tweak the spacing of all the things!\n dd\n margin-left: 2rem\n > :first-child\n margin-top: 0.125rem\n > :last-child\n margin-bottom: 0.75rem\n\n // This is used for the arguments\n .field-list\n margin-bottom: 0.75rem\n\n // \"Headings\" (like \"Parameters\" and \"Return\")\n > dt\n text-transform: uppercase\n font-size: var(--font-size--small)\n\n dd:empty\n margin-bottom: 0.5rem\n dd > ul\n margin-left: -1.2rem\n > li\n > p:nth-child(2)\n margin-top: 0\n // When the last-empty-paragraph follows a paragraph, it doesn't need\n // to augument the existing spacing.\n > p + p:last-child:empty\n margin-top: 0\n margin-bottom: 0\n\n // Colorize the elements\n > dt\n color: var(--color-api-overall)\n\n.sig:not(.sig-inline)\n font-weight: bold\n\n font-size: var(--api-font-size)\n font-family: var(--font-stack--monospace)\n\n margin-left: -0.25rem\n margin-right: -0.25rem\n padding-top: 0.25rem\n padding-bottom: 0.25rem\n padding-right: 0.5rem\n\n // These are intentionally em, to properly match the font size.\n padding-left: 3em\n text-indent: -2.5em\n\n border-radius: 0.25rem\n\n background: var(--color-api-background)\n transition: background 100ms ease-out\n\n &:hover\n background: var(--color-api-background-hover)\n\n // adjust the size of the [source] link on the right.\n a.reference\n .viewcode-link\n font-weight: normal\n width: 3.5rem\n\nem.property\n font-style: normal\n &:first-child\n color: var(--color-api-keyword)\n.sig-name\n color: var(--color-api-name)\n.sig-prename\n font-weight: normal\n color: var(--color-api-pre-name)\n.sig-paren\n color: var(--color-api-paren)\n.sig-param\n font-style: normal\n\n.versionmodified\n font-style: italic\ndiv.versionadded, div.versionchanged, div.deprecated\n p\n margin-top: 0.125rem\n margin-bottom: 0.125rem\n\n// Align the [docs] and [source] to the right.\n.viewcode-link, .viewcode-back\n float: right\n text-align: right\n",".line-block\n margin-top: 0.5rem\n margin-bottom: 0.75rem\n .line-block\n margin-top: 0rem\n margin-bottom: 0rem\n padding-left: 1rem\n","// Captions\narticle p.caption,\ntable > caption,\n.code-block-caption\n font-size: var(--font-size--small)\n text-align: center\n\n// Caption above a TOCTree\n.toctree-wrapper.compound\n .caption, :not(.caption) > .caption-text\n font-size: var(--font-size--small)\n text-transform: uppercase\n\n text-align: initial\n margin-bottom: 0\n\n > ul\n margin-top: 0\n margin-bottom: 0\n","// Inline code\ncode.literal, .sig-inline\n background: var(--color-inline-code-background)\n border-radius: 0.2em\n // Make the font smaller, and use padding to recover.\n font-size: var(--font-size--small--2)\n padding: 0.1em 0.2em\n\n pre.literal-block &\n font-size: inherit\n padding: 0\n\n p &\n border: 1px solid var(--color-background-border)\n\n.sig-inline\n font-family: var(--font-stack--monospace)\n\n// Code and Literal Blocks\n$code-spacing-vertical: 0.625rem\n$code-spacing-horizontal: 0.875rem\n\n// Wraps every literal block + line numbers.\ndiv[class*=\" highlight-\"],\ndiv[class^=\"highlight-\"]\n margin: 1em 0\n display: flex\n\n .table-wrapper\n margin: 0\n padding: 0\n\npre\n margin: 0\n padding: 0\n overflow: auto\n\n // Needed to have more specificity than pygments' \"pre\" selector. :(\n article[role=\"main\"] .highlight &\n line-height: 1.5\n\n &.literal-block,\n .highlight &\n font-size: var(--code-font-size)\n padding: $code-spacing-vertical $code-spacing-horizontal\n\n // Make it look like all the other blocks.\n &.literal-block\n margin-top: 1rem\n margin-bottom: 1rem\n\n border-radius: 0.2rem\n background-color: var(--color-code-background)\n color: var(--color-code-foreground)\n\n// All code is always contained in this.\n.highlight\n width: 100%\n border-radius: 0.2rem\n\n // Make line numbers and prompts un-selectable.\n .gp, span.linenos\n user-select: none\n pointer-events: none\n\n // Expand the line-highlighting.\n .hll\n display: block\n margin-left: -$code-spacing-horizontal\n margin-right: -$code-spacing-horizontal\n padding-left: $code-spacing-horizontal\n padding-right: $code-spacing-horizontal\n\n/* Make code block captions be nicely integrated */\n.code-block-caption\n display: flex\n padding: $code-spacing-vertical $code-spacing-horizontal\n\n border-radius: 0.25rem\n border-bottom-left-radius: 0\n border-bottom-right-radius: 0\n font-weight: 300\n border-bottom: 1px solid\n\n background-color: var(--color-code-background)\n color: var(--color-code-foreground)\n border-color: var(--color-background-border)\n\n + div[class]\n margin-top: 0\n pre\n border-top-left-radius: 0\n border-top-right-radius: 0\n\n// When `html_codeblock_linenos_style` is table.\n.highlighttable\n width: 100%\n display: block\n tbody\n display: block\n\n tr\n display: flex\n\n // Line numbers\n td.linenos\n background-color: var(--color-code-background)\n color: var(--color-code-foreground)\n padding: $code-spacing-vertical $code-spacing-horizontal\n padding-right: 0\n border-top-left-radius: 0.2rem\n border-bottom-left-radius: 0.2rem\n\n .linenodiv\n padding-right: $code-spacing-horizontal\n font-size: var(--code-font-size)\n box-shadow: -0.0625rem 0 var(--color-foreground-border) inset\n\n // Actual code\n td.code\n padding: 0\n display: block\n flex: 1\n overflow: hidden\n\n .highlight\n border-top-left-radius: 0\n border-bottom-left-radius: 0\n\n// When `html_codeblock_linenos_style` is inline.\n.highlight\n span.linenos\n display: inline-block\n padding-left: 0\n padding-right: $code-spacing-horizontal\n margin-right: $code-spacing-horizontal\n box-shadow: -0.0625rem 0 var(--color-foreground-border) inset\n","// Inline Footnote Reference\n.footnote-reference\n font-size: var(--font-size--small--4)\n vertical-align: super\n\n// Definition list, listing the content of each note.\n// docutils <= 0.17\ndl.footnote.brackets\n font-size: var(--font-size--small)\n color: var(--color-foreground-secondary)\n\n display: grid\n grid-template-columns: max-content auto\n dt\n margin: 0\n > .fn-backref\n margin-left: 0.25rem\n\n &:after\n content: \":\"\n\n .brackets\n &:before\n content: \"[\"\n &:after\n content: \"]\"\n\n dd\n margin: 0\n padding: 0 1rem\n\n// docutils >= 0.18\naside.footnote\n font-size: var(--font-size--small)\n color: var(--color-foreground-secondary)\n\naside.footnote > span,\ndiv.citation > span\n float: left\n font-weight: 500\n padding-right: 0.25rem\n\naside.footnote > p,\ndiv.citation > p\n margin-left: 2rem\n","//\n// Figures\n//\nimg\n box-sizing: border-box\n max-width: 100%\n height: auto\n\narticle\n figure, .figure\n border-radius: 0.2rem\n\n margin: 0\n :last-child\n margin-bottom: 0\n\n .align-left\n float: left\n clear: left\n margin: 0 1rem 1rem\n\n .align-right\n float: right\n clear: right\n margin: 0 1rem 1rem\n\n .align-default,\n .align-center\n display: block\n text-align: center\n margin-left: auto\n margin-right: auto\n\n // WELL, table needs to be stylised like a table.\n table.align-default\n display: table\n text-align: initial\n",".genindex-jumpbox, .domainindex-jumpbox\n border-top: 1px solid var(--color-background-border)\n border-bottom: 1px solid var(--color-background-border)\n padding: 0.25rem\n\n.genindex-section, .domainindex-section\n h2\n margin-top: 0.75rem\n margin-bottom: 0.5rem\n ul\n margin-top: 0\n margin-bottom: 0\n","ul,\nol\n padding-left: 1.2rem\n\n // Space lists out like paragraphs\n margin-top: 1rem\n margin-bottom: 1rem\n // reduce margins within li.\n li\n > p:first-child\n margin-top: 0.25rem\n margin-bottom: 0.25rem\n\n > p:last-child\n margin-top: 0.25rem\n\n > ul,\n > ol\n margin-top: 0.5rem\n margin-bottom: 0.5rem\n\nol\n &.arabic\n list-style: decimal\n &.loweralpha\n list-style: lower-alpha\n &.upperalpha\n list-style: upper-alpha\n &.lowerroman\n list-style: lower-roman\n &.upperroman\n list-style: upper-roman\n\n// Don't space lists out when they're \"simple\" or in a `.. toctree::`\n.simple,\n.toctree-wrapper\n li\n > ul,\n > ol\n margin-top: 0\n margin-bottom: 0\n\n// Definition Lists\n.field-list,\n.option-list,\ndl:not([class]),\ndl.simple,\ndl.footnote,\ndl.glossary\n dt\n font-weight: 500\n margin-top: 0.25rem\n + dt\n margin-top: 0\n\n .classifier::before\n content: \":\"\n margin-left: 0.2rem\n margin-right: 0.2rem\n\n dd\n > p:first-child,\n ul\n margin-top: 0.125rem\n\n ul\n margin-bottom: 0.125rem\n",".math-wrapper\n width: 100%\n overflow-x: auto\n\ndiv.math\n position: relative\n text-align: center\n\n .headerlink,\n &:focus .headerlink\n display: none\n\n &:hover .headerlink\n display: inline-block\n\n span.eqno\n position: absolute\n right: 0.5rem\n top: 50%\n transform: translate(0, -50%)\n z-index: 1\n","// Abbreviations\nabbr[title]\n cursor: help\n\n// \"Problematic\" content, as identified by Sphinx\n.problematic\n color: var(--color-problematic)\n\n// Keyboard / Mouse \"instructions\"\nkbd:not(.compound)\n margin: 0 0.2rem\n padding: 0 0.2rem\n border-radius: 0.2rem\n border: 1px solid var(--color-foreground-border)\n color: var(--color-foreground-primary)\n vertical-align: text-bottom\n\n font-size: var(--font-size--small--3)\n display: inline-block\n\n box-shadow: 0 0.0625rem 0 rgba(0, 0, 0, 0.2), inset 0 0 0 0.125rem var(--color-background-primary)\n\n background-color: var(--color-background-secondary)\n\n// Blockquote\nblockquote\n border-left: 4px solid var(--color-background-border)\n background: var(--color-background-secondary)\n\n margin-left: 0\n margin-right: 0\n padding: 0.5rem 1rem\n\n .attribution\n font-weight: 600\n text-align: right\n\n &.pull-quote,\n &.highlights\n font-size: 1.25em\n\n &.epigraph,\n &.pull-quote\n border-left-width: 0\n border-radius: 0.5rem\n\n &.highlights\n border-left-width: 0\n background: transparent\n\n// Center align embedded-in-text images\np .reference img\n vertical-align: middle\n","p.rubric\n line-height: 1.25\n font-weight: bold\n font-size: 1.125em\n\n // For Numpy-style documentation that's got rubrics within it.\n // https://github.com/pradyunsg/furo/discussions/505\n dd &\n line-height: inherit\n font-weight: inherit\n\n font-size: var(--font-size--small)\n text-transform: uppercase\n","article .sidebar\n float: right\n clear: right\n width: 30%\n\n margin-left: 1rem\n margin-right: 0\n\n border-radius: 0.2rem\n background-color: var(--color-background-secondary)\n border: var(--color-background-border) 1px solid\n\n > *\n padding-left: 1rem\n padding-right: 1rem\n\n > ul, > ol // lists need additional padding, because bullets.\n padding-left: 2.2rem\n\n .sidebar-title\n margin: 0\n padding: 0.5rem 1rem\n border-bottom: var(--color-background-border) 1px solid\n\n font-weight: 500\n\n// TODO: subtitle\n// TODO: dedicated variables?\n",".table-wrapper\n width: 100%\n overflow-x: auto\n margin-top: 1rem\n margin-bottom: 0.5rem\n padding: 0.2rem 0.2rem 0.75rem\n\ntable.docutils\n border-radius: 0.2rem\n border-spacing: 0\n border-collapse: collapse\n\n box-shadow: 0 0.2rem 0.5rem rgba(0, 0, 0, 0.05), 0 0 0.0625rem rgba(0, 0, 0, 0.1)\n\n th\n background: var(--color-table-header-background)\n\n td,\n th\n // Space things out properly\n padding: 0 0.25rem\n\n // Get the borders looking just-right.\n border-left: 1px solid var(--color-table-border)\n border-right: 1px solid var(--color-table-border)\n border-bottom: 1px solid var(--color-table-border)\n\n p\n margin: 0.25rem\n\n &:first-child\n border-left: none\n &:last-child\n border-right: none\n\n // MyST-parser tables set these classes for control of column alignment\n &.text-left\n text-align: left\n &.text-right\n text-align: right\n &.text-center\n text-align: center\n",":target\n scroll-margin-top: 0.5rem\n\n@media (max-width: $full-width - $sidebar-width)\n :target\n scroll-margin-top: calc(0.5rem + var(--header-height))\n\n // When a heading is selected\n section > span:target\n scroll-margin-top: calc(0.8rem + var(--header-height))\n\n// Permalinks\n.headerlink\n font-weight: 100\n user-select: none\n\nh1,\nh2,\nh3,\nh4,\nh5,\nh6,\ndl dt,\np.caption,\nfigcaption p,\ntable > caption,\n.code-block-caption\n > .headerlink\n margin-left: 0.5rem\n visibility: hidden\n &:hover > .headerlink\n visibility: visible\n\n // Don't change to link-like, if someone adds the contents directive.\n > .toc-backref\n color: inherit\n text-decoration-line: none\n\n// Figure and table captions are special.\nfigure:hover > figcaption > p > .headerlink,\ntable:hover > caption > .headerlink\n visibility: visible\n\n:target >, // Regular section[id] style anchors\nspan:target ~ // Non-regular span[id] style \"extra\" anchors\n h1,\n h2,\n h3,\n h4,\n h5,\n h6\n &:nth-of-type(1)\n background-color: var(--color-highlight-on-target)\n // .headerlink\n // visibility: visible\n code.literal\n background-color: transparent\n\ntable:target > caption,\nfigure:target\n background-color: var(--color-highlight-on-target)\n\n// Inline page contents\n.this-will-duplicate-information-and-it-is-still-useful-here li :target\n background-color: var(--color-highlight-on-target)\n\n// Code block permalinks\n.literal-block-wrapper:target .code-block-caption\n background-color: var(--color-highlight-on-target)\n\n// When a definition list item is selected\n//\n// There isn't really an alternative to !important here, due to the\n// high-specificity of API documentation's selector.\ndt:target\n background-color: var(--color-highlight-on-target) !important\n\n// When a footnote reference is selected\n.footnote > dt:target + dd,\n.footnote-reference:target\n background-color: var(--color-highlight-on-target)\n",".guilabel\n background-color: var(--color-guilabel-background)\n border: 1px solid var(--color-guilabel-border)\n color: var(--color-guilabel-text)\n\n padding: 0 0.3em\n border-radius: 0.5em\n font-size: 0.9em\n","// This file contains the styles used for stylizing the footer that's shown\n// below the content.\n\nfooter\n font-size: var(--font-size--small)\n display: flex\n flex-direction: column\n\n margin-top: 2rem\n\n// Bottom of page information\n.bottom-of-page\n display: flex\n align-items: center\n justify-content: space-between\n\n margin-top: 1rem\n padding-top: 1rem\n padding-bottom: 1rem\n\n color: var(--color-foreground-secondary)\n border-top: 1px solid var(--color-background-border)\n\n line-height: 1.5\n\n @media (max-width: $content-width)\n text-align: center\n flex-direction: column-reverse\n gap: 0.25rem\n\n .left-details\n font-size: var(--font-size--small)\n\n .right-details\n display: flex\n flex-direction: column\n gap: 0.25rem\n text-align: right\n\n .icons\n display: flex\n justify-content: flex-end\n gap: 0.25rem\n font-size: 1rem\n\n a\n text-decoration: none\n\n svg,\n img\n font-size: 1.125rem\n height: 1em\n width: 1em\n\n// Next/Prev page information\n.related-pages\n a\n display: flex\n align-items: center\n\n text-decoration: none\n &:hover .page-info .title\n text-decoration: underline\n color: var(--color-link)\n text-decoration-color: var(--color-link-underline)\n\n svg.furo-related-icon,\n svg.furo-related-icon > use\n flex-shrink: 0\n\n color: var(--color-foreground-border)\n\n width: 0.75rem\n height: 0.75rem\n margin: 0 0.5rem\n\n &.next-page\n max-width: 50%\n\n float: right\n clear: right\n text-align: right\n\n &.prev-page\n max-width: 50%\n\n float: left\n clear: left\n\n svg\n transform: rotate(180deg)\n\n.page-info\n display: flex\n flex-direction: column\n overflow-wrap: anywhere\n\n .next-page &\n align-items: flex-end\n\n .context\n display: flex\n align-items: center\n\n padding-bottom: 0.1rem\n\n color: var(--color-foreground-muted)\n font-size: var(--font-size--small)\n text-decoration: none\n","// This file contains the styles for the contents of the left sidebar, which\n// contains the navigation tree, logo, search etc.\n\n////////////////////////////////////////////////////////////////////////////////\n// Brand on top of the scrollable tree.\n////////////////////////////////////////////////////////////////////////////////\n.sidebar-brand\n display: flex\n flex-direction: column\n flex-shrink: 0\n\n padding: var(--sidebar-item-spacing-vertical) var(--sidebar-item-spacing-horizontal)\n text-decoration: none\n\n.sidebar-brand-text\n color: var(--color-sidebar-brand-text)\n overflow-wrap: break-word\n margin: var(--sidebar-item-spacing-vertical) 0\n font-size: 1.5rem\n\n.sidebar-logo-container\n margin: var(--sidebar-item-spacing-vertical) 0\n\n.sidebar-logo\n margin: 0 auto\n display: block\n max-width: 100%\n\n////////////////////////////////////////////////////////////////////////////////\n// Search\n////////////////////////////////////////////////////////////////////////////////\n.sidebar-search-container\n display: flex\n align-items: center\n margin-top: var(--sidebar-search-space-above)\n\n position: relative\n\n background: var(--color-sidebar-search-background)\n &:hover,\n &:focus-within\n background: var(--color-sidebar-search-background--focus)\n\n &::before\n content: \"\"\n position: absolute\n left: var(--sidebar-item-spacing-horizontal)\n width: var(--sidebar-search-icon-size)\n height: var(--sidebar-search-icon-size)\n\n background-color: var(--color-sidebar-search-icon)\n mask-image: var(--icon-search)\n\n.sidebar-search\n box-sizing: border-box\n\n border: none\n border-top: 1px solid var(--color-sidebar-search-border)\n border-bottom: 1px solid var(--color-sidebar-search-border)\n\n padding-top: var(--sidebar-search-input-spacing-vertical)\n padding-bottom: var(--sidebar-search-input-spacing-vertical)\n padding-right: var(--sidebar-search-input-spacing-horizontal)\n padding-left: calc(var(--sidebar-item-spacing-horizontal) + var(--sidebar-search-input-spacing-horizontal) + var(--sidebar-search-icon-size))\n\n width: 100%\n\n color: var(--color-sidebar-search-foreground)\n background: transparent\n z-index: 10\n\n &:focus\n outline: none\n\n &::placeholder\n font-size: var(--sidebar-search-input-font-size)\n\n//\n// Hide Search Matches link\n//\n#searchbox .highlight-link\n padding: var(--sidebar-item-spacing-vertical) var(--sidebar-item-spacing-horizontal) 0\n margin: 0\n text-align: center\n\n a\n color: var(--color-sidebar-search-icon)\n font-size: var(--font-size--small--2)\n\n////////////////////////////////////////////////////////////////////////////////\n// Structure/Skeleton of the navigation tree (left)\n////////////////////////////////////////////////////////////////////////////////\n.sidebar-tree\n font-size: var(--sidebar-item-font-size)\n margin-top: var(--sidebar-tree-space-above)\n margin-bottom: var(--sidebar-item-spacing-vertical)\n\n ul\n padding: 0\n margin-top: 0\n margin-bottom: 0\n\n display: flex\n flex-direction: column\n\n list-style: none\n\n li\n position: relative\n margin: 0\n\n > ul\n margin-left: var(--sidebar-item-spacing-horizontal)\n\n .icon\n color: var(--color-sidebar-link-text)\n\n .reference\n box-sizing: border-box\n color: var(--color-sidebar-link-text)\n\n // Fill the parent.\n display: inline-block\n line-height: var(--sidebar-item-line-height)\n text-decoration: none\n\n // Don't allow long words to cause wrapping.\n overflow-wrap: anywhere\n\n height: 100%\n width: 100%\n\n padding: var(--sidebar-item-spacing-vertical) var(--sidebar-item-spacing-horizontal)\n\n &:hover\n background: var(--color-sidebar-item-background--hover)\n\n // Add a nice little \"external-link\" arrow here.\n &.external::after\n content: url('data:image/svg+xml,')\n margin: 0 0.25rem\n vertical-align: middle\n color: var(--color-sidebar-link-text)\n\n // Make the current page reference bold.\n .current-page > .reference\n font-weight: bold\n\n label\n position: absolute\n top: 0\n right: 0\n height: var(--sidebar-item-height)\n width: var(--sidebar-expander-width)\n\n cursor: pointer\n user-select: none\n\n display: flex\n justify-content: center\n align-items: center\n\n .caption, :not(.caption) > .caption-text\n font-size: var(--sidebar-caption-font-size)\n color: var(--color-sidebar-caption-text)\n\n font-weight: bold\n text-transform: uppercase\n\n margin: var(--sidebar-caption-space-above) 0 0 0\n padding: var(--sidebar-item-spacing-vertical) var(--sidebar-item-spacing-horizontal)\n\n // If it has children, add a bit more padding to wrap the content to avoid\n // overlapping with the