mirror of
				https://github.com/imfing/hextra.git
				synced 2025-10-25 14:30:14 -04:00 
			
		
		
		
	Compare commits
	
		
			366 Commits
		
	
	
		
			v0.2.3
			...
			copilot/ad
		
	
	| Author | SHA1 | Date | |
|---|---|---|---|
| ![copilot-swe-agent[bot]](/assets/img/avatar_default.png)  | e56169291c | ||
|   | 3551a56b8c | ||
|   | bfeae19076 | ||
|   | b7f4bffce6 | ||
|   | 708358de80 | ||
|   | 8699deb1dd | ||
|   | a03dbf463f | ||
|   | 1c06ae5580 | ||
|   | ccb63d60f1 | ||
|   | 3bc454bbf6 | ||
|   | 1b536e27a5 | ||
|   | 0e919e77f8 | ||
|   | 83f3b5052e | ||
|   | f8eae96c11 | ||
|   | ec97808b69 | ||
|   | 334158af7a | ||
|   | 184ee25011 | ||
|   | cc5884dd2a | ||
|   | 493cfba523 | ||
|   | 5846274db7 | ||
|   | 4635bdc846 | ||
|   | 6d0e59b16f | ||
|   | 3abcde4f8e | ||
|   | 82e25c0b0d | ||
|   | bbffff1f52 | ||
|   | f9a94f02a6 | ||
|   | 22f81e2470 | ||
|   | c149af0f74 | ||
|   | 524af14bd1 | ||
|   | fee0481a6c | ||
|   | 546bcc2e26 | ||
|   | a19de798b6 | ||
|   | f297d24189 | ||
|   | 990d24906b | ||
|   | 22c1a4f9df | ||
|   | 7b8e1bdfd1 | ||
|   | 5b6f4218be | ||
|   | f4d75a4e5b | ||
|   | a3635ea638 | ||
|   | 201ce3f763 | ||
|   | 18a9335d4b | ||
|   | 363b1e50ff | ||
|   | b2ff662c8e | ||
|   | e3ef6bcebb | ||
|   | 48bae073cb | ||
|   | 6613f94b75 | ||
|   | 880084b091 | ||
|   | f79bd1a8cf | ||
|   | de97b0ec16 | ||
|   | d0cdd29ee5 | ||
|   | 91cc6b53d8 | ||
|   | 2033d50005 | ||
|   | 80ada64da0 | ||
|   | 776c758825 | ||
|   | ec007d73c0 | ||
|   | eca7665571 | ||
|   | 4d861d7175 | ||
|   | 8025de9d9d | ||
|   | 9f70e8a87b | ||
|   | 6823cc1fe8 | ||
|   | 953042a0c0 | ||
|   | b4d7d982f9 | ||
|   | 9a20f07d4e | ||
|   | b6864a0c19 | ||
|   | 532cbcce10 | ||
|   | 30866e328c | ||
|   | 0bb59d6f49 | ||
|   | 025dd1f211 | ||
|   | af78002014 | ||
|   | 096f0d9c22 | ||
|   | 7ac1d59e9f | ||
|   | ea17ae6cbd | ||
|   | 1081ab8d7f | ||
|   | 46290e10e7 | ||
|   | 7b27743159 | ||
|   | c9795867c4 | ||
|   | 7610118c04 | ||
|   | 05fd0129c2 | ||
|   | 7031718449 | ||
|   | e22b8d5c0e | ||
|   | 32a55bb8ee | ||
|   | b43870a538 | ||
|   | 9c2a9f600b | ||
|   | 7385fe9e2a | ||
|   | b1d40c4a2d | ||
|   | 40b1c5f2f1 | ||
|   | 3a13d44d3c | ||
|   | c24d55ee40 | ||
|   | a44de285b2 | ||
|   | c8a231b650 | ||
|   | 5a6fa55d0a | ||
|   | c497ef700e | ||
|   | 41140af6fa | ||
|   | 33129ca59f | ||
|   | 72c383ef5c | ||
|   | 51b5de23c7 | ||
|   | befce4cd9a | ||
|   | 128235e7e2 | ||
|   | f194bc64e4 | ||
|   | 1cc02a6931 | ||
|   | 1eb4b9ea23 | ||
|   | d08b077acd | ||
|   | 1f88cff7d4 | ||
|   | 83fda0109f | ||
|   | 7fdb6a3a07 | ||
|   | 8021437f77 | ||
|   | b700825943 | ||
|   | c74d44492e | ||
|   | 32f7f6d33f | ||
|   | 38c8ee1168 | ||
|   | 17de708c9f | ||
|   | a338c363ed | ||
|   | c3ce3b67e6 | ||
|   | 9d57dbd9cd | ||
|   | a2718d8aa3 | ||
|   | 2b83a3762f | ||
|   | aad859d72e | ||
|   | 6a2f11d780 | ||
|   | 662d9202dc | ||
|   | 9f9ddd69ab | ||
|   | 96b2f6145d | ||
|   | c2286c9dd1 | ||
|   | 49b1cd11ee | ||
|   | b2e6c30c7f | ||
|   | 8eb348323d | ||
|   | ab56c66ae5 | ||
|   | ddc017b8dc | ||
|   | ce837dca42 | ||
|   | c60d1f5de7 | ||
|   | bbe3c46320 | ||
|   | cb3373d500 | ||
|   | 56f28e6f14 | ||
|   | 14036ffea6 | ||
|   | a1232ecf9f | ||
|   | a933f464f5 | ||
|   | 709a407b2e | ||
|   | 876eb3abff | ||
|   | a27f6eef5e | ||
|   | 323f4c4b44 | ||
|   | b8f617f1b0 | ||
|   | 9632c4d05a | ||
|   | 594b1f190c | ||
|   | 12815aaddc | ||
|   | e532637cbc | ||
|   | 79bb4504a0 | ||
|   | 26a298da5d | ||
|   | cf61e606c1 | ||
|   | 0716533699 | ||
|   | 9efcda2fdd | ||
|   | 655148f329 | ||
|   | b6d14afca3 | ||
|   | bc778ee243 | ||
|   | f377609eba | ||
|   | a9b992436e | ||
| ![dependabot[bot]](/assets/img/avatar_default.png)  | 7f5a7f2f5a | ||
| ![dependabot[bot]](/assets/img/avatar_default.png)  | 30fddec3fa | ||
|   | 68dd327312 | ||
|   | 0c90c1aa50 | ||
|   | fe2271b60b | ||
|   | bd34a5bad3 | ||
|   | 0dcf7e7a40 | ||
|   | 86a1f3fd96 | ||
|   | 4c4f43779c | ||
|   | ff85e6951d | ||
|   | ec37876f4d | ||
|   | f65aca556d | ||
|   | 7b7eb0f1f3 | ||
|   | 80fae9f86d | ||
|   | 1358c5b945 | ||
|   | b4d292010b | ||
|   | 37089d237a | ||
|   | 2565f372d1 | ||
|   | a97a1791cc | ||
|   | 97ea67198b | ||
|   | 94624bcac6 | ||
|   | f1f84b1bf9 | ||
|   | d367a443f1 | ||
|   | 36ab5287b5 | ||
|   | 9173f59392 | ||
|   | c70900c25f | ||
|   | cabdb421e3 | ||
|   | de9f9b312e | ||
|   | 2af73b3d7e | ||
|   | d1c3c40a95 | ||
| ![dependabot[bot]](/assets/img/avatar_default.png)  | 086af4d173 | ||
|   | c6de4b5b6b | ||
|   | 66d2bf57ba | ||
|   | 526be88d7b | ||
|   | 2863a3a029 | ||
|   | 852a07b15e | ||
|   | e83c11f31a | ||
|   | f439e6bb87 | ||
|   | 9c0ba06db4 | ||
|   | 5c6ed19c34 | ||
|   | 0986b9ee84 | ||
|   | d2d2a62d5a | ||
|   | e3b582676e | ||
|   | 56f6f19978 | ||
|   | 1313415c8b | ||
|   | 3cba6b9820 | ||
|   | 6ee6ddeacb | ||
|   | db92de0f1b | ||
|   | b2bc4f7098 | ||
|   | 857c4e4ca1 | ||
|   | d3f251b621 | ||
|   | d43ac66494 | ||
|   | 4eca719b0b | ||
|   | daaf281012 | ||
|   | ba7707d4d9 | ||
|   | 25da2baf7f | ||
|   | 234fd39254 | ||
|   | 4a7ab16695 | ||
|   | 19852552c6 | ||
|   | 80a1692ade | ||
|   | 81720c7727 | ||
|   | f98fa8c389 | ||
|   | 3503b4c4f1 | ||
|   | bf38320797 | ||
|   | 6a9117ae7c | ||
|   | be7e0d3f40 | ||
|   | e4cdabff12 | ||
|   | 2f127a2f2f | ||
|   | 781e7000c4 | ||
|   | 8e98791082 | ||
|   | 6d4bbac085 | ||
|   | 5103da4cd5 | ||
|   | 9f2b67c08c | ||
|   | 0d6cbba9df | ||
|   | 1936b46af4 | ||
|   | ab9e40effd | ||
|   | 5080877576 | ||
|   | 24fb13b221 | ||
|   | a5b59b61f1 | ||
|   | ef536af9e8 | ||
|   | d8351aa432 | ||
|   | 741a640b1a | ||
|   | 935ff7f719 | ||
|   | 29e70c8b21 | ||
|   | c634cb83eb | ||
|   | 22d4737b99 | ||
|   | fba95d5336 | ||
|   | 07b67ff112 | ||
|   | 5c7303bee7 | ||
|   | 0e312d3476 | ||
|   | 98546e66b9 | ||
|   | b1f49c091f | ||
|   | 8e6cc68e88 | ||
|   | 9995617c66 | ||
|   | 10907cdc25 | ||
|   | 6bd2cfbd6b | ||
|   | 2d8f42f4c5 | ||
|   | 716af59393 | ||
|   | 7191e25958 | ||
|   | c630805511 | ||
|   | cb274c8ac5 | ||
|   | fbe1aac123 | ||
|   | ff8f2537ca | ||
|   | 728fe21ef1 | ||
|   | fc964f11a6 | ||
|   | 09e9f52019 | ||
|   | f5e4283961 | ||
|   | 10f1d85ab4 | ||
|   | fd9a87a86a | ||
|   | 1f9e7c8b39 | ||
|   | 7772313a53 | ||
|   | 7f0a35ab48 | ||
|   | 35c733b7ee | ||
|   | aad68d8afd | ||
|   | d58a8b5469 | ||
|   | 64ac97b2d6 | ||
|   | 27c976bcc1 | ||
|   | d675d3bc7b | ||
|   | 456c96921a | ||
|   | c2d7ba8ce5 | ||
|   | 8801a04ebe | ||
|   | 21b0acdec5 | ||
|   | c0a1bc32dd | ||
|   | 4ea18168e3 | ||
|   | 23c84e124c | ||
|   | 678f0b86ee | ||
|   | defc9bc11b | ||
|   | 363c8c91ee | ||
|   | 088e9f7821 | ||
|   | 7be079f504 | ||
|   | 10ba5533e5 | ||
|   | 433beed666 | ||
|   | 56a87e559d | ||
|   | 9d68f7c71a | ||
|   | e3f6069968 | ||
|   | de1286c7fc | ||
|   | c0a1d5cb67 | ||
|   | 50f1b084ae | ||
|   | 0d629b67b9 | ||
|   | 7c902b4eb9 | ||
|   | 9744b4d727 | ||
|   | 34aecec9d4 | ||
|   | 15ea31c389 | ||
|   | 141e0d8f8c | ||
|   | 2d2e8aec4c | ||
|   | c6f432566d | ||
|   | ae01ac08b6 | ||
|   | b5ab4ecdcb | ||
|   | cf7b669278 | ||
|   | 53b688f014 | ||
| ![dependabot[bot]](/assets/img/avatar_default.png)  | 8ad5a0cf0e | ||
|   | e135f5a6b4 | ||
|   | 97e6945c04 | ||
|   | 93cb788e52 | ||
|   | 88b0f1b2ab | ||
|   | a31b46f5e3 | ||
|   | 6641d36b98 | ||
|   | e42d01898a | ||
|   | 6cd4c55613 | ||
|   | cb09b7ce1e | ||
|   | 96c6ff073f | ||
|   | 28a20e1e7e | ||
|   | 5f4c7423d0 | ||
|   | 2bc4ed19e3 | ||
|   | 8aa6439132 | ||
|   | b7558aca44 | ||
|   | 55ff819dae | ||
|   | 924d8508d0 | ||
|   | 1b932f260a | ||
|   | 5768ed4695 | ||
|   | f4cea168b1 | ||
|   | e2d00fdcd0 | ||
|   | 103faa24f3 | ||
|   | d1bed05843 | ||
|   | 2df3c563bf | ||
|   | ec02eb34fe | ||
|   | 46dea718e6 | ||
|   | adf5a113fc | ||
|   | 6a19ac31c0 | ||
|   | 3c4ede96df | ||
|   | 01f7e3a425 | ||
|   | da5a087891 | ||
|   | 79883dc7cc | ||
|   | b283227046 | ||
|   | 0e9cf1a519 | ||
|   | fdc30c6cd5 | ||
|   | 3632294706 | ||
|   | 929578192b | ||
|   | c18d5def26 | ||
|   | 4e63aa4f14 | ||
|   | b51bfa3177 | ||
|   | c799160e86 | ||
|   | 00d26dee2c | ||
|   | e9ea9786e9 | ||
|   | 84ac7fe773 | ||
|   | a184cfd41e | ||
|   | 76ac694542 | ||
|   | f70ba59ca0 | ||
|   | 4a9a2850fc | ||
|   | 4553a8eda2 | ||
|   | 237d890f67 | ||
|   | 04e131f93a | ||
|   | 61e41f247b | ||
|   | 6d00cb32b0 | ||
|   | 939acc02a8 | ||
|   | e4c36236df | ||
|   | 4381f31085 | ||
|   | 171399889d | ||
|   | 3bcdf84ad4 | ||
|   | 8e8f7f23c9 | ||
|   | 5b71912ab2 | ||
|   | 34c6f6c7f3 | 
| @@ -3,7 +3,7 @@ | |||||||
|   "features": { |   "features": { | ||||||
|     "ghcr.io/devcontainers/features/hugo:1": { |     "ghcr.io/devcontainers/features/hugo:1": { | ||||||
|       "extended": true, |       "extended": true, | ||||||
|       "version": "0.116.1" |       "version": "0.147.7" | ||||||
|     }, |     }, | ||||||
|     "ghcr.io/devcontainers/features/node:1": {} |     "ghcr.io/devcontainers/features/node:1": {} | ||||||
|   }, |   }, | ||||||
| @@ -18,5 +18,7 @@ | |||||||
|     } |     } | ||||||
|   }, |   }, | ||||||
|   "postCreateCommand": "npm install", |   "postCreateCommand": "npm install", | ||||||
|   "forwardPorts": [1313] |   "forwardPorts": [ | ||||||
|  |     1313 | ||||||
|  |   ] | ||||||
| } | } | ||||||
							
								
								
									
										110
									
								
								.github/CONTRIBUTING.md
									
									
									
									
										vendored
									
									
										Normal file
									
								
							
							
						
						
									
										110
									
								
								.github/CONTRIBUTING.md
									
									
									
									
										vendored
									
									
										Normal file
									
								
							| @@ -0,0 +1,110 @@ | |||||||
|  | # Contribute to Hextra | ||||||
|  |  | ||||||
|  | 👋 Thank you for being interested in contributing to Hextra! As an open source project, we welcome contributions of many forms including bug reports, feature requests, documentation improvements, and code contributions. | ||||||
|  |  | ||||||
|  | <!-- omit in toc --> | ||||||
|  | ## Table of Contents | ||||||
|  |  | ||||||
|  | - [Guidelines](#guidelines) | ||||||
|  |   - [Contributing Code](#contributing-code) | ||||||
|  |   - [Contributing Documentation](#contributing-documentation) | ||||||
|  |   - [💬 GitHub Discussions](#-github-discussions) | ||||||
|  |   - [GitHub Issues](#github-issues) | ||||||
|  | - [Development](#development) | ||||||
|  |   - [Local development setup](#local-development-setup) | ||||||
|  |   - [Project structure](#project-structure) | ||||||
|  |   - [Start the development server](#start-the-development-server) | ||||||
|  |   - [Compile the styles](#compile-the-styles) | ||||||
|  |  | ||||||
|  |  | ||||||
|  | ## Guidelines | ||||||
|  |  | ||||||
|  | ### Contributing Code | ||||||
|  |  | ||||||
|  | To contribute, please follow the ["Fork and Pull Request"][fork and pull] workflow: | ||||||
|  |  | ||||||
|  | Fork the repository, make your changes, and then submit a pull request. | ||||||
|  | Please make sure to include a description of the changes you made and why you made them. | ||||||
|  | Use [Conventional Commits][conventional commits] message to make it easier to understand the changes you made. | ||||||
|  |  | ||||||
|  | ### Contributing Documentation | ||||||
|  |  | ||||||
|  | Similar to contributing code, you can also contribute to the documentation by submitting a pull request. | ||||||
|  |  | ||||||
|  | The documentation site is located in the [`docs`](../docs/) folder. | ||||||
|  | You can make changes to the documentation and create a pull request. A preview of the new documentation will be automatically generated and displayed in the pull request comment via [Netlify][netlify deploy preview]. | ||||||
|  |  | ||||||
|  | ### 💬 GitHub Discussions | ||||||
|  |  | ||||||
|  | We’re using [Discussions][discussions] as a place to connect with other members using Hextra: | ||||||
|  |  | ||||||
|  | - Ask questions you’re wondering about. | ||||||
|  | - Share ideas. | ||||||
|  | - Engage with other users. | ||||||
|  |  | ||||||
|  | ### GitHub Issues | ||||||
|  |  | ||||||
|  | If you find a bug or have a feature request, please [open an issue][issues]. | ||||||
|  |  | ||||||
|  | Please make sure to include a description of the bug or feature you are requesting. If you are reporting a bug, please include steps to reproduce the bug. | ||||||
|  |  | ||||||
|  | We recommend that you search existing [issues][issues] or discussions before opening a new one to prevent duplicates. | ||||||
|  |  | ||||||
|  | ## Development | ||||||
|  |  | ||||||
|  | > **Note** | ||||||
|  | > You can start developing on [GitHub Codespaces][open in codespaces] or use [devcontainer][devcontainer] locally without installing any dependencies. | ||||||
|  |  | ||||||
|  | ### Local development setup | ||||||
|  |  | ||||||
|  | - [Hugo][hugo] >= v0.124.0 (extended version) | ||||||
|  | - [Node.js][nodejs] | ||||||
|  | - [Go][go] | ||||||
|  |  | ||||||
|  | Install dependencies: | ||||||
|  |  | ||||||
|  | ```bash | ||||||
|  | npm i | ||||||
|  | ``` | ||||||
|  |  | ||||||
|  | ### Project structure | ||||||
|  |  | ||||||
|  | - [`assets`](../assets/): CSS styles and JavaScript files. | ||||||
|  | - [`data`](../data/): The theme data files. Now only contains the `icons.yaml` file. | ||||||
|  | - [`docs`](../docs/): The documentation site for the theme. | ||||||
|  | - [`i18n`](../i18n/): The theme translation files. | ||||||
|  | - [`layouts`](../layouts/): The theme layouts. | ||||||
|  | - [`static`](../static/): The static files for the theme. For example, the favicon and the site logo. | ||||||
|  |  | ||||||
|  | Please refer to the [Hugo documentation][hugo] for more information. | ||||||
|  |  | ||||||
|  | ### Start the development server | ||||||
|  |  | ||||||
|  | ```bash | ||||||
|  | npm run dev:theme | ||||||
|  | ``` | ||||||
|  |  | ||||||
|  | It starts the Hugo server on `http://localhost:1313/` for the `docs` content. | ||||||
|  |  | ||||||
|  | ### Compile the styles | ||||||
|  |  | ||||||
|  | For development preview, we compile the Tailwind CSS styles on the fly. But for production, we need to compile the styles first. | ||||||
|  |  | ||||||
|  | ```bash | ||||||
|  | npm run build:css | ||||||
|  | ``` | ||||||
|  |  | ||||||
|  | It will compile the Tailwind CSS styles and generate the `assets/css/compiled/main.css` file. | ||||||
|  |  | ||||||
|  | <!--links--> | ||||||
|  |  | ||||||
|  | [fork and pull]: https://docs.github.com/en/get-started/quickstart/contributing-to-projects | ||||||
|  | [conventional commits]: https://www.conventionalcommits.org | ||||||
|  | [issues]: https://github.com/imfing/hextra/issues | ||||||
|  | [discussions]: https://github.com/imfing/hextra/discussions | ||||||
|  | [nodejs]: https://nodejs.org/en/ | ||||||
|  | [hugo]: https://gohugo.io/ | ||||||
|  | [go]: https://golang.org/doc/install | ||||||
|  | [devcontainer]: https://code.visualstudio.com/docs/devcontainers/containers | ||||||
|  | [open in codespaces]: https://codespaces.new/imfing/hextra | ||||||
|  | [netlify deploy preview]: https://docs.netlify.com/site-deploys/deploy-previews/ | ||||||
							
								
								
									
										15
									
								
								.github/FUNDING.yml
									
									
									
									
										vendored
									
									
										Normal file
									
								
							
							
						
						
									
										15
									
								
								.github/FUNDING.yml
									
									
									
									
										vendored
									
									
										Normal file
									
								
							| @@ -0,0 +1,15 @@ | |||||||
|  | # These are supported funding model platforms | ||||||
|  |  | ||||||
|  | github: imfing | ||||||
|  | patreon: # Replace with a single Patreon username | ||||||
|  | open_collective: # Replace with a single Open Collective username | ||||||
|  | ko_fi: # Replace with a single Ko-fi username | ||||||
|  | tidelift: # Replace with a single Tidelift platform-name/package-name e.g., npm/babel | ||||||
|  | community_bridge: # Replace with a single Community Bridge project-name e.g., cloud-foundry | ||||||
|  | liberapay: # Replace with a single Liberapay username | ||||||
|  | issuehunt: # Replace with a single IssueHunt username | ||||||
|  | lfx_crowdfunding: # Replace with a single LFX Crowdfunding project-name e.g., cloud-foundry | ||||||
|  | polar: # Replace with a single Polar username | ||||||
|  | buy_me_a_coffee: # Replace with a single Buy Me a Coffee username | ||||||
|  | thanks_dev: # Replace with a single thanks.dev username | ||||||
|  | custom: # Replace with up to 4 custom sponsorship URLs e.g., ['link1', 'link2'] | ||||||
							
								
								
									
										42
									
								
								.github/ISSUE_TEMPLATE/bug_report.md
									
									
									
									
										vendored
									
									
										Normal file
									
								
							
							
						
						
									
										42
									
								
								.github/ISSUE_TEMPLATE/bug_report.md
									
									
									
									
										vendored
									
									
										Normal file
									
								
							| @@ -0,0 +1,42 @@ | |||||||
|  | --- | ||||||
|  | name: Bug report | ||||||
|  | about: Create a report to help us improve | ||||||
|  | title: '' | ||||||
|  | labels: '' | ||||||
|  | assignees: '' | ||||||
|  |  | ||||||
|  | --- | ||||||
|  |  | ||||||
|  | **Description** | ||||||
|  |  | ||||||
|  | <!-- Provide a clear and concise description of the bug --> | ||||||
|  |  | ||||||
|  | **Steps To Reproduce** | ||||||
|  |  | ||||||
|  | 1.  | ||||||
|  | 2.  | ||||||
|  | 3.  | ||||||
|  |  | ||||||
|  | <!-- Provide a minimal example or link to a repository that reproduces the bug --> | ||||||
|  |  | ||||||
|  | **Expected Behavior** | ||||||
|  |  | ||||||
|  | <!-- What should have happened? --> | ||||||
|  |  | ||||||
|  | **Actual Behavior** | ||||||
|  |  | ||||||
|  | <!-- What happened instead? --> | ||||||
|  |  | ||||||
|  | **Screenshots** | ||||||
|  |  | ||||||
|  | <!-- If applicable, add screenshots to help explain your problem --> | ||||||
|  |  | ||||||
|  | **Environment** | ||||||
|  |  | ||||||
|  | - Hugo Version: [e.g., 0.85.0] | ||||||
|  | - Browser/OS: [e.g., Chrome, MacOS] | ||||||
|  | - Theme Version: [e.g., v2.0] | ||||||
|  |  | ||||||
|  | **Additional Context** | ||||||
|  |  | ||||||
|  | <!-- Add any other context about the problem here --> | ||||||
							
								
								
									
										24
									
								
								.github/ISSUE_TEMPLATE/feature_request.md
									
									
									
									
										vendored
									
									
										Normal file
									
								
							
							
						
						
									
										24
									
								
								.github/ISSUE_TEMPLATE/feature_request.md
									
									
									
									
										vendored
									
									
										Normal file
									
								
							| @@ -0,0 +1,24 @@ | |||||||
|  | --- | ||||||
|  | name: Feature request | ||||||
|  | about: Suggest an idea for this project | ||||||
|  | title: '' | ||||||
|  | labels: '' | ||||||
|  | assignees: '' | ||||||
|  |  | ||||||
|  | --- | ||||||
|  |  | ||||||
|  | **Feature Description** | ||||||
|  |  | ||||||
|  | <!-- Provide a clear and concise description of the feature --> | ||||||
|  |  | ||||||
|  | **Problem/Solution** | ||||||
|  |  | ||||||
|  | <!-- What problem will this feature solve? Or what new capability will it add? --> | ||||||
|  |  | ||||||
|  | **Alternatives Considered** | ||||||
|  |  | ||||||
|  | <!-- Have you considered any alternative solutions or workarounds? --> | ||||||
|  |  | ||||||
|  | **Additional Context** | ||||||
|  |  | ||||||
|  | <!-- Add any other context or screenshots about the feature request here --> | ||||||
							
								
								
									
										48
									
								
								.github/workflows/pages.yml
									
									
									
									
										vendored
									
									
								
							
							
						
						
									
										48
									
								
								.github/workflows/pages.yml
									
									
									
									
										vendored
									
									
								
							| @@ -1,5 +1,5 @@ | |||||||
| # Sample workflow for building and deploying a Hugo site to GitHub Pages | # Build and deploy Hextra docs site to GitHub Pages | ||||||
| name: Deploy Hugo site to Pages | name: Deploy Hextra docs site to Pages | ||||||
|  |  | ||||||
| on: | on: | ||||||
|   # Runs on pushes targeting the default branch |   # Runs on pushes targeting the default branch | ||||||
| @@ -31,33 +31,43 @@ jobs: | |||||||
|   build: |   build: | ||||||
|     runs-on: ubuntu-latest |     runs-on: ubuntu-latest | ||||||
|     env: |     env: | ||||||
|       HUGO_VERSION: 0.117.0 |       HUGO_VERSION: 0.147.7 | ||||||
|     steps: |     steps: | ||||||
|       - name: Checkout |       - name: Checkout | ||||||
|         uses: actions/checkout@v3 |         uses: actions/checkout@v4 | ||||||
|  |         with: | ||||||
|  |           fetch-depth: 0 # fetch all history for .GitInfo and .Lastmod | ||||||
|  |           fetch-tags: true | ||||||
|  |           submodules: recursive | ||||||
|  |  | ||||||
|       - name: Setup Go |       - name: Setup Go | ||||||
|         uses: actions/setup-go@v4 |         uses: actions/setup-go@v5 | ||||||
|         with: |         with: | ||||||
|           go-version: '1.20' |           go-version: "1.24" | ||||||
|  |  | ||||||
|  |       - name: Setup Pages | ||||||
|  |         id: pages | ||||||
|  |         uses: actions/configure-pages@v5 | ||||||
|  |  | ||||||
|       - name: Setup Hugo |       - name: Setup Hugo | ||||||
|         uses: peaceiris/actions-hugo@v2 |         run: | | ||||||
|         with: |           wget -O ${{ runner.temp }}/hugo.deb https://github.com/gohugoio/hugo/releases/download/v${HUGO_VERSION}/hugo_extended_${HUGO_VERSION}_linux-amd64.deb \ | ||||||
|           hugo-version: ${{ env.HUGO_VERSION }} |           && sudo dpkg -i ${{ runner.temp }}/hugo.deb | ||||||
|           extended: true |  | ||||||
|       - name: Build with Hugo |       - name: Make build script executable | ||||||
|  |         run: chmod +x ./build.sh | ||||||
|  |  | ||||||
|  |       - name: Build all site versions | ||||||
|         env: |         env: | ||||||
|           # For maximum backward compatibility with Hugo modules |  | ||||||
|           HUGO_ENVIRONMENT: production |           HUGO_ENVIRONMENT: production | ||||||
|           HUGO_ENV: production |           HUGO_ENV: production | ||||||
|         run: | |         run: | | ||||||
|           hugo \ |           ./build.sh "${{ steps.pages.outputs.base_url }}" | ||||||
|             --minify \ |  | ||||||
|             --themesDir=../.. --source=exampleSite \ |  | ||||||
|             --baseURL "https://imfing.github.io/hextra/" |  | ||||||
|       - name: Upload artifact |       - name: Upload artifact | ||||||
|         uses: actions/upload-pages-artifact@v2 |         uses: actions/upload-pages-artifact@v3 | ||||||
|         with: |         with: | ||||||
|           path: ./exampleSite/public |           path: ./public | ||||||
|  |  | ||||||
|   # Deployment job |   # Deployment job | ||||||
|   deploy: |   deploy: | ||||||
| @@ -69,4 +79,4 @@ jobs: | |||||||
|     steps: |     steps: | ||||||
|       - name: Deploy to GitHub Pages |       - name: Deploy to GitHub Pages | ||||||
|         id: deployment |         id: deployment | ||||||
|         uses: actions/deploy-pages@v2 |         uses: actions/deploy-pages@v4 | ||||||
|   | |||||||
| @@ -1,4 +1,7 @@ | |||||||
| { | { | ||||||
|  |   "plugins": [ | ||||||
|  |     "prettier-plugin-go-template" | ||||||
|  |   ], | ||||||
|   "goTemplateBracketSpacing": true, |   "goTemplateBracketSpacing": true, | ||||||
|   "htmlWhitespaceSensitivity": "css", |   "htmlWhitespaceSensitivity": "css", | ||||||
|   "printWidth": 200, |   "printWidth": 200, | ||||||
|   | |||||||
							
								
								
									
										3
									
								
								.vscode/settings.json
									
									
									
									
										vendored
									
									
								
							
							
						
						
									
										3
									
								
								.vscode/settings.json
									
									
									
									
										vendored
									
									
								
							| @@ -1,4 +1,5 @@ | |||||||
| { | { | ||||||
|   "editor.tabSize": 2, |   "editor.tabSize": 2, | ||||||
|   "css.customData": [".vscode/tailwind.json"] |   "css.customData": [".vscode/tailwind.json"], | ||||||
|  |   "markdown.extension.toc.levels": "2..6" | ||||||
| } | } | ||||||
|   | |||||||
							
								
								
									
										161
									
								
								CLAUDE.md
									
									
									
									
									
										Normal file
									
								
							
							
						
						
									
										161
									
								
								CLAUDE.md
									
									
									
									
									
										Normal file
									
								
							| @@ -0,0 +1,161 @@ | |||||||
|  | # CLAUDE.md | ||||||
|  |  | ||||||
|  | This file provides guidance to [Claude Code](claude.ai/code) when working with code in this repository. | ||||||
|  |  | ||||||
|  | ## Project Overview | ||||||
|  |  | ||||||
|  | Hextra is a modern, responsive Hugo theme designed for creating documentation websites, technical blogs, and static sites. Built with Tailwind CSS and inspired by Nextra, it offers features like full-text search, dark mode, multi-language support, and extensive customization options. | ||||||
|  |  | ||||||
|  | ## Development Commands | ||||||
|  |  | ||||||
|  | ### Development Server | ||||||
|  |  | ||||||
|  | ```bash | ||||||
|  | # Start development server with theme reloading (recommended for theme development) | ||||||
|  | npm run dev:theme | ||||||
|  |  | ||||||
|  | # Using Task runner | ||||||
|  | task dev | ||||||
|  | ``` | ||||||
|  |  | ||||||
|  | ### Building | ||||||
|  |  | ||||||
|  | ```bash | ||||||
|  | # Build the example site | ||||||
|  | npm run build | ||||||
|  | # or | ||||||
|  | task build | ||||||
|  |  | ||||||
|  | # Build CSS assets only | ||||||
|  | npm run build:css | ||||||
|  | # or | ||||||
|  | task css | ||||||
|  | ``` | ||||||
|  |  | ||||||
|  | ### Alternative Task Commands | ||||||
|  |  | ||||||
|  | The project uses Task runner (`taskfile.yaml`) for simplified commands: | ||||||
|  |  | ||||||
|  | - `task dev` - Start development server (runs `npm run dev:theme`) | ||||||
|  | - `task build` - Build example site | ||||||
|  | - `task css` - Compile CSS (depends on build) | ||||||
|  |  | ||||||
|  | ## Architecture Overview | ||||||
|  |  | ||||||
|  | ### Hugo Theme Structure | ||||||
|  |  | ||||||
|  | - **Base Layout**: `layouts/baseof.html` wraps all pages | ||||||
|  | - **Specialized Layouts**: `layouts/docs/`, `layouts/blog/`, `layouts/hextra-home.html` | ||||||
|  | - **Partials**: Reusable components in `layouts/_partials/` | ||||||
|  |   - Core UI: `navbar.html`, `sidebar.html`, `footer.html`, `breadcrumb.html`, `toc.html` | ||||||
|  |   - Utilities: `layouts/_partials/utils/` for helper functions | ||||||
|  |   - Custom overrides: `layouts/_partials/custom/` for user customizations | ||||||
|  | - **Shortcodes**: Custom Markdown extensions in `layouts/_shortcodes/` | ||||||
|  | - **Render Hooks**: Custom Markdown rendering in `layouts/_markup/` for codeblocks, headings, images, and links | ||||||
|  |  | ||||||
|  | ### Asset Organization | ||||||
|  |  | ||||||
|  | ``` | ||||||
|  | assets/ | ||||||
|  | ├── css/ | ||||||
|  | │   ├── styles.css              # Main stylesheet | ||||||
|  | │   ├── compiled/main.css       # Built CSS output (generated) | ||||||
|  | │   ├── components/             # Component-specific styles | ||||||
|  | │   └── chroma/                 # Syntax highlighting themes | ||||||
|  | ├── js/                         # JavaScript components | ||||||
|  | └── lib/                        # External libraries | ||||||
|  | ``` | ||||||
|  |  | ||||||
|  | ### Key Components | ||||||
|  |  | ||||||
|  | - **Search**: FlexSearch-powered full-text search (`assets/js/flexsearch.js`) | ||||||
|  | - **Navigation**: Responsive navbar and auto-generated sidebar | ||||||
|  | - **Theme Toggle**: Dark/light mode switching | ||||||
|  | - **Internationalization**: 20+ language support in `i18n/` | ||||||
|  |  | ||||||
|  | ### Content Features | ||||||
|  |  | ||||||
|  | - **Shortcodes**: `callout`, `card`, `cards`, `tabs`, `details`, `steps`, `jupyter`, `filetree` | ||||||
|  | - **Code Features**: Syntax highlighting (Chroma), copy buttons, line numbers via render hooks | ||||||
|  | - **SEO**: Open Graph, Twitter Cards, structured data | ||||||
|  | - **Performance**: Minimal JavaScript, optimized CSS with Tailwind | ||||||
|  |  | ||||||
|  | ## Development Workflow | ||||||
|  |  | ||||||
|  | ### Example Site Development | ||||||
|  |  | ||||||
|  | The `docs/` directory serves as both documentation and testing ground: | ||||||
|  |  | ||||||
|  | - Test new features here before releasing | ||||||
|  | - Configuration examples in `docs/hugo.yaml` showing multi-language setup | ||||||
|  | - Content examples demonstrate all theme capabilities | ||||||
|  | - Run from docs with: `hugo server --themesDir=../..` | ||||||
|  |  | ||||||
|  | ### CSS Development Workflow | ||||||
|  |  | ||||||
|  | - Source: `assets/css/styles.css` (main stylesheet) | ||||||
|  | - Build process: Tailwind CSS → PostCSS → `assets/css/compiled/main.css` | ||||||
|  | - Component styles organized in `assets/css/components/` | ||||||
|  | - Chroma syntax highlighting themes in `assets/css/chroma/` | ||||||
|  | - CSS compilation requires Node.js dependencies (PostCSS, Tailwind CSS v4+) | ||||||
|  |  | ||||||
|  | ### Customization Points | ||||||
|  |  | ||||||
|  | - Custom partials: `layouts/_partials/custom/` | ||||||
|  | - Custom CSS: `assets/css/custom.css` | ||||||
|  | - Site-specific overrides: Copy any layout to your site's `layouts/` directory | ||||||
|  |  | ||||||
|  | ## Configuration & Requirements | ||||||
|  |  | ||||||
|  | ### Theme Requirements | ||||||
|  |  | ||||||
|  | - Hugo minimum version: 0.146.0 (extended version required - see `theme.toml`) | ||||||
|  | - Go 1.20+ (as specified in `go.mod`) | ||||||
|  | - Node.js for CSS compilation (PostCSS, Tailwind CSS v4+) | ||||||
|  |  | ||||||
|  | ### Key Configuration Files | ||||||
|  |  | ||||||
|  | - `docs/hugo.yaml` - Example Hugo configuration with multi-language setup | ||||||
|  | - `postcss.config.mjs` - PostCSS configuration for CSS processing | ||||||
|  | - `package.json` - Node.js dependencies and build scripts | ||||||
|  | - `taskfile.yaml` - Task runner configuration | ||||||
|  |  | ||||||
|  | ### Development Environment | ||||||
|  |  | ||||||
|  | - Default Hugo development server: Port 1313 | ||||||
|  | - Development server runs with `--disableFastRender -D` for better development experience | ||||||
|  | - Theme development uses `--logLevel=debug` for detailed logging | ||||||
|  |  | ||||||
|  | ### Multi-language Support | ||||||
|  |  | ||||||
|  | - Configure languages in `hugo.yaml` (supports 20+ languages including RTL) | ||||||
|  | - Translation files in `i18n/` directory (e.g., `en.yaml`, `fa.yaml`, `ja.yaml`, `zh-cn.yaml`) | ||||||
|  | - Example supports English, Persian (RTL), Japanese, and Simplified Chinese | ||||||
|  |  | ||||||
|  | ## Theme Development Guidelines | ||||||
|  |  | ||||||
|  | ### Hugo Theme Conventions | ||||||
|  |  | ||||||
|  | - Theme files in this repository override Hugo defaults | ||||||
|  | - Follow Hugo's theme development guidelines for compatibility | ||||||
|  | - Maintain backward compatibility with existing configurations | ||||||
|  |  | ||||||
|  | ### JavaScript & Performance | ||||||
|  |  | ||||||
|  | - All JavaScript components are designed to have minimal footprint | ||||||
|  | - Core JS components: `theme.js`, `search.js`, `nav-menu.js`, `code-copy.js` | ||||||
|  | - FlexSearch powers offline full-text search functionality | ||||||
|  |  | ||||||
|  | ### CSS Architecture | ||||||
|  |  | ||||||
|  | - Uses Tailwind CSS v4+ with PostCSS processing | ||||||
|  | - Component-based CSS organization in `assets/css/components/` | ||||||
|  | - Compiled output goes to `assets/css/compiled/main.css` | ||||||
|  | - Prettier formatting for Go templates and code consistency | ||||||
|  |  | ||||||
|  | ### Testing & Quality Assurance | ||||||
|  |  | ||||||
|  | - Test all changes in `docs/` before releasing | ||||||
|  | - Use `npm run dev:theme` for theme development with hot reloading | ||||||
|  | - Format code with `npx prettier --write .` before committing | ||||||
|  | - Verify multi-language functionality across supported languages | ||||||
							
								
								
									
										48
									
								
								README.fa.md
									
									
									
									
									
										Normal file
									
								
							
							
						
						
									
										48
									
								
								README.fa.md
									
									
									
									
									
										Normal file
									
								
							| @@ -0,0 +1,48 @@ | |||||||
|  | <div align="center"> | ||||||
|  |   <h1 align="center">هگزترا</h1> | ||||||
|  |   <sup align="center"><a href="README.md">English</a> | <a href="README.zh-cn.md">简体中文</a> | <a href="README.fa.md">فارسی</a></sup> | ||||||
|  |   <p align="center">تم هیوگو مدرن، پاسخگو و دارای امکانات کامل برای ایجاد وبسایتهای استاتیک زیبا.</p> | ||||||
|  |  | ||||||
|  | نسخهی نمایشی → [imfing.github.io/hextra](https://imfing.github.io/hextra/fa) | ||||||
|  | </div> | ||||||
|  |  | ||||||
|  | <picture> | ||||||
|  |   <source media="(prefers-color-scheme: dark)" srcset="https://user-images.githubusercontent.com/5097752/263550533-c18343ca-3848-4230-b5c0-ee989d7916da.png"> | ||||||
|  |   <img alt="Hextra" src="https://user-images.githubusercontent.com/5097752/263550528-663599f9-17a1-4686-b5c4-3da233b5034d.png"> | ||||||
|  | </picture> | ||||||
|  |  | ||||||
|  | <div align="right"> | ||||||
|  | <a href="https://github.com/imfing/hextra/actions/workflows/pages.yml"><img alt="GitHub Actions Status" src="https://github.com/imfing/hextra/actions/workflows/pages.yml/badge.svg"></a> <a href="https://app.netlify.com/sites/hugo-hextra/deploys"><img alt="Netlify Status" src="https://api.netlify.com/api/v1/badges/61d6e55a-2447-487e-b59f-c9537e5df175/deploy-status"></a> | ||||||
|  | </div> | ||||||
|  |  | ||||||
|  | ## ویژگیها | ||||||
|  |  | ||||||
|  | - **طراحی زیبا** - با الهام از Nextra، هگزترا از Tailwind CSS برای ارائه یک طراحی مدرن که سایت شما را برجسته میکند، استفاده میکند. | ||||||
|  | - **طراحی واکنشگرا و حالت تیره** - در تمام دستگاهها، از تلفن همراه، تبلت تا دسکتاپ، عالی به نظر میرسد. حالت تیره نیز برای انطباق با شرایط مختلف روشنایی پشتیبانی میشود. | ||||||
|  | - **سریع و سبک** - طراحی شده توسط Hugo، یک ایجادکننده سایت استاتیک سریع مثل رعد و برق که در یک فایل باینری قرار گرفته است، هگزترا ردپای خود را به حداقل میرساند. برای استفاده از آن به جاوااسکریپت یا Node.js نیازی ندارید. | ||||||
|  | - **جستجوی متن کامل** - جستجوی متن کاملا آفلاین داخلی طراحی شده توسط FlexSearch، بدون نیاز به پیکربندی اضافی. | ||||||
|  | - **امکانات کامل** - برای بهتر کردن محتوای شما مارکداون، برجستهکردن سینتکس، فرمولهای ریاضی LaTeX، نمودارها و عناصر Shortcodeها را شامل میشه. فهرست مطالب، بردکرامب، صفحهبندی، پیمایش نوار کناری و موارد دیگر همه به صورت خودکار تولید میشوند. | ||||||
|  | - **چند زبانه و سئو آماده** - سایتهای چند زبانه با حالت چند زبانه Hugo راحت ساخته میشوند. پشتیبانی خارج از جعبه برای برچسبهای سئو، Open Graph و کارتهای توییتر گنجانده شده است. | ||||||
|  |  | ||||||
|  | ## شروع کنید | ||||||
|  |  | ||||||
|  | ### شروع سریع از طریق Template | ||||||
|  |  | ||||||
|  | استفاده از [Hextra Starter Template](https://github.com/imfing/hextra-starter-template) سادهترین روش برای راهاندازی سریع یک وبسایت جدید با تم هگزترا است. با کلیک بر روی دکمه "Use this template" در بالای صفحه مخزن شروع کنید. | ||||||
|  |  | ||||||
|  | مخزن تم همچنین شامل یک [گردش کار گیتهاب Actions](https://docs.github.com/en/pages/getting-started-with-github-pages/configuring-a-publishing-source-for-your-github-pages-site#publishing-with-a-custom-github-actions-workflow) رای بهکاراندازی وبسایت شما در گیتهاب Pages است. | ||||||
|  |  | ||||||
|  | <img alt="Hextra Starter Template" src="https://user-images.githubusercontent.com/5097752/263551418-c403b9a9-a76c-47a6-8466-513d772ef0b7.jpg" width=600/> | ||||||
|  |  | ||||||
|  | ### استفاده | ||||||
|  |  | ||||||
|  | برای اطلاعات بیشتر به بخش [مستندات](https://imfing.github.io/hextra/fa/docs) مراجعه کنید. | ||||||
|  |  | ||||||
|  | ## مشارکت کردن | ||||||
|  |  | ||||||
|  | از مشارکت افراد جدید استقبال میکنیم. | ||||||
|  |  برای شروع، [راهنمای مشارکت](.github/CONTRIBUTING.md) را بررسی کنید. | ||||||
|  |  | ||||||
|  | ## مجوز | ||||||
|  |  | ||||||
|  | [مجوز MIT](./LICENSE) | ||||||
							
								
								
									
										14
									
								
								README.md
									
									
									
									
									
								
							
							
						
						
									
										14
									
								
								README.md
									
									
									
									
									
								
							| @@ -1,5 +1,6 @@ | |||||||
| <div align="center"> | <div align="center"> | ||||||
|   <h1 align="center">Hextra</h1> |   <h1 align="center">Hextra</h1> | ||||||
|  |   <sup align="center"><a href="README.md">English</a> | <a href="README.zh-cn.md">简体中文</a> | <a href="README.fa.md">فارسی</a></sup> | ||||||
|   <p align="center">Modern, responsive, batteries-included Hugo theme for creating beautiful static websites.</p> |   <p align="center">Modern, responsive, batteries-included Hugo theme for creating beautiful static websites.</p> | ||||||
|  |  | ||||||
| Demo → [imfing.github.io/hextra](https://imfing.github.io/hextra/) | Demo → [imfing.github.io/hextra](https://imfing.github.io/hextra/) | ||||||
| @@ -17,10 +18,10 @@ Demo → [imfing.github.io/hextra](https://imfing.github.io/hextra/) | |||||||
| ## Features | ## Features | ||||||
|  |  | ||||||
| - **Beautiful Design** - Inspired by Nextra, Hextra utilizes Tailwind CSS to offer a modern design that makes your site look outstanding. | - **Beautiful Design** - Inspired by Nextra, Hextra utilizes Tailwind CSS to offer a modern design that makes your site look outstanding. | ||||||
| - **Responsive Layout and Dark Mode** - It looks great on all devices, from mobile, tablet to desktop. Dark mode is also supported to accomodate various lighting conditions. | - **Responsive Layout and Dark Mode** - It looks great on all devices, from mobile to desktop. Dark mode is also supported to accommodate various lighting conditions. | ||||||
| - **Fast and Lightweight** - Powered by Hugo, a lightning-fast static-site generator housed in a single binary file, Hextra keeps its footprint minimal. No Javascript or Node.js are needed to use it. | - **Fast and Lightweight** - Powered by Hugo, a lightning-fast static-site generator housed in a single binary file, Hextra keeps its footprint minimal. No JavaScript or Node.js are needed to use it. | ||||||
| - **Full-text Search** - Built-in offline full-text search powered by FlexSearch, no additional configuration required. | - **Full-text Search** - Built-in offline full-text search powered by FlexSearch, no extra configuration required. | ||||||
| - **Battery-included** - Markdown, syntax highlighting, LaTeX math formulae, diagrams and Shortcodes elements to enhance your content. Table of contents, breadcumbs, pagination, sidebar navigation and more are all automatically generated. | - **Battery-included** - Markdown, syntax highlighting, LaTeX math formulae, diagrams and Shortcodes elements to enhance your content. Table of contents, breadcrumbs, pagination, sidebar navigation and more are all automatically generated. | ||||||
| - **Multi-language and SEO Ready** - Multi-language sites made easy with Hugo's multilingual mode. Out-of-the-box support is included for SEO tags, Open Graph, and Twitter Cards. | - **Multi-language and SEO Ready** - Multi-language sites made easy with Hugo's multilingual mode. Out-of-the-box support is included for SEO tags, Open Graph, and Twitter Cards. | ||||||
|  |  | ||||||
| ## Quick Start | ## Quick Start | ||||||
| @@ -31,13 +32,16 @@ Using the [Hextra Starter Template](https://github.com/imfing/hextra-starter-tem | |||||||
|  |  | ||||||
| The template repository also includes a [GitHub Actions workflow](https://docs.github.com/en/pages/getting-started-with-github-pages/configuring-a-publishing-source-for-your-github-pages-site#publishing-with-a-custom-github-actions-workflow) for deploying your website to GitHub Pages. | The template repository also includes a [GitHub Actions workflow](https://docs.github.com/en/pages/getting-started-with-github-pages/configuring-a-publishing-source-for-your-github-pages-site#publishing-with-a-custom-github-actions-workflow) for deploying your website to GitHub Pages. | ||||||
|  |  | ||||||
|  | <img alt="Hextra Starter Template" src="https://user-images.githubusercontent.com/5097752/263551418-c403b9a9-a76c-47a6-8466-513d772ef0b7.jpg" width=600/> | ||||||
|  |  | ||||||
| ### Usage | ### Usage | ||||||
|  |  | ||||||
| Refer to the [documentation](https://imfing.github.io/hextra/docs) for more information. | Refer to the [documentation](https://imfing.github.io/hextra/docs) for more information. | ||||||
|  |  | ||||||
| ## Contributing | ## Contributing | ||||||
|  |  | ||||||
| This project is actively under development. Contributions are welcome! | Contributions are welcome. | ||||||
|  | Check out the [contributing guide](.github/CONTRIBUTING.md) to get started. | ||||||
|  |  | ||||||
| ## License | ## License | ||||||
|  |  | ||||||
|   | |||||||
| @@ -1,5 +1,6 @@ | |||||||
| <div align="center"> | <div align="center"> | ||||||
|   <h1 align="center">Hextra</h1> |   <h1 align="center">Hextra</h1> | ||||||
|  |   <sup align="center"><a href="README.md">English</a> | <a href="README.zh-cn.md">简体中文</a> | <a href="README.fa.md">فارسی</a></sup> | ||||||
|   <p align="center">用于创建美观的静态站点的现代化, 响应式, 功能强大的 Hugo 主题.</p> |   <p align="center">用于创建美观的静态站点的现代化, 响应式, 功能强大的 Hugo 主题.</p> | ||||||
|  |  | ||||||
| 演示 → [imfing.github.io/hextra](https://imfing.github.io/hextra/) | 演示 → [imfing.github.io/hextra](https://imfing.github.io/hextra/) | ||||||
| @@ -20,7 +21,7 @@ | |||||||
| - **响应式布局和深色模式支持** - 在任何设备上看起来都足够美观, 无论是手机, 平板电脑或者电脑. 深色模式的支持使 Hextra 可以应对各种照明环境. | - **响应式布局和深色模式支持** - 在任何设备上看起来都足够美观, 无论是手机, 平板电脑或者电脑. 深色模式的支持使 Hextra 可以应对各种照明环境. | ||||||
| - **快速且轻量** - 由 Hugo 强力支持, Hugo 是一个快如闪电的静态站点生成器, 这一切都只需一个可执行文件, Hextra 始终保持最小化, 无需 Javascript 或者 Node.js. | - **快速且轻量** - 由 Hugo 强力支持, Hugo 是一个快如闪电的静态站点生成器, 这一切都只需一个可执行文件, Hextra 始终保持最小化, 无需 Javascript 或者 Node.js. | ||||||
| - **全文搜索** - 集成了 Flexsearch 的全文搜索, 无需额外的配置. | - **全文搜索** - 集成了 Flexsearch 的全文搜索, 无需额外的配置. | ||||||
| - **网站中的瑞士军刀** - Markdown, 代码高亮, LaTex 数学公式, diagrams 图表和 Shortcodes 都可以用于丰富你的内容. 目录, 面包屑导航, 分页, 侧边栏等均由 Hextra 自动生成。 | - **功能齐全** - Markdown, 代码高亮, LaTex 数学公式, diagrams 图表和 Shortcodes 都可以用于丰富你的内容. 目录, 面包屑导航, 分页, 侧边栏等均由 Hextra 自动生成。 | ||||||
| - **多语言和 SEO Ready** - Hugo 的多语言模式使得构建多语言网站更简单. 具有 SEO tags, Open Graph, 和 Twitter Cards 等诸多开箱即用的功能. | - **多语言和 SEO Ready** - Hugo 的多语言模式使得构建多语言网站更简单. 具有 SEO tags, Open Graph, 和 Twitter Cards 等诸多开箱即用的功能. | ||||||
|  |  | ||||||
| ## 快速开始 | ## 快速开始 | ||||||
|   | |||||||
| @@ -74,7 +74,7 @@ | |||||||
|   /* CommentPreprocFile */ .chroma .cpf { color: #8b949e; font-weight: bold; font-style: italic } |   /* CommentPreprocFile */ .chroma .cpf { color: #8b949e; font-weight: bold; font-style: italic } | ||||||
|   /* Generic */ .chroma .g {  } |   /* Generic */ .chroma .g {  } | ||||||
|   /* GenericDeleted */ .chroma .gd { color: #ffa198; background-color: #490202 } |   /* GenericDeleted */ .chroma .gd { color: #ffa198; background-color: #490202 } | ||||||
|   /* GenericEmph */ .chroma .ge { font-style: italic } |   /* GenericEmph */ .chroma .ge { color: inherit; font-style: italic } | ||||||
|   /* GenericError */ .chroma .gr { color: #ffa198 } |   /* GenericError */ .chroma .gr { color: #ffa198 } | ||||||
|   /* GenericHeading */ .chroma .gh { color: #79c0ff; font-weight: bold } |   /* GenericHeading */ .chroma .gh { color: #79c0ff; font-weight: bold } | ||||||
|   /* GenericInserted */ .chroma .gi { color: #56d364; background-color: #0f5323 } |   /* GenericInserted */ .chroma .gi { color: #56d364; background-color: #0f5323 } | ||||||
|   | |||||||
										
											
												File diff suppressed because one or more lines are too long
											
										
									
								
							
							
								
								
									
										3
									
								
								assets/css/components/badge.css
									
									
									
									
									
										Normal file
									
								
							
							
						
						
									
										3
									
								
								assets/css/components/badge.css
									
									
									
									
									
										Normal file
									
								
							| @@ -0,0 +1,3 @@ | |||||||
|  | .hextra-badge { | ||||||
|  |   @apply hx:inline-flex hx:items-center; | ||||||
|  | } | ||||||
							
								
								
									
										12
									
								
								assets/css/components/banner.css
									
									
									
									
									
										Normal file
									
								
							
							
						
						
									
										12
									
								
								assets/css/components/banner.css
									
									
									
									
									
										Normal file
									
								
							| @@ -0,0 +1,12 @@ | |||||||
|  | .hextra-banner-hidden .hextra-banner { | ||||||
|  |   display: none; | ||||||
|  | } | ||||||
|  |  | ||||||
|  | .hextra-banner { | ||||||
|  |   :where(a):not(:where([class~=not-prose],[class~=not-prose] *)) { | ||||||
|  |     @apply hx:underline hx:decoration-from-font; | ||||||
|  |   } | ||||||
|  |   :where(p):not(:where([class~=not-prose],[class~=not-prose] *)) { | ||||||
|  |     @apply hx:leading-7 hx:first:mt-0; | ||||||
|  |   } | ||||||
|  | } | ||||||
| @@ -1,16 +1,20 @@ | |||||||
| .hextra-cards { | .hextra-cards { | ||||||
|   grid-template-columns: repeat(auto-fill, minmax(max(250px, calc((100% - 1rem * 2) / var(--rows))), 1fr)); |   grid-template-columns: repeat(auto-fill, minmax(max(250px, calc((100% - 1rem * 2) / var(--hextra-cards-grid-cols))), 1fr)); | ||||||
|  | } | ||||||
|  |  | ||||||
|  | .hextra-card { | ||||||
|  |   position: relative; | ||||||
| } | } | ||||||
|  |  | ||||||
| .hextra-card img { | .hextra-card img { | ||||||
|   user-select: none; |   user-select: none; | ||||||
| } | } | ||||||
|  |  | ||||||
| .hextra-card:hover svg { | .hextra-card:hover .hextra-card-icon svg { | ||||||
|   color: currentColor; |   color: currentColor; | ||||||
| } | } | ||||||
|  |  | ||||||
| .hextra-card svg { | .hextra-card .hextra-card-icon svg { | ||||||
|   width: 1.5rem; |   width: 1.5rem; | ||||||
|   color: #00000033; |   color: #00000033; | ||||||
|   transition: color 0.3s ease; |   transition: color 0.3s ease; | ||||||
| @@ -18,12 +22,25 @@ | |||||||
|  |  | ||||||
| .hextra-card p { | .hextra-card p { | ||||||
|   margin-top: 0.5rem; |   margin-top: 0.5rem; | ||||||
|  |   position: relative; | ||||||
| } | } | ||||||
|  |  | ||||||
| .dark .hextra-card svg { | .dark .hextra-card .hextra-card-icon svg { | ||||||
|   color: #ffffff66; |   color: #ffffff66; | ||||||
| } | } | ||||||
|  |  | ||||||
| .dark .hextra-card:hover svg { | .dark .hextra-card:hover .hextra-card-icon svg { | ||||||
|   color: currentColor; |   color: currentColor; | ||||||
| } | } | ||||||
|  |  | ||||||
|  | .hextra-card-tag { | ||||||
|  |   position: absolute; | ||||||
|  |   z-index: 10; | ||||||
|  |   top: 5px; | ||||||
|  |   &:where(:dir(ltr)) { | ||||||
|  |     right: 5px; | ||||||
|  |   } | ||||||
|  |   &:where(:dir(rtl)) { | ||||||
|  |     left: 5px; | ||||||
|  |   } | ||||||
|  | } | ||||||
|   | |||||||
							
								
								
									
										7
									
								
								assets/css/components/code-copy.css
									
									
									
									
									
										Normal file
									
								
							
							
						
						
									
										7
									
								
								assets/css/components/code-copy.css
									
									
									
									
									
										Normal file
									
								
							| @@ -0,0 +1,7 @@ | |||||||
|  | @supports ( | ||||||
|  |   (-webkit-backdrop-filter: blur(1px)) or (backdrop-filter: blur(1px)) | ||||||
|  | ) { | ||||||
|  |   .hextra-code-copy-btn { | ||||||
|  |     @apply hx:backdrop-blur-md hx:opacity-85 hx:dark:opacity-80; | ||||||
|  |   } | ||||||
|  | } | ||||||
							
								
								
									
										5
									
								
								assets/css/components/hextra/feature-grid.css
									
									
									
									
									
										Normal file
									
								
							
							
						
						
									
										5
									
								
								assets/css/components/hextra/feature-grid.css
									
									
									
									
									
										Normal file
									
								
							| @@ -0,0 +1,5 @@ | |||||||
|  | .hextra-feature-grid { | ||||||
|  |   @media (min-width: 1024px) { | ||||||
|  |     grid-template-columns: repeat(var(--hextra-feature-grid-cols), minmax(0, 1fr)); | ||||||
|  |   } | ||||||
|  | } | ||||||
							
								
								
									
										17
									
								
								assets/css/components/jupyter.css
									
									
									
									
									
										Normal file
									
								
							
							
						
						
									
										17
									
								
								assets/css/components/jupyter.css
									
									
									
									
									
										Normal file
									
								
							| @@ -0,0 +1,17 @@ | |||||||
|  | .hextra-jupyter-code-cell { | ||||||
|  |   scrollbar-gutter: auto; | ||||||
|  |  | ||||||
|  |   @apply hx:mt-6; | ||||||
|  |  | ||||||
|  |   .hextra-jupyter-code-cell-outputs-container { | ||||||
|  |     @apply hx:text-xs hx:overflow-hidden; | ||||||
|  |  | ||||||
|  |     .hextra-jupyter-code-cell-outputs { | ||||||
|  |       @apply hx:overflow-auto hx:max-h-[50vh]; | ||||||
|  |  | ||||||
|  |       pre { | ||||||
|  |         @apply hx:text-xs hx:overflow-auto hx:max-w-full; | ||||||
|  |       } | ||||||
|  |     } | ||||||
|  |   } | ||||||
|  | } | ||||||
| @@ -1,47 +1,50 @@ | |||||||
| nav { | nav { | ||||||
|   .search-wrapper { |   .hextra-search-wrapper { | ||||||
|     @apply hidden md:inline-block; |     @apply hx:hidden hx:md:inline-block; | ||||||
|   } |   } | ||||||
| } | } | ||||||
|  |  | ||||||
| .hamburger-menu svg { | @supports ( | ||||||
|   g { |   (-webkit-backdrop-filter: blur(1px)) or (backdrop-filter: blur(1px)) | ||||||
|     @apply origin-center; | ) { | ||||||
|     transition: transform 0.2s cubic-bezier(0.25, 1, 0.5, 1); |   .hextra-nav-container-blur { | ||||||
|   } |     @apply hx:backdrop-blur-md hx:bg-white/[.85] hx:dark:bg-dark/80!; | ||||||
|   path { |  | ||||||
|     opacity: 1; |  | ||||||
|     transition: |  | ||||||
|       transform 0.2s cubic-bezier(0.25, 1, 0.5, 1) 0.2s, |  | ||||||
|       opacity 0.2s ease 0.2s; |  | ||||||
|   } |  | ||||||
|  |  | ||||||
|   &.open { |  | ||||||
|     path { |  | ||||||
|       transition: |  | ||||||
|         transform 0.2s cubic-bezier(0.25, 1, 0.5, 1), |  | ||||||
|         opacity 0s ease 0.2s; |  | ||||||
|     } |  | ||||||
|     g { |  | ||||||
|       transition: transform 0.2s cubic-bezier(0.25, 1, 0.5, 1) 0.2s; |  | ||||||
|   } |   } | ||||||
| } | } | ||||||
|  |  | ||||||
|   &.open > { | /* Hamburger Menu - Flattened Structure */ | ||||||
|     path { | .hextra-hamburger-menu svg g { | ||||||
|       @apply opacity-0; |   @apply hx:origin-center hx:transition-all hx:duration-100 hx:ease-out; | ||||||
| } | } | ||||||
|     g:nth-of-type(1) { |  | ||||||
|       @apply rotate-45; | .hextra-hamburger-menu svg path { | ||||||
|       path { |   @apply hx:opacity-100 hx:transition-all hx:duration-100 hx:ease-out hx:delay-100; | ||||||
|         transform: translate3d(0, 4px, 0); |  | ||||||
| } | } | ||||||
|  |  | ||||||
|  | .hextra-hamburger-menu svg.open path { | ||||||
|  |   @apply hx:transition-transform hx:duration-100 hx:ease-out hx:delay-0; | ||||||
| } | } | ||||||
|     g:nth-of-type(2) { |  | ||||||
|       @apply -rotate-45; | .hextra-hamburger-menu svg.open g { | ||||||
|       path { |   @apply hx:transition-transform hx:duration-100 hx:ease-out hx:delay-100; | ||||||
|         transform: translate3d(0, -4px, 0); |  | ||||||
| } | } | ||||||
|  |  | ||||||
|  | .hextra-hamburger-menu svg.open > path { | ||||||
|  |   @apply hx:opacity-0; | ||||||
| } | } | ||||||
|  |  | ||||||
|  | .hextra-hamburger-menu svg.open > g:nth-of-type(1) { | ||||||
|  |   @apply hx:rotate-45; | ||||||
| } | } | ||||||
|  |  | ||||||
|  | .hextra-hamburger-menu svg.open > g:nth-of-type(1) path { | ||||||
|  |   @apply hx:translate-y-1; | ||||||
|  | } | ||||||
|  |  | ||||||
|  | .hextra-hamburger-menu svg.open > g:nth-of-type(2) { | ||||||
|  |   @apply hx:-rotate-45; | ||||||
|  | } | ||||||
|  |  | ||||||
|  | .hextra-hamburger-menu svg.open > g:nth-of-type(2) path { | ||||||
|  |   @apply hx:-translate-y-1; | ||||||
| } | } | ||||||
|   | |||||||
							
								
								
									
										21
									
								
								assets/css/components/scrollbar.css
									
									
									
									
									
										Normal file
									
								
							
							
						
						
									
										21
									
								
								assets/css/components/scrollbar.css
									
									
									
									
									
										Normal file
									
								
							| @@ -0,0 +1,21 @@ | |||||||
|  | .hextra-scrollbar, .hextra-scrollbar * { | ||||||
|  |   scrollbar-width: thin; /* Firefox */ | ||||||
|  |   scrollbar-color: oklch(55.55% 0 0 / 40%) transparent; /* Firefox */ | ||||||
|  |  | ||||||
|  |   scrollbar-gutter: stable; | ||||||
|  |   &::-webkit-scrollbar { | ||||||
|  |     @apply hx:w-3 hx:h-3; | ||||||
|  |   } | ||||||
|  |   &::-webkit-scrollbar-track { | ||||||
|  |     @apply hx:bg-transparent; | ||||||
|  |   } | ||||||
|  |   &::-webkit-scrollbar-thumb { | ||||||
|  |     @apply hx:rounded-[10px]; | ||||||
|  |   } | ||||||
|  |   &:hover::-webkit-scrollbar-thumb { | ||||||
|  |     border: 3px solid transparent; | ||||||
|  |     background-color: var(--tw-shadow-color); | ||||||
|  |     background-clip: content-box; | ||||||
|  |     @apply hx:shadow-neutral-500/20 hx:hover:shadow-neutral-500/40; | ||||||
|  |   } | ||||||
|  | } | ||||||
| @@ -1,38 +1,38 @@ | |||||||
| .search-wrapper { | .hextra-search-wrapper { | ||||||
|   li { |   li { | ||||||
|     @apply mx-2.5 break-words rounded-md contrast-more:border text-gray-800 contrast-more:border-transparent dark:text-gray-300; |     @apply hx:mx-2.5 hx:break-words hx:rounded-md hx:contrast-more:border hx:text-gray-800 hx:contrast-more:border-transparent hx:dark:text-gray-300; | ||||||
|     a { |     a { | ||||||
|       @apply block scroll-m-12 px-2.5 py-2; |       @apply hx:focus-visible:outline-none hx:focus:outline-none hx:block hx:scroll-m-12 hx:px-2.5 hx:py-2; | ||||||
|     } |     } | ||||||
|  |  | ||||||
|     .title { |     .hextra-search-title { | ||||||
|       @apply text-base font-semibold leading-5; |       @apply hx:text-base hx:font-semibold hx:leading-5; | ||||||
|     } |     } | ||||||
|  |  | ||||||
|     .active { |     .hextra-search-active { | ||||||
|       @apply rounded-md bg-primary-500/10 contrast-more:border-primary-500; |       @apply hx:rounded-md hx:bg-primary-500/10 hx:contrast-more:border-primary-500; | ||||||
|     } |     } | ||||||
|   } |   } | ||||||
|  |  | ||||||
|   .no-result { |   .hextra-search-no-result { | ||||||
|     @apply block select-none p-8 text-center text-sm text-gray-400; |     @apply hx:block hx:select-none hx:p-8 hx:text-center hx:text-sm hx:text-gray-400; | ||||||
|   } |   } | ||||||
|  |  | ||||||
|   .prefix { |   .hextra-search-prefix { | ||||||
|     @apply mx-2.5 mb-2 mt-6 select-none border-b border-black/10 px-2.5 pb-1.5 text-xs font-semibold |     @apply hx:mx-2.5 hx:mb-2 hx:mt-6 hx:select-none hx:border-b hx:border-black/10 hx:px-2.5 hx:pb-1.5 hx:text-xs hx:font-semibold | ||||||
|     uppercase text-gray-500 first:mt-0 dark:border-white/20 dark:text-gray-300 contrast-more:border-gray-600 |     hx:uppercase hx:text-gray-500 hx:first:mt-0 hx:dark:border-white/20 hx:dark:text-gray-300 hx:contrast-more:border-gray-600 | ||||||
|     contrast-more:text-gray-900 contrast-more:dark:border-gray-50 contrast-more:dark:text-gray-50; |     hx:contrast-more:text-gray-900 hx:contrast-more:dark:border-gray-50 hx:contrast-more:dark:text-gray-50; | ||||||
|   } |   } | ||||||
|  |  | ||||||
|   .excerpt { |   .hextra-search-excerpt { | ||||||
|     @apply overflow-hidden text-ellipsis mt-1 text-sm leading-[1.35rem] text-gray-600 dark:text-gray-400 contrast-more:dark:text-gray-50; |     @apply hx:overflow-hidden hx:text-ellipsis hx:mt-1 hx:text-sm hx:leading-[1.35rem] hx:text-gray-600 hx:dark:text-gray-400 hx:contrast-more:dark:text-gray-50; | ||||||
|     display: -webkit-box; |     display: -webkit-box; | ||||||
|     line-clamp: 1; |     line-clamp: 1; | ||||||
|     -webkit-line-clamp: 1; |     -webkit-line-clamp: 1; | ||||||
|     -webkit-box-orient: vertical; |     -webkit-box-orient: vertical; | ||||||
|   } |   } | ||||||
|  |  | ||||||
|   .match { |   .hextra-search-match { | ||||||
|     @apply text-primary-600; |     @apply hx:text-primary-600; | ||||||
|   } |   } | ||||||
| } | } | ||||||
|   | |||||||
| @@ -1,21 +1,21 @@ | |||||||
| @media (max-width: 767px) { | @media (max-width: 48rem) { | ||||||
|   .sidebar-container { |   .hextra-sidebar-container { | ||||||
|     @apply fixed pt-[calc(var(--navbar-height))] top-0 w-full bottom-0 z-[15] overscroll-contain bg-white dark:bg-dark; |     @apply hx:fixed hx:pt-[calc(var(--navbar-height)+var(--hextra-banner-height))] hx:top-0 hx:w-full hx:bottom-0 hx:z-[15] hx:overscroll-contain hx:bg-white hx:dark:bg-dark; | ||||||
|     transition: transform 0.8s cubic-bezier(0.52, 0.16, 0.04, 1); |     transition: transform 0.4s cubic-bezier(0.52, 0.16, 0.04, 1); | ||||||
|     will-change: transform, opacity; |     will-change: transform, opacity; | ||||||
|     contain: layout style; |     contain: layout style; | ||||||
|     backface-visibility: hidden; |     backface-visibility: hidden; | ||||||
|   } |   } | ||||||
| } | } | ||||||
|  |  | ||||||
| .sidebar-container { | .hextra-sidebar-container { | ||||||
|   li > div { |   li > div { | ||||||
|     @apply h-0; |     @apply hx:h-0; | ||||||
|   } |   } | ||||||
|   li.open > div { |   li.open > div { | ||||||
|     @apply h-auto; |     @apply hx:h-auto hx:pt-1; | ||||||
|   } |   } | ||||||
|   li.open > a > span > svg > path { |   li.open > a > span > svg > path { | ||||||
|     @apply rotate-90; |     @apply hx:rotate-90; | ||||||
|   } |   } | ||||||
| } | } | ||||||
|   | |||||||
| @@ -1,11 +1,22 @@ | |||||||
| .steps h3 { | .hextra-steps { | ||||||
|  |   :where(h2, h3, h4, h5, h6):not(.no-step-marker) { | ||||||
|     counter-increment: step; |     counter-increment: step; | ||||||
|  |     @apply hx:ltr:before:ml-[-41px] hx:rtl:before:mr-[-44px]; | ||||||
|  |     /* https://github.com/tailwindlabs/tailwindcss/issues/15597#issuecomment-2582673546 */ | ||||||
|  |     @apply hx:before:bg-gray-100 hx:dark:before:bg-neutral-800; | ||||||
|  |     @apply hx:before:border-4 hx:before:border-white hx:dark:before:border-dark; | ||||||
|     &:before { |     &:before { | ||||||
|     @apply absolute w-[33px] h-[33px]; |  | ||||||
|     @apply border-4 border-white bg-gray-100 dark:border-dark dark:bg-neutral-800; |  | ||||||
|     @apply rounded-full text-neutral-400 text-base font-normal text-center -indent-px; |  | ||||||
|     @apply mt-[3px] ml-[-41px]; |  | ||||||
|       content: counter(step); |       content: counter(step); | ||||||
|  |       @apply hx:absolute hx:size-[33px]; | ||||||
|  |       @apply hx:rounded-full hx:text-neutral-400 hx:text-base hx:font-normal hx:text-center hx:-indent-px; | ||||||
|  |     } | ||||||
|  |   } | ||||||
|  | } | ||||||
|  |  | ||||||
|  | :lang(fa) .hextra-steps { | ||||||
|  |   :where(h2, h3, h4, h5, h6):not(.no-step-marker) { | ||||||
|  |     &:before { | ||||||
|  |       content: counter(step, persian); | ||||||
|  |     } | ||||||
|   } |   } | ||||||
| } | } | ||||||
|   | |||||||
							
								
								
									
										4
									
								
								assets/css/components/toc.css
									
									
									
									
									
										Normal file
									
								
							
							
						
						
									
										4
									
								
								assets/css/components/toc.css
									
									
									
									
									
										Normal file
									
								
							| @@ -0,0 +1,4 @@ | |||||||
|  | /* Table of Contents Scroll Spy Styles */ | ||||||
|  | .hextra-toc a.hextra-toc-active { | ||||||
|  |   @apply hx:text-gray-900! hx:dark:text-gray-50! hx:transition-all hx:duration-200; | ||||||
|  | } | ||||||
| @@ -1,48 +1,53 @@ | |||||||
| /* Code syntax highlight */ | /* Code syntax highlight */ | ||||||
| @import "chroma/light.css"; | @import "./chroma/light.css"; | ||||||
| @import "chroma/dark.css"; | @import "./chroma/dark.css"; | ||||||
|  |  | ||||||
| .code-block { | .hextra-code-block { | ||||||
|   @apply text-[.9em] leading-5; |   @apply hx:text-[.9em] hx:leading-5; | ||||||
|  |  | ||||||
|   pre { |   pre { | ||||||
|     @apply text-[.9em] bg-primary-700/5 overflow-x-auto font-medium subpixel-antialiased dark:bg-primary-300/10 contrast-more:border contrast-more:border-primary-900/20 contrast-more:contrast-150 contrast-more:dark:border-primary-100/40; |     @apply hx:text-[.9em] hx:bg-primary-700/5 hx:overflow-x-auto hx:font-medium hx:subpixel-antialiased hx:dark:bg-primary-300/10 hx:contrast-more:border hx:contrast-more:border-primary-900/20 hx:contrast-more:contrast-150 hx:contrast-more:dark:border-primary-100/40; | ||||||
|   } |   } | ||||||
|  |  | ||||||
|   .filename { |   .hextra-code-filename { | ||||||
|     @apply absolute top-0 z-[1] w-full truncate rounded-t-xl bg-primary-700/5 py-2 px-4 text-xs text-gray-700 dark:bg-primary-300/10 dark:text-gray-200; |     @apply hx:absolute hx:top-0 hx:z-[1] hx:w-full hx:truncate hx:rounded-t-xl hx:bg-primary-700/5 hx:py-2 hx:px-4 hx:text-xs hx:text-gray-700 hx:dark:bg-primary-300/10 hx:dark:text-gray-200; | ||||||
|  |   } | ||||||
|  |  | ||||||
|  |   .hextra-code-filename + pre:not(.lntable pre) { | ||||||
|  |     /* Override padding for code blocks with filename but no highlight */ | ||||||
|  |     @apply hx:pt-12; | ||||||
|   } |   } | ||||||
| } | } | ||||||
|  |  | ||||||
| .code-block pre:not(.lntable pre) { | .hextra-code-block pre:not(.lntable pre) { | ||||||
|   @apply px-4 mb-4 py-4 rounded-xl; |   @apply hx:px-4 hx:mb-4 hx:py-4 hx:rounded-xl; | ||||||
| } | } | ||||||
|  |  | ||||||
| .code-block div:nth-of-type(2) pre { | .hextra-code-block div:nth-of-type(2) pre { | ||||||
|   @apply pt-12 pb-4; |   @apply hx:pt-12 hx:pb-4; | ||||||
| } | } | ||||||
|  |  | ||||||
| .chroma { | .chroma { | ||||||
|   .lntable { |   .lntable { | ||||||
|     @apply m-0 block w-auto overflow-auto rounded-xl; |     @apply hx:m-0 hx:block hx:w-auto hx:overflow-auto hx:rounded-xl; | ||||||
|  |  | ||||||
|     pre { |     pre { | ||||||
|       @apply pt-4 pb-4; |       @apply hx:pt-4 hx:pb-4; | ||||||
|     } |     } | ||||||
|   } |   } | ||||||
|   .ln, |   .ln, | ||||||
|   .lnt:not(.hl > .lnt), |   .lnt:not(.hl > .lnt), | ||||||
|   .hl { |   .hl:not(.line) { | ||||||
|     @apply pl-4 pr-4 min-w-[2.6rem] text-neutral-600 dark:text-neutral-300; |     @apply hx:pl-4 hx:pr-4 hx:min-w-[2.6rem] hx:text-neutral-600 hx:dark:text-neutral-300; | ||||||
|   } |   } | ||||||
|   .lntd { |   .lntd { | ||||||
|     @apply p-0 align-top; |     @apply hx:p-0 hx:align-top; | ||||||
|   } |   } | ||||||
|   .lntd:last-of-type { |   .lntd:last-of-type { | ||||||
|     @apply w-full; |     @apply hx:w-full; | ||||||
|   } |   } | ||||||
|   /* LineHighlight */ |   /* LineHighlight */ | ||||||
|   .hl { |   .hl { | ||||||
|     @apply block w-full bg-primary-800/10; |     @apply hx:block hx:w-full hx:bg-primary-800/10; | ||||||
|   } |   } | ||||||
| } | } | ||||||
|   | |||||||
							
								
								
									
										1
									
								
								assets/css/safelist.txt
									
									
									
									
									
										Normal file
									
								
							
							
						
						
									
										1
									
								
								assets/css/safelist.txt
									
									
									
									
									
										Normal file
									
								
							| @@ -0,0 +1 @@ | |||||||
|  | hx:max-w-full | ||||||
| @@ -1,31 +1,59 @@ | |||||||
| @import "typography.css"; | @import "tailwindcss" prefix(hx); | ||||||
| @import "highlight.css"; |  | ||||||
| @import "components/cards.css"; |  | ||||||
| @import "components/steps.css"; |  | ||||||
| @import "components/search.css"; |  | ||||||
| @import "components/sidebar.css"; |  | ||||||
| @import "components/navbar.css"; |  | ||||||
|  |  | ||||||
| @tailwind base; | @custom-variant dark (&:where(.dark, .dark *)); | ||||||
| @tailwind components; |  | ||||||
| @tailwind utilities; | @theme { | ||||||
|  |   --color-primary-50: hsl(var(--primary-hue) var(--primary-saturation) calc(var(--primary-lightness) + calc(calc(100% - var(--primary-lightness)) / 50) * 47)); | ||||||
|  |   --color-primary-100: hsl(var(--primary-hue) var(--primary-saturation) calc(var(--primary-lightness) + calc(calc(100% - var(--primary-lightness)) / 50) * 44)); | ||||||
|  |   --color-primary-200: hsl(var(--primary-hue) var(--primary-saturation) calc(var(--primary-lightness) + calc(calc(100% - var(--primary-lightness)) / 50) * 36)); | ||||||
|  |   --color-primary-300: hsl(var(--primary-hue) var(--primary-saturation) calc(var(--primary-lightness) + calc(calc(100% - var(--primary-lightness)) / 50) * 27)); | ||||||
|  |   --color-primary-400: hsl(var(--primary-hue) var(--primary-saturation) calc(var(--primary-lightness) + calc(calc(100% - var(--primary-lightness)) / 50) * 16)); | ||||||
|  |   --color-primary-500: hsl(var(--primary-hue) var(--primary-saturation) var(--primary-lightness)); | ||||||
|  |   --color-primary-600: hsl(var(--primary-hue) var(--primary-saturation) calc(calc(var(--primary-lightness) / 50) * 45)); | ||||||
|  |   --color-primary-700: hsl(var(--primary-hue) var(--primary-saturation) calc(calc(var(--primary-lightness) / 50) * 39)); | ||||||
|  |   --color-primary-800: hsl(var(--primary-hue) var(--primary-saturation) calc(calc(var(--primary-lightness) / 50) * 32)); | ||||||
|  |   --color-primary-900: hsl(var(--primary-hue) var(--primary-saturation) calc(calc(var(--primary-lightness) / 50) * 24)); | ||||||
|  |   --color-dark: #111; | ||||||
|  | } | ||||||
|  |  | ||||||
| html { | html { | ||||||
|   @apply text-base antialiased; |   @apply hx:text-base hx:antialiased; | ||||||
|   font-feature-settings: "rlig" 1, "calt" 1, "ss01" 1; |  | ||||||
|   -webkit-tap-highlight-color: transparent; |  | ||||||
| } | } | ||||||
|  |  | ||||||
| body { | body { | ||||||
|   @apply w-full bg-white dark:bg-dark dark:text-gray-100; |   @apply hx:w-full hx:bg-white hx:dark:bg-dark hx:dark:text-gray-100; | ||||||
| } | } | ||||||
|  |  | ||||||
| :root { | :root { | ||||||
|   --primary-hue: 212deg; |   --primary-hue: 212deg; | ||||||
|  |   --primary-saturation: 100%; | ||||||
|  |   --primary-lightness: 50%; | ||||||
|   --navbar-height: 4rem; |   --navbar-height: 4rem; | ||||||
|   --menu-height: 3.75rem; |   --hextra-banner-height: 2rem; | ||||||
|  |   --menu-height: 3.75rem; /* 60px */ | ||||||
| } | } | ||||||
|  |  | ||||||
| .dark { | .dark { | ||||||
|   --primary-hue: 204deg; |   --primary-hue: 204deg; | ||||||
|  |   --primary-saturation: 100%; | ||||||
|  |   --primary-lightness: 50%; | ||||||
| } | } | ||||||
|  |  | ||||||
|  | @utility hextra-focus { | ||||||
|  |   @apply hx:outline-none hx:ring-2 hx:ring-primary-200 hx:ring-offset-1 hx:ring-offset-primary-300 hx:dark:ring-primary-800 hx:dark:ring-offset-primary-700; | ||||||
|  | } | ||||||
|  |  | ||||||
|  | @import "./typography.css"; | ||||||
|  | @import "./highlight.css"; | ||||||
|  | @import "./components/cards.css"; | ||||||
|  | @import "./components/steps.css"; | ||||||
|  | @import "./components/search.css"; | ||||||
|  | @import "./components/sidebar.css"; | ||||||
|  | @import "./components/banner.css"; | ||||||
|  | @import "./components/navbar.css"; | ||||||
|  | @import "./components/scrollbar.css"; | ||||||
|  | @import "./components/code-copy.css"; | ||||||
|  | @import "./components/hextra/feature-grid.css"; | ||||||
|  | @import "./components/jupyter.css"; | ||||||
|  | @import "./components/badge.css"; | ||||||
|  | @import "./components/toc.css"; | ||||||
|   | |||||||
| @@ -1,154 +1,147 @@ | |||||||
| .content { | .content { | ||||||
|   h1 { |   :where(h1):not(:where([class~=not-prose],[class~=not-prose] *)) { | ||||||
|     @apply mt-2 text-4xl font-bold tracking-tight text-slate-900 dark:text-slate-100; |     @apply hx:mt-2 hx:text-4xl hx:font-bold hx:tracking-tight hx:text-slate-900 hx:dark:text-slate-100; | ||||||
|   } |   } | ||||||
|   h2 { |   :where(h2):not(:where([class~=not-prose],[class~=not-prose] *)) { | ||||||
|     @apply font-semibold tracking-tight text-slate-900 dark:text-slate-100 mt-10 border-b pb-1 text-3xl border-neutral-200/70 contrast-more:border-neutral-400 dark:border-primary-100/10 contrast-more:dark:border-neutral-400; |     @apply hx:font-semibold hx:tracking-tight hx:text-slate-900 hx:dark:text-slate-100 hx:mt-10 hx:border-b hx:pb-1 hx:text-3xl hx:border-neutral-200/70 hx:contrast-more:border-neutral-400 hx:dark:border-primary-100/10 hx:contrast-more:dark:border-neutral-400; | ||||||
|   } |   } | ||||||
|   h3 { |   :where(h3):not(:where([class~=not-prose],[class~=not-prose] *)) { | ||||||
|     @apply font-semibold tracking-tight text-slate-900 dark:text-slate-100 mt-8 text-2xl; |     @apply hx:font-semibold hx:tracking-tight hx:text-slate-900 hx:dark:text-slate-100 hx:mt-8 hx:text-2xl; | ||||||
|   } |   } | ||||||
|   h4 { |   :where(h4):not(:where([class~=not-prose],[class~=not-prose] *)) { | ||||||
|     @apply font-semibold tracking-tight text-slate-900 dark:text-slate-100 mt-8 text-xl; |     @apply hx:font-semibold hx:tracking-tight hx:text-slate-900 hx:dark:text-slate-100 hx:mt-8 hx:text-xl; | ||||||
|   } |   } | ||||||
|   h5 { |   :where(h5):not(:where([class~=not-prose],[class~=not-prose] *)) { | ||||||
|     @apply font-semibold tracking-tight text-slate-900 dark:text-slate-100 mt-8 text-lg; |     @apply hx:font-semibold hx:tracking-tight hx:text-slate-900 hx:dark:text-slate-100 hx:mt-8 hx:text-lg; | ||||||
|   } |   } | ||||||
|   h6 { |   :where(h6):not(:where([class~=not-prose],[class~=not-prose] *)) { | ||||||
|     @apply font-semibold tracking-tight text-slate-900 dark:text-slate-100 mt-8 text-base; |     @apply hx:font-semibold hx:tracking-tight hx:text-slate-900 hx:dark:text-slate-100 hx:mt-8 hx:text-base; | ||||||
|   } |   } | ||||||
|   p { |   :where(p):not(:where([class~=not-prose],[class~=not-prose] *)) { | ||||||
|     @apply mt-6 leading-7 first:mt-0; |     @apply hx:mt-6 hx:leading-7 hx:first:mt-0; | ||||||
|   } |   } | ||||||
|   a { |   :where(a):not(:where([class~=not-prose],[class~=not-prose] *)) { | ||||||
|     @apply text-primary-600 underline decoration-from-font [text-underline-position:from-font]; |     @apply hx:text-primary-600 hx:underline hx:decoration-from-font; | ||||||
|   } |   } | ||||||
|   .not-prose a { |   :where(blockquote):not(:where([class~=not-prose],[class~=not-prose] *)) { | ||||||
|     @apply text-current no-underline; |     @apply hx:mt-6 hx:border-gray-300 hx:italic hx:text-gray-700 hx:dark:border-gray-700 hx:dark:text-gray-400 hx:first:mt-0 hx:ltr:border-l-2 hx:ltr:pl-6 hx:rtl:border-r-2 hx:rtl:pr-6; | ||||||
|   } |   } | ||||||
|   blockquote { |   :where(pre):not(:where(.hextra-code-block pre, [class~=not-prose],[class~=not-prose] *)) { | ||||||
|     @apply mt-6 border-gray-300 italic text-gray-700 dark:border-gray-700 dark:text-gray-400 first:mt-0 ltr:border-l-2 ltr:pl-6 rtl:border-r-2 rtl:pr-6; |     @apply hx:bg-primary-700/5 hx:mb-4 hx:overflow-x-auto hx:rounded-xl hx:font-medium hx:subpixel-antialiased hx:dark:bg-primary-300/10 hx:text-[.9em] hx:contrast-more:border hx:contrast-more:border-primary-900/20 hx:contrast-more:contrast-150 hx:contrast-more:dark:border-primary-100/40 hx:py-4; | ||||||
|   } |   } | ||||||
|   pre:not(.code-block pre) { |   :where(code):not(:where(.hextra-code-block code, [class~=not-prose],[class~=not-prose] *)) { | ||||||
|     @apply bg-primary-700/5 mb-4 overflow-x-auto rounded-xl font-medium subpixel-antialiased dark:bg-primary-300/10 text-[.9em] contrast-more:border contrast-more:border-primary-900/20 contrast-more:contrast-150 contrast-more:dark:border-primary-100/40 py-4; |     @apply hx:border-black/4 hx:bg-black/3 hx:break-words hx:rounded-md hx:border hx:py-0.5 hx:px-[.25em] hx:text-[.9em] hx:dark:border-white/10 hx:dark:bg-white/10; | ||||||
|   } |   } | ||||||
|   code:not(.code-block code) { |   :where(table):not(:where(.hextra-code-block table, [class~=not-prose],[class~=not-prose] *)) { | ||||||
|     @apply border-black border-opacity-[0.04] bg-opacity-[0.03] bg-black break-words rounded-md border py-0.5 px-[.25em] text-[.9em] dark:border-white/10 dark:bg-white/10; |     @apply hx:block hx:overflow-x-auto hx:my-6 hx:p-0 hx:first:mt-0 hx:w-full hx:text-sm hx:leading-5 hx:border-collapse; | ||||||
|   } |  | ||||||
|   table:not(.code-block table) { |  | ||||||
|     @apply block overflow-x-auto mt-6 p-0 first:mt-0; |  | ||||||
|  |  | ||||||
|  |     thead { | ||||||
|  |       @apply hx:bg-gray-50 hx:dark:bg-gray-600/20; | ||||||
|  |     } | ||||||
|     tr { |     tr { | ||||||
|       @apply m-0 border-t border-gray-300 p-0 dark:border-gray-600 even:bg-gray-100 even:dark:bg-gray-600/20; |       @apply hx:m-0 hx:border-t hx:border-gray-300 hx:p-0 hx:dark:border-gray-600; | ||||||
|     } |     } | ||||||
|     th { |     th { | ||||||
|       @apply m-0 border border-gray-300 px-4 py-2 font-semibold dark:border-gray-600; |       @apply hx:m-0 hx:border hx:border-gray-300 hx:p-2 hx:font-semibold hx:dark:border-gray-600; | ||||||
|     } |     } | ||||||
|     td { |     td { | ||||||
|       @apply m-0 border border-gray-300 px-4 py-2 dark:border-gray-600; |       @apply hx:m-0 hx:border hx:border-gray-300 hx:p-2 hx:dark:border-gray-600; | ||||||
|     } |     } | ||||||
|   } |   } | ||||||
|   ol { |   :where(ol):not(:where([class~=not-prose],[class~=not-prose] *)) { | ||||||
|     @apply mt-6 list-decimal first:mt-0 ltr:ml-6 rtl:mr-6; |     @apply hx:mt-6 hx:list-decimal hx:first:mt-0 hx:ltr:ml-6 hx:rtl:mr-6; | ||||||
|     li { |     li { | ||||||
|       @apply my-2; |       @apply hx:my-2; | ||||||
|     } |     } | ||||||
|   } |   } | ||||||
|   ul { |   :where(ul):not(:where([class~=not-prose],[class~=not-prose] *)) { | ||||||
|     @apply mt-6 list-disc first:mt-0 ltr:ml-6 rtl:mr-6; |     @apply hx:mt-6 hx:list-disc hx:first:mt-0 hx:ltr:ml-6 hx:rtl:mr-6; | ||||||
|     li { |     li { | ||||||
|       @apply my-2; |       @apply hx:my-2; | ||||||
|     } |     } | ||||||
|   } |   } | ||||||
|   .not-prose ul, .not-prose ol { |   /* Task lists - hide list markers for lists containing checkboxes */ | ||||||
|     @apply m-0 list-none; |   :where(ul):not(:where([class~=not-prose],[class~=not-prose] *)):has(li input[type="checkbox"]) { | ||||||
|     li { |     @apply hx:list-none; | ||||||
|       @apply m-0; |  | ||||||
|   } |   } | ||||||
|  |   /* This CSS rule targets the first nested unordered (ul) or ordered (ol) list | ||||||
|  |      inside the list item (li) of any parent ul or ol. | ||||||
|  |      The rule sets the top margin of the selected list to zero. */ | ||||||
|  |   :where(ul, ol) > li > :where(ul, ol):not(:where([class~=not-prose],[class~=not-prose] *)) { | ||||||
|  |     @apply hx:mt-0; | ||||||
|   } |   } | ||||||
|   kbd { |   :where(kbd):not(:where([class~=not-prose],[class~=not-prose] *)) { | ||||||
|     @apply border-black border-opacity-[0.04] bg-opacity-[0.03] bg-black break-words rounded-md border py-0.5 px-[.25em] text-[.9em] dark:border-white/10 dark:bg-white/10; |     @apply hx:border-black/4 hx:bg-black/3 hx:break-words hx:rounded-md hx:border hx:py-0.5 hx:px-[.25em] hx:text-[.9em] hx:dark:border-white/10 hx:dark:bg-white/10; | ||||||
|   } |   } | ||||||
|   pre:not(.code-block pre).mermaid { |   :where(pre.mermaid):not(:where(.hextra-code-block pre, [class~=not-prose],[class~=not-prose] *)) { | ||||||
|     @apply bg-transparent rounded-none dark:bg-transparent; |     @apply hx:bg-transparent hx:rounded-none hx:dark:bg-transparent; | ||||||
|   } |   } | ||||||
|   img { |   :where(img):not(:where([class~=not-prose],[class~=not-prose] *)) { | ||||||
|     @apply mx-auto my-4 rounded-md; |     @apply hx:mx-auto hx:my-4 hx:rounded-md; | ||||||
|   } |   } | ||||||
|   .not-prose img { |   :where(figure):not(:where([class~=not-prose],[class~=not-prose] *)) { | ||||||
|     @apply m-0 rounded-none; |  | ||||||
|   } |  | ||||||
|   figure { |  | ||||||
|     figcaption { |     figcaption { | ||||||
|       @apply text-sm text-gray-500 dark:text-gray-400 mt-2 block text-center; |       @apply hx:text-sm hx:text-gray-500 hx:dark:text-gray-400 hx:mt-2 hx:block hx:text-center; | ||||||
|     } |     } | ||||||
|   } |   } | ||||||
|  |   /* Definition list */ | ||||||
|  |   :where(dl):not(:where([class~=not-prose],[class~=not-prose] *)) { | ||||||
|  |     dt { | ||||||
|  |       @apply hx:mt-6 hx:font-semibold; | ||||||
|  |     } | ||||||
|  |     dd { | ||||||
|  |       @apply hx:my-2 hx:ps-6; | ||||||
|  |     } | ||||||
|  |   } | ||||||
|  |   /* Horizontal line */ | ||||||
|  |   :where(hr):not(:where([class~=not-prose],[class~=not-prose] *)) { | ||||||
|  |     @apply hx:my-10 hx:first:mt-0 hx:last:mb-0 hx:border-gray-200 hx:dark:border-neutral-800; | ||||||
|  |   } | ||||||
|   .footnotes { |   .footnotes { | ||||||
|     @apply mt-12 text-sm; |     @apply hx:mt-12 hx:text-sm; | ||||||
|   } |  | ||||||
| } |  | ||||||
|  |  | ||||||
|  |     hr { | ||||||
|  |       @apply hx:border-gray-200 hx:dark:border-neutral-800; | ||||||
|  |     } | ||||||
|  |   } | ||||||
|   .subheading-anchor { |   .subheading-anchor { | ||||||
|   @apply opacity-0 transition-opacity ltr:ml-1 rtl:mr-1; |     @apply hx:opacity-0 hx:transition-opacity hx:ltr:ml-1 hx:rtl:mr-1; | ||||||
|    |    | ||||||
|     span:target + &, |     span:target + &, | ||||||
|     :hover > &, |     :hover > &, | ||||||
|     &:focus { |     &:focus { | ||||||
|     @apply opacity-100; |       @apply hx:opacity-100; | ||||||
|     } |     } | ||||||
|    |    | ||||||
|     span + &, |     span + &, | ||||||
|     :hover > & { |     :hover > & { | ||||||
|     @apply !no-underline; |       @apply hx:no-underline!; | ||||||
|     } |     } | ||||||
|    |    | ||||||
|  |     @apply hx:after:text-gray-300 hx:dark:after:text-neutral-700; | ||||||
|     &:after { |     &:after { | ||||||
|     @apply content-['#'] px-1; |       @apply hx:content-['#'] hx:px-1; | ||||||
|     @apply text-gray-300 dark:text-neutral-700; |  | ||||||
|       span:target + & { |       span:target + & { | ||||||
|       @apply text-gray-400; |         @apply hx:text-gray-400; | ||||||
|       @apply dark:text-neutral-500; |         @apply hx:dark:text-neutral-500; | ||||||
|       } |       } | ||||||
|     } |     } | ||||||
|   } |   } | ||||||
|  |  | ||||||
| article.typesetting-article { |  | ||||||
|   font-size: 17px; |  | ||||||
|   font-feature-settings: |  | ||||||
|     'rlig' 1, |  | ||||||
|     'calt' 1; |  | ||||||
|   h1 { |  | ||||||
|     @apply mt-6 mb-4 text-center; |  | ||||||
|     font-size: 2.5rem; |  | ||||||
|   } |  | ||||||
|   h2 { |  | ||||||
|     @apply border-none; |  | ||||||
|   } |  | ||||||
|   a { |  | ||||||
|     @apply no-underline hover:underline; |  | ||||||
|   } |  | ||||||
|   p { |  | ||||||
|     @apply leading-8; |  | ||||||
|   } |  | ||||||
|   code { |  | ||||||
|     @apply border-none dark:bg-neutral-700; |  | ||||||
|   } |  | ||||||
|   pre code { |  | ||||||
|     @apply dark:bg-transparent; |  | ||||||
|   } |  | ||||||
|   .subheading-anchor + a { |  | ||||||
|     @apply no-underline hover:no-underline after:hidden; |  | ||||||
|   } |  | ||||||
| } | } | ||||||
|  |  | ||||||
| article details > summary { | article details > summary { | ||||||
|   &::-webkit-details-marker { |   &::-webkit-details-marker { | ||||||
|     @apply hidden; |     @apply hx:hidden; | ||||||
|   } |   } | ||||||
|   &::before { |   &::before { | ||||||
|     background-image: url("data:image/svg+xml,%3Csvg xmlns='http://www.w3.org/2000/svg' class='h-5 w-5' viewBox='0 0 20 20' fill='currentColor'%3E%3Cpath fill-rule='evenodd' d='M7.293 14.707a1 1 0 010-1.414L10.586 10 7.293 6.707a1 1 0 011.414-1.414l4 4a1 1 0 010 1.414l-4 4a1 1 0 01-1.414 0z' clip-rule='evenodd' /%3E%3C/svg%3E"); |     background-image: url("data:image/svg+xml,%3Csvg xmlns='http://www.w3.org/2000/svg' class='hx:h-5 hx:w-5' viewBox='0 0 20 20' fill='currentColor'%3E%3Cpath fill-rule='evenodd' d='M7.293 14.707a1 1 0 010-1.414L10.586 10 7.293 6.707a1 1 0 011.414-1.414l4 4a1 1 0 010 1.414l-4 4a1 1 0 01-1.414 0z' clip-rule='evenodd' /%3E%3C/svg%3E"); | ||||||
|     height: 1.2em; |     height: 1.2em; | ||||||
|     width: 1.2em; |     width: 1.2em; | ||||||
|     vertical-align: -4px; |     vertical-align: -4px; | ||||||
|  |     padding: 0 0.6em; | ||||||
|   } |   } | ||||||
| } | } | ||||||
|  |  | ||||||
|  | :lang(fa) ol { | ||||||
|  |   list-style-type: persian; | ||||||
|  | } | ||||||
|   | |||||||
							
								
								
									
										30
									
								
								assets/css/variables.css
									
									
									
									
									
										Normal file
									
								
							
							
						
						
									
										30
									
								
								assets/css/variables.css
									
									
									
									
									
										Normal file
									
								
							| @@ -0,0 +1,30 @@ | |||||||
|  | /* Hugo template to derive CSS variables from site and page parameters */ | ||||||
|  |  | ||||||
|  | /* Do not remove the following comment. It is used by Hugo to render CSS variables. | ||||||
|  | {{- $pageWidth := site.Params.page.width -}} | ||||||
|  | {{- $maxPageWidth := cond (eq $pageWidth "wide") "90rem" (cond (eq $pageWidth "full") "100%" "80rem") -}} | ||||||
|  |  | ||||||
|  | {{- $navbarWidth := site.Params.navbar.width -}} | ||||||
|  | {{- $maxNavbarWidth := cond (eq $navbarWidth "wide") "90rem" (cond (eq $navbarWidth "full") "100%" "80rem") -}} | ||||||
|  |  | ||||||
|  | {{- $footerWidth := site.Params.footer.width -}} | ||||||
|  | {{- $maxFooterWidth := cond (eq $footerWidth "wide") "90rem" (cond (eq $footerWidth "full") "100%" "80rem") -}} | ||||||
|  | */ | ||||||
|  |  | ||||||
|  | :root { | ||||||
|  |   --hextra-max-page-width: {{ $maxPageWidth }}; | ||||||
|  |   --hextra-max-navbar-width: {{ $maxNavbarWidth }}; | ||||||
|  |   --hextra-max-footer-width: {{ $maxFooterWidth }}; | ||||||
|  | } | ||||||
|  |  | ||||||
|  | .hextra-max-page-width { | ||||||
|  |   max-width: var(--hextra-max-page-width); | ||||||
|  | } | ||||||
|  |  | ||||||
|  | .hextra-max-navbar-width { | ||||||
|  |   max-width: var(--hextra-max-navbar-width); | ||||||
|  | } | ||||||
|  |  | ||||||
|  | .hextra-max-footer-width { | ||||||
|  |   max-width: var(--hextra-max-footer-width); | ||||||
|  | } | ||||||
| @@ -1,30 +0,0 @@ | |||||||
| document.querySelectorAll('.code-copy-btn').forEach(function (button) { |  | ||||||
|   button.addEventListener('click', function (e) { |  | ||||||
|     e.preventDefault(); |  | ||||||
|     const targetId = button.getAttribute('data-clipboard-target'); |  | ||||||
|     const target = document.querySelector(targetId); |  | ||||||
|     let codeElement; |  | ||||||
|     if (target.tagName === 'CODE') { |  | ||||||
|       codeElement = target; |  | ||||||
|     } else { |  | ||||||
|       // Select the last code element in case line numbers are present |  | ||||||
|       const codeElements = target.querySelectorAll('code'); |  | ||||||
|       codeElement = codeElements[codeElements.length - 1]; |  | ||||||
|     } |  | ||||||
|     if (codeElement) { |  | ||||||
|       // Replace double newlines with single newlines in the innerText |  | ||||||
|       // as each line inside <span> has trailing newline '\n' |  | ||||||
|       const code = codeElement.innerText.replace(/\n\n/g, '\n'); |  | ||||||
|       navigator.clipboard.writeText(code).then(function () { |  | ||||||
|         button.classList.add('copied'); |  | ||||||
|         setTimeout(function () { |  | ||||||
|           button.classList.remove('copied'); |  | ||||||
|         }, 500); |  | ||||||
|       }).catch(function (err) { |  | ||||||
|         console.error('Failed to copy text: ', err); |  | ||||||
|       }); |  | ||||||
|     } else { |  | ||||||
|       console.error('Target element not found'); |  | ||||||
|     } |  | ||||||
|   }); |  | ||||||
| }); |  | ||||||
							
								
								
									
										22
									
								
								assets/js/core/back-to-top.js
									
									
									
									
									
										Normal file
									
								
							
							
						
						
									
										22
									
								
								assets/js/core/back-to-top.js
									
									
									
									
									
										Normal file
									
								
							| @@ -0,0 +1,22 @@ | |||||||
|  | // Back to top button | ||||||
|  |  | ||||||
|  | document.addEventListener("DOMContentLoaded", function () { | ||||||
|  |   const backToTop = document.querySelector("#backToTop"); | ||||||
|  |   if (backToTop) { | ||||||
|  |     document.addEventListener("scroll", (e) => { | ||||||
|  |       if (window.scrollY > 300) { | ||||||
|  |         backToTop.classList.remove("hx:opacity-0"); | ||||||
|  |       } else { | ||||||
|  |         backToTop.classList.add("hx:opacity-0"); | ||||||
|  |       } | ||||||
|  |     }); | ||||||
|  |   } | ||||||
|  | }); | ||||||
|  |  | ||||||
|  | function scrollUp() { | ||||||
|  |   window.scroll({ | ||||||
|  |     top: 0, | ||||||
|  |     left: 0, | ||||||
|  |     behavior: "smooth", | ||||||
|  |   }); | ||||||
|  | } | ||||||
							
								
								
									
										15
									
								
								assets/js/core/banner.js
									
									
									
									
									
										Normal file
									
								
							
							
						
						
									
										15
									
								
								assets/js/core/banner.js
									
									
									
									
									
										Normal file
									
								
							| @@ -0,0 +1,15 @@ | |||||||
|  | // {{- if site.Params.banner }} | ||||||
|  | (function () { | ||||||
|  |   const banner = document.querySelector(".hextra-banner") | ||||||
|  |   document.documentElement.style.setProperty("--hextra-banner-height", banner.clientHeight+"px"); | ||||||
|  |  | ||||||
|  |   const closeBtn = banner.querySelector(".hextra-banner-close-button"); | ||||||
|  |  | ||||||
|  |   closeBtn.addEventListener("click", () => { | ||||||
|  |     document.documentElement.classList.add("hextra-banner-hidden"); | ||||||
|  |     document.documentElement.style.setProperty("--hextra-banner-height", "0px"); | ||||||
|  |  | ||||||
|  |     localStorage.setItem('{{ site.Params.banner.key | default `banner-closed` }}', "0"); | ||||||
|  |   }); | ||||||
|  | })(); | ||||||
|  | // {{- end -}} | ||||||
							
								
								
									
										66
									
								
								assets/js/core/code-copy.js
									
									
									
									
									
										Normal file
									
								
							
							
						
						
									
										66
									
								
								assets/js/core/code-copy.js
									
									
									
									
									
										Normal file
									
								
							| @@ -0,0 +1,66 @@ | |||||||
|  | // Copy button for code blocks | ||||||
|  |  | ||||||
|  | document.addEventListener('DOMContentLoaded', function () { | ||||||
|  |   const getCopyIcon = () => { | ||||||
|  |     const svg = document.createElementNS('http://www.w3.org/2000/svg', 'svg'); | ||||||
|  |     svg.innerHTML = ` | ||||||
|  |       <path stroke-linecap="round" stroke-linejoin="round" d="M8 16H6a2 2 0 01-2-2V6a2 2 0 012-2h8a2 2 0 012 2v2m-6 12h8a2 2 0 002-2v-8a2 2 0 00-2-2h-8a2 2 0 00-2 2v8a2 2 0 002 2z" /> | ||||||
|  |     `; | ||||||
|  |     svg.setAttribute('fill', 'none'); | ||||||
|  |     svg.setAttribute('viewBox', '0 0 24 24'); | ||||||
|  |     svg.setAttribute('stroke', 'currentColor'); | ||||||
|  |     svg.setAttribute('stroke-width', '2'); | ||||||
|  |     return svg; | ||||||
|  |   } | ||||||
|  |  | ||||||
|  |   const getSuccessIcon = () => { | ||||||
|  |     const svg = document.createElementNS('http://www.w3.org/2000/svg', 'svg'); | ||||||
|  |     svg.innerHTML = ` | ||||||
|  |       <path stroke-linecap="round" stroke-linejoin="round" d="M5 13l4 4L19 7" /> | ||||||
|  |     `; | ||||||
|  |     svg.setAttribute('fill', 'none'); | ||||||
|  |     svg.setAttribute('viewBox', '0 0 24 24'); | ||||||
|  |     svg.setAttribute('stroke', 'currentColor'); | ||||||
|  |     svg.setAttribute('stroke-width', '2'); | ||||||
|  |     return svg; | ||||||
|  |   } | ||||||
|  |  | ||||||
|  |   document.querySelectorAll('.hextra-code-copy-btn').forEach(function (button) { | ||||||
|  |     // Add copy and success icons | ||||||
|  |     button.querySelector('.hextra-copy-icon')?.appendChild(getCopyIcon()); | ||||||
|  |     button.querySelector('.hextra-success-icon')?.appendChild(getSuccessIcon()); | ||||||
|  |  | ||||||
|  |     // Add click event listener for copy button | ||||||
|  |     button.addEventListener('click', function (e) { | ||||||
|  |       e.preventDefault(); | ||||||
|  |       // Get the code target | ||||||
|  |       const target = button.parentElement.previousElementSibling; | ||||||
|  |       let codeElement; | ||||||
|  |       if (target.tagName === 'CODE') { | ||||||
|  |         codeElement = target; | ||||||
|  |       } else { | ||||||
|  |         // Select the last code element in case line numbers are present | ||||||
|  |         const codeElements = target.querySelectorAll('code'); | ||||||
|  |         codeElement = codeElements[codeElements.length - 1]; | ||||||
|  |       } | ||||||
|  |       if (codeElement) { | ||||||
|  |         let code = codeElement.innerText; | ||||||
|  |         // Replace double newlines with single newlines in the innerText | ||||||
|  |         // as each line inside <span> has trailing newline '\n' | ||||||
|  |         if ("lang" in codeElement.dataset) { | ||||||
|  |           code = code.replace(/\n\n/g, '\n'); | ||||||
|  |         } | ||||||
|  |         navigator.clipboard.writeText(code).then(function () { | ||||||
|  |           button.classList.add('copied'); | ||||||
|  |           setTimeout(function () { | ||||||
|  |             button.classList.remove('copied'); | ||||||
|  |           }, 1000); | ||||||
|  |         }).catch(function (err) { | ||||||
|  |           console.error('Failed to copy text: ', err); | ||||||
|  |         }); | ||||||
|  |       } else { | ||||||
|  |         console.error('Target element not found'); | ||||||
|  |       } | ||||||
|  |     }); | ||||||
|  |   }); | ||||||
|  | }); | ||||||
							
								
								
									
										22
									
								
								assets/js/core/favicon.js
									
									
									
									
									
										Normal file
									
								
							
							
						
						
									
										22
									
								
								assets/js/core/favicon.js
									
									
									
									
									
										Normal file
									
								
							| @@ -0,0 +1,22 @@ | |||||||
|  | // {{ $faviconDarkExists := fileExists (path.Join "static" "favicon-dark.svg") }} | ||||||
|  | (function () { | ||||||
|  |   const faviconEl = document.getElementById("favicon-svg"); | ||||||
|  |   const faviconDarkExists = "{{ $faviconDarkExists }}" === "true"; | ||||||
|  |  | ||||||
|  |   if (faviconEl && faviconDarkExists) { | ||||||
|  |     const lightFavicon = '{{ "favicon.svg" | relURL }}'; | ||||||
|  |     const darkFavicon = '{{ "favicon-dark.svg" | relURL }}'; | ||||||
|  |  | ||||||
|  |     const darkModeQuery = window.matchMedia("(prefers-color-scheme: dark)"); | ||||||
|  |  | ||||||
|  |     function updateFavicon(e) { | ||||||
|  |       faviconEl.href = e.matches ? darkFavicon : lightFavicon; | ||||||
|  |     } | ||||||
|  |  | ||||||
|  |     // Set favicon on load | ||||||
|  |     updateFavicon(darkModeQuery); | ||||||
|  |  | ||||||
|  |     // Listen for system preference changes | ||||||
|  |     darkModeQuery.addEventListener("change", updateFavicon); | ||||||
|  |   } | ||||||
|  | })(); | ||||||
							
								
								
									
										26
									
								
								assets/js/core/lang.js
									
									
									
									
									
										Normal file
									
								
							
							
						
						
									
										26
									
								
								assets/js/core/lang.js
									
									
									
									
									
										Normal file
									
								
							| @@ -0,0 +1,26 @@ | |||||||
|  | (function () { | ||||||
|  |   const languageSwitchers = document.querySelectorAll('.hextra-language-switcher'); | ||||||
|  |  | ||||||
|  |   languageSwitchers.forEach((switcher) => { | ||||||
|  |     switcher.addEventListener('click', (e) => { | ||||||
|  |       e.preventDefault(); | ||||||
|  |  | ||||||
|  |       switcher.dataset.state = switcher.dataset.state === 'open' ? 'closed' : 'open'; | ||||||
|  |  | ||||||
|  |       toggleMenu(switcher); | ||||||
|  |     }); | ||||||
|  |   }); | ||||||
|  |  | ||||||
|  |   window.addEventListener("resize", () => languageSwitchers.forEach(resizeMenu)) | ||||||
|  |  | ||||||
|  |   // Dismiss language switcher when clicking outside | ||||||
|  |   document.addEventListener('click', (e) => { | ||||||
|  |     if (e.target.closest('.hextra-language-switcher') === null) { | ||||||
|  |       languageSwitchers.forEach((switcher) => { | ||||||
|  |         switcher.dataset.state = 'closed'; | ||||||
|  |         const optionsElement = switcher.nextElementSibling; | ||||||
|  |         optionsElement.classList.add('hx:hidden'); | ||||||
|  |       }); | ||||||
|  |     } | ||||||
|  |   }); | ||||||
|  | })(); | ||||||
							
								
								
									
										40
									
								
								assets/js/core/menu.js
									
									
									
									
									
										Normal file
									
								
							
							
						
						
									
										40
									
								
								assets/js/core/menu.js
									
									
									
									
									
										Normal file
									
								
							| @@ -0,0 +1,40 @@ | |||||||
|  | // Hamburger menu for mobile navigation | ||||||
|  |  | ||||||
|  | document.addEventListener('DOMContentLoaded', function () { | ||||||
|  |   const menu = document.querySelector('.hextra-hamburger-menu'); | ||||||
|  |   const sidebarContainer = document.querySelector('.hextra-sidebar-container'); | ||||||
|  |  | ||||||
|  |   function toggleMenu() { | ||||||
|  |     // Toggle the hamburger menu | ||||||
|  |     menu.querySelector('svg').classList.toggle('open'); | ||||||
|  |  | ||||||
|  |     // When the menu is open, we want to show the navigation sidebar | ||||||
|  |     sidebarContainer.classList.toggle('hx:max-md:[transform:translate3d(0,-100%,0)]'); | ||||||
|  |     sidebarContainer.classList.toggle('hx:max-md:[transform:translate3d(0,0,0)]'); | ||||||
|  |  | ||||||
|  |     // When the menu is open, we want to prevent the body from scrolling | ||||||
|  |     document.body.classList.toggle('hx:overflow-hidden'); | ||||||
|  |     document.body.classList.toggle('hx:md:overflow-auto'); | ||||||
|  |   } | ||||||
|  |  | ||||||
|  |   menu.addEventListener('click', (e) => { | ||||||
|  |     e.preventDefault(); | ||||||
|  |     toggleMenu(); | ||||||
|  |   }); | ||||||
|  |  | ||||||
|  |   // Select all anchor tags in the sidebar container | ||||||
|  |   const sidebarLinks = sidebarContainer.querySelectorAll('a'); | ||||||
|  |  | ||||||
|  |   // Add click event listener to each anchor tag | ||||||
|  |   sidebarLinks.forEach(link => { | ||||||
|  |     link.addEventListener('click', (e) => { | ||||||
|  |       // Check if the href attribute contains a hash symbol (links to a heading) | ||||||
|  |       if (link.getAttribute('href') && link.getAttribute('href').startsWith('#')) { | ||||||
|  |         // Only dismiss overlay on mobile view | ||||||
|  |         if (window.innerWidth < 768) { | ||||||
|  |           toggleMenu(); | ||||||
|  |         } | ||||||
|  |       } | ||||||
|  |     }); | ||||||
|  |   }); | ||||||
|  | }); | ||||||
							
								
								
									
										62
									
								
								assets/js/core/nav-menu.js
									
									
									
									
									
										Normal file
									
								
							
							
						
						
									
										62
									
								
								assets/js/core/nav-menu.js
									
									
									
									
									
										Normal file
									
								
							| @@ -0,0 +1,62 @@ | |||||||
|  | (function () { | ||||||
|  |   const hiddenClass = "hx:hidden"; | ||||||
|  |   const dropdownToggles = document.querySelectorAll(".hextra-nav-menu-toggle"); | ||||||
|  |  | ||||||
|  |   dropdownToggles.forEach((toggle) => { | ||||||
|  |     toggle.addEventListener("click", (e) => { | ||||||
|  |       e.preventDefault(); | ||||||
|  |       e.stopPropagation(); | ||||||
|  |  | ||||||
|  |       // Close all other dropdowns first | ||||||
|  |       dropdownToggles.forEach((otherToggle) => { | ||||||
|  |         if (otherToggle !== toggle) { | ||||||
|  |           otherToggle.dataset.state = "closed"; | ||||||
|  |           const otherMenuItems = otherToggle.nextElementSibling; | ||||||
|  |           otherMenuItems.classList.add(hiddenClass); | ||||||
|  |         } | ||||||
|  |       }); | ||||||
|  |  | ||||||
|  |       // Toggle current dropdown | ||||||
|  |       const isOpen = toggle.dataset.state === "open"; | ||||||
|  |       toggle.dataset.state = isOpen ? "closed" : "open"; | ||||||
|  |       const menuItemsElement = toggle.nextElementSibling; | ||||||
|  |  | ||||||
|  |       if (!isOpen) { | ||||||
|  |         // Position dropdown centered with toggle | ||||||
|  |         menuItemsElement.style.position = "absolute"; | ||||||
|  |         menuItemsElement.style.top = "100%"; | ||||||
|  |         menuItemsElement.style.left = "50%"; | ||||||
|  |         menuItemsElement.style.transform = "translateX(-50%)"; | ||||||
|  |         menuItemsElement.style.zIndex = "1000"; | ||||||
|  |  | ||||||
|  |         // Show dropdown | ||||||
|  |         menuItemsElement.classList.remove(hiddenClass); | ||||||
|  |       } else { | ||||||
|  |         // Hide dropdown | ||||||
|  |         menuItemsElement.classList.add(hiddenClass); | ||||||
|  |       } | ||||||
|  |     }); | ||||||
|  |   }); | ||||||
|  |  | ||||||
|  |   // Dismiss dropdown when clicking outside | ||||||
|  |   document.addEventListener("click", (e) => { | ||||||
|  |     if (e.target.closest(".hextra-nav-menu-toggle") === null) { | ||||||
|  |       dropdownToggles.forEach((toggle) => { | ||||||
|  |         toggle.dataset.state = "closed"; | ||||||
|  |         const menuItemsElement = toggle.nextElementSibling; | ||||||
|  |         menuItemsElement.classList.add(hiddenClass); | ||||||
|  |       }); | ||||||
|  |     } | ||||||
|  |   }); | ||||||
|  |  | ||||||
|  |   // Close dropdowns on escape key | ||||||
|  |   document.addEventListener("keydown", (e) => { | ||||||
|  |     if (e.key === "Escape") { | ||||||
|  |       dropdownToggles.forEach((toggle) => { | ||||||
|  |         toggle.dataset.state = "closed"; | ||||||
|  |         const menuItemsElement = toggle.nextElementSibling; | ||||||
|  |         menuItemsElement.classList.add(hiddenClass); | ||||||
|  |       }); | ||||||
|  |     } | ||||||
|  |   }); | ||||||
|  | })(); | ||||||
							
								
								
									
										36
									
								
								assets/js/core/sidebar.js
									
									
									
									
									
										Normal file
									
								
							
							
						
						
									
										36
									
								
								assets/js/core/sidebar.js
									
									
									
									
									
										Normal file
									
								
							| @@ -0,0 +1,36 @@ | |||||||
|  | document.addEventListener("DOMContentLoaded", function () { | ||||||
|  |   scrollToActiveItem(); | ||||||
|  |   enableCollapsibles(); | ||||||
|  | }); | ||||||
|  |  | ||||||
|  | function enableCollapsibles() { | ||||||
|  |   const buttons = document.querySelectorAll(".hextra-sidebar-collapsible-button"); | ||||||
|  |   buttons.forEach(function (button) { | ||||||
|  |     button.addEventListener("click", function (e) { | ||||||
|  |       e.preventDefault(); | ||||||
|  |       const list = button.parentElement.parentElement; | ||||||
|  |       if (list) { | ||||||
|  |         list.classList.toggle("open") | ||||||
|  |       } | ||||||
|  |     }); | ||||||
|  |   }); | ||||||
|  | } | ||||||
|  |  | ||||||
|  | function scrollToActiveItem() { | ||||||
|  |   const sidebarScrollbar = document.querySelector("aside.hextra-sidebar-container > .hextra-scrollbar"); | ||||||
|  |   const activeItems = document.querySelectorAll(".hextra-sidebar-active-item"); | ||||||
|  |   const visibleActiveItem = Array.from(activeItems).find(function (activeItem) { | ||||||
|  |     return activeItem.getBoundingClientRect().height > 0; | ||||||
|  |   }); | ||||||
|  |  | ||||||
|  |   if (!visibleActiveItem) { | ||||||
|  |     return; | ||||||
|  |   } | ||||||
|  |  | ||||||
|  |   const yOffset = visibleActiveItem.clientHeight; | ||||||
|  |   const yDistance = visibleActiveItem.getBoundingClientRect().top - sidebarScrollbar.getBoundingClientRect().top; | ||||||
|  |   sidebarScrollbar.scrollTo({ | ||||||
|  |     behavior: "instant", | ||||||
|  |     top: yDistance - yOffset | ||||||
|  |   }); | ||||||
|  | } | ||||||
							
								
								
									
										52
									
								
								assets/js/core/switcher-menu.js
									
									
									
									
									
										Normal file
									
								
							
							
						
						
									
										52
									
								
								assets/js/core/switcher-menu.js
									
									
									
									
									
										Normal file
									
								
							| @@ -0,0 +1,52 @@ | |||||||
|  | function computeMenuTranslation(switcher, optionsElement) { | ||||||
|  |   // Calculate the position of a language options element. | ||||||
|  |   const switcherRect = switcher.getBoundingClientRect(); | ||||||
|  |  | ||||||
|  |   // Must be called before optionsElement.clientWidth. | ||||||
|  |   optionsElement.style.minWidth = `${Math.max(switcherRect.width, 50)}px`; | ||||||
|  |  | ||||||
|  |   const isOnTop = switcher.dataset.location === 'top'; | ||||||
|  |   const isOnBottom = switcher.dataset.location === 'bottom'; | ||||||
|  |   const isOnBottomRight = switcher.dataset.location === 'bottom-right'; | ||||||
|  |   const isRTL = document.documentElement.dir === 'rtl' | ||||||
|  |  | ||||||
|  |   // Stuck on the left side of the switcher. | ||||||
|  |   let x = switcherRect.left; | ||||||
|  |  | ||||||
|  |   if (isOnTop && !isRTL || isOnBottom && isRTL || isOnBottomRight && !isRTL) { | ||||||
|  |     // Stuck on the right side of the switcher. | ||||||
|  |     x = switcherRect.right - optionsElement.clientWidth; | ||||||
|  |   } | ||||||
|  |  | ||||||
|  |   // Stuck on the top of the switcher. | ||||||
|  |   let y = switcherRect.top - window.innerHeight - 10; | ||||||
|  |  | ||||||
|  |   if (isOnTop) { | ||||||
|  |     // Stuck on the bottom of the switcher. | ||||||
|  |     y = switcherRect.top - window.innerHeight + optionsElement.clientHeight + switcher.clientHeight + 4; | ||||||
|  |   } | ||||||
|  |  | ||||||
|  |   return { x: x, y: y }; | ||||||
|  | } | ||||||
|  |  | ||||||
|  | function toggleMenu(switcher) { | ||||||
|  |   const optionsElement = switcher.nextElementSibling; | ||||||
|  |  | ||||||
|  |   optionsElement.classList.toggle('hx:hidden'); | ||||||
|  |  | ||||||
|  |   // Calculate the position of a language options element. | ||||||
|  |   const translate = computeMenuTranslation(switcher, optionsElement); | ||||||
|  |  | ||||||
|  |   optionsElement.style.transform = `translate3d(${translate.x}px, ${translate.y}px, 0)`; | ||||||
|  | } | ||||||
|  |  | ||||||
|  | function resizeMenu(switcher) { | ||||||
|  |   const optionsElement = switcher.nextElementSibling; | ||||||
|  |  | ||||||
|  |   if (optionsElement.classList.contains('hx:hidden')) return; | ||||||
|  |  | ||||||
|  |   // Calculate the position of a language options element. | ||||||
|  |   const translate = computeMenuTranslation(switcher, optionsElement); | ||||||
|  |  | ||||||
|  |   optionsElement.style.transform = `translate3d(${translate.x}px, ${translate.y}px, 0)`; | ||||||
|  | } | ||||||
							
								
								
									
										57
									
								
								assets/js/core/tabs.js
									
									
									
									
									
										Normal file
									
								
							
							
						
						
									
										57
									
								
								assets/js/core/tabs.js
									
									
									
									
									
										Normal file
									
								
							| @@ -0,0 +1,57 @@ | |||||||
|  | (function () { | ||||||
|  |   function updateGroup(container, index) { | ||||||
|  |     const tabs = Array.from(container.querySelectorAll('.hextra-tabs-toggle')); | ||||||
|  |     tabs.forEach((tab, i) => { | ||||||
|  |       tab.dataset.state = i === index ? 'selected' : ''; | ||||||
|  |       if (i === index) { | ||||||
|  |         tab.setAttribute('aria-selected', 'true'); | ||||||
|  |         tab.tabIndex = 0; | ||||||
|  |       } else { | ||||||
|  |         tab.removeAttribute('aria-selected'); | ||||||
|  |         tab.removeAttribute('tabindex'); | ||||||
|  |       } | ||||||
|  |     }); | ||||||
|  |     const panelsContainer = container.parentElement.nextElementSibling; | ||||||
|  |     if (!panelsContainer) return; | ||||||
|  |     Array.from(panelsContainer.children).forEach((panel, i) => { | ||||||
|  |       panel.dataset.state = i === index ? 'selected' : ''; | ||||||
|  |       if (i === index) { | ||||||
|  |         panel.tabIndex = 0; | ||||||
|  |       } else { | ||||||
|  |         panel.removeAttribute('tabindex'); | ||||||
|  |       } | ||||||
|  |     }); | ||||||
|  |   } | ||||||
|  |  | ||||||
|  |   const syncGroups = document.querySelectorAll('[data-tab-group]'); | ||||||
|  |  | ||||||
|  |   syncGroups.forEach((group) => { | ||||||
|  |     const key = encodeURIComponent(group.dataset.tabGroup); | ||||||
|  |     const saved = localStorage.getItem('hextra-tab-' + key); | ||||||
|  |     if (saved !== null) { | ||||||
|  |       updateGroup(group, parseInt(saved, 10)); | ||||||
|  |     } | ||||||
|  |   }); | ||||||
|  |  | ||||||
|  |   document.querySelectorAll('.hextra-tabs-toggle').forEach((button) => { | ||||||
|  |     button.addEventListener('click', function (e) { | ||||||
|  |       const container = e.target.parentElement; | ||||||
|  |       const index = Array.from(container.querySelectorAll('.hextra-tabs-toggle')).indexOf( | ||||||
|  |         e.target | ||||||
|  |       ); | ||||||
|  |        | ||||||
|  |       if (container.dataset.tabGroup) { | ||||||
|  |         // Sync behavior: update all tab groups with the same name | ||||||
|  |         const tabGroupValue = container.dataset.tabGroup; | ||||||
|  |         const key = encodeURIComponent(tabGroupValue); | ||||||
|  |         document | ||||||
|  |           .querySelectorAll('[data-tab-group="' + tabGroupValue + '"]') | ||||||
|  |           .forEach((grp) => updateGroup(grp, index)); | ||||||
|  |         localStorage.setItem('hextra-tab-' + key, index.toString()); | ||||||
|  |       } else { | ||||||
|  |         // Non-sync behavior: update only this specific tab group | ||||||
|  |         updateGroup(container, index); | ||||||
|  |       } | ||||||
|  |     }); | ||||||
|  |   }); | ||||||
|  | })(); | ||||||
							
								
								
									
										61
									
								
								assets/js/core/theme.js
									
									
									
									
									
										Normal file
									
								
							
							
						
						
									
										61
									
								
								assets/js/core/theme.js
									
									
									
									
									
										Normal file
									
								
							| @@ -0,0 +1,61 @@ | |||||||
|  | // Light / Dark theme toggle | ||||||
|  | (function () { | ||||||
|  |   const defaultTheme = '{{ site.Params.theme.default | default `system`}}' | ||||||
|  |   const themes = ["light", "dark"]; | ||||||
|  |  | ||||||
|  |   const themeToggleButtons = document.querySelectorAll(".hextra-theme-toggle"); | ||||||
|  |   const themeToggleOptions = document.querySelectorAll(".hextra-theme-toggle-options p"); | ||||||
|  |  | ||||||
|  |   function applyTheme(theme) { | ||||||
|  |     theme = themes.includes(theme) ? theme : "system"; | ||||||
|  |  | ||||||
|  |     themeToggleButtons.forEach((btn) => btn.parentElement.dataset.theme = theme ); | ||||||
|  |  | ||||||
|  |     localStorage.setItem("color-theme", theme); | ||||||
|  |   } | ||||||
|  |  | ||||||
|  |   function switchTheme(theme) { | ||||||
|  |     setTheme(theme); | ||||||
|  |     applyTheme(theme); | ||||||
|  |   } | ||||||
|  |  | ||||||
|  |   const colorTheme = "color-theme" in localStorage ? localStorage.getItem("color-theme") : defaultTheme; | ||||||
|  |   switchTheme(colorTheme); | ||||||
|  |  | ||||||
|  |   // Add click event handler to the menu items. | ||||||
|  |   themeToggleOptions.forEach((option) => { | ||||||
|  |     option.addEventListener("click", function (e) { | ||||||
|  |       e.preventDefault(); | ||||||
|  |  | ||||||
|  |       switchTheme(option.dataset.item); | ||||||
|  |     }) | ||||||
|  |   }) | ||||||
|  |  | ||||||
|  |   // Add click event handler to the buttons | ||||||
|  |   themeToggleButtons.forEach((toggler) => { | ||||||
|  |     toggler.addEventListener("click", function (e) { | ||||||
|  |       e.preventDefault(); | ||||||
|  |  | ||||||
|  |       toggleMenu(toggler); | ||||||
|  |     }); | ||||||
|  |   }); | ||||||
|  |  | ||||||
|  |   window.addEventListener("resize", () => themeToggleButtons.forEach(resizeMenu)) | ||||||
|  |  | ||||||
|  |   // Dismiss the menu when clicking outside | ||||||
|  |   document.addEventListener('click', (e) => { | ||||||
|  |     if (e.target.closest('.hextra-theme-toggle') === null) { | ||||||
|  |       themeToggleButtons.forEach((toggler) => { | ||||||
|  |         toggler.dataset.state = 'closed'; | ||||||
|  |         toggler.nextElementSibling.classList.add('hx:hidden'); | ||||||
|  |       }); | ||||||
|  |     } | ||||||
|  |   }); | ||||||
|  |  | ||||||
|  |   // Listen for system theme changes | ||||||
|  |   window.matchMedia("(prefers-color-scheme: dark)").addEventListener("change", () => { | ||||||
|  |     if (localStorage.getItem("color-theme") === "system") { | ||||||
|  |       setTheme("system"); | ||||||
|  |     } | ||||||
|  |   }); | ||||||
|  | })(); | ||||||
							
								
								
									
										93
									
								
								assets/js/core/toc-scroll.js
									
									
									
									
									
										Normal file
									
								
							
							
						
						
									
										93
									
								
								assets/js/core/toc-scroll.js
									
									
									
									
									
										Normal file
									
								
							| @@ -0,0 +1,93 @@ | |||||||
|  | /** | ||||||
|  |  * TOC Scroll - Highlights active TOC links based on visible headings | ||||||
|  |  *  | ||||||
|  |  * Uses Intersection Observer to track heading visibility and applies | ||||||
|  |  * 'hextra-toc-active' class to corresponding TOC links. Selects the | ||||||
|  |  * topmost heading when multiple are visible. | ||||||
|  |  *  | ||||||
|  |  * Requires: .hextra-toc element, matching heading IDs, toc.css styles | ||||||
|  |  */ | ||||||
|  | document.addEventListener("DOMContentLoaded", function () { | ||||||
|  |   const toc = document.querySelector(".hextra-toc"); | ||||||
|  |   if (!toc) return; | ||||||
|  |  | ||||||
|  |   const tocLinks = toc.querySelectorAll('a[href^="#"]'); | ||||||
|  |   if (tocLinks.length === 0) return; | ||||||
|  |  | ||||||
|  |   const headingIds = Array.from(tocLinks).map((link) => link.getAttribute("href").substring(1)); | ||||||
|  |  | ||||||
|  |   const headings = headingIds.map((id) => document.getElementById(decodeURIComponent(id))).filter(Boolean); | ||||||
|  |   if (headings.length === 0) return; | ||||||
|  |  | ||||||
|  |   let currentActiveLink = null; | ||||||
|  |   let isHashNavigation = false; | ||||||
|  |  | ||||||
|  |   // Create intersection observer | ||||||
|  |   const observer = new IntersectionObserver( | ||||||
|  |     (entries) => { | ||||||
|  |       // Skip observer updates during hash navigation | ||||||
|  |       if (isHashNavigation) return; | ||||||
|  |  | ||||||
|  |       const visibleHeadings = entries.filter((entry) => entry.isIntersecting).map((entry) => entry.target); | ||||||
|  |  | ||||||
|  |       if (visibleHeadings.length === 0) return; | ||||||
|  |  | ||||||
|  |       // Find the heading closest to the top of the viewport | ||||||
|  |       const topMostHeading = visibleHeadings.reduce((closest, heading) => { | ||||||
|  |         const headingTop = heading.getBoundingClientRect().top; | ||||||
|  |         const closestTop = closest.getBoundingClientRect().top; | ||||||
|  |         return Math.abs(headingTop) < Math.abs(closestTop) ? heading : closest; | ||||||
|  |       }); | ||||||
|  |  | ||||||
|  |       // Encode the id and make it lowercase to match the TOC link | ||||||
|  |       const targetId = encodeURIComponent(topMostHeading.id).toLowerCase(); | ||||||
|  |       const targetLink = toc.querySelector(`a[href="#${targetId}"]`); | ||||||
|  |  | ||||||
|  |       if (targetLink && targetLink !== currentActiveLink) { | ||||||
|  |         // Remove active class from previous link | ||||||
|  |         if (currentActiveLink) { | ||||||
|  |           currentActiveLink.classList.remove("hextra-toc-active"); | ||||||
|  |         } | ||||||
|  |  | ||||||
|  |         // Add active class to current link | ||||||
|  |         targetLink.classList.add("hextra-toc-active"); | ||||||
|  |         currentActiveLink = targetLink; | ||||||
|  |       } | ||||||
|  |     }, | ||||||
|  |     { | ||||||
|  |       rootMargin: "-20px 0px -60% 0px", // Adjust sensitivity | ||||||
|  |       threshold: [0, 0.1, 0.5, 1], | ||||||
|  |     } | ||||||
|  |   ); | ||||||
|  |  | ||||||
|  |   // Observe all headings | ||||||
|  |   headings.forEach((heading) => observer.observe(heading)); | ||||||
|  |  | ||||||
|  |   // Handle direct navigation to page with hash | ||||||
|  |   function handleHashNavigation() { | ||||||
|  |     const hash = window.location.hash; // already url encoded | ||||||
|  |     if (hash) { | ||||||
|  |       const targetLink = toc.querySelector(`a[href="${hash}"]`); | ||||||
|  |       if (targetLink) { | ||||||
|  |         // Disable observer temporarily during hash navigation | ||||||
|  |         isHashNavigation = true; | ||||||
|  |  | ||||||
|  |         if (currentActiveLink) { | ||||||
|  |           currentActiveLink.classList.remove("hextra-toc-active"); | ||||||
|  |         } | ||||||
|  |         targetLink.classList.add("hextra-toc-active"); | ||||||
|  |         currentActiveLink = targetLink; | ||||||
|  |  | ||||||
|  |         // Re-enable observer after scroll settles | ||||||
|  |         setTimeout(() => { isHashNavigation = false; }, 500); | ||||||
|  |         return; | ||||||
|  |       } | ||||||
|  |     } | ||||||
|  |   } | ||||||
|  |  | ||||||
|  |   // Handle hash changes navigation | ||||||
|  |   window.addEventListener("hashchange", handleHashNavigation); | ||||||
|  |  | ||||||
|  |   // Handle initial load | ||||||
|  |   setTimeout(handleHashNavigation, 100); | ||||||
|  | }); | ||||||
| @@ -1,5 +1,16 @@ | |||||||
| // Search functionality using FlexSearch. | // Search functionality using FlexSearch. | ||||||
|  |  | ||||||
|  | // Change shortcut key to cmd+k on Mac, iPad or iPhone. | ||||||
|  | document.addEventListener("DOMContentLoaded", function () { | ||||||
|  |   if (/iPad|iPhone|Macintosh/.test(navigator.userAgent)) { | ||||||
|  |     // select the kbd element under the .hextra-search-wrapper class | ||||||
|  |     const keys = document.querySelectorAll(".hextra-search-wrapper kbd"); | ||||||
|  |     keys.forEach(key => { | ||||||
|  |       key.innerHTML = '<span class="hx:text-xs">⌘</span>K'; | ||||||
|  |     }); | ||||||
|  |   } | ||||||
|  | }); | ||||||
|  |  | ||||||
| // Render the search data as JSON. | // Render the search data as JSON. | ||||||
| // {{ $searchDataFile := printf "%s.search-data.json" .Language.Lang }} | // {{ $searchDataFile := printf "%s.search-data.json" .Language.Lang }} | ||||||
| // {{ $searchData := resources.Get "json/search-data.json" | resources.ExecuteAsTemplate $searchDataFile . }} | // {{ $searchData := resources.Get "json/search-data.json" | resources.ExecuteAsTemplate $searchDataFile . }} | ||||||
| @@ -11,21 +22,35 @@ | |||||||
| (function () { | (function () { | ||||||
|   const searchDataURL = '{{ $searchData.RelPermalink }}'; |   const searchDataURL = '{{ $searchData.RelPermalink }}'; | ||||||
|  |  | ||||||
|   const inputElements = document.querySelectorAll('.search-input'); |   const inputElements = document.querySelectorAll('.hextra-search-input'); | ||||||
|   for (const el of inputElements) { |   for (const el of inputElements) { | ||||||
|     el.addEventListener('focus', init); |     el.addEventListener('focus', init); | ||||||
|     el.addEventListener('keyup', search); |     el.addEventListener('keyup', search); | ||||||
|     el.addEventListener('keydown', handleKeyDown); |     el.addEventListener('keydown', handleKeyDown); | ||||||
|  |     el.addEventListener('input', handleInputChange); | ||||||
|  |   } | ||||||
|  |  | ||||||
|  |   const shortcutElements = document.querySelectorAll('.hextra-search-wrapper kbd'); | ||||||
|  |  | ||||||
|  |   function setShortcutElementsOpacity(opacity) { | ||||||
|  |     shortcutElements.forEach(el => { | ||||||
|  |       el.style.opacity = opacity; | ||||||
|  |     }); | ||||||
|  |   } | ||||||
|  |  | ||||||
|  |   function handleInputChange(e) { | ||||||
|  |     const opacity = e.target.value.length > 0 ? 0 : 100; | ||||||
|  |     setShortcutElementsOpacity(opacity); | ||||||
|   } |   } | ||||||
|  |  | ||||||
|   // Get the search wrapper, input, and results elements. |   // Get the search wrapper, input, and results elements. | ||||||
|   function getActiveSearchElement() { |   function getActiveSearchElement() { | ||||||
|     const inputs = Array.from(document.querySelectorAll('.search-wrapper')).filter(el => el.clientHeight > 0); |     const inputs = Array.from(document.querySelectorAll('.hextra-search-wrapper')).filter(el => el.clientHeight > 0); | ||||||
|     if (inputs.length === 1) { |     if (inputs.length === 1) { | ||||||
|       return { |       return { | ||||||
|         wrapper: inputs[0], |         wrapper: inputs[0], | ||||||
|         inputElement: inputs[0].querySelector('.search-input'), |         inputElement: inputs[0].querySelector('.hextra-search-input'), | ||||||
|         resultsElement: inputs[0].querySelector('.search-results') |         resultsElement: inputs[0].querySelector('.hextra-search-results') | ||||||
|       }; |       }; | ||||||
|     } |     } | ||||||
|     return undefined; |     return undefined; | ||||||
| @@ -68,6 +93,7 @@ | |||||||
|       e.target !== resultsElement && |       e.target !== resultsElement && | ||||||
|       !resultsElement.contains(e.target) |       !resultsElement.contains(e.target) | ||||||
|     ) { |     ) { | ||||||
|  |       setShortcutElementsOpacity(100); | ||||||
|       hideSearchResults(); |       hideSearchResults(); | ||||||
|     } |     } | ||||||
|   }); |   }); | ||||||
| @@ -77,7 +103,7 @@ | |||||||
|     const { resultsElement } = getActiveSearchElement(); |     const { resultsElement } = getActiveSearchElement(); | ||||||
|     if (!resultsElement) return { result: undefined, index: -1 }; |     if (!resultsElement) return { result: undefined, index: -1 }; | ||||||
|  |  | ||||||
|     const result = resultsElement.querySelector('.active'); |     const result = resultsElement.querySelector('.hextra-search-active'); | ||||||
|     if (!result) return { result: undefined, index: -1 }; |     if (!result) return { result: undefined, index: -1 }; | ||||||
|  |  | ||||||
|     const index = parseInt(result.dataset.index, 10); |     const index = parseInt(result.dataset.index, 10); | ||||||
| @@ -90,10 +116,10 @@ | |||||||
|     if (!resultsElement) return; |     if (!resultsElement) return; | ||||||
|  |  | ||||||
|     const { result: activeResult } = getActiveResult(); |     const { result: activeResult } = getActiveResult(); | ||||||
|     activeResult && activeResult.classList.remove('active'); |     activeResult && activeResult.classList.remove('hextra-search-active'); | ||||||
|     const result = resultsElement.querySelector(`[data-index="${index}"]`); |     const result = resultsElement.querySelector(`[data-index="${index}"]`); | ||||||
|     if (result) { |     if (result) { | ||||||
|       result.classList.add('active'); |       result.classList.add('hextra-search-active'); | ||||||
|       result.focus(); |       result.focus(); | ||||||
|     } |     } | ||||||
|   } |   } | ||||||
| @@ -117,7 +143,7 @@ | |||||||
|   function hideSearchResults() { |   function hideSearchResults() { | ||||||
|     const { resultsElement } = getActiveSearchElement(); |     const { resultsElement } = getActiveSearchElement(); | ||||||
|     if (!resultsElement) return; |     if (!resultsElement) return; | ||||||
|     resultsElement.classList.add('hidden'); |     resultsElement.classList.add('hx:hidden'); | ||||||
|   } |   } | ||||||
|  |  | ||||||
|   // Handle keyboard events. |   // Handle keyboard events. | ||||||
| @@ -146,6 +172,10 @@ | |||||||
|       case 'Escape': |       case 'Escape': | ||||||
|         e.preventDefault(); |         e.preventDefault(); | ||||||
|         hideSearchResults(); |         hideSearchResults(); | ||||||
|  |         // Clear the input when pressing escape | ||||||
|  |         inputElement.value = ''; | ||||||
|  |         inputElement.dispatchEvent(new Event('input')); | ||||||
|  |         // Remove focus from the input | ||||||
|         inputElement.blur(); |         inputElement.blur(); | ||||||
|         break; |         break; | ||||||
|     } |     } | ||||||
| @@ -159,26 +189,42 @@ | |||||||
|     } |     } | ||||||
|   } |   } | ||||||
|  |  | ||||||
|   // Preload the search index. |   /** | ||||||
|  |    * Preloads the search index by fetching data and adding it to the FlexSearch index. | ||||||
|  |    * @returns {Promise<void>} A promise that resolves when the index is preloaded. | ||||||
|  |    */ | ||||||
|   async function preloadIndex() { |   async function preloadIndex() { | ||||||
|  |     const tokenize = '{{- site.Params.search.flexsearch.tokenize | default  "forward" -}}'; | ||||||
|  |  | ||||||
|  |     // https://github.com/TryGhost/Ghost/pull/21148 | ||||||
|  |     const regex = new RegExp( | ||||||
|  |       `[\u{4E00}-\u{9FFF}\u{3040}-\u{309F}\u{30A0}-\u{30FF}\u{AC00}-\u{D7A3}\u{3400}-\u{4DBF}\u{20000}-\u{2A6DF}\u{2A700}-\u{2B73F}\u{2B740}-\u{2B81F}\u{2B820}-\u{2CEAF}\u{2CEB0}-\u{2EBEF}\u{30000}-\u{3134F}\u{31350}-\u{323AF}\u{2EBF0}-\u{2EE5F}\u{F900}-\u{FAFF}\u{2F800}-\u{2FA1F}]|[0-9A-Za-zа-я\u00C0-\u017F\u0400-\u04FF\u0600-\u06FF\u0980-\u09FF\u1E00-\u1EFF\u0590-\u05FF]+`, | ||||||
|  |       'mug' | ||||||
|  |     ); | ||||||
|  |     const encode = (str) => { return ('' + str).toLowerCase().match(regex) ?? []; } | ||||||
|  |  | ||||||
|     window.pageIndex = new FlexSearch.Document({ |     window.pageIndex = new FlexSearch.Document({ | ||||||
|       tokenize: 'forward', |       tokenize, | ||||||
|  |       encode, | ||||||
|       cache: 100, |       cache: 100, | ||||||
|       document: { |       document: { | ||||||
|         id: 'id', |         id: 'id', | ||||||
|         store: ['title'], |         store: ['title', 'crumb'], | ||||||
|         index: "content" |         index: "content" | ||||||
|       } |       } | ||||||
|     }); |     }); | ||||||
|  |  | ||||||
|     window.sectionIndex = new FlexSearch.Document({ |     window.sectionIndex = new FlexSearch.Document({ | ||||||
|       tokenize: 'forward', |       tokenize, | ||||||
|  |       encode, | ||||||
|       cache: 100, |       cache: 100, | ||||||
|       document: { |       document: { | ||||||
|         id: 'id', |         id: 'id', | ||||||
|         store: ['title', 'content', 'url', 'display'], |         store: ['title', 'content', 'url', 'display', 'crumb'], | ||||||
|         index: "content", |         index: "content", | ||||||
|         tag: 'pageId' |         tag: [{ | ||||||
|  |           field: "pageId" | ||||||
|  |         }] | ||||||
|       } |       } | ||||||
|     }); |     }); | ||||||
|  |  | ||||||
| @@ -188,6 +234,30 @@ | |||||||
|     for (const route in data) { |     for (const route in data) { | ||||||
|       let pageContent = ''; |       let pageContent = ''; | ||||||
|       ++pageId; |       ++pageId; | ||||||
|  |       const urlParts = route.split('/').filter(x => x != "" && !x.startsWith('#')); | ||||||
|  |  | ||||||
|  |       let crumb = ''; | ||||||
|  |       let searchUrl = '/'; | ||||||
|  |       for (let i = 0; i < urlParts.length; i++) { | ||||||
|  |         const urlPart = urlParts[i]; | ||||||
|  |         searchUrl += urlPart + '/' | ||||||
|  |  | ||||||
|  |         const crumbData = data[searchUrl]; | ||||||
|  |         if (!crumbData) { | ||||||
|  |           console.warn('Excluded page', searchUrl, '- will not be included for search result breadcrumb for', route); | ||||||
|  |           continue; | ||||||
|  |         } | ||||||
|  |  | ||||||
|  |         let title = data[searchUrl].title; | ||||||
|  |         if (title == "_index") { | ||||||
|  |           title = urlPart.split("-").map(x => x).join(" "); | ||||||
|  |         } | ||||||
|  |         crumb += title; | ||||||
|  |  | ||||||
|  |         if (i < urlParts.length - 1) { | ||||||
|  |           crumb += ' > '; | ||||||
|  |         } | ||||||
|  |       } | ||||||
|  |  | ||||||
|       for (const heading in data[route].data) { |       for (const heading in data[route].data) { | ||||||
|         const [hash, text] = heading.split('#'); |         const [hash, text] = heading.split('#'); | ||||||
| @@ -201,6 +271,7 @@ | |||||||
|           id: url, |           id: url, | ||||||
|           url, |           url, | ||||||
|           title, |           title, | ||||||
|  |           crumb, | ||||||
|           pageId: `page_${pageId}`, |           pageId: `page_${pageId}`, | ||||||
|           content: title, |           content: title, | ||||||
|           ...(paragraphs[0] && { display: paragraphs[0] }) |           ...(paragraphs[0] && { display: paragraphs[0] }) | ||||||
| @@ -211,6 +282,7 @@ | |||||||
|             id: `${url}_${i}`, |             id: `${url}_${i}`, | ||||||
|             url, |             url, | ||||||
|             title, |             title, | ||||||
|  |             crumb, | ||||||
|             pageId: `page_${pageId}`, |             pageId: `page_${pageId}`, | ||||||
|             content: paragraphs[i] |             content: paragraphs[i] | ||||||
|           }); |           }); | ||||||
| @@ -222,12 +294,17 @@ | |||||||
|       window.pageIndex.add({ |       window.pageIndex.add({ | ||||||
|         id: pageId, |         id: pageId, | ||||||
|         title: data[route].title, |         title: data[route].title, | ||||||
|  |         crumb, | ||||||
|         content: pageContent |         content: pageContent | ||||||
|       }); |       }); | ||||||
|  |  | ||||||
|     } |     } | ||||||
|   } |   } | ||||||
|  |  | ||||||
|  |   /** | ||||||
|  |    * Performs a search based on the provided query and displays the results. | ||||||
|  |    * @param {Event} e - The event object. | ||||||
|  |    */ | ||||||
|   function search(e) { |   function search(e) { | ||||||
|     const query = e.target.value; |     const query = e.target.value; | ||||||
|     if (!e.target.value) { |     if (!e.target.value) { | ||||||
| @@ -239,7 +316,7 @@ | |||||||
|     while (resultsElement.firstChild) { |     while (resultsElement.firstChild) { | ||||||
|       resultsElement.removeChild(resultsElement.firstChild); |       resultsElement.removeChild(resultsElement.firstChild); | ||||||
|     } |     } | ||||||
|     resultsElement.classList.remove('hidden'); |     resultsElement.classList.remove('hx:hidden'); | ||||||
|  |  | ||||||
|     const pageResults = window.pageIndex.search(query, 5, { enrich: true, suggest: true })[0]?.result || []; |     const pageResults = window.pageIndex.search(query, 5, { enrich: true, suggest: true })[0]?.result || []; | ||||||
|  |  | ||||||
| @@ -251,7 +328,7 @@ | |||||||
|       pageTitleMatches[i] = 0; |       pageTitleMatches[i] = 0; | ||||||
|  |  | ||||||
|       // Show the top 5 results for each page |       // Show the top 5 results for each page | ||||||
|       const sectionResults = window.sectionIndex.search(query, 5, { enrich: true, suggest: true, tag: `page_${result.id}` })[0]?.result || []; |       const sectionResults = window.sectionIndex.search(query, 5, { enrich: true, suggest: true, tag: { 'pageId': `page_${result.id}` } })[0]?.result || []; | ||||||
|       let isFirstItemOfPage = true |       let isFirstItemOfPage = true | ||||||
|       const occurred = {} |       const occurred = {} | ||||||
|  |  | ||||||
| @@ -270,7 +347,7 @@ | |||||||
|           _page_rk: i, |           _page_rk: i, | ||||||
|           _section_rk: j, |           _section_rk: j, | ||||||
|           route: url, |           route: url, | ||||||
|           prefix: isFirstItemOfPage ? result.doc.title : undefined, |           prefix: isFirstItemOfPage ? result.doc.crumb : undefined, | ||||||
|           children: { title, content } |           children: { title, content } | ||||||
|         }) |         }) | ||||||
|         isFirstItemOfPage = false |         isFirstItemOfPage = false | ||||||
| @@ -296,12 +373,18 @@ | |||||||
|     displayResults(sortedResults, query); |     displayResults(sortedResults, query); | ||||||
|   } |   } | ||||||
|  |  | ||||||
|  |   /** | ||||||
|  |    * Displays the search results on the page. | ||||||
|  |    * | ||||||
|  |    * @param {Array} results - The array of search results. | ||||||
|  |    * @param {string} query - The search query. | ||||||
|  |    */ | ||||||
|   function displayResults(results, query) { |   function displayResults(results, query) { | ||||||
|     const { resultsElement } = getActiveSearchElement(); |     const { resultsElement } = getActiveSearchElement(); | ||||||
|     if (!resultsElement) return; |     if (!resultsElement) return; | ||||||
|  |  | ||||||
|     if (!results.length) { |     if (!results.length) { | ||||||
|       resultsElement.innerHTML = `<span class="no-result">{{ $noResultsFound | safeHTML }}</span>`; |       resultsElement.innerHTML = `<span class="hextra-search-no-result">{{ $noResultsFound | safeHTML }}</span>`; | ||||||
|       return; |       return; | ||||||
|     } |     } | ||||||
|  |  | ||||||
| @@ -309,7 +392,7 @@ | |||||||
|     function highlightMatches(text, query) { |     function highlightMatches(text, query) { | ||||||
|       const escapedQuery = query.replace(/[-\\^$*+?.()|[\]{}]/g, '\\$&'); |       const escapedQuery = query.replace(/[-\\^$*+?.()|[\]{}]/g, '\\$&'); | ||||||
|       const regex = new RegExp(escapedQuery, 'gi'); |       const regex = new RegExp(escapedQuery, 'gi'); | ||||||
|       return text.replace(regex, (match) => `<span class="match">${match}</span>`); |       return text.replace(regex, (match) => `<span class="hextra-search-match">${match}</span>`); | ||||||
|     } |     } | ||||||
|  |  | ||||||
|     // Create a DOM element from the HTML string. |     // Create a DOM element from the HTML string. | ||||||
| @@ -322,11 +405,11 @@ | |||||||
|     function handleMouseMove(e) { |     function handleMouseMove(e) { | ||||||
|       const target = e.target.closest('a'); |       const target = e.target.closest('a'); | ||||||
|       if (target) { |       if (target) { | ||||||
|         const active = resultsElement.querySelector('a.active'); |         const active = resultsElement.querySelector('a.hextra-search-active'); | ||||||
|         if (active) { |         if (active) { | ||||||
|           active.classList.remove('active'); |           active.classList.remove('hextra-search-active'); | ||||||
|         } |         } | ||||||
|         target.classList.add('active'); |         target.classList.add('hextra-search-active'); | ||||||
|       } |       } | ||||||
|     } |     } | ||||||
|  |  | ||||||
| @@ -335,14 +418,14 @@ | |||||||
|       const result = results[i]; |       const result = results[i]; | ||||||
|       if (result.prefix) { |       if (result.prefix) { | ||||||
|         fragment.appendChild(createElement(` |         fragment.appendChild(createElement(` | ||||||
|           <div class="prefix">${result.prefix}</div>`)); |           <div class="hextra-search-prefix">${result.prefix}</div>`)); | ||||||
|       } |       } | ||||||
|         let li = createElement(` |         let li = createElement(` | ||||||
|         <li> |         <li> | ||||||
|           <a data-index="${i}" href="${result.route}" class=${i === 0 ? "active" : ""}> |           <a data-index="${i}" href="${result.route}" class=${i === 0 ? "hextra-search-active" : ""}> | ||||||
|             <div class="title">`+ highlightMatches(result.children.title, query) + `</div>` + |             <div class="hextra-search-title">`+ highlightMatches(result.children.title, query) + `</div>` + | ||||||
|         (result.children.content ? |         (result.children.content ? | ||||||
|           `<div class="excerpt">` + highlightMatches(result.children.content, query) + `</div>` : '') + ` |             `<div class="hextra-search-excerpt">` + highlightMatches(result.children.content, query) + `</div>` : '') + ` | ||||||
|           </a> |           </a> | ||||||
|         </li>`); |         </li>`); | ||||||
|       li.addEventListener('mousemove', handleMouseMove); |       li.addEventListener('mousemove', handleMouseMove); | ||||||
|   | |||||||
							
								
								
									
										6
									
								
								assets/js/head/banner.js
									
									
									
									
									
										Normal file
									
								
							
							
						
						
									
										6
									
								
								assets/js/head/banner.js
									
									
									
									
									
										Normal file
									
								
							| @@ -0,0 +1,6 @@ | |||||||
|  | // The section must not be in the banner.js (body) file because it can create a quick flash. | ||||||
|  |  | ||||||
|  | if (localStorage.getItem('{{ site.Params.banner.key | default `banner-closed` }}')) { | ||||||
|  |   document.documentElement.style.setProperty("--hextra-banner-height", "0px"); | ||||||
|  |   document.documentElement.classList.add("hextra-banner-hidden"); | ||||||
|  | } | ||||||
							
								
								
									
										14
									
								
								assets/js/head/theme.js
									
									
									
									
									
										Normal file
									
								
							
							
						
						
									
										14
									
								
								assets/js/head/theme.js
									
									
									
									
									
										Normal file
									
								
							| @@ -0,0 +1,14 @@ | |||||||
|  | // The section must not be in the theme.js (body) file because it can create a quick flash (switch between light and dark). | ||||||
|  |  | ||||||
|  | function setTheme(theme) { | ||||||
|  |   document.documentElement.classList.remove("light", "dark"); | ||||||
|  |  | ||||||
|  |   if (theme !== "light" && theme !== "dark") { | ||||||
|  |     theme = window.matchMedia("(prefers-color-scheme: dark)").matches ? "dark" : "light"; | ||||||
|  |   } | ||||||
|  |  | ||||||
|  |   document.documentElement.classList.add(theme); | ||||||
|  |   document.documentElement.style.colorScheme = theme; | ||||||
|  | } | ||||||
|  |  | ||||||
|  | setTheme("color-theme" in localStorage ? localStorage.getItem("color-theme") : '{{ site.Params.theme.default | default `system`}}') | ||||||
| @@ -1,28 +0,0 @@ | |||||||
| (function () { |  | ||||||
|   const languageSwitchers = document.querySelectorAll('.language-switcher'); |  | ||||||
|   languageSwitchers.forEach((switcher) => { |  | ||||||
|     switcher.addEventListener('click', (e) => { |  | ||||||
|       e.preventDefault(); |  | ||||||
|       switcher.dataset.state = switcher.dataset.state === 'open' ? 'closed' : 'open'; |  | ||||||
|       const optionsElement = switcher.nextElementSibling; |  | ||||||
|       optionsElement.classList.toggle('hidden'); |  | ||||||
|  |  | ||||||
|       // Calculate position of language options element |  | ||||||
|       const switcherRect = switcher.getBoundingClientRect(); |  | ||||||
|       const translateY = switcherRect.top - window.innerHeight - 15; |  | ||||||
|       optionsElement.style.transform = `translate3d(${switcherRect.left}px, ${translateY}px, 0)`; |  | ||||||
|       optionsElement.style.minWidth = `${Math.max(switcherRect.width, 50)}px`; |  | ||||||
|     }); |  | ||||||
|   }); |  | ||||||
|  |  | ||||||
|   // Dismiss language switcher when clicking outside |  | ||||||
|   document.addEventListener('click', (e) => { |  | ||||||
|     if (e.target.closest('.language-switcher') === null) { |  | ||||||
|       languageSwitchers.forEach((switcher) => { |  | ||||||
|         switcher.dataset.state = 'closed'; |  | ||||||
|         const optionsElement = switcher.nextElementSibling; |  | ||||||
|         optionsElement.classList.add('hidden'); |  | ||||||
|       }); |  | ||||||
|     } |  | ||||||
|   }); |  | ||||||
| })(); |  | ||||||
| @@ -1,19 +0,0 @@ | |||||||
| // Hamburger menu for mobile navigation |  | ||||||
|  |  | ||||||
| const menu = document.querySelector('.hamburger-menu'); |  | ||||||
|  |  | ||||||
| menu.addEventListener('click', (e) => { |  | ||||||
|   e.preventDefault(); |  | ||||||
|   const sidebarContainer = document.querySelector('.sidebar-container'); |  | ||||||
|  |  | ||||||
|   // Toggle the hamburger menu |  | ||||||
|   menu.querySelector('svg').classList.toggle('open'); |  | ||||||
|  |  | ||||||
|   // When the menu is open, we want to show the navigation sidebar |  | ||||||
|   sidebarContainer.classList.toggle('max-md:[transform:translate3d(0,-100%,0)]'); |  | ||||||
|   sidebarContainer.classList.toggle('max-md:[transform:translate3d(0,0,0)]'); |  | ||||||
|  |  | ||||||
|   // When the menu is open, we want to prevent the body from scrolling |  | ||||||
|   document.body.classList.toggle('overflow-hidden'); |  | ||||||
|   document.body.classList.toggle('md:overflow-auto'); |  | ||||||
| }); |  | ||||||
| @@ -1,12 +0,0 @@ | |||||||
| document.addEventListener("DOMContentLoaded", function () { |  | ||||||
|   const buttons = document.querySelectorAll(".hextra-sidebar-collapsible-button"); |  | ||||||
|   buttons.forEach(function (button) { |  | ||||||
|     button.addEventListener("click", function (e) { |  | ||||||
|       e.preventDefault(); |  | ||||||
|       const list = button.parentElement.parentElement; |  | ||||||
|       if (list) { |  | ||||||
|         list.classList.toggle("open") |  | ||||||
|       } |  | ||||||
|     }); |  | ||||||
|   }); |  | ||||||
| }); |  | ||||||
| @@ -1,20 +0,0 @@ | |||||||
| document.querySelectorAll('.tabs-toggle').forEach(function (button) { |  | ||||||
|   button.addEventListener('click', function (e) { |  | ||||||
|     // set parent tabs to unselected |  | ||||||
|     const tabs = Array.from(e.target.parentElement.querySelectorAll('.tabs-toggle')); |  | ||||||
|     tabs.map(tab => tab.dataset.state = ''); |  | ||||||
|  |  | ||||||
|     // set current tab to selected |  | ||||||
|     e.target.dataset.state = 'selected'; |  | ||||||
|  |  | ||||||
|     // set all panels to unselected |  | ||||||
|     const panelsContainer = e.target.parentElement.nextElementSibling; |  | ||||||
|     Array.from(panelsContainer.children).forEach(function (panel) { |  | ||||||
|       panel.dataset.state = ''; |  | ||||||
|     }); |  | ||||||
|  |  | ||||||
|     const panelId = e.target.getAttribute('aria-controls'); |  | ||||||
|     const panel = panelsContainer.querySelector(`#${panelId}`); |  | ||||||
|     panel.dataset.state = 'selected'; |  | ||||||
|   }); |  | ||||||
| }); |  | ||||||
| @@ -1,36 +0,0 @@ | |||||||
| // Dark theme toggle |  | ||||||
|  |  | ||||||
| const themeToggleButtons = document.querySelectorAll(".theme-toggle"); |  | ||||||
|  |  | ||||||
| // Change the icons inside the button based on previous settings |  | ||||||
| if ( |  | ||||||
|   localStorage.getItem("color-theme") === "dark" || |  | ||||||
|   (!("color-theme" in localStorage) && window.matchMedia("(prefers-color-scheme: dark)").matches) |  | ||||||
| ) { |  | ||||||
|   themeToggleButtons.forEach((el) => el.dataset.theme = "dark"); |  | ||||||
| } else { |  | ||||||
|   themeToggleButtons.forEach((el) => el.dataset.theme = "light"); |  | ||||||
| } |  | ||||||
|  |  | ||||||
| themeToggleButtons.forEach((el) => { |  | ||||||
|   el.addEventListener("click", function () { |  | ||||||
|     if (localStorage.getItem("color-theme")) { |  | ||||||
|       if (localStorage.getItem("color-theme") === "light") { |  | ||||||
|         document.documentElement.classList.add("dark"); |  | ||||||
|         localStorage.setItem("color-theme", "dark"); |  | ||||||
|       } else { |  | ||||||
|         document.documentElement.classList.remove("dark"); |  | ||||||
|         localStorage.setItem("color-theme", "light"); |  | ||||||
|       } |  | ||||||
|     } else { |  | ||||||
|       if (document.documentElement.classList.contains("dark")) { |  | ||||||
|         document.documentElement.classList.remove("dark"); |  | ||||||
|         localStorage.setItem("color-theme", "light"); |  | ||||||
|       } else { |  | ||||||
|         document.documentElement.classList.add("dark"); |  | ||||||
|         localStorage.setItem("color-theme", "dark"); |  | ||||||
|       } |  | ||||||
|     } |  | ||||||
|     el.dataset.theme = document.documentElement.classList.contains("dark") ? "dark" : "light"; |  | ||||||
|   }); |  | ||||||
| }); |  | ||||||
| @@ -1,3 +1,10 @@ | |||||||
|  | {{/* FlexSearch Index Data */}} | ||||||
|  | {{- $indexType := site.Params.search.flexsearch.index | default "content" -}} | ||||||
|  |  | ||||||
|  | {{- if not (in (slice "content" "summary" "heading" "title" ) $indexType) -}} | ||||||
|  |   {{- errorf "unknown flexsearch index type: %s" $indexType -}} | ||||||
|  | {{- end -}} | ||||||
|  |  | ||||||
| {{- $pages := where .Site.Pages "Kind" "in" (slice "page" "section") -}} | {{- $pages := where .Site.Pages "Kind" "in" (slice "page" "section") -}} | ||||||
| {{- $pages = where $pages "Params.excludeSearch" "!=" true -}} | {{- $pages = where $pages "Params.excludeSearch" "!=" true -}} | ||||||
| {{- $pages = where $pages "Content" "!=" "" -}} | {{- $pages = where $pages "Content" "!=" "" -}} | ||||||
| @@ -7,7 +14,7 @@ | |||||||
| {{- range $index, $page := $pages -}} | {{- range $index, $page := $pages -}} | ||||||
|   {{- $pageTitle := $page.LinkTitle | default $page.File.BaseFileName -}} |   {{- $pageTitle := $page.LinkTitle | default $page.File.BaseFileName -}} | ||||||
|   {{- $pageLink := $page.RelPermalink -}} |   {{- $pageLink := $page.RelPermalink -}} | ||||||
|   {{- $data := partial "utils/fragments" $page -}} |   {{- $data := partial "utils/fragments" (dict "context" $page "type" $indexType) -}} | ||||||
|   {{- $output = $output | merge (dict $pageLink (dict "title" $pageTitle "data" $data)) -}} |   {{- $output = $output | merge (dict $pageLink (dict "title" $pageTitle "data" $data)) -}} | ||||||
| {{- end -}} | {{- end -}} | ||||||
|  |  | ||||||
|   | |||||||
							
								
								
									
										39
									
								
								assets/lib/flexsearch/flexsearch.bundle.min.js
									
									
									
									
										vendored
									
									
								
							
							
						
						
									
										39
									
								
								assets/lib/flexsearch/flexsearch.bundle.min.js
									
									
									
									
										vendored
									
									
								
							| @@ -1,39 +0,0 @@ | |||||||
| /** |  | ||||||
|  * Skipped minification because the original files appears to be already minified. |  | ||||||
|  * Original file: /npm/flexsearch@0.7.31/dist/flexsearch.bundle.js |  | ||||||
|  * |  | ||||||
|  * Do NOT use SRI with dynamically generated files! More information: https://www.jsdelivr.com/using-sri-with-dynamic-files |  | ||||||
|  */ |  | ||||||
| /**! |  | ||||||
|  * FlexSearch.js v0.7.31 (Bundle) |  | ||||||
|  * Copyright 2018-2022 Nextapps GmbH |  | ||||||
|  * Author: Thomas Wilkerling |  | ||||||
|  * Licence: Apache-2.0 |  | ||||||
|  * https://github.com/nextapps-de/flexsearch |  | ||||||
|  */ |  | ||||||
| (function _f(self){'use strict';try{if(module)self=module}catch(e){}self._factory=_f;var t;function u(a){return"undefined"!==typeof a?a:!0}function aa(a){const b=Array(a);for(let c=0;c<a;c++)b[c]=v();return b}function v(){return Object.create(null)}function ba(a,b){return b.length-a.length}function x(a){return"string"===typeof a}function C(a){return"object"===typeof a}function D(a){return"function"===typeof a};function ca(a,b){var c=da;if(a&&(b&&(a=E(a,b)),this.H&&(a=E(a,this.H)),this.J&&1<a.length&&(a=E(a,this.J)),c||""===c)){a=a.split(c);if(this.filter){b=this.filter;c=a.length;const d=[];for(let e=0,f=0;e<c;e++){const g=a[e];g&&!b[g]&&(d[f++]=g)}a=d}return a}return a}const da=/[\p{Z}\p{S}\p{P}\p{C}]+/u,ea=/[\u0300-\u036f]/g; |  | ||||||
| function fa(a,b){const c=Object.keys(a),d=c.length,e=[];let f="",g=0;for(let h=0,k,m;h<d;h++)k=c[h],(m=a[k])?(e[g++]=F(b?"(?!\\b)"+k+"(\\b|_)":k),e[g++]=m):f+=(f?"|":"")+k;f&&(e[g++]=F(b?"(?!\\b)("+f+")(\\b|_)":"("+f+")"),e[g]="");return e}function E(a,b){for(let c=0,d=b.length;c<d&&(a=a.replace(b[c],b[c+1]),a);c+=2);return a}function F(a){return new RegExp(a,"g")}function ha(a){let b="",c="";for(let d=0,e=a.length,f;d<e;d++)(f=a[d])!==c&&(b+=c=f);return b};var ja={encode:ia,F:!1,G:""};function ia(a){return ca.call(this,(""+a).toLowerCase(),!1)};const ka={},G={};function la(a){I(a,"add");I(a,"append");I(a,"search");I(a,"update");I(a,"remove")}function I(a,b){a[b+"Async"]=function(){const c=this,d=arguments;var e=d[d.length-1];let f;D(e)&&(f=e,delete d[d.length-1]);e=new Promise(function(g){setTimeout(function(){c.async=!0;const h=c[b].apply(c,d);c.async=!1;g(h)})});return f?(e.then(f),this):e}};function ma(a,b,c,d){const e=a.length;let f=[],g,h,k=0;d&&(d=[]);for(let m=e-1;0<=m;m--){const n=a[m],w=n.length,q=v();let r=!g;for(let l=0;l<w;l++){const p=n[l],z=p.length;if(z)for(let B=0,A,y;B<z;B++)if(y=p[B],g){if(g[y]){if(!m)if(c)c--;else if(f[k++]=y,k===b)return f;if(m||d)q[y]=1;r=!0}if(d&&(A=(h[y]||0)+1,h[y]=A,A<e)){const H=d[A-2]||(d[A-2]=[]);H[H.length]=y}}else q[y]=1}if(d)g||(h=q);else if(!r)return[];g=q}if(d)for(let m=d.length-1,n,w;0<=m;m--){n=d[m];w=n.length;for(let q=0,r;q<w;q++)if(r= |  | ||||||
| n[q],!g[r]){if(c)c--;else if(f[k++]=r,k===b)return f;g[r]=1}}return f}function na(a,b){const c=v(),d=v(),e=[];for(let f=0;f<a.length;f++)c[a[f]]=1;for(let f=0,g;f<b.length;f++){g=b[f];for(let h=0,k;h<g.length;h++)k=g[h],c[k]&&!d[k]&&(d[k]=1,e[e.length]=k)}return e};function J(a){this.l=!0!==a&&a;this.cache=v();this.h=[]}function oa(a,b,c){C(a)&&(a=a.query);let d=this.cache.get(a);d||(d=this.search(a,b,c),this.cache.set(a,d));return d}J.prototype.set=function(a,b){if(!this.cache[a]){var c=this.h.length;c===this.l?delete this.cache[this.h[c-1]]:c++;for(--c;0<c;c--)this.h[c]=this.h[c-1];this.h[0]=a}this.cache[a]=b};J.prototype.get=function(a){const b=this.cache[a];if(this.l&&b&&(a=this.h.indexOf(a))){const c=this.h[a-1];this.h[a-1]=this.h[a];this.h[a]=c}return b};const qa={memory:{charset:"latin:extra",D:3,B:4,m:!1},performance:{D:3,B:3,s:!1,context:{depth:2,D:1}},match:{charset:"latin:extra",G:"reverse"},score:{charset:"latin:advanced",D:20,B:3,context:{depth:3,D:9}},"default":{}};function ra(a,b,c,d,e,f,g){setTimeout(function(){const h=a(c?c+"."+d:d,JSON.stringify(g));h&&h.then?h.then(function(){b.export(a,b,c,e,f+1)}):b.export(a,b,c,e,f+1)})};function K(a,b){if(!(this instanceof K))return new K(a);var c;if(a){x(a)?a=qa[a]:(c=a.preset)&&(a=Object.assign({},c[c],a));c=a.charset;var d=a.lang;x(c)&&(-1===c.indexOf(":")&&(c+=":default"),c=G[c]);x(d)&&(d=ka[d])}else a={};let e,f,g=a.context||{};this.encode=a.encode||c&&c.encode||ia;this.register=b||v();this.D=e=a.resolution||9;this.G=b=c&&c.G||a.tokenize||"strict";this.depth="strict"===b&&g.depth;this.l=u(g.bidirectional);this.s=f=u(a.optimize);this.m=u(a.fastupdate);this.B=a.minlength||1;this.C= |  | ||||||
| a.boost;this.map=f?aa(e):v();this.A=e=g.resolution||1;this.h=f?aa(e):v();this.F=c&&c.F||a.rtl;this.H=(b=a.matcher||d&&d.H)&&fa(b,!1);this.J=(b=a.stemmer||d&&d.J)&&fa(b,!0);if(c=b=a.filter||d&&d.filter){c=b;d=v();for(let h=0,k=c.length;h<k;h++)d[c[h]]=1;c=d}this.filter=c;this.cache=(b=a.cache)&&new J(b)}t=K.prototype;t.append=function(a,b){return this.add(a,b,!0)}; |  | ||||||
| t.add=function(a,b,c,d){if(b&&(a||0===a)){if(!d&&!c&&this.register[a])return this.update(a,b);b=this.encode(b);if(d=b.length){const m=v(),n=v(),w=this.depth,q=this.D;for(let r=0;r<d;r++){let l=b[this.F?d-1-r:r];var e=l.length;if(l&&e>=this.B&&(w||!n[l])){var f=L(q,d,r),g="";switch(this.G){case "full":if(2<e){for(f=0;f<e;f++)for(var h=e;h>f;h--)if(h-f>=this.B){var k=L(q,d,r,e,f);g=l.substring(f,h);M(this,n,g,k,a,c)}break}case "reverse":if(1<e){for(h=e-1;0<h;h--)g=l[h]+g,g.length>=this.B&&M(this,n, |  | ||||||
| g,L(q,d,r,e,h),a,c);g=""}case "forward":if(1<e){for(h=0;h<e;h++)g+=l[h],g.length>=this.B&&M(this,n,g,f,a,c);break}default:if(this.C&&(f=Math.min(f/this.C(b,l,r)|0,q-1)),M(this,n,l,f,a,c),w&&1<d&&r<d-1)for(e=v(),g=this.A,f=l,h=Math.min(w+1,d-r),e[f]=1,k=1;k<h;k++)if((l=b[this.F?d-1-r-k:r+k])&&l.length>=this.B&&!e[l]){e[l]=1;const p=this.l&&l>f;M(this,m,p?f:l,L(g+(d/2>g?0:1),d,r,h-1,k-1),a,c,p?l:f)}}}}this.m||(this.register[a]=1)}}return this}; |  | ||||||
| function L(a,b,c,d,e){return c&&1<a?b+(d||0)<=a?c+(e||0):(a-1)/(b+(d||0))*(c+(e||0))+1|0:0}function M(a,b,c,d,e,f,g){let h=g?a.h:a.map;if(!b[c]||g&&!b[c][g])a.s&&(h=h[d]),g?(b=b[c]||(b[c]=v()),b[g]=1,h=h[g]||(h[g]=v())):b[c]=1,h=h[c]||(h[c]=[]),a.s||(h=h[d]||(h[d]=[])),f&&h.includes(e)||(h[h.length]=e,a.m&&(a=a.register[e]||(a.register[e]=[]),a[a.length]=h))} |  | ||||||
| t.search=function(a,b,c){c||(!b&&C(a)?(c=a,a=c.query):C(b)&&(c=b));let d=[],e;let f,g=0;if(c){a=c.query||a;b=c.limit;g=c.offset||0;var h=c.context;f=c.suggest}if(a&&(a=this.encode(""+a),e=a.length,1<e)){c=v();var k=[];for(let n=0,w=0,q;n<e;n++)if((q=a[n])&&q.length>=this.B&&!c[q])if(this.s||f||this.map[q])k[w++]=q,c[q]=1;else return d;a=k;e=a.length}if(!e)return d;b||(b=100);h=this.depth&&1<e&&!1!==h;c=0;let m;h?(m=a[0],c=1):1<e&&a.sort(ba);for(let n,w;c<e;c++){w=a[c];h?(n=sa(this,d,f,b,g,2===e,w, |  | ||||||
| m),f&&!1===n&&d.length||(m=w)):n=sa(this,d,f,b,g,1===e,w);if(n)return n;if(f&&c===e-1){k=d.length;if(!k){if(h){h=0;c=-1;continue}return d}if(1===k)return ta(d[0],b,g)}}return ma(d,b,g,f)}; |  | ||||||
| function sa(a,b,c,d,e,f,g,h){let k=[],m=h?a.h:a.map;a.s||(m=ua(m,g,h,a.l));if(m){let n=0;const w=Math.min(m.length,h?a.A:a.D);for(let q=0,r=0,l,p;q<w;q++)if(l=m[q])if(a.s&&(l=ua(l,g,h,a.l)),e&&l&&f&&(p=l.length,p<=e?(e-=p,l=null):(l=l.slice(e),e=0)),l&&(k[n++]=l,f&&(r+=l.length,r>=d)))break;if(n){if(f)return ta(k,d,0);b[b.length]=k;return}}return!c&&k}function ta(a,b,c){a=1===a.length?a[0]:[].concat.apply([],a);return c||a.length>b?a.slice(c,c+b):a} |  | ||||||
| function ua(a,b,c,d){c?(d=d&&b>c,a=(a=a[d?b:c])&&a[d?c:b]):a=a[b];return a}t.contain=function(a){return!!this.register[a]};t.update=function(a,b){return this.remove(a).add(a,b)}; |  | ||||||
| t.remove=function(a,b){const c=this.register[a];if(c){if(this.m)for(let d=0,e;d<c.length;d++)e=c[d],e.splice(e.indexOf(a),1);else N(this.map,a,this.D,this.s),this.depth&&N(this.h,a,this.A,this.s);b||delete this.register[a];if(this.cache){b=this.cache;for(let d=0,e,f;d<b.h.length;d++)f=b.h[d],e=b.cache[f],e.includes(a)&&(b.h.splice(d--,1),delete b.cache[f])}}return this}; |  | ||||||
| function N(a,b,c,d,e){let f=0;if(a.constructor===Array)if(e)b=a.indexOf(b),-1!==b?1<a.length&&(a.splice(b,1),f++):f++;else{e=Math.min(a.length,c);for(let g=0,h;g<e;g++)if(h=a[g])f=N(h,b,c,d,e),d||f||delete a[g]}else for(let g in a)(f=N(a[g],b,c,d,e))||delete a[g];return f}t.searchCache=oa; |  | ||||||
| t.export=function(a,b,c,d,e){let f,g;switch(e||(e=0)){case 0:f="reg";if(this.m){g=v();for(let h in this.register)g[h]=1}else g=this.register;break;case 1:f="cfg";g={doc:0,opt:this.s?1:0};break;case 2:f="map";g=this.map;break;case 3:f="ctx";g=this.h;break;default:return}ra(a,b||this,c,f,d,e,g);return!0};t.import=function(a,b){if(b)switch(x(b)&&(b=JSON.parse(b)),a){case "cfg":this.s=!!b.opt;break;case "reg":this.m=!1;this.register=b;break;case "map":this.map=b;break;case "ctx":this.h=b}};la(K.prototype);function va(a){a=a.data;var b=self._index;const c=a.args;var d=a.task;switch(d){case "init":d=a.options||{};a=a.factory;b=d.encode;d.cache=!1;b&&0===b.indexOf("function")&&(d.encode=Function("return "+b)());a?(Function("return "+a)()(self),self._index=new self.FlexSearch.Index(d),delete self.FlexSearch):self._index=new K(d);break;default:a=a.id,b=b[d].apply(b,c),postMessage("search"===d?{id:a,msg:b}:{id:a})}};let wa=0;function O(a){if(!(this instanceof O))return new O(a);var b;a?D(b=a.encode)&&(a.encode=b.toString()):a={};(b=(self||window)._factory)&&(b=b.toString());const c="undefined"===typeof window&&self.exports,d=this;this.o=xa(b,c,a.worker);this.h=v();if(this.o){if(c)this.o.on("message",function(e){d.h[e.id](e.msg);delete d.h[e.id]});else this.o.onmessage=function(e){e=e.data;d.h[e.id](e.msg);delete d.h[e.id]};this.o.postMessage({task:"init",factory:b,options:a})}}P("add");P("append");P("search"); |  | ||||||
| P("update");P("remove");function P(a){O.prototype[a]=O.prototype[a+"Async"]=function(){const b=this,c=[].slice.call(arguments);var d=c[c.length-1];let e;D(d)&&(e=d,c.splice(c.length-1,1));d=new Promise(function(f){setTimeout(function(){b.h[++wa]=f;b.o.postMessage({task:a,id:wa,args:c})})});return e?(d.then(e),this):d}} |  | ||||||
| function xa(a,b,c){let d;try{d=b?eval('new (require("worker_threads")["Worker"])("../dist/node/node.js")'):a?new Worker(URL.createObjectURL(new Blob(["onmessage="+va.toString()],{type:"text/javascript"}))):new Worker(x(c)?c:"worker/worker.js",{type:"module"})}catch(e){}return d};function Q(a){if(!(this instanceof Q))return new Q(a);var b=a.document||a.doc||a,c;this.K=[];this.h=[];this.A=[];this.register=v();this.key=(c=b.key||b.id)&&S(c,this.A)||"id";this.m=u(a.fastupdate);this.C=(c=b.store)&&!0!==c&&[];this.store=c&&v();this.I=(c=b.tag)&&S(c,this.A);this.l=c&&v();this.cache=(c=a.cache)&&new J(c);a.cache=!1;this.o=a.worker;this.async=!1;c=v();let d=b.index||b.field||b;x(d)&&(d=[d]);for(let e=0,f,g;e<d.length;e++)f=d[e],x(f)||(g=f,f=f.field),g=C(g)?Object.assign({},a,g):a, |  | ||||||
| this.o&&(c[f]=new O(g),c[f].o||(this.o=!1)),this.o||(c[f]=new K(g,this.register)),this.K[e]=S(f,this.A),this.h[e]=f;if(this.C)for(a=b.store,x(a)&&(a=[a]),b=0;b<a.length;b++)this.C[b]=S(a[b],this.A);this.index=c}function S(a,b){const c=a.split(":");let d=0;for(let e=0;e<c.length;e++)a=c[e],0<=a.indexOf("[]")&&(a=a.substring(0,a.length-2))&&(b[d]=!0),a&&(c[d++]=a);d<c.length&&(c.length=d);return 1<d?c:c[0]}function T(a,b){if(x(b))a=a[b];else for(let c=0;a&&c<b.length;c++)a=a[b[c]];return a} |  | ||||||
| function U(a,b,c,d,e){a=a[e];if(d===c.length-1)b[e]=a;else if(a)if(a.constructor===Array)for(b=b[e]=Array(a.length),e=0;e<a.length;e++)U(a,b,c,d,e);else b=b[e]||(b[e]=v()),e=c[++d],U(a,b,c,d,e)}function V(a,b,c,d,e,f,g,h){if(a=a[g])if(d===b.length-1){if(a.constructor===Array){if(c[d]){for(b=0;b<a.length;b++)e.add(f,a[b],!0,!0);return}a=a.join(" ")}e.add(f,a,h,!0)}else if(a.constructor===Array)for(g=0;g<a.length;g++)V(a,b,c,d,e,f,g,h);else g=b[++d],V(a,b,c,d,e,f,g,h)}t=Q.prototype; |  | ||||||
| t.add=function(a,b,c){C(a)&&(b=a,a=T(b,this.key));if(b&&(a||0===a)){if(!c&&this.register[a])return this.update(a,b);for(let d=0,e,f;d<this.h.length;d++)f=this.h[d],e=this.K[d],x(e)&&(e=[e]),V(b,e,this.A,0,this.index[f],a,e[0],c);if(this.I){let d=T(b,this.I),e=v();x(d)&&(d=[d]);for(let f=0,g,h;f<d.length;f++)if(g=d[f],!e[g]&&(e[g]=1,h=this.l[g]||(this.l[g]=[]),!c||!h.includes(a)))if(h[h.length]=a,this.m){const k=this.register[a]||(this.register[a]=[]);k[k.length]=h}}if(this.store&&(!c||!this.store[a])){let d; |  | ||||||
| if(this.C){d=v();for(let e=0,f;e<this.C.length;e++)f=this.C[e],x(f)?d[f]=b[f]:U(b,d,f,0,f[0])}this.store[a]=d||b}}return this};t.append=function(a,b){return this.add(a,b,!0)};t.update=function(a,b){return this.remove(a).add(a,b)}; |  | ||||||
| t.remove=function(a){C(a)&&(a=T(a,this.key));if(this.register[a]){for(var b=0;b<this.h.length&&(this.index[this.h[b]].remove(a,!this.o),!this.m);b++);if(this.I&&!this.m)for(let c in this.l){b=this.l[c];const d=b.indexOf(a);-1!==d&&(1<b.length?b.splice(d,1):delete this.l[c])}this.store&&delete this.store[a];delete this.register[a]}return this}; |  | ||||||
| t.search=function(a,b,c,d){c||(!b&&C(a)?(c=a,a=""):C(b)&&(c=b,b=0));let e=[],f=[],g,h,k,m,n,w,q=0;if(c)if(c.constructor===Array)k=c,c=null;else{a=c.query||a;k=(g=c.pluck)||c.index||c.field;m=c.tag;h=this.store&&c.enrich;n="and"===c.bool;b=c.limit||b||100;w=c.offset||0;if(m&&(x(m)&&(m=[m]),!a)){for(let l=0,p;l<m.length;l++)if(p=ya.call(this,m[l],b,w,h))e[e.length]=p,q++;return q?e:[]}x(k)&&(k=[k])}k||(k=this.h);n=n&&(1<k.length||m&&1<m.length);const r=!d&&(this.o||this.async)&&[];for(let l=0,p,z,B;l< |  | ||||||
| k.length;l++){let A;z=k[l];x(z)||(A=z,z=A.field,a=A.query||a,b=A.limit||b);if(r)r[l]=this.index[z].searchAsync(a,b,A||c);else{d?p=d[l]:p=this.index[z].search(a,b,A||c);B=p&&p.length;if(m&&B){const y=[];let H=0;n&&(y[0]=[p]);for(let X=0,pa,R;X<m.length;X++)if(pa=m[X],B=(R=this.l[pa])&&R.length)H++,y[y.length]=n?[R]:R;H&&(p=n?ma(y,b||100,w||0):na(p,y),B=p.length)}if(B)f[q]=z,e[q++]=p;else if(n)return[]}}if(r){const l=this;return new Promise(function(p){Promise.all(r).then(function(z){p(l.search(a,b, |  | ||||||
| c,z))})})}if(!q)return[];if(g&&(!h||!this.store))return e[0];for(let l=0,p;l<f.length;l++){p=e[l];p.length&&h&&(p=za.call(this,p));if(g)return p;e[l]={field:f[l],result:p}}return e};function ya(a,b,c,d){let e=this.l[a],f=e&&e.length-c;if(f&&0<f){if(f>b||c)e=e.slice(c,c+b);d&&(e=za.call(this,e));return{tag:a,result:e}}}function za(a){const b=Array(a.length);for(let c=0,d;c<a.length;c++)d=a[c],b[c]={id:d,doc:this.store[d]};return b}t.contain=function(a){return!!this.register[a]};t.get=function(a){return this.store[a]}; |  | ||||||
| t.set=function(a,b){this.store[a]=b;return this};t.searchCache=oa;t.export=function(a,b,c,d,e){e||(e=0);d||(d=0);if(d<this.h.length){const f=this.h[d],g=this.index[f];b=this;setTimeout(function(){g.export(a,b,e?f:"",d,e++)||(d++,e=1,b.export(a,b,f,d,e))})}else{let f,g;switch(e){case 1:f="tag";g=this.l;break;case 2:f="store";g=this.store;break;default:return}ra(a,this,c,f,d,e,g)}}; |  | ||||||
| t.import=function(a,b){if(b)switch(x(b)&&(b=JSON.parse(b)),a){case "tag":this.l=b;break;case "reg":this.m=!1;this.register=b;for(let d=0,e;d<this.h.length;d++)e=this.index[this.h[d]],e.register=b,e.m=!1;break;case "store":this.store=b;break;default:a=a.split(".");const c=a[0];a=a[1];c&&a&&this.index[c].import(a,b)}};la(Q.prototype);var Ba={encode:Aa,F:!1,G:""};const Ca=[F("[\u00e0\u00e1\u00e2\u00e3\u00e4\u00e5]"),"a",F("[\u00e8\u00e9\u00ea\u00eb]"),"e",F("[\u00ec\u00ed\u00ee\u00ef]"),"i",F("[\u00f2\u00f3\u00f4\u00f5\u00f6\u0151]"),"o",F("[\u00f9\u00fa\u00fb\u00fc\u0171]"),"u",F("[\u00fd\u0177\u00ff]"),"y",F("\u00f1"),"n",F("[\u00e7c]"),"k",F("\u00df"),"s",F(" & ")," and "];function Aa(a){var b=a=""+a;b.normalize&&(b=b.normalize("NFD").replace(ea,""));return ca.call(this,b.toLowerCase(),!a.normalize&&Ca)};var Ea={encode:Da,F:!1,G:"strict"};const Fa=/[^a-z0-9]+/,Ga={b:"p",v:"f",w:"f",z:"s",x:"s","\u00df":"s",d:"t",n:"m",c:"k",g:"k",j:"k",q:"k",i:"e",y:"e",u:"o"};function Da(a){a=Aa.call(this,a).join(" ");const b=[];if(a){const c=a.split(Fa),d=c.length;for(let e=0,f,g=0;e<d;e++)if((a=c[e])&&(!this.filter||!this.filter[a])){f=a[0];let h=Ga[f]||f,k=h;for(let m=1;m<a.length;m++){f=a[m];const n=Ga[f]||f;n&&n!==k&&(h+=n,k=n)}b[g++]=h}}return b};var Ia={encode:Ha,F:!1,G:""};const Ja=[F("ae"),"a",F("oe"),"o",F("sh"),"s",F("th"),"t",F("ph"),"f",F("pf"),"f",F("(?![aeo])h(?![aeo])"),"",F("(?!^[aeo])h(?!^[aeo])"),""];function Ha(a,b){a&&(a=Da.call(this,a).join(" "),2<a.length&&(a=E(a,Ja)),b||(1<a.length&&(a=ha(a)),a&&(a=a.split(" "))));return a||[]};var La={encode:Ka,F:!1,G:""};const Ma=F("(?!\\b)[aeo]");function Ka(a){a&&(a=Ha.call(this,a,!0),1<a.length&&(a=a.replace(Ma,"")),1<a.length&&(a=ha(a)),a&&(a=a.split(" ")));return a||[]};G["latin:default"]=ja;G["latin:simple"]=Ba;G["latin:balance"]=Ea;G["latin:advanced"]=Ia;G["latin:extra"]=La;const W=self;let Y;const Z={Index:K,Document:Q,Worker:O,registerCharset:function(a,b){G[a]=b},registerLanguage:function(a,b){ka[a]=b}};(Y=W.define)&&Y.amd?Y([],function(){return Z}):W.exports?W.exports=Z:W.FlexSearch=Z;}(this)); |  | ||||||
							
								
								
									
										1
									
								
								assets/lib/katex/auto-render.min.js
									
									
									
									
										vendored
									
									
								
							
							
						
						
									
										1
									
								
								assets/lib/katex/auto-render.min.js
									
									
									
									
										vendored
									
									
								
							| @@ -1 +0,0 @@ | |||||||
| !function(e,t){"object"==typeof exports&&"object"==typeof module?module.exports=t(require("katex")):"function"==typeof define&&define.amd?define(["katex"],t):"object"==typeof exports?exports.renderMathInElement=t(require("katex")):e.renderMathInElement=t(e.katex)}("undefined"!=typeof self?self:this,(function(e){return function(){"use strict";var t={771:function(t){t.exports=e}},r={};function n(e){var i=r[e];if(void 0!==i)return i.exports;var a=r[e]={exports:{}};return t[e](a,a.exports,n),a.exports}n.n=function(e){var t=e&&e.__esModule?function(){return e.default}:function(){return e};return n.d(t,{a:t}),t},n.d=function(e,t){for(var r in t)n.o(t,r)&&!n.o(e,r)&&Object.defineProperty(e,r,{enumerable:!0,get:t[r]})},n.o=function(e,t){return Object.prototype.hasOwnProperty.call(e,t)};var i={};return function(){n.d(i,{default:function(){return s}});var e=n(771),t=n.n(e),r=function(e,t,r){for(var n=r,i=0,a=e.length;n<t.length;){var o=t[n];if(i<=0&&t.slice(n,n+a)===e)return n;"\\"===o?n++:"{"===o?i++:"}"===o&&i--,n++}return-1},a=/^\\begin{/,o=function(e,t){for(var n,i=[],o=new RegExp("("+t.map((function(e){return e.left.replace(/[-/\\^$*+?.()|[\]{}]/g,"\\$&")})).join("|")+")");-1!==(n=e.search(o));){n>0&&(i.push({type:"text",data:e.slice(0,n)}),e=e.slice(n));var l=t.findIndex((function(t){return e.startsWith(t.left)}));if(-1===(n=r(t[l].right,e,t[l].left.length)))break;var d=e.slice(0,n+t[l].right.length),s=a.test(d)?d:e.slice(t[l].left.length,n);i.push({type:"math",data:s,rawData:d,display:t[l].display}),e=e.slice(n+t[l].right.length)}return""!==e&&i.push({type:"text",data:e}),i},l=function(e,r){var n=o(e,r.delimiters);if(1===n.length&&"text"===n[0].type)return null;for(var i=document.createDocumentFragment(),a=0;a<n.length;a++)if("text"===n[a].type)i.appendChild(document.createTextNode(n[a].data));else{var l=document.createElement("span"),d=n[a].data;r.displayMode=n[a].display;try{r.preProcess&&(d=r.preProcess(d)),t().render(d,l,r)}catch(e){if(!(e instanceof t().ParseError))throw e;r.errorCallback("KaTeX auto-render: Failed to parse `"+n[a].data+"` with ",e),i.appendChild(document.createTextNode(n[a].rawData));continue}i.appendChild(l)}return i},d=function e(t,r){for(var n=0;n<t.childNodes.length;n++){var i=t.childNodes[n];if(3===i.nodeType){for(var a=i.textContent,o=i.nextSibling,d=0;o&&o.nodeType===Node.TEXT_NODE;)a+=o.textContent,o=o.nextSibling,d++;var s=l(a,r);if(s){for(var f=0;f<d;f++)i.nextSibling.remove();n+=s.childNodes.length-1,t.replaceChild(s,i)}else n+=d}else 1===i.nodeType&&function(){var t=" "+i.className+" ";-1===r.ignoredTags.indexOf(i.nodeName.toLowerCase())&&r.ignoredClasses.every((function(e){return-1===t.indexOf(" "+e+" ")}))&&e(i,r)}()}},s=function(e,t){if(!e)throw new Error("No element provided to render");var r={};for(var n in t)t.hasOwnProperty(n)&&(r[n]=t[n]);r.delimiters=r.delimiters||[{left:"$$",right:"$$",display:!0},{left:"\\(",right:"\\)",display:!1},{left:"\\begin{equation}",right:"\\end{equation}",display:!0},{left:"\\begin{align}",right:"\\end{align}",display:!0},{left:"\\begin{alignat}",right:"\\end{alignat}",display:!0},{left:"\\begin{gather}",right:"\\end{gather}",display:!0},{left:"\\begin{CD}",right:"\\end{CD}",display:!0},{left:"\\[",right:"\\]",display:!0}],r.ignoredTags=r.ignoredTags||["script","noscript","style","textarea","pre","code","option"],r.ignoredClasses=r.ignoredClasses||[],r.errorCallback=r.errorCallback||console.error,r.macros=r.macros||{},d(e,r)}}(),i=i.default}()})); |  | ||||||
										
											Binary file not shown.
										
									
								
							
										
											Binary file not shown.
										
									
								
							
										
											Binary file not shown.
										
									
								
							
										
											Binary file not shown.
										
									
								
							
										
											Binary file not shown.
										
									
								
							
										
											Binary file not shown.
										
									
								
							
										
											Binary file not shown.
										
									
								
							
										
											Binary file not shown.
										
									
								
							
										
											Binary file not shown.
										
									
								
							
										
											Binary file not shown.
										
									
								
							
										
											Binary file not shown.
										
									
								
							
										
											Binary file not shown.
										
									
								
							
										
											Binary file not shown.
										
									
								
							
										
											Binary file not shown.
										
									
								
							
										
											Binary file not shown.
										
									
								
							
										
											Binary file not shown.
										
									
								
							
										
											Binary file not shown.
										
									
								
							
										
											Binary file not shown.
										
									
								
							
										
											Binary file not shown.
										
									
								
							
										
											Binary file not shown.
										
									
								
							
										
											Binary file not shown.
										
									
								
							
										
											Binary file not shown.
										
									
								
							
										
											Binary file not shown.
										
									
								
							
										
											Binary file not shown.
										
									
								
							
										
											Binary file not shown.
										
									
								
							
										
											Binary file not shown.
										
									
								
							
										
											Binary file not shown.
										
									
								
							
										
											Binary file not shown.
										
									
								
							
										
											Binary file not shown.
										
									
								
							
										
											Binary file not shown.
										
									
								
							
										
											Binary file not shown.
										
									
								
							
										
											Binary file not shown.
										
									
								
							
										
											Binary file not shown.
										
									
								
							
										
											Binary file not shown.
										
									
								
							
										
											Binary file not shown.
										
									
								
							
										
											Binary file not shown.
										
									
								
							
										
											Binary file not shown.
										
									
								
							
										
											Binary file not shown.
										
									
								
							
										
											Binary file not shown.
										
									
								
							
										
											Binary file not shown.
										
									
								
							
										
											Binary file not shown.
										
									
								
							
										
											Binary file not shown.
										
									
								
							
										
											Binary file not shown.
										
									
								
							
										
											Binary file not shown.
										
									
								
							Some files were not shown because too many files have changed in this diff Show More
		Reference in New Issue
	
	Block a user