[
  {
    "path": ".githooks/commit-msg",
    "content": "#!/usr/bin/env bash\n\n# Commit-msg hook generated by git-sumi.\n# For more information and documentation, visit: https://sumi.rs\n\nset -e  # Exit on any error.\n\n# Get the current branch name.\ncurrent_branch=$(git rev-parse --abbrev-ref HEAD)\n\n# Check if the current branch is 'main'.\nif [ \"$current_branch\" != \"main\" ]; then\n    exit 0  # Exit successfully without running git-sumi.\nfi\n\n# Check if git-sumi is installed.\nif ! command -v git-sumi &> /dev/null\nthen\n    echo \"git-sumi is not installed. Please install it. See https://sumi.rs for instructions.\"\n    echo \"Alternatively, edit or remove the commit-msg hook in .git/hooks/commit-msg.\"\n    exit 1\nfi\n\n# Run git-sumi on the commit message if on the 'main' branch.\ngit-sumi -- \"$(cat $1)\"  # Exit with error if linting fails.\n"
  },
  {
    "path": ".githooks/pre-commit",
    "content": "#!/usr/bin/env bash\n\n#################################################################################\n# This script is run by git before a commit is made.                            #\n# To use it, copy it to .git/hooks/pre-commit and make it executable.           #\n# Alternatively, run the following command from the root of the repo:           #\n# git config core.hooksPath .githooks                                           #\n#                                                                               #\n#                                  FEATURES                                     #\n# Updates the \"updated\" field in the front matter of .md files.                 #\n# Compresses PNG files with either oxipng or optipng if available.              #\n# Runs subset_font if config.toml has been modified.                            #\n#                                                                               #\n# Stops you from commiting:                                                     #\n# - a draft .md file                                                            #\n# - a file with a \"TODO\"                                                        #\n# - a JS file without a minified version                                        #\n# - a minified JS file that isn't as small as it can be                         #\n# - a config.toml and theme.toml with different amounts of lines in [extra]     #\n#################################################################################\n\n# Function to exit the script with an error message.\nfunction error_exit() {\n    echo \"ERROR: $1\" >&2\n    exit \"${2:-1}\"\n}\n\n# Function to extract the date from the front matter.\nfunction extract_date() {\n    local file=\"$1\"\n    local field=\"$2\"\n    grep -m 1 \"^$field =\" \"$file\" | sed -e \"s/$field = //\" -e 's/ *$//'\n}\n\n# Function to check if the .md file is a draft.\nfunction is_draft() {\n    local file=\"$1\"\n    awk '/^\\+\\+\\+$/,/^\\+\\+\\+$/ { if(/draft = true/) exit 0 } END { exit 1 }' \"$file\"\n}\n\n# Check if the file contains \"TODO\".\nfunction contains_todo() {\n    local file=\"$1\"\n    grep -q \"TODO\" \"$file\"\n}\n\n# Check for changes outside of the front matter.\nfunction has_body_changes() {\n    local file=\"$1\"\n    local in_front_matter=1\n    local triple_plus_count=0\n\n    diff_output=$(git diff --unified=999 --cached --output-indicator-new='%' --output-indicator-old='&' \"$file\")\n\n    while read -r line; do\n        if [[ \"$line\" =~ ^\\+\\+\\+$ ]]; then\n            triple_plus_count=$((triple_plus_count + 1))\n            if [[ $triple_plus_count -eq 2 ]]; then\n                in_front_matter=0\n            fi\n        elif [[ $in_front_matter -eq 0 ]]; then\n            if [[ \"$line\" =~ ^[\\%\\&] ]]; then\n                return 0\n            fi\n        fi\n    done <<< \"$diff_output\"\n    return 1\n}\n\n# Function to update the social media card for a post or section.\nfunction generate_and_commit_card {\n    local file=$1\n    social_media_card=$(social-cards-zola -o static/img/social_cards -b http://127.0.0.1:1111 -u -p -i \"$file\") || {\n        echo \"Failed to update social media card for $file\"\n        exit 1\n    }\n\n    git add \"$social_media_card\" || {\n        echo \"Failed to add social media card $social_media_card\"\n        exit 1\n    }\n\n    git add \"$file\" || {\n        echo \"Failed to add $file\"\n        exit 1\n    }\n}\n\nexport -f generate_and_commit_card\n\nfunction has_minified_version() {\n    local file=\"$1\"\n    local extension=\"${file##*.}\"\n    local minified_file=\"${file%.*}.min.$extension\"\n    [ -f \"$minified_file\" ]\n}\n\nfunction is_minified() {\n    local file=\"$1\"\n\n    # Check if terser and uglifyjs are installed.\n    if ! command -v terser &> /dev/null || ! command -v uglifyjs &> /dev/null; then\n        echo \"Either terser or uglifyjs is not installed. Skipping minification check.\"\n        return 0\n    fi\n\n    # Original file size.\n    local original_size=$(wc -c < \"$file\")\n\n    # File size after compression with terser.\n    local terser_size=$(terser --compress --mangle -- \"$file\" | wc -c)\n\n    # File size after compression with uglifyjs.\n    local uglifyjs_size=$(uglifyjs --compress --mangle -- \"$file\" | wc -c)\n\n    # Check if the file is already as small as or smaller than both minified versions.\n    if (( original_size <= terser_size && original_size <= uglifyjs_size )); then\n        return 0\n    fi\n\n    # If the file isn't as small as it can be, suggest the better compressor in the error message\n    if (( terser_size < uglifyjs_size )); then\n        error_exit \"Minified JS file $file isn't as small as it can be! Try using terser for better compression.\"\n    else\n        error_exit \"Minified JS file $file isn't as small as it can be! Try using uglifyjs for better compression.\"\n    fi\n}\n\n# Check if the script is being run from the root of the repo.\nif [[ ! $(git rev-parse --show-toplevel) == $(pwd) ]]; then\n    error_exit \"This script must be run from the root of the repo.\"\nfi\n\n# Check if oxipng is installed.\npng_compressor=\"\"\nif command -v oxipng &> /dev/null; then\n    png_compressor=\"oxipng -o max\"\nelif command -v optipng &> /dev/null; then\n    png_compressor=\"optipng -o 7\"\nfi\n\n##################################################################\n# Compress PNG files with either oxipng or optipng if available. #\n# Update the \"updated\" field in the front matter of .md files.   #\n#          https://osc.garden/blog/zola-date-git-hook/           #\n# Ensure the [extra] section from config.toml and theme.toml     #\n# have the same amount of lines.                                 #\n# Ensure JavaScript files are minified.                          #\n##################################################################\n\n# Get the newly added and modified files, but not deleted files.\nmapfile -t all_changed_files < <(git diff --cached --name-only --diff-filter=d)\n\nscript_name=$(basename \"$0\")\n# Loop through all newly added or modified files.\nfor file in \"${all_changed_files[@]}\"; do\n    file_name=$(basename \"$file\")\n\n    # Ignore this script and the changelog.\n    if [[ \"$file_name\" == \"$script_name\" ]] || [[ \"$file_name\" == \"CHANGELOG.md\" ]]; then\n        continue\n    fi\n\n    # If the file is a PNG and png_compressor is set, compress it and add it to the commit.\n    if [[ \"$file\" == *.png ]] && [[ -n \"$png_compressor\" ]]; then\n        $png_compressor \"$file\" || error_exit \"Failed to compress PNG file $file\"\n        git add --force \"$file\" || error_exit \"Failed to add compressed PNG file $file\"\n        continue\n    fi\n\n    # If the file contains \"TODO\", abort the commit.\n    if contains_todo \"$file\"; then\n        error_exit \"File $file contains TODO! Remove or complete the TODO before committing.\"\n    fi\n\n    # If the file is a JS file and it doesn't have a minified version, abort the commit.\n    if [[ \"$file\" == *.js ]] && [[ \"$file\" != *.min.js ]] && ! has_minified_version \"$file\"; then\n        error_exit \"JS file $file doesn't have a minified version!\"\n    fi\n\n    # If the file is a minified JS file and it isn't as small as it can be, abort the commit.\n    # Error message shows which minifier is best for the file.\n    if [[ \"$file\" == *.min.js ]]; then\n        is_minified \"$file\"\n    fi\n\n    # Ensure the [extra] section from config.toml and theme.toml have the same amount of lines.\n    if [[ \"$file\" == \"config.toml\" ]] || [[ \"$file\" == \"theme.toml\" ]]; then\n        # Get the line number where [extra] starts in config.toml.\n        extra_line_config=$(grep -n \"^\\[extra\\]$\" \"config.toml\" | cut -d: -f1)\n\n        # Get the line number where [extra] starts in theme.toml.\n        extra_line_theme=$(grep -n \"^\\[extra\\]$\" \"theme.toml\" | cut -d: -f1)\n\n        # Get the number of lines in the [extra] section in config.toml.\n        extra_lines_config=$(tail -n +\"$extra_line_config\" \"config.toml\" | wc -l)\n\n        # Get the number of lines in the [extra] section in theme.toml.\n        extra_lines_theme=$(tail -n +\"$extra_line_theme\" \"theme.toml\" | wc -l)\n\n        # Abort the commit if the number of lines in the [extra] section don't match.\n        if [[ \"$extra_lines_config\" -ne \"$extra_lines_theme\" ]]; then\n            error_exit \"The [extra] section in config.toml and theme.toml don't have the same amount of lines!\"\n        fi\n    fi\ndone\n\n# Get the modified .md to update the \"updated\" field in the front matter.\nmodified_md_files=$(git diff --cached --name-only --diff-filter=M | grep -E '\\.md$' | grep -v '_index.md$')\n\n# Loop through each modified .md file.\nfor file in $modified_md_files; do\necho $file\n    # If the file is an .md file and it's a draft, abort the commit.\n    if is_draft \"$file\"; then\n        error_exit \"Draft file $file is being committed!\"\n    fi\n\n    # If changes are only in the front matter, skip the file.\n    if ! has_body_changes \"$file\"; then\n        continue\n    fi\n\n    # Modify the \"updated\" date, if necessary.\n    # Get the last modified date from the filesystem.\n    last_modified_date=$(date -r \"$file\" +'%Y-%m-%d')\n\n    # Extract the \"date\" field from the front matter.\n    date_value=$(extract_date \"$file\" \"date\")\n\n    # Skip the file if the last modified date is the same as the \"date\" field.\n    if [[ \"$last_modified_date\" == \"$date_value\" ]]; then\n        continue\n    fi\n\n    # Update the \"updated\" field with the last modified date.\n    # If the \"updated\" field doesn't exist, create it below the \"date\" field.\n    if ! awk -v date_line=\"$last_modified_date\" 'BEGIN{FS=OFS=\" = \"; first = 1} {\n        if (/^date =/ && first) {\n            print; getline;\n            if (!/^updated =/) print \"updated\" OFS date_line;\n            first=0;\n        }\n        if (/^updated =/ && !first) gsub(/[^ ]*$/, date_line, $2);\n        print;\n    }' \"$file\" > \"${file}.tmp\"\n    then\n        error_exit \"Failed to process $file with AWK\"\n    fi\n\n    mv \"${file}.tmp\" \"$file\" || error_exit \"Failed to overwrite $file with updated content\"\n\n    # Stage the changes.\n    git add \"$file\"\n\ndone\n\n# Use `social-cards-zola` to create/update the social media card for Markdown files.\n# See https://osc.garden/blog/automating-social-media-cards-zola/ for context.\n# Use parallel to create the social media cards in parallel and commit them.\n# echo \"$modified_md_files\" | parallel -j 8 generate_and_commit_card\n\n#########################################################\n# Run subset_font if config.toml has been modified.     #\n# https://welpo.github.io/tabi/blog/custom-font-subset/ #\n#########################################################\nif git diff --cached --name-only | grep -q \"config.toml\"; then\n    echo \"config.toml modified. Attempting to run subset_font…\"\n\n    # Check if subset_font is available and exit early if not.\n    if ! command -v subset_font &> /dev/null; then\n        echo \"subset_font command not found. Skipping this step.\"\n        exit 0\n    fi\n\n    # Call the subset_font script.\n    ~/bin/subset_font -c config.toml -f static/fonts/Inter4.woff2 -o static/\n\n    # Add the generated subset.css file to the commit.\n    git add static/custom_subset.css\nfi\n"
  },
  {
    "path": ".github/CODEOWNERS",
    "content": "* @welpo\ni18n/ar.toml @TheAwiteb\n"
  },
  {
    "path": ".github/FUNDING.yml",
    "content": "github: welpo\n"
  },
  {
    "path": ".github/ISSUE_TEMPLATE/2_bug_report.yml",
    "content": "name: \"🐛 Bug report\"\ndescription: \"Did you run into an issue while using tabi?\"\nlabels: [\"bug\"]\nbody:\n- type: textarea\n  attributes:\n      label: \"System information\"\n      description: |\n          Please provide the following information:\n          - Zola version: Run `zola --version`\n          - tabi version or commit hash\n      placeholder: |\n          Zola version: \n          tabi version or commit: \n  validations:\n      required: true\n\n- type: textarea\n  attributes:\n      label: \"Expected behaviour\"\n      description: \"Tell us what should have happened.\"\n      placeholder: \"Describe what you expected tabi to do…\"\n  validations:\n      required: true\n\n- type: textarea\n  attributes:\n      label: \"Actual behaviour\"\n      description: \"Tell us what happens instead of the expected behavior.\"\n      placeholder: \"Describe what actually happened…\"\n  validations:\n      required: true\n\n- type: textarea\n  attributes:\n      label: \"Steps to reproduce\"\n      description: \"Please provide detailed steps to reproduce the issue.\"\n      placeholder: |\n          1. Set up `config.toml` with these settings: …\n          2. Create content with this structure: …\n          3. Run Zola with these arguments: …\n          4. See the following error: …\n  validations:\n      required: true\n\n- type: textarea\n  attributes:\n      label: \"Additional context\"\n      description: >\n          Please provide any relevant configuration files, error messages, or screenshots that might help us understand the issue.\n          You can drag and drop files here to attach them.\n  validations:\n      required: false\n\n- type: checkboxes\n  attributes:\n      label: \"Final checklist\"\n      options:\n          - label: \"I've checked that the issue isn't already reported.\"\n            required: true\n          - label: \"I've tested with the latest version of tabi to check if the issue has already been fixed.\"\n            required: true\n"
  },
  {
    "path": ".github/ISSUE_TEMPLATE/3_feature_request.yml",
    "content": "name: \"✨ Feature request\"\ndescription: \"Suggest an idea for tabi\"\nlabels: [\"enhancement\"]\nbody:\n- type: textarea\n  attributes:\n      label: \"Summary and motivation\"\n      description: \"Briefly describe the feature and why it would be valuable for tabi users.\"\n      placeholder: |\n          Describe:\n          - What the feature is\n          - Why it would be useful\n          - Any examples from other projects\n  validations:\n      required: true\n\n- type: textarea\n  attributes:\n      label: \"Implementation details\"\n      description: \"Share any ideas you have about how this could be implemented.\"\n      placeholder: \"Technical suggestions, potential approaches, or specific requirements.\"\n  validations:\n      required: false\n\n- type: checkboxes\n  attributes:\n      label: \"Checklist\"\n      options:\n          - label: \"I've searched existing issues to make sure this feature hasn't already been requested.\"\n            required: true\n          - label: \"This feature aligns with tabi's philosophy (minimal JS, accessible…)\"\n            required: true\n          - label: \"I'm willing to contribute to the implementation of this feature.\"\n            required: false\n"
  },
  {
    "path": ".github/ISSUE_TEMPLATE/config.yml",
    "content": "blank_issues_enabled: false\ncontact_links:\n  - name: 💬 tabi discussions\n    url: https://github.com/welpo/tabi/discussions\n    about: If you have questions or need help, feel free to ask here~\n"
  },
  {
    "path": ".github/PULL_REQUEST_TEMPLATE.md",
    "content": "<!--\n  Thank you for contributing to tabi!\n\n  This template is designed to guide you through the pull request process.\n  Please fill out the sections below as applicable.\n\n  Don't worry if your PR is not complete or you're unsure about something;\n  feel free to submit it and ask for feedback. We appreciate all contributions, big or small!\n\n  Feel free to remove any section or checklist item that does not apply to your changes.\n  If it's a quick fix (for example, fixing a typo), a Summary is enough.\n-->\n\n## Summary\n\n<!-- Please provide a brief description of the changes made in this PR -->\n\n### Related issue\n\n<!-- Mention any relevant issues like #123 -->\n\n## Changes\n\n<!-- Please provide some more detail regarding the changes.\nAdd any additional information, configuration, or data that might be necessary for the review -->\n\n### Accessibility\n\n<!-- Have you taken steps to make your changes accessible? This could include:\n- Semantic HTML\n- ARIA attributes\n- Keyboard navigation\n- Voiceover or screen reader compatibility\n- Colour contrast\nPlease elaborate on the actions taken.\nIf you need help, please ask! -->\n\n### Screenshots\n\n<!-- If applicable, add screenshots to help explain the changes made. Consider if a before/after is helpful -->\n\n### Type of change\n\n<!-- Mark the relevant option with an `x` like so: `[x]` (no spaces) -->\n\n- [ ] Bug fix (fixes an issue without altering functionality)\n- [ ] New feature (adds non-breaking functionality)\n- [ ] Breaking change (alters existing functionality)\n- [ ] UI/UX improvement (enhances user interface without altering functionality)\n- [ ] Refactor (improves code quality without altering functionality)\n- [ ] Documentation update\n- [ ] Other (please describe below)\n\n---\n\n## Checklist\n\n- [ ] I have verified the accessibility of my changes\n- [ ] I have tested all possible scenarios for this change\n- [ ] I have updated `theme.toml` with a sane default for the feature\n- [ ] I have updated `config.toml` in [tabi-start](https://github.com/welpo/tabi-start)\n- [ ] I have made corresponding changes to the documentation:\n  - [ ] Updated `config.toml` comments\n  - [ ] Updated `theme.toml` comments\n  - [ ] Updated \"Mastering tabi\" post in English\n  - [ ] (Optional) Updated \"Mastering tabi\" post in Spanish\n  - [ ] (Optional) Updated \"Mastering tabi\" post in Catalan\n"
  },
  {
    "path": ".github/renovate.json",
    "content": "{\n  \"$schema\": \"https://docs.renovatebot.com/renovate-schema.json\",\n  \"extends\": [\"config:recommended\", \":automergeMinor\", \":disableDependencyDashboard\"],\n  \"commitMessagePrefix\": \"⬆️\",\n  \"commitMessageAction\": \"chore(deps): update\",\n  \"commitMessageTopic\": \"{{{depName}}}\",\n  \"labels\": [\"dependencies\"],\n  \"git-submodules\": {\n    \"enabled\": true\n  },\n  \"packageRules\": [\n    {\n      \"updateTypes\": [\"pin\"],\n      \"commitMessagePrefix\": \"📌\"\n    },\n    {\n      \"updateTypes\": [\"major\", \"minor\", \"patch\", \"digest\", \"bump\"],\n      \"commitMessagePrefix\": \"⬆️\"\n    },\n    {\n      \"updateTypes\": [\"rollback\"],\n      \"commitMessagePrefix\": \"⬇️\"\n    },\n    {\n      \"matchFileNames\": [\"scripts/release/**\"],\n      \"automerge\": true\n    }\n  ]\n}\n"
  },
  {
    "path": ".github/workflows/cd.yml",
    "content": "name: Continuous Deployment\n\non:\n  push:\n    tags:\n      - 'v*.*.*'\n\njobs:\n  deploy:\n    name: Deploy and release\n    runs-on: ubuntu-24.04\n    permissions:\n      contents: write\n    steps:\n      - name: Checkout\n        uses: actions/checkout@v6\n        with:\n          fetch-depth: 0\n\n      - name: Generate the changelog\n        uses: orhun/git-cliff-action@main\n        id: git-cliff\n        with:\n          config: cliff.toml\n          args: --latest --strip all\n        env:\n          GITHUB_TOKEN: ${{ secrets.GITHUB_TOKEN }}\n          OUTPUT: CHANGES.md\n\n      - name: Create GitHub release\n        run: |\n          gh release create ${{ github.ref_name }} \\\n            --title \"Release ${{ github.ref_name }}\" \\\n            --notes-file CHANGES.md\n        env:\n          GH_TOKEN: ${{ secrets.GITHUB_TOKEN }}\n"
  },
  {
    "path": ".github/workflows/ci.yml",
    "content": "name: Build Site\n\non:\n  push:\n    branches:\n      - main\n  pull_request:\n    branches:\n      - main\n\njobs:\n  check_and_build_pr:\n    name: Check and Build for Pull Requests\n    runs-on: ubuntu-latest\n    if: github.event_name == 'pull_request'\n    steps:\n      - name: Checkout Code\n        uses: actions/checkout@v6\n\n      - name: Zola Build\n        uses: shalzz/zola-deploy-action@v0.22.0\n        env:\n          BUILD_ONLY: true\n\n      - name: Zola Check\n        uses: shalzz/zola-deploy-action@v0.22.0\n        env:\n          BUILD_ONLY: true\n          CHECK_LINKS: true\n\n  build_and_deploy:\n    name: Build and Deploy on Main Push\n    runs-on: ubuntu-24.04\n    if: github.event_name == 'push' && github.ref == 'refs/heads/main'\n    steps:\n      - name: Checkout Code\n        uses: actions/checkout@v6\n\n      - name: Build and Deploy\n        uses: shalzz/zola-deploy-action@v0.22.0\n        env:\n          PAGES_BRANCH: gh-pages\n          TOKEN: ${{ secrets.TOKEN }}\n          BUILD_THEMES: false\n"
  },
  {
    "path": ".github/workflows/git-sumi.yml",
    "content": "name: Lint pull request title\n\non:\n  pull_request:\n    types:\n      - opened\n      - edited\n      - synchronize\n      - ready_for_review\n\nconcurrency:\n  group: ${{ github.workflow }}-${{ github.event.pull_request.number || github.ref }}\n  cancel-in-progress: true\n\npermissions:\n  pull-requests: read\n\njobs:\n  main:\n    name: Run git-sumi\n    runs-on: ubuntu-24.04\n    steps:\n      - name: Set header length for dependency updates\n        if: contains(github.event.pull_request.title, 'chore(deps)')\n        run: echo \"GIT_SUMI_MAX_HEADER_LENGTH=80\" >> $GITHUB_ENV\n      - uses: welpo/git-sumi-action@main\n        env:\n          GITHUB_TOKEN: ${{ secrets.GITHUB_TOKEN }}\n"
  },
  {
    "path": ".github/workflows/upgrade-deps.yml",
    "content": "name: Dependency upgrade\non:\n  workflow_dispatch:\n    inputs:\n      dependency:\n        description: 'Dependency to upgrade'\n        required: true\n        type: choice\n        options:\n          - all\n          - mermaid\n          - katex\n  schedule:\n    - cron: '32 4 * * *'\n\njobs:\n  upgrade-dependency:\n    name: Upgrade dependency\n    runs-on: ubuntu-24.04\n    permissions:\n      contents: write\n      pull-requests: write\n    strategy:\n      matrix:\n        dependency: ${{ github.event_name == 'schedule' && fromJson('[\"mermaid\", \"katex\"]') || fromJson(format('[\"{0}\"]', github.event.inputs.dependency)) }}\n    steps:\n      - name: Checkout repository\n        uses: actions/checkout@v6\n        with:\n          fetch-depth: 0\n\n      - name: Set up environment\n        run: |\n          sudo apt-get update\n          sudo apt-get install -y jq npm curl git\n          npm install uglify-js -g\n          uglifyjs --version\n\n      - name: Configure GPG key\n        run: |\n          echo -n ${{ secrets.GPG_PRIVATE_KEY }} | base64 --decode | gpg --import\n\n      - name: Configure Git\n        run: |\n          git config --global user.signingkey 33EACFE956484C3940BFEEDCE4EC28F8DFB57474\n          git config --global commit.gpgsign true\n          git config --global user.name \"welpo\"\n          git config --global user.email \"welpo@users.noreply.github.com\"\n\n      - name: Check for existing branch\n        id: check_branch\n        run: |\n          if git ls-remote --heads origin deps/upgrade-${{ matrix.dependency }} | grep -q deps/upgrade-${{ matrix.dependency }}; then\n            echo \"branch_exists=true\" >> $GITHUB_OUTPUT\n          else\n            echo \"branch_exists=false\" >> $GITHUB_OUTPUT\n          fi\n\n      - name: Handle existing branch\n        if: steps.check_branch.outputs.branch_exists == 'true'\n        run: |\n          echo \"Branch deps/upgrade-${{ matrix.dependency }} already exists.\"\n          echo \"Skipping upgrade as there's already an open PR\"\n          exit 0\n\n      - name: Create and switch to new branch\n        run: |\n          git checkout -b deps/upgrade-${{ matrix.dependency }}\n\n      - name: Run upgrade script\n        shell: bash\n        run: |\n          if [[ \"${{ matrix.dependency }}\" == \"all\" ]]; then\n            bash scripts/upgrade-deps --all\n          else\n            bash scripts/upgrade-deps --${{ matrix.dependency }}\n          fi\n\n      - name: Push changes and create PR\n        shell: bash\n        run: |\n          if git diff --quiet HEAD origin/main; then\n            echo \"No changes to push for ${{ matrix.dependency }}\"\n            exit 0\n          fi\n          git push -u origin deps/upgrade-${{ matrix.dependency }}\n          gh pr create --fill --base main --head deps/upgrade-${{ matrix.dependency }} --label \"dependencies\"\n        env:\n          GH_TOKEN: ${{ secrets.GITHUB_TOKEN }}\n"
  },
  {
    "path": ".gitignore",
    "content": "public"
  },
  {
    "path": ".gitmodules",
    "content": "[submodule \"scripts/release\"]\n\tpath = scripts/release\n\turl = https://github.com/welpo/release.git\n"
  },
  {
    "path": ".prettierignore",
    "content": "*.min.*\n"
  },
  {
    "path": ".prettierrc.toml",
    "content": "semi = true\ntrailingComma = \"es5\"\nsingleQuote = true\nprintWidth = 88\ntabWidth = 4\nuseTabs = false\narrowParens = \"always\"\nbracketSpacing = true\njsxBracketSameLine = false\njsxSingleQuote = true\nendOfLine = \"lf\"\n"
  },
  {
    "path": "CHANGELOG.md",
    "content": "# Changelog\n\nWelcome to the changelog for tabi. This document aims to provide a comprehensive list of all notable changes made to the project, organised chronologically by release version.\n\nWe use Semantic Versioning (SemVer) for our version numbers, formatted as MAJOR.MINOR.PATCH. Major version changes involve significant (breaking) changes, minor versions introduce features and improvements in a backward compatible manner, and patch versions are for bug fixes and minor tweaks.\n\n## [4.1.0](https://github.com/welpo/tabi/compare/v4.0.0..v4.1.0) - 2025-06-14\n\n### ✨ Features\n\n- *(feed)* Add feed icon tag pages ([#522](https://github.com/welpo/tabi/issues/522)) by [@tzinm](https://github.com/tzinm) and [@welpo](https://github.com/welpo)\n- *(i18n)* Add Finnish language ([#505](https://github.com/welpo/tabi/issues/505)) by [@axelkar](https://github.com/axelkar) and [@welpo](https://github.com/welpo)\n- *(indieweb)* Add hidden h-card ([#506](https://github.com/welpo/tabi/issues/506)) by [@mmai](https://github.com/mmai) and [@welpo](https://github.com/welpo)\n- *(socials)* Add itch.io icon ([#520](https://github.com/welpo/tabi/issues/520)) by [@mystal](https://github.com/mystal)\n- *(webmentions)* Add hcard in post page ([#524](https://github.com/welpo/tabi/issues/524)) by [@mmai](https://github.com/mmai) and [@welpo](https://github.com/welpo)\n- Add support for webmentions ([#485](https://github.com/welpo/tabi/issues/485)) by [@gortavoher](https://github.com/gortavoher), [@Jeremiah](https://github.com/Jeremiah) Russell, [@Henri](https://github.com/Henri) Bourcereau and [@welpo](https://github.com/welpo)\n- Extend tabi `<head>` and `<body>` elements ([#528](https://github.com/welpo/tabi/issues/528)) by [@axis7818](https://github.com/axis7818) and [@welpo](https://github.com/welpo)\n- Parse markdown in post summary & description ([#517](https://github.com/welpo/tabi/issues/517)) by [@rktjump](https://github.com/rktjump) and [@welpo](https://github.com/welpo)\n- Cache bust images in shortcodes ([#504](https://github.com/welpo/tabi/issues/504)) by [@axelkar](https://github.com/axelkar)\n\n### 🐛 Bug fixes\n\n- *(feed)* Prioritise description > summary ([#525](https://github.com/welpo/tabi/issues/525)) by [@tzinm](https://github.com/tzinm)\n- Render author HTML on page metadata ([7321fc5](https://github.com/welpo/tabi/commit/7321fc5a43fa50dc24c9fa87714612a2295b904a)) by [@welpo](https://github.com/welpo)\n- Allow feed icon to be hidden ([#533](https://github.com/welpo/tabi/issues/533)) by [@bajacc](https://github.com/bajacc)\n- Fix hcard html tags ([#508](https://github.com/welpo/tabi/issues/508)) by [@mmai](https://github.com/mmai)\n- Fix external link icon breaking word wrapping ([#498](https://github.com/welpo/tabi/issues/498)) by [@welpo](https://github.com/welpo)\n- Fix padding hover on links with code (custom font) ([#494](https://github.com/welpo/tabi/issues/494)) by [@clement-escolano](https://github.com/clement-escolano)\n\n### 💄 Styling\n\n- Remove extra spacing after author ([2e29782](https://github.com/welpo/tabi/commit/2e29782279d8154cb9c6f1df9a2401612ec77948)) by [@welpo](https://github.com/welpo)\n\n### ♻️ Refactor\n\n- Add \"target_attribute\" macro ([#515](https://github.com/welpo/tabi/issues/515)) by [@tzinm](https://github.com/tzinm) and @Óscar\n- Improve error message when title is unset ([a384ac8](https://github.com/welpo/tabi/commit/a384ac80c8e6b8be46d4a335930fba697a523af9)) by [@welpo](https://github.com/welpo)\n\n### 🔧 Miscellaneous tasks\n\n- *(CI)* Fix Mermaid version detection ([e11186a](https://github.com/welpo/tabi/commit/e11186a8448f94494b350402f76a2cc5e5a98a97)) by [@welpo](https://github.com/welpo)\n- *(README)* Add b1n.io to 'sites using tabi' ([#531](https://github.com/welpo/tabi/issues/531)) by [@b1nhack](https://github.com/b1nhack)\n- *(README)* Fix broken link tag ([#516](https://github.com/welpo/tabi/issues/516)) by [@Olexandr88](https://github.com/Olexandr88)\n- *(README)* Add tzinm's blog to 'sites using tabi' ([#496](https://github.com/welpo/tabi/issues/496)) by [@tzinm](https://github.com/tzinm)\n\n### 👥 New contributors\n\n🫶 [@gortavoher](https://github.com/gortavoher) made their first contribution in [#485](https://github.com/welpo/tabi/pull/485)\n\n🫶 [@bajacc](https://github.com/bajacc) made their first contribution in [#533](https://github.com/welpo/tabi/pull/533)\n\n🫶 [@axis7818](https://github.com/axis7818) made their first contribution in [#528](https://github.com/welpo/tabi/pull/528)\n\n🫶 [@b1nhack](https://github.com/b1nhack) made their first contribution in [#531](https://github.com/welpo/tabi/pull/531)\n\n🫶 [@tzinm](https://github.com/tzinm) made their first contribution in [#525](https://github.com/welpo/tabi/pull/525)\n\n🫶 [@mmai](https://github.com/mmai) made their first contribution in [#524](https://github.com/welpo/tabi/pull/524)\n\n🫶 [@rktjump](https://github.com/rktjump) made their first contribution in [#517](https://github.com/welpo/tabi/pull/517)\n\n🫶 [@mystal](https://github.com/mystal) made their first contribution in [#520](https://github.com/welpo/tabi/pull/520)\n\n🫶 [@axelkar](https://github.com/axelkar) made their first contribution in [#504](https://github.com/welpo/tabi/pull/504)\n\n🫶 [@clement-escolano](https://github.com/clement-escolano) made their first contribution in [#494](https://github.com/welpo/tabi/pull/494)\n\n## [4.0.0](https://github.com/welpo/tabi/compare/v3.1.0..v4.0.0) - 2025-02-16\n\n### 💥 BREAKING CHANGES 💥\n\n- *(archive)* Enforce chronological sorting ([#483](https://github.com/welpo/tabi/issues/483)) by [@welpo](https://github.com/welpo)\n- Add tag filtering for projects ([#431](https://github.com/welpo/tabi/issues/431)) ([a783329](https://github.com/welpo/tabi/commit/a7833299fff4c753e35fa1f00667d8801f541f9a)) by [@welpo](https://github.com/welpo)\n\n### ✨ Features\n\n- *(archive)* [**‼️BREAKING‼️**] Enforce chronological sorting ([#483](https://github.com/welpo/tabi/issues/483)) by [@welpo](https://github.com/welpo)\n- *(shortcodes)* Add 'aside' shortcode for side notes ([#452](https://github.com/welpo/tabi/issues/452)) by [@welpo](https://github.com/welpo)\n- *(socials)* Add more Fediverse social icons ([89ad44c](https://github.com/welpo/tabi/commit/89ad44ce26d002f2f2c3baa6cf769fb8972551af)) by [@welpo](https://github.com/welpo)\n- *(socials)* Add Letterboxd social icon ([0e0a391](https://github.com/welpo/tabi/commit/0e0a391ea84c3ed8b66931158ddef1e4cdd161f4)) by [@welpo](https://github.com/welpo)\n- *(socials)* Add ORCID icon ([cf63855](https://github.com/welpo/tabi/commit/cf6385553c779710d1164509f3fc0d480e858ebd)) by [@welpo](https://github.com/welpo)\n- Add `hide_from_main_feed` support ([#490](https://github.com/welpo/tabi/issues/490)) by [@welpo](https://github.com/welpo)\n- Support Zola's native code block names ([#489](https://github.com/welpo/tabi/issues/489)) by [@welpo](https://github.com/welpo)\n- Render markdown in page titles & descriptions ([#486](https://github.com/welpo/tabi/issues/486)) by [@stalkerGH](https://github.com/stalkerGH) and [@welpo](https://github.com/welpo)\n- Add indicator to external links ([#443](https://github.com/welpo/tabi/issues/443)) by [@welpo](https://github.com/welpo)\n- Allow setting post_listing_date per section ([a3ae897](https://github.com/welpo/tabi/commit/a3ae897c8310ba6a1da72db0fc81f19ca8993d6f)) by [@welpo](https://github.com/welpo)\n- Render section content ([#484](https://github.com/welpo/tabi/issues/484)) by [@welpo](https://github.com/welpo)\n- Support disabling header font-subset ([#458](https://github.com/welpo/tabi/issues/458)) by [@Nizzlay](https://github.com/Nizzlay) and [@welpo](https://github.com/welpo)\n- Support using info-page.html for sections ([#455](https://github.com/welpo/tabi/issues/455)) by [@wischi-chr](https://github.com/wischi-chr) and [@welpo](https://github.com/welpo)\n- Add `raw_path` parameter to image shortcodes ([#439](https://github.com/welpo/tabi/issues/439)) by [@welpo](https://github.com/welpo)\n- [**‼️BREAKING‼️**] Add tag filtering for projects ([#431](https://github.com/welpo/tabi/issues/431)) ([a783329](https://github.com/welpo/tabi/commit/a7833299fff4c753e35fa1f00667d8801f541f9a)) by [@welpo](https://github.com/welpo)\n\n### 🐛 Bug fixes\n\n- *(CSP)* Improve CSP coverage ([#471](https://github.com/welpo/tabi/issues/471)) by [@stekershaw](https://github.com/stekershaw)\n- *(analytics)* Update umami domain ([#448](https://github.com/welpo/tabi/issues/448)) by [@arichtman](https://github.com/arichtman)\n- *(feed)* Styling now works in all website paths ([45fe170](https://github.com/welpo/tabi/commit/45fe17046704e249be3b5fbb94a3355d21149d4d)) by [@welpo](https://github.com/welpo)\n- *(feed)* Load CSS in subfolder setups ([#467](https://github.com/welpo/tabi/issues/467)) by [@welpo](https://github.com/welpo)\n- Adapt footnotes styling for zola 0.19.x & 0.20.x ([1c067e7](https://github.com/welpo/tabi/commit/1c067e708eac4282a2191507f046eacd9f18e0f6)) by [@welpo](https://github.com/welpo)\n- Improve menu accessibility ([#482](https://github.com/welpo/tabi/issues/482)) by [@welpo](https://github.com/welpo)\n- Support external URLs in menu ([#481](https://github.com/welpo/tabi/issues/481)) by [@welpo](https://github.com/welpo)\n- Remove `nofollow` robots meta tag ([#465](https://github.com/welpo/tabi/issues/465)) by [@Nizzlay](https://github.com/Nizzlay)\n- Show feed icon without socials ([de6fa58](https://github.com/welpo/tabi/commit/de6fa58fe3b276c12f16f9ddc1cf9f93269161f9)) by [@welpo](https://github.com/welpo)\n- Add separator between update date & remote changes link ([2cc336d](https://github.com/welpo/tabi/commit/2cc336d9d0f7019015a356242230f0568658a523)) by [@welpo](https://github.com/welpo)\n- Support relative paths in inherited social media card ([#432](https://github.com/welpo/tabi/issues/432)) by [@welpo](https://github.com/welpo)\n\n### 💄 Styling\n\n- *(admonitions)* Fix external link icon ([2b106b7](https://github.com/welpo/tabi/commit/2b106b7c628e01046495a3add7a9ec4cc00b9fa3)) by [@welpo](https://github.com/welpo)\n- Increase emoji favicon size ([d777d5d](https://github.com/welpo/tabi/commit/d777d5d99679ab609bebb89726aef80562b30d87)) by [@welpo](https://github.com/welpo)\n- Remove extra whitespace in post metadata ([dbb0e82](https://github.com/welpo/tabi/commit/dbb0e826b6572531d4c44bdb031d68207a906666)) by [@welpo](https://github.com/welpo)\n- Use text-colour for hover on `rt` in links ([a6b6c32](https://github.com/welpo/tabi/commit/a6b6c3205096fd718757a6f32ff07d60bcc22808)) by [@welpo](https://github.com/welpo)\n- Fix home banner layout for short intro ([#438](https://github.com/welpo/tabi/issues/438)) by [@welpo](https://github.com/welpo)\n- Fix numbered code blocks on iOS Safari ([4be8a56](https://github.com/welpo/tabi/commit/4be8a5634fa2990825e9bc785029c4dddb340c39)) by [@welpo](https://github.com/welpo)\n- Fix numbered code blocks ([#437](https://github.com/welpo/tabi/issues/437)) by [@welpo](https://github.com/welpo)\n- Retain base code block style in admonitions ([557ea77](https://github.com/welpo/tabi/commit/557ea7786ffbad75fafdd69d864d6073fd1dadbc)) by [@welpo](https://github.com/welpo)\n\n### 📝 Documentation\n\n- *(README)* Fix typo ([#457](https://github.com/welpo/tabi/issues/457)) by [@Olexandr88](https://github.com/Olexandr88)\n- *(README)* Add maintenance badges ([4834a9e](https://github.com/welpo/tabi/commit/4834a9e85ff89f1678374752289116bce6f3a076)) by [@welpo](https://github.com/welpo)\n- *(demo)* Basic Arabic translation ([#390](https://github.com/welpo/tabi/issues/390)) by [@TheAwiteb](https://github.com/TheAwiteb) and [@welpo](https://github.com/welpo)\n- *(series)* Add tip for custom text permalinks ([4029657](https://github.com/welpo/tabi/commit/402965786beb942aab0922de68cd98952f9a9be0)) by [@welpo](https://github.com/welpo)\n- *(shortcodes)* Clarify aside usage ([1dcd615](https://github.com/welpo/tabi/commit/1dcd615615b492dffc00c34bd7e1bbc680cf95fd)) by [@welpo](https://github.com/welpo)\n- *(shortcodes)* Clarify `position` parameter usage for \"aside\" ([da1b6bb](https://github.com/welpo/tabi/commit/da1b6bb0cc7719f3de23a6855d80dc0262d5c025)) by [@welpo](https://github.com/welpo)\n- Suggest MathML in lieu of KaTeX for JS-free math ([eb03953](https://github.com/welpo/tabi/commit/eb0395352caffab67f1ad9b81aab51824e9feb54)) by [@welpo](https://github.com/welpo)\n- Add tabi-start integration ([#441](https://github.com/welpo/tabi/issues/441)) by [@welpo](https://github.com/welpo)\n\n### ♻️ Refactor\n\n- Only show project tag filter with 2 or more tags ([180fc53](https://github.com/welpo/tabi/commit/180fc533844ec3427c447f7040ea837520c4fe5d)) by [@welpo](https://github.com/welpo)\n\n### 🔧 Miscellaneous tasks\n\n- *(CI)* Add concurrency control to sumi action ([47ea26d](https://github.com/welpo/tabi/commit/47ea26db97cfbd87518ab00b892fdbb245577535)) by [@welpo](https://github.com/welpo)\n- *(CI)* Add dependencies label to upgrade PRs ([551b3d2](https://github.com/welpo/tabi/commit/551b3d28c6852c3a1e7acf3c645e0d7b84e079cb)) by [@welpo](https://github.com/welpo)\n- *(CI)* Ignore deleted files in pre-commit ([f93def0](https://github.com/welpo/tabi/commit/f93def0692146a15b63d19406ec1180b9e41ba3d)) by [@welpo](https://github.com/welpo)\n- *(README)* Add idle-ti.me blog to sites using tabi ([#479](https://github.com/welpo/tabi/issues/479)) by [@be-next](https://github.com/be-next)\n- *(README)* Add zzmzaw's blog to sites using tabi ([#469](https://github.com/welpo/tabi/issues/469)) by [@ZzMzaw](https://github.com/ZzMzaw)\n- *(README)* Add nizzlay.com to Sites using tabi ([#464](https://github.com/welpo/tabi/issues/464)) by [@Nizzlay](https://github.com/Nizzlay)\n- *(README)* Add szabolcs.me to Sites using tabi ([#451](https://github.com/welpo/tabi/issues/451)) by [@szabolcsf](https://github.com/szabolcsf)\n- *(README)* Set original colours for docs|up badge ([69713c9](https://github.com/welpo/tabi/commit/69713c973c11d7646c8fe56476261d96cc1978d8)) by [@welpo](https://github.com/welpo)\n- *(deps)* Check for existing branch upgrades ([29bbd2e](https://github.com/welpo/tabi/commit/29bbd2eecc97aaaa95c61f8fb736d1301042aa91)) by [@welpo](https://github.com/welpo)\n- *(docs)* Fix Series section headings ([0778212](https://github.com/welpo/tabi/commit/077821208f161e813da90c56db817b743bc211c4)) by [@welpo](https://github.com/welpo)\n- *(docs)* Use internal links in home description ([0b10560](https://github.com/welpo/tabi/commit/0b10560a94f4a0a4bcb8536ee0595e8c4f4df800)) by [@welpo](https://github.com/welpo)\n- *(projects)* Add nemui project ([11eb774](https://github.com/welpo/tabi/commit/11eb774bd74562a4628f87a130575281aa469918)) by [@welpo](https://github.com/welpo)\n- *(upgrade-deps)* Improve version detection ([0688628](https://github.com/welpo/tabi/commit/06886284202f670be2034dcedbf6130446b91436)) by [@welpo](https://github.com/welpo)\n- Add \"shuku\" project ([e1b6054](https://github.com/welpo/tabi/commit/e1b60542dfc3eb9bed39a92d3b7acfae88f32b84)) by [@welpo](https://github.com/welpo)\n- Add ずつ (zutsu) project page ([c6d1c3d](https://github.com/welpo/tabi/commit/c6d1c3dff844461c7c774882dec8a87d7d3ea548)) by [@welpo](https://github.com/welpo)\n- Fix dōteki docs link ([c15098a](https://github.com/welpo/tabi/commit/c15098a959242765353c7fab0906d1fc596c64c0)) by [@welpo](https://github.com/welpo)\n- Update git-sumi & dōteki projects ([6d44320](https://github.com/welpo/tabi/commit/6d44320f45bc381c65e370f8f03bf3525b59e05a)) by [@welpo](https://github.com/welpo)\n\n### 👥 New contributors\n\n🫶 [@stalkerGH](https://github.com/stalkerGH) made their first contribution in [#486](https://github.com/welpo/tabi/pull/486)\n\n🫶 [@stekershaw](https://github.com/stekershaw) made their first contribution in [#471](https://github.com/welpo/tabi/pull/471)\n\n🫶 [@Nizzlay](https://github.com/Nizzlay) made their first contribution in [#465](https://github.com/welpo/tabi/pull/465)\n\n🫶 [@Olexandr88](https://github.com/Olexandr88) made their first contribution in [#457](https://github.com/welpo/tabi/pull/457)\n\n🫶 [@wischi-chr](https://github.com/wischi-chr) made their first contribution in [#455](https://github.com/welpo/tabi/pull/455)\n\n🫶 [@szabolcsf](https://github.com/szabolcsf) made their first contribution in [#451](https://github.com/welpo/tabi/pull/451)\n\n## [3.1.0](https://github.com/welpo/tabi/compare/v3.0.0..v3.1.0) - 2024-11-14\n\n### ✨ Features\n\n- *(shortcodes)* Support body admonitions ([#423](https://github.com/welpo/tabi/issues/423)) by [@welpo](https://github.com/welpo)\n- Allow pinned posts with pagination ([#428](https://github.com/welpo/tabi/issues/428)) by [@welpo](https://github.com/welpo)\n- Add pinned posts functionality ([#424](https://github.com/welpo/tabi/issues/424)) by [@welpo](https://github.com/welpo)\n- Add series functionality ([#406](https://github.com/welpo/tabi/issues/406)) by [@ZzMzaw](https://github.com/ZzMzaw) and [@welpo](https://github.com/welpo)\n\n### 🐛 Bug fixes\n\n- Link to \"All posts\" in non-default-language root ([3b7fd3d](https://github.com/welpo/tabi/commit/3b7fd3db9c33fd3ca6114ca00fffa2a91f682a0a)) by [@welpo](https://github.com/welpo)\n- Isso comments in multilingual setups ([#427](https://github.com/welpo/tabi/issues/427)) by [@welpo](https://github.com/welpo)\n\n### 💄 Styling\n\n- Colour scrollbar to match primary colour ([#430](https://github.com/welpo/tabi/issues/430)) by [@welpo](https://github.com/welpo)\n- Set accent color to match theme ([#429](https://github.com/welpo/tabi/issues/429)) by [@welpo](https://github.com/welpo)\n- Fix info admonition code color ([5927409](https://github.com/welpo/tabi/commit/5927409c41e71d6943deee426fc75a94037cee36)) by [@welpo](https://github.com/welpo)\n- Improve navigation bar & metadata wrapping ([#425](https://github.com/welpo/tabi/issues/425)) by [@welpo](https://github.com/welpo)\n- Fix hover color for <rt> element in links ([87f1099](https://github.com/welpo/tabi/commit/87f1099caa8741bcacadf2aae2677a529395e454)) by [@welpo](https://github.com/welpo)\n- Add styling for <kbd> keyboard input ([51fee5d](https://github.com/welpo/tabi/commit/51fee5d660232a1eafeb39f439fe5377f3f63406)) by [@welpo](https://github.com/welpo)\n\n### 📝 Documentation\n\n- *(README)* Fix typo ([#421](https://github.com/welpo/tabi/issues/421)) by [@ZzMzaw](https://github.com/ZzMzaw)\n- Highlight pinned posts feature ([c3c344a](https://github.com/welpo/tabi/commit/c3c344a76fb894ec87fc9896e7a0202788e001c5)) by [@welpo](https://github.com/welpo)\n- Clarify instructions for listing recent posts ([#418](https://github.com/welpo/tabi/issues/418)) ([3442fd9](https://github.com/welpo/tabi/commit/3442fd9ff0c0a093f08bb27aaa0d92e5993df29b)) by [@welpo](https://github.com/welpo)\n\n### ♻️ Refactor\n\n- *(GitHub)* Update issue templates ([5687f3b](https://github.com/welpo/tabi/commit/5687f3bacb6ed69ffbe4143e93b6c95ce8fb5c59)) by [@welpo](https://github.com/welpo)\n\n### 🔧 Miscellaneous tasks\n\n- *(CI)* Allow longer PR titles for dep updates ([e2c5c6e](https://github.com/welpo/tabi/commit/e2c5c6e9b749c3591af6a822eee65c1ee1854c01)) by [@welpo](https://github.com/welpo)\n- *(CI)* Allow longer PR titles for dep updates ([fa9f160](https://github.com/welpo/tabi/commit/fa9f16034241ccee6229ef0279b3b9bc65249cfa)) by [@welpo](https://github.com/welpo)\n- *(CI)* Allow longer PR titles for dep updates ([bd45a1a](https://github.com/welpo/tabi/commit/bd45a1ae45997b950c703d1ebf1e34b3b82173eb)) by [@welpo](https://github.com/welpo)\n- *(deps)* Remove local release script ([2e3cff2](https://github.com/welpo/tabi/commit/2e3cff2ef1ca8396c632de48b21174601d5ff46c)) by [@welpo](https://github.com/welpo)\n- *(projects)* Add ラム (ramu) project ([f300129](https://github.com/welpo/tabi/commit/f3001298c24d49d234247c9cce2a6c771778c1c3)) by [@welpo](https://github.com/welpo)\n- Change codeblock language from plaintext to txt ([57a0a8e](https://github.com/welpo/tabi/commit/57a0a8e1a09d586acdf5c72464d79328e0c9b27f)) by [@welpo](https://github.com/welpo)\n\n## [3.0.0](https://github.com/welpo/tabi/compare/v2.17.0..v3.0.0) - 2024-10-18\n\n### 💥 BREAKING CHANGES 💥\n\n- Force code blocks LTR rendering ([#412](https://github.com/welpo/tabi/issues/412)) ([092ccdd](https://github.com/welpo/tabi/commit/092ccdd1ba8bf08cc738664ba4fbe9e7a0ff282e)) by [@welpo](https://github.com/welpo)\n\n### ✨ Features\n\n- Add force_text_direction shortcode ([#414](https://github.com/welpo/tabi/issues/414)) ([c9f8d27](https://github.com/welpo/tabi/commit/c9f8d27b962217ff0fb09b30ec14aee65fe8518d)) by [@welpo](https://github.com/welpo)\n- [**‼️BREAKING‼️**] Force code blocks LTR rendering ([#412](https://github.com/welpo/tabi/issues/412)) ([092ccdd](https://github.com/welpo/tabi/commit/092ccdd1ba8bf08cc738664ba4fbe9e7a0ff282e)) by [@welpo](https://github.com/welpo)\n\n### 🐛 Bug fixes\n\n- Update CSP for GoatCounter ([#413](https://github.com/welpo/tabi/issues/413)) by [@noeddl](https://github.com/noeddl)\n\n### 📝 Documentation\n\n- *(CONTRIBUTING)* Fix broken link to commit message style ([11e246a](https://github.com/welpo/tabi/commit/11e246a876eef846e8c1e26add7331f25beb9c27)) by [@welpo](https://github.com/welpo)\n- *(mastering tabi)* Mention social links rel='me' ([5db70b7](https://github.com/welpo/tabi/commit/5db70b7978e7141be9396d3f69375a87b2d44ed2)) by [@welpo](https://github.com/welpo)\n\n### 🔧 Miscellaneous tasks\n\n- Update git-cliff variables to `commit.remote` ([0a36ef5](https://github.com/welpo/tabi/commit/0a36ef55ee092fe7bb1f8803c04a4cb8245b2a13)) by [@welpo](https://github.com/welpo)\n\n### 👥 New contributors\n\n🫶 [@noeddl](https://github.com/noeddl) made their first contribution in [#413](https://github.com/welpo/tabi/pull/413)\n\n## [2.17.0](https://github.com/welpo/tabi/compare/v2.16.0..v2.17.0) - 2024-10-10\n\n### ✨ Features\n\n- Add fediverse creator metadata support ([#409](https://github.com/welpo/tabi/issues/409)) by [@arichtman](https://github.com/arichtman), [@Ariel](https://github.com/Ariel) Richtman and [@welpo](https://github.com/welpo)\n\n### 🐛 Bug fixes\n\n- *(search)* Restore highlighting functionality ([#401](https://github.com/welpo/tabi/issues/401)) by [@welpo](https://github.com/welpo)\n- Set proper URL for self-hosted Umami ([#402](https://github.com/welpo/tabi/issues/402)) by [@soumendrak](https://github.com/soumendrak)\n\n### 📝 Documentation\n\n- Add mermaid shortcode usage ([#407](https://github.com/welpo/tabi/issues/407)) by [@ZzMzaw](https://github.com/ZzMzaw) and [@welpo](https://github.com/welpo)\n- Update comment in config.toml RE: #402 ([c50edbd](https://github.com/welpo/tabi/commit/c50edbd453eca2f031bc97a5a25caa206644d568)) by [@welpo](https://github.com/welpo)\n\n### ♻️ Refactor\n\n- *(search)* Reduce search lag on mobile ([7ceada9](https://github.com/welpo/tabi/commit/7ceada974b7c3f2c52aaa5688af631c806901aed)) by [@welpo](https://github.com/welpo)\n\n### 🔧 Miscellaneous tasks\n\n- *(README)* Add jmbhughes.com to showcase ([#404](https://github.com/welpo/tabi/issues/404)) by [@jmbhughes](https://github.com/jmbhughes)\n- *(deps)* Replace local release script w/ git submodule ([66239be](https://github.com/welpo/tabi/commit/66239bee016406c87a227b685eb70a464546e199)) by [@welpo](https://github.com/welpo)\n- *(deps)* Automate KaTeX/mermaid upgrades ([fc04ab4](https://github.com/welpo/tabi/commit/fc04ab4e40977bafa1b821c77e38554d287f69bc)) by [@welpo](https://github.com/welpo)\n- Reformat social card creation tip ([47fcee8](https://github.com/welpo/tabi/commit/47fcee8f81fb10bcd7cc042cc4a80dcada244b6d)) by [@welpo](https://github.com/welpo)\n\n### 👥 New contributors\n\n🫶 [@github-actions](https://github.com/github-actions)[bot] made their first contribution in [#405](https://github.com/welpo/tabi/pull/405)\n\n🫶 [@jmbhughes](https://github.com/jmbhughes) made their first contribution in [#404](https://github.com/welpo/tabi/pull/404)\n\n🫶 [@soumendrak](https://github.com/soumendrak) made their first contribution in [#402](https://github.com/welpo/tabi/pull/402)\n\n## [2.16.0](https://github.com/welpo/tabi/compare/v2.15.0..v2.16.0) - 2024-09-22\n\n### ✨ Features\n\n- *(remote_text shortcode)* Support line ranges ([#399](https://github.com/welpo/tabi/issues/399)) ([008b976](https://github.com/welpo/tabi/commit/008b976e06e0b93e021e2ca641eda42521e58f98)) by [@welpo](https://github.com/welpo)\n- *(remote_text shortcode)* Support relative paths ([#398](https://github.com/welpo/tabi/issues/398)) by [@welpo](https://github.com/welpo)\n\n## [2.15.0](https://github.com/welpo/tabi/compare/v2.14.0..v2.15.0) - 2024-09-20\n\n### ✨ Features\n\n- *(feed)* Make \"Visit website\" link context-aware ([#394](https://github.com/welpo/tabi/issues/394)) by [@welpo](https://github.com/welpo)\n- *(search)* Hide \"clear search\" icon if input is empty ([#388](https://github.com/welpo/tabi/issues/388)) by [@welpo](https://github.com/welpo)\n\n### 🐛 Bug fixes\n\n- *(feed)* Resolve Atom feed validation issues ([#393](https://github.com/welpo/tabi/issues/393)) by [@welpo](https://github.com/welpo)\n- Allow pages within pages ([#385](https://github.com/welpo/tabi/issues/385)) by [@welpo](https://github.com/welpo)\n- Improve dark mode and OS theme handling ([#380](https://github.com/welpo/tabi/issues/380)) by [@welpo](https://github.com/welpo)\n\n### 💄 Styling\n\n- Improve RTL styling consistency ([#381](https://github.com/welpo/tabi/issues/381)) by [@welpo](https://github.com/welpo)\n\n### 🔧 Miscellaneous tasks\n\n- *(README)* Add Ponderosa Games to showcase ([#395](https://github.com/welpo/tabi/issues/395)) by [@JVimes](https://github.com/JVimes)\n- *(deps)* Avoid masking return values ([1b11f4b](https://github.com/welpo/tabi/commit/1b11f4b321d0ae52179d88cc3ecd1b72ef2b37ae)) by [@welpo](https://github.com/welpo)\n- *(deps)* Check local deps version to early exit ([b5cbad4](https://github.com/welpo/tabi/commit/b5cbad422bc14f44a620bcbb87449f2425c16af6)) by [@welpo](https://github.com/welpo)\n- Set [@TheAwiteb](https://github.com/TheAwiteb) as owner of Arabic translation ([edb0873](https://github.com/welpo/tabi/commit/edb087392f2989b4f08122c81fa3e51d17fbb6b8)) by [@welpo](https://github.com/welpo)\n\n### 👥 New contributors\n\n🫶 [@JVimes](https://github.com/JVimes) made their first contribution in [#395](https://github.com/welpo/tabi/pull/395)\n\n## [2.14.0](https://github.com/welpo/tabi/compare/v2.13.0..v2.14.0) - 2024-09-08\n\n### ✨ Features\n\n- *(i18n)* Add Odia language ([#372](https://github.com/welpo/tabi/issues/372))\n- *(i18n)* Add Estonian language ([#365](https://github.com/welpo/tabi/issues/365)) by [@NippleOfAnApe](https://github.com/NippleOfAnApe)\n- Add Mermaid diagram support ([#370](https://github.com/welpo/tabi/issues/370)) by [@welpo](https://github.com/welpo)\n\n### 🐛 Bug fixes\n\n- *(RTL)* Fix blockquote style ([#368](https://github.com/welpo/tabi/issues/368)) by [@TheAwiteb](https://github.com/TheAwiteb) and [@welpo](https://github.com/welpo)\n- Reduce main page header/list sizes ([#375](https://github.com/welpo/tabi/issues/375)) by [@welpo](https://github.com/welpo)\n- Add missing quotes to HTML attributes ([#367](https://github.com/welpo/tabi/issues/367)) by [@DataTriny](https://github.com/DataTriny)\n\n### 💄 Styling\n\n- Use note admonition for TL;DR ([b9bf4b2](https://github.com/welpo/tabi/commit/b9bf4b2fd67c71647c0d2a1bd77dee8cb9e0ec49)) by [@welpo](https://github.com/welpo)\n\n### 📝 Documentation\n\n- *(mastering tabi)* Clarify prev/next links requirements ([6219b92](https://github.com/welpo/tabi/commit/6219b9200cfd544b999b064fabfa0ca37d4815d8)) by [@welpo](https://github.com/welpo)\n\n### ♻️ Refactor\n\n- *(footer)* Lazy load social icons ([9e7b845](https://github.com/welpo/tabi/commit/9e7b845e544758792831da520379e04089909b78)) by [@welpo](https://github.com/welpo)\n\n### 🔧 Miscellaneous tasks\n\n- *(deps)* Add KaTeX to dependency upgrade script ([001ec8f](https://github.com/welpo/tabi/commit/001ec8fc9a9524efbe4c3c04669fb64be4ed8efd)) by [@welpo](https://github.com/welpo)\n- *(deps)* Add script to upgrade mermaid ([d73c4bd](https://github.com/welpo/tabi/commit/d73c4bde5a12d434a595be877f4452fa0c20ae30)) by [@welpo](https://github.com/welpo)\n- *(release)* Ensure script is ran from default branch ([82f7a98](https://github.com/welpo/tabi/commit/82f7a984d54046a4f884461b3c5f52ac830661bf)) by [@welpo](https://github.com/welpo)\n- Add funding information ([8d22a42](https://github.com/welpo/tabi/commit/8d22a42e3fd5d909eaf90f7132e818b9c45f7b07)) by [@welpo](https://github.com/welpo)\n\n### 👥 New contributors\n\n🫶 @ made their first contribution in [#372](https://github.com/welpo/tabi/pull/372)\n\n🫶 [@DataTriny](https://github.com/DataTriny) made their first contribution in [#367](https://github.com/welpo/tabi/pull/367)\n\n🫶 [@NippleOfAnApe](https://github.com/NippleOfAnApe) made their first contribution in [#365](https://github.com/welpo/tabi/pull/365)\n\n## [2.13.0](https://github.com/welpo/tabi/compare/v2.12.0..v2.13.0) - 2024-07-24\n\n### ✨ Features\n\n- Follow user OS theme with JS disabled ([#357](https://github.com/welpo/tabi/issues/357)) by [@welpo](https://github.com/welpo)\n- Add date visibility options to post list ([#330](https://github.com/welpo/tabi/issues/330)) by [@sam9032](https://github.com/sam9032) and [@welpo](https://github.com/welpo)\n\n### 🐛 Bug fixes\n\n- *(header)* Render feed links based on config ([#358](https://github.com/welpo/tabi/issues/358)) by [@welpo](https://github.com/welpo)\n- *(post_listing_date)* Show original date for unedited articles ([#355](https://github.com/welpo/tabi/issues/355)) by [@sam9032](https://github.com/sam9032)\n- *(search)* Close modal when clicking current page result ([#350](https://github.com/welpo/tabi/issues/350)) by [@welpo](https://github.com/welpo)\n\n### 💄 Styling\n\n- *(error)* Enhance visibility of Zola errors ([#359](https://github.com/welpo/tabi/issues/359)) by [@welpo](https://github.com/welpo)\n- Match footnote backlinks size on Zola v0.19 ([0f18ac2](https://github.com/welpo/tabi/commit/0f18ac264df8b9766c4d431dfd37f5f5799f3834)) by [@welpo](https://github.com/welpo)\n\n### 📝 Documentation\n\n- *(mastering tabi)* Fix broken image ([1111631](https://github.com/welpo/tabi/commit/1111631325e3ebf8ae3a2f6cd18492cfc5aa44c2)) by [@welpo](https://github.com/welpo)\n- Mention custom Twitter card & automatic OG tags ([fbb81f9](https://github.com/welpo/tabi/commit/fbb81f976433d93ee3787f4a9d5c6c3049f815a2)) by [@welpo](https://github.com/welpo)\n- Deprecate JS footnote backlinks ([ced0cb4](https://github.com/welpo/tabi/commit/ced0cb43b6b4b4f6fba583990e428ddedbf52af3)) by [@welpo](https://github.com/welpo)\n- Fix broken internal links ([4145364](https://github.com/welpo/tabi/commit/4145364e085929ae9c09ccb908a0bae77a0b7520)) by [@welpo](https://github.com/welpo)\n\n### 🔧 Miscellaneous tasks\n\n- *(release)* Add error handling function ([9585843](https://github.com/welpo/tabi/commit/9585843b14131843775e41df67fe9cc60c95a2ea)) by [@welpo](https://github.com/welpo)\n- *(release)* Ensure local repository is in good state ([617a940](https://github.com/welpo/tabi/commit/617a940cf823917bc86df5f05350236c40560dc0)) by [@welpo](https://github.com/welpo)\n\n### 👥 New contributors\n\n🫶 [@sam9032](https://github.com/sam9032) made their first contribution in [#355](https://github.com/welpo/tabi/pull/355)\n\n🫶 [@renovate](https://github.com/renovate)[bot] made their first contribution in [#345](https://github.com/welpo/tabi/pull/345)\n\n## [2.12.0](https://github.com/welpo/tabi/compare/v2.11.0..v2.12.0) - 2024-06-29\n\n### ✨ Features\n\n- *(admonitions)* Allow swapping admonition icons ([#340](https://github.com/welpo/tabi/issues/340)) by [@welpo](https://github.com/welpo)\n- *(socials)* Add Debian icon ([82ecbc1](https://github.com/welpo/tabi/commit/82ecbc145489bf4415d7a3500be85191524d1b53)) by [@welpo](https://github.com/welpo)\n- Allow sorting tags based on post count ([#344](https://github.com/welpo/tabi/issues/344)) by [@metaleap](https://github.com/metaleap) and [@welpo](https://github.com/welpo)\n- Add option to use sans-serif font everywhere ([a907f56](https://github.com/welpo/tabi/commit/a907f56bd568378b099c0706fa4c92d7102b1688)) by [@welpo](https://github.com/welpo)\n\n### 📝 Documentation\n\n- *(mastering tabi)* Clarify post sorting in Archive ([c9c8ed8](https://github.com/welpo/tabi/commit/c9c8ed84a00136469a53d30b6e694b6cb845c5fa)) by [@welpo](https://github.com/welpo)\n\n### ♻️ Refactor\n\n- Improve \"Last updated on\" format ([#342](https://github.com/welpo/tabi/issues/342)) by [@welpo](https://github.com/welpo)\n\n### 👥 New contributors\n\n🫶 [@metaleap](https://github.com/metaleap) made their first contribution in [#344](https://github.com/welpo/tabi/pull/344)\n\n## [2.11.0](https://github.com/welpo/tabi/compare/v2.10.0..v2.11.0) - 2024-06-24\n\n### ✨ Features\n\n- *(socials)* Add bluesky icon ([#333](https://github.com/welpo/tabi/issues/333)) by [@andreacfromtheapp](https://github.com/andreacfromtheapp)\n- Allow HTML tags in multilingual quote author ([be7628a](https://github.com/welpo/tabi/commit/be7628aeaa573b69739a2751e93d77da0b587124)) by [@welpo](https://github.com/welpo)\n\n### 🐛 Bug fixes\n\n- *(footer)* Show feed icon on Zola 0.19.0 ([#336](https://github.com/welpo/tabi/issues/336)) by [@welpo](https://github.com/welpo)\n\n### 💄 Styling\n\n- *(admonitions)* Reduce thickness of strong tag ([faa9633](https://github.com/welpo/tabi/commit/faa963353170f42ab2b05b2a35aae1431ff5c319)) by [@welpo](https://github.com/welpo)\n\n### 📝 Documentation\n\n- *(README)* Fix command to update tabi submodule ([7a3380c](https://github.com/welpo/tabi/commit/7a3380c467d3bf77cfa1d9548a0b063964d73f37)) by [@welpo](https://github.com/welpo)\n- *(README)* Clarify section_path format ([7d38628](https://github.com/welpo/tabi/commit/7d386283ff808576a7149a96223be3564984d783)) by [@welpo](https://github.com/welpo)\n- *(README)* Add commands to update the theme ([4a47e0a](https://github.com/welpo/tabi/commit/4a47e0ad2375a9b160ab0424719f04702653970d)) by [@welpo](https://github.com/welpo)\n- Explain how to use custom social icons ([5ae629d](https://github.com/welpo/tabi/commit/5ae629d171d12ed262d4862da65e7e8c8a21d87a)) by [@welpo](https://github.com/welpo)\n\n### 🔧 Miscellaneous tasks\n\n- *(pre-commit hook)* Handle filenames with spaces ([448250f](https://github.com/welpo/tabi/commit/448250ff571d442d4608e0d7a8550c66cbe1bc2b)) by [@welpo](https://github.com/welpo)\n\n### 👥 New contributors\n\n🫶 [@andreacfromtheapp](https://github.com/andreacfromtheapp) made their first contribution in [#333](https://github.com/welpo/tabi/pull/333)\n\n## [2.10.0](https://github.com/welpo/tabi/compare/v2.9.0..v2.10.0) - 2024-05-30\n\n### ✨ Features\n\n- *(homepage)* Allow hiding posts listing ([#317](https://github.com/welpo/tabi/issues/317)) by [@welpo](https://github.com/welpo)\n- *(shortcodes)* Add inline option to image shortcodes ([#315](https://github.com/welpo/tabi/issues/315)) by [@welpo](https://github.com/welpo)\n- Add option to show author in post metadata ([#322](https://github.com/welpo/tabi/issues/322)) by [@welpo](https://github.com/welpo)\n- Add inline image CSS class ([6a4f424](https://github.com/welpo/tabi/commit/6a4f4244e0ae16127f207680061fa4dcb1c782dd)) by [@welpo](https://github.com/welpo)\n\n### 🐛 Bug fixes\n\n- *(footer)* Allow showing only email/feed icons ([aef49bb](https://github.com/welpo/tabi/commit/aef49bb2251c95e1fb70f05f149ecc8af7184145)) by [@welpo](https://github.com/welpo)\n- Allow quotation in page summary ([22925ca](https://github.com/welpo/tabi/commit/22925caf060d122630c02437c5fc73bc6b82e457)) by [@welpo](https://github.com/welpo)\n\n### 💄 Styling\n\n- Increase font size of h1 ([94b3b1d](https://github.com/welpo/tabi/commit/94b3b1d8045753187044174a4d127fdc94608923)) by [@welpo](https://github.com/welpo)\n\n### 📝 Documentation\n\n- *(mastering tabi)* Fix typo ([349663f](https://github.com/welpo/tabi/commit/349663f28fc61bb19beec8ba8d42d5724c062bee)) by [@welpo](https://github.com/welpo)\n- CSP requirements to use built-in syntax highlighting ([3b0d7e5](https://github.com/welpo/tabi/commit/3b0d7e5a668aa087af512f0ecb204acf61723a63)) by [@welpo](https://github.com/welpo)\n\n### ♻️ Refactor\n\n- Improve settings hierarchy performance ([#325](https://github.com/welpo/tabi/issues/325)) by [@faassen](https://github.com/faassen)\n\n### 🔧 Miscellaneous tasks\n\n- *(analytics)* Stop collecting Language data ([1894d55](https://github.com/welpo/tabi/commit/1894d550bcbd8b664c3865576aa65abd0b1db0c3)) by [@welpo](https://github.com/welpo)\n- Update nani logo ([#323](https://github.com/welpo/tabi/issues/323)) by [@faassen](https://github.com/faassen)\n\n### 👥 New contributors\n\n🫶 [@faassen](https://github.com/faassen) made their first contribution in [#325](https://github.com/welpo/tabi/pull/325)\n\n## [2.9.0](https://github.com/welpo/tabi/compare/v2.8.1..v2.9.0) - 2024-05-11\n\n### ✨ Features\n\n- *(shortcodes)* Add remote_text shortcode ([#305](https://github.com/welpo/tabi/issues/305)) by [@welpo](https://github.com/welpo)\n- Enable adding URL/path to code blocks ([#307](https://github.com/welpo/tabi/issues/307)) ([b70efd1](https://github.com/welpo/tabi/commit/b70efd1642db11b060665ecf6d1bf0f11e3ffd06)) by [@welpo](https://github.com/welpo)\n- Add option to display/hide date in metadata ([#306](https://github.com/welpo/tabi/issues/306)) by [@welpo](https://github.com/welpo)\n\n### 🐛 Bug fixes\n\n- *(add_src_to_code_block)* Avoid overlapping text ([f081a92](https://github.com/welpo/tabi/commit/f081a92f5939e141b6dbd7afe8ad1a9519d47f30)) by [@welpo](https://github.com/welpo)\n- *(analytics)* Allow Umami through CSP ([#310](https://github.com/welpo/tabi/issues/310)) by [@arichtman](https://github.com/arichtman)\n\n## [2.8.1](https://github.com/welpo/tabi/compare/v2.8.0..v2.8.1) - 2024-05-01\n\n### 🐛 Bug fixes\n\n- Ensure spacing when all projects are shown ([50123c0](https://github.com/welpo/tabi/commit/50123c09ef7ee544d812a58a607d8a1e92369c4f)) by [@welpo](https://github.com/welpo)\n\n## [2.8.0](https://github.com/welpo/tabi/compare/v2.7.0..v2.8.0) - 2024-04-29\n\n### ✨ Features\n\n- Allow showcasing featured projects on homepage ([#297](https://github.com/welpo/tabi/issues/297)) by [@welpo](https://github.com/welpo)\n\n### 🐛 Bug fixes\n\n- *(CSP)* Add Umami API URLs to CSP ([#301](https://github.com/welpo/tabi/issues/301)) by [@arichtman](https://github.com/arichtman)\n- Align projects and posts properly ([69094d6](https://github.com/welpo/tabi/commit/69094d633579a1ea4bee1fcadd77e1511d4020ca)) by [@welpo](https://github.com/welpo)\n\n### 📝 Documentation\n\n- *(README)* Add arichtman blog to showcase ([#298](https://github.com/welpo/tabi/issues/298)) by [@arichtman](https://github.com/arichtman)\n\n### ♻️ Refactor\n\n- *(search)* Close unclosed div ([b6b48cc](https://github.com/welpo/tabi/commit/b6b48cc3b4a2ad17c2ee331664b0b2c9f1fcecc2)) by [@welpo](https://github.com/welpo)\n\n### 🔧 Miscellaneous tasks\n\n- *(README)* Remove sites no longer using tabi ([ba5acca](https://github.com/welpo/tabi/commit/ba5acca7340a940092a9be196302b01441617720)) by [@welpo](https://github.com/welpo)\n\n## [2.7.0](https://github.com/welpo/tabi/compare/v2.6.1..v2.7.0) - 2024-04-18\n\n### ✨ Features\n\n- *(i18n)* Add Dutch language ([#290](https://github.com/welpo/tabi/issues/290)) by [@Myrdincx](https://github.com/Myrdincx) and [@welpo](https://github.com/welpo)\n\n### 🐛 Bug fixes\n\n- Link 'All posts' to proper section ([#295](https://github.com/welpo/tabi/issues/295)) by [@Dalker](https://github.com/Dalker), [@Daniel](https://github.com/Daniel) Kessler (a.k.a. Dalker) and [@welpo](https://github.com/welpo)\n- Fix history url for GitLab & Codeberg ([#294](https://github.com/welpo/tabi/issues/294)) by [@TheAwiteb](https://github.com/TheAwiteb)\n\n### 💄 Styling\n\n- *(shortcodes)* Use colourful admonition icons ([e965694](https://github.com/welpo/tabi/commit/e965694cdf1c588d40e167ec98961a854e2236d0)) by [@welpo](https://github.com/welpo)\n\n### 🔧 Miscellaneous tasks\n\n- *(projects)* List Streaming Royalties Calculator ([ad289f1](https://github.com/welpo/tabi/commit/ad289f1d2caaf7781a011d610e46bd7de41e7f33)) by [@welpo](https://github.com/welpo)\n- Update nani logo ([ce1a98e](https://github.com/welpo/tabi/commit/ce1a98ed93ad38214f6455ba262289a3969cbf52)) by [@welpo](https://github.com/welpo)\n- Compress dōteki & git-sumi social media cards ([ec6ba01](https://github.com/welpo/tabi/commit/ec6ba01bbcbeecc684e2bfb49c5e1f7fac847aa5)) by [@welpo](https://github.com/welpo)\n\n### 👥 New contributors\n\n🫶 [@Dalker](https://github.com/Dalker) made their first contribution in [#295](https://github.com/welpo/tabi/pull/295)\n\n🫶 [@Myrdincx](https://github.com/Myrdincx) made their first contribution in [#290](https://github.com/welpo/tabi/pull/290)\n\n## [2.6.1](https://github.com/welpo/tabi/compare/v2.6.0..v2.6.1) - 2024-03-21\n\n### 🔧 Miscellaneous tasks\n\n- *(CI)* Use CHANGES.md for GitHub release notes ([38f3aaf](https://github.com/welpo/tabi/commit/38f3aafdd8a34ae5452ae9bb879881b807e5a7b4)) by [@welpo](https://github.com/welpo)\n\n## [2.6.0](https://github.com/welpo/tabi/compare/v2.5.0..v2.6.0) - 2024-03-21\n\n### ✨ Features\n\n- *(shortcodes)* Add admonition shortcode ([#285](https://github.com/welpo/tabi/issues/285)) ([d016229](https://github.com/welpo/tabi/commit/d0162291d961734c3e5f115348655f5755b8aa54)) by [@welpo](https://github.com/welpo)\n\n### 🐛 Bug fixes\n\n- *(feed)* Fix \"Visit website\" link with skin config ([#286](https://github.com/welpo/tabi/issues/286)) by [@be-next](https://github.com/be-next)\n\n## [2.5.0](https://github.com/welpo/tabi/compare/v2.4.0..v2.5.0) - 2024-03-17\n\n### ✨ Features\n\n- Allow disabling CSP, following the hierarchy ([5e0cbdd](https://github.com/welpo/tabi/commit/5e0cbdd67dbe4c17d635040959133edcc5066aff)) by [@welpo](https://github.com/welpo)\n\n### 🐛 Bug fixes\n\n- *(i18n)* Improve FR translations and correct typos ([#284](https://github.com/welpo/tabi/issues/284)) by [@be-next](https://github.com/be-next)\n\n### ♻️ Refactor\n\n- Set visibility hidden in \"hidden\" class ([8537bbc](https://github.com/welpo/tabi/commit/8537bbc10ef7a142128f8d471ee19814cb0d904c)) by [@welpo](https://github.com/welpo)\n- Correct invalid attributes in search input ([d825bef](https://github.com/welpo/tabi/commit/d825bef73065c7343ec952f938f7d4a17b4e363e)) by [@welpo](https://github.com/welpo)\n\n### 🔧 Miscellaneous tasks\n\n- *(release)* Replace both pull and issue links ([e951e95](https://github.com/welpo/tabi/commit/e951e951102249707a4c899f0d194b781c3466d2)) by [@welpo](https://github.com/welpo)\n\n### 👥 New contributors\n\n🫶 [@be-next](https://github.com/be-next) made their first contribution in [#284](https://github.com/welpo/tabi/pull/284)\n\n## [2.4.0](https://github.com/welpo/tabi/compare/v2.3.0..v2.4.0) - 2024-03-02\n\n### ✨ Features\n\n- *(i18n)* Add Persian language ([#283](https://github.com/welpo/tabi/issues/283)) by [@mehr32](https://github.com/mehr32) and [@welpo](https://github.com/welpo)\n- *(socials)* Add Signal icon ([4aaa234](https://github.com/welpo/tabi/commit/4aaa2346d0c10ef0322b0033358812520e2137af)) by [@welpo](https://github.com/welpo)\n\n### 🐛 Bug fixes\n\n- Allow translated drafts ([#282](https://github.com/welpo/tabi/issues/282)) by [@Smtbook](https://github.com/Smtbook) and [@welpo](https://github.com/welpo)\n\n### 🔧 Miscellaneous tasks\n\n- *(CI)* Update git-cliff's github integration ([2ede035](https://github.com/welpo/tabi/commit/2ede03550aa1ce2d2c17063fff3d5a676c3ea2be)) by [@welpo](https://github.com/welpo)\n- *(release)* Fix cross-compatibility of sed command ([0192b3e](https://github.com/welpo/tabi/commit/0192b3ed9f6df26f28320fd771f445bb74750812)) by [@welpo](https://github.com/welpo)\n- *(release)* Add author link only on full changelog ([8de049a](https://github.com/welpo/tabi/commit/8de049a2b588b7e3b1d09ed361bb28ee2ea65af4)) by [@welpo](https://github.com/welpo)\n\n### 👥 New contributors\n\n🫶 [@mehr32](https://github.com/mehr32) made their first contribution in [#283](https://github.com/welpo/tabi/pull/283)\n\n## [2.3.0](https://github.com/welpo/tabi/compare/v2.2.0..v2.3.0) - 2024-02-19\n\n### ✨ Features\n\n- *(i18n)* Implement pluralization logic ([#277](https://github.com/welpo/tabi/issues/277)) by [@TheAwiteb](https://github.com/TheAwiteb) and [@welpo](https://github.com/welpo)\n- *(i18n)* Add Arabic language ([#265](https://github.com/welpo/tabi/issues/265)) by [@TheAwiteb](https://github.com/TheAwiteb)\n- *(shortcodes)* Support URLs for image source ([#280](https://github.com/welpo/tabi/issues/280)) by [@welpo](https://github.com/welpo)\n\n### 🐛 Bug fixes\n\n- *(404)* Fix broken language selection from 404 page ([90def02](https://github.com/welpo/tabi/commit/90def02dd7d54e62cbfe8f4e78b6e1e198b631be)) by [@welpo](https://github.com/welpo)\n- *(article-navigation)* Adjust spacing for RTL ([#275](https://github.com/welpo/tabi/issues/275)) by [@welpo](https://github.com/welpo)\n- *(i18n)* Reverse arrows in RTL languages ([#272](https://github.com/welpo/tabi/issues/272)) by [@TheAwiteb](https://github.com/TheAwiteb) and [@welpo](https://github.com/welpo)\n- *(i18n)* Fix header anchor for RTL ([#270](https://github.com/welpo/tabi/issues/270)) by [@TheAwiteb](https://github.com/TheAwiteb) and [@welpo](https://github.com/welpo)\n- *(i18n)* Atom feed page direction for RTL langs ([#268](https://github.com/welpo/tabi/issues/268)) by [@TheAwiteb](https://github.com/TheAwiteb) and [@welpo](https://github.com/welpo)\n\n### 💄 Styling\n\n- *(projects)* Set entire card as link target ([22d4d68](https://github.com/welpo/tabi/commit/22d4d68a690239dfe664e8513eaaf15b525e45ec)) by [@welpo](https://github.com/welpo)\n- *(projects)* Minor redesign ([#278](https://github.com/welpo/tabi/issues/278)) by [@welpo](https://github.com/welpo)\n\n### 📝 Documentation\n\n- *(i18n)* Add Arabic to list of supported languages ([e5442fa](https://github.com/welpo/tabi/commit/e5442fa29f32f9c1294d980df692b1ce096ce3fc)) by [@welpo](https://github.com/welpo)\n- *(mastering-tabi)* Fix unclosed codeblock ([274c414](https://github.com/welpo/tabi/commit/274c4146b2dd04c6ca188c096002a02d4b073273)) by [@welpo](https://github.com/welpo)\n- Fix various broken links ([6efe2ee](https://github.com/welpo/tabi/commit/6efe2ee57db4202fb0898b12e1f5e952e97ddc3a)) by [@welpo](https://github.com/welpo)\n- Update tabi screenshots ([238487b](https://github.com/welpo/tabi/commit/238487b4b2d7e04c8176d35d69e4925dc4b98d9c)) by [@welpo](https://github.com/welpo)\n\n### 🔧 Miscellaneous tasks\n\n- *(CI)* Build before checking links ([daef403](https://github.com/welpo/tabi/commit/daef40338ccf03250da2ec2b4382d644f277e4b4)) by [@welpo](https://github.com/welpo)\n- *(CI)* Improve CI to check and build for PRs ([262fcd9](https://github.com/welpo/tabi/commit/262fcd95f18e90dca17522ce2de4f07df6febedd)) by [@welpo](https://github.com/welpo)\n- *(demo)* Update projects section ([d543560](https://github.com/welpo/tabi/commit/d543560431e1ddd4624a684e5e8cca7f11ea0b7f)) by [@welpo](https://github.com/welpo)\n- *(git-sumi)* Improve emoji matching ([e373482](https://github.com/welpo/tabi/commit/e3734829460d4f93643e7d8fc4a33670f5bb9c6e)) by [@welpo](https://github.com/welpo)\n- *(link_checker)* Skip vultr.com links ([0158097](https://github.com/welpo/tabi/commit/0158097594f1d19653f2b68c4fedf802342aa8ad)) by [@welpo](https://github.com/welpo)\n- *(projects)* Load git-sumi demo from its repo ([7b13798](https://github.com/welpo/tabi/commit/7b13798a68110d1da63870ea95230ece84526a8b)) by [@welpo](https://github.com/welpo)\n- *(release)* Remove link comparison ([e2358f7](https://github.com/welpo/tabi/commit/e2358f7edfc2944e34bc59cbe4b1cbff64412f21)) by [@welpo](https://github.com/welpo)\n- *(release)* Update CHANGELOG format ([5b5d1fa](https://github.com/welpo/tabi/commit/5b5d1fa592b72b50e1fccf172d9245a9e087bfd3)) by [@welpo](https://github.com/welpo)\n- Add renovate config ([3672a94](https://github.com/welpo/tabi/commit/3672a94e863a44e88b1ec0e1896ac6ef38833ef7)) by [@welpo](https://github.com/welpo)\n\n## [2.2.0](https://github.com/welpo/tabi/compare/v2.1.0..v2.2.0) - 2024-02-10\n\n### ✨ Features\n\n- *(i18n)* Add Right-to-left script support ([#262](https://github.com/welpo/tabi/issues/262)) by [@TheAwiteb](https://github.com/TheAwiteb) and [@welpo](https://github.com/welpo)\n\n### 🐛 Bug fixes\n\n- *(tags)* Arabic tags stuck together ([#264](https://github.com/welpo/tabi/issues/264)) by [@TheAwiteb](https://github.com/TheAwiteb) and [@welpo](https://github.com/welpo)\n\n### 📝 Documentation\n\n- *(README)* Add git-sumi badge ([8b5a131](https://github.com/welpo/tabi/commit/8b5a131001cbb29656ab2339cafa777938353ea3)) by [@welpo](https://github.com/welpo)\n- *(README)* Add IPA notation for pronunciation ([f7c0de4](https://github.com/welpo/tabi/commit/f7c0de412daf32f14374e282863c6f7be651e0c8)) by [@welpo](https://github.com/welpo)\n- *(README)* Add project badges ([bcde187](https://github.com/welpo/tabi/commit/bcde187f73a278cb534282204f9aa183cec2ec0a)) by [@welpo](https://github.com/welpo)\n- *(README)* Remove quick start section ([1382b44](https://github.com/welpo/tabi/commit/1382b44c182827a5757b7aaf50b1f036ccda12b4)) by [@welpo](https://github.com/welpo)\n- *(README)* Recommend installing tabi as a submodule or cloning it ([a05170c](https://github.com/welpo/tabi/commit/a05170c6895ad7f7e8d71a482e6ea9efaa69030e)) by [@welpo](https://github.com/welpo)\n- *(contributing)* Enrich PR information ([28d0f70](https://github.com/welpo/tabi/commit/28d0f70e18c392b0c57cc0e818ee1b763aa0137e)) by [@welpo](https://github.com/welpo)\n\n### 🔧 Miscellaneous tasks\n\n- *(CHANGELOG)* Improve emoji pattern ([1bb3c91](https://github.com/welpo/tabi/commit/1bb3c91b91585bb6317ed0d82e6d0c0fbc36e0b6)) by [@welpo](https://github.com/welpo)\n- *(CI)* Update git-sumi config ([1e468f4](https://github.com/welpo/tabi/commit/1e468f4acb9d671c630d4a13e2b6e11dce13ed78)) by [@welpo](https://github.com/welpo)\n- *(CI)* Use git-sumi hook to lint commit messages ([71f09a8](https://github.com/welpo/tabi/commit/71f09a81b0c190d69e28a6c122b559560fc72a26)) by [@welpo](https://github.com/welpo)\n- *(CI)* Fix links in tag description ([7cc74d1](https://github.com/welpo/tabi/commit/7cc74d1fd9fe10d671c240029660c0164e821db5)) by [@welpo](https://github.com/welpo)\n- *(git-sumi)* Require a space after the gitmoji ([119533b](https://github.com/welpo/tabi/commit/119533b176da6bb182b1349eacc049bea85c2bac)) by [@welpo](https://github.com/welpo)\n- *(release)* Verify version tag format on release ([fcf16a4](https://github.com/welpo/tabi/commit/fcf16a457eac2f6daa15c9e7159713e2c78de709)) by [@welpo](https://github.com/welpo)\n- Use git-sumi to lint commit messages ([c193d4d](https://github.com/welpo/tabi/commit/c193d4d4e898def908ff923958ccab7af4023664)) by [@welpo](https://github.com/welpo)\n- Update changelog sections ([0e6c5c2](https://github.com/welpo/tabi/commit/0e6c5c269177712ae58649fe24662b407315d3f1)) by [@welpo](https://github.com/welpo)\n- Add continuous deployment workflow ([9b8b139](https://github.com/welpo/tabi/commit/9b8b1396cfaeefd61050cdf49bf81f01977e5899)) by [@welpo](https://github.com/welpo)\n\n### 👥 New contributors\n\n🫶 [@TheAwiteb](https://github.com/TheAwiteb) made their first contribution in [#264](https://github.com/welpo/tabi/pull/264)\n\n## [2.1.0](https://github.com/welpo/tabi/compare/v2.0.0..v2.1.0) - 2024-01-29\n\n### ✨ Features\n\n- Allow inverting previous/next article links ([#261](https://github.com/welpo/tabi/issues/261)) by [@ZzMzaw](https://github.com/ZzMzaw) and [@welpo](https://github.com/welpo)\n\n### 📝 Documentation\n\n- *(README)* Add CSS code block setup in required config ([c9d6b62](https://github.com/welpo/tabi/commit/c9d6b629078f70f4392818a17a4a7ac6b11c0480)) by [@welpo](https://github.com/welpo)\n\n### 👥 New contributors\n\n🫶 [@andwati](https://github.com/andwati) made their first contribution in [#257](https://github.com/welpo/tabi/pull/257)\n\n## [2.0.0](https://github.com/welpo/tabi/compare/v1.0.0..v2.0.0) - 2024-01-15\n\n### 💥 BREAKING CHANGES 💥\n\n- *(i18n)* Force config.toml copyright translation ([#255](https://github.com/welpo/tabi/issues/255)) ([9f39b7f](https://github.com/welpo/tabi/commit/9f39b7fe47721eb7aaea04d8b9ee66b557b7eabf)) by [@welpo](https://github.com/welpo)\n- *(home-banner)* Force proper header.img path ([#254](https://github.com/welpo/tabi/issues/254)) by [@welpo](https://github.com/welpo)\n\n### ✨ Features\n\n- *(a11y)* Add keyboard accessibility to theme toggler ([dce495e](https://github.com/welpo/tabi/commit/dce495e71d13c50fb5d2d57698fb350d72708609)) by [@welpo](https://github.com/welpo)\n- *(archive)* Allow multiple sources in Archive ([#249](https://github.com/welpo/tabi/issues/249)) by [@welpo](https://github.com/welpo)\n- *(socials)* Add Keybase icon ([#231](https://github.com/welpo/tabi/issues/231)) by [@arichtman](https://github.com/arichtman)\n- *(tags)* Dynamically adjust tag list columns based on tag count ([#233](https://github.com/welpo/tabi/issues/233)) by [@welpo](https://github.com/welpo)\n- *(tags)* Add `compact_tags` option ([#232](https://github.com/welpo/tabi/issues/232)) by [@welpo](https://github.com/welpo) and [@arichtman](https://github.com/arichtman)\n- Add search functionality ([#250](https://github.com/welpo/tabi/issues/250)) by [@welpo](https://github.com/welpo)\n- Adjust width of prev/next link section ([#248](https://github.com/welpo/tabi/issues/248)) by [@welpo](https://github.com/welpo)\n- Add previous/next article navigation ([#246](https://github.com/welpo/tabi/issues/246)) by [@welpo](https://github.com/welpo)\n- Add taxonomy list & single term templates ([#239](https://github.com/welpo/tabi/issues/239)) by [@welpo](https://github.com/welpo)\n\n### 🐛 Bug fixes\n\n- *(archive)* Group multi-sourced posts by year ([2c8a21d](https://github.com/welpo/tabi/commit/2c8a21d558f5f6b3a208a3caf9a88b01c92ca0e1)) by [@welpo](https://github.com/welpo)\n- *(i18n)* Localise date in single taxonomy listing ([0aef23b](https://github.com/welpo/tabi/commit/0aef23b3703ddedeff080525cbdede94451e7817)) by [@welpo](https://github.com/welpo)\n\n### 📝 Documentation\n\n- *(faq-languages)* Clarify setting default language ([c4e5a65](https://github.com/welpo/tabi/commit/c4e5a65908431188c0b95038ac10bd0d4a1cff8d)) by [@welpo](https://github.com/welpo)\n- *(mastering-tabi)* Fix `browser_theme_color` spelling ([e2baa36](https://github.com/welpo/tabi/commit/e2baa36ddc229e2a6a4519d312045a5b89669b0d)) by [@welpo](https://github.com/welpo)\n- *(mastering-tabi)* Favour GitHub discussions over comment section ([cd2cbf3](https://github.com/welpo/tabi/commit/cd2cbf36fa6cc4eafff925c7907ce2598227a88c)) by [@welpo](https://github.com/welpo)\n\n### ♻️ Refactor\n\n- Format JS with Prettier ([#240](https://github.com/welpo/tabi/issues/240)) by [@welpo](https://github.com/welpo)\n\n### 🗑️️ Deprecations\n\n- *(home-banner)* [**‼️BREAKING‼️**] Force proper header.img path ([#254](https://github.com/welpo/tabi/issues/254)) by [@welpo](https://github.com/welpo)\n- *(i18n)* [**‼️BREAKING‼️**] Force config.toml copyright translation ([#255](https://github.com/welpo/tabi/issues/255)) ([9f39b7f](https://github.com/welpo/tabi/commit/9f39b7fe47721eb7aaea04d8b9ee66b557b7eabf)) by [@welpo](https://github.com/welpo)\n\n### 👥 New contributors\n\n🫶 [@arichtman](https://github.com/arichtman) made their first contribution in [#231](https://github.com/welpo/tabi/pull/231)\n\n## 1.0.0 - 2023-12-29\n\n### 💥 BREAKING CHANGES 💥\n\n- *(theme-switcher)* Respect `theme_default` when JS is enabled ([#224](https://github.com/welpo/tabi/issues/224)) by [@welpo](https://github.com/welpo)\n\n### ✨ Features\n\n- *(HTML)* Respect `external_links_…` config ([#126](https://github.com/welpo/tabi/issues/126)) by [@welpo](https://github.com/welpo)\n- *(SEO)* Add description support for sections ([#179](https://github.com/welpo/tabi/issues/179)) by [@donovanglover](https://github.com/donovanglover)\n- *(ToC)* Implement setting hierarchy for ToC ([695e40b](https://github.com/welpo/tabi/commit/695e40be706a57023f9a4b69f82fdb05db77b9ec)) by [@welpo](https://github.com/welpo)\n- *(alias)* Style redirect page ([#156](https://github.com/welpo/tabi/issues/156)) by [@welpo](https://github.com/welpo)\n- *(archive)* Allow customising section for archive ([b987439](https://github.com/welpo/tabi/commit/b9874393b448e75b90945f32c5ea41aa5465fe25)) by [@welpo](https://github.com/welpo)\n- *(atom)* Add `full_content_in_feed` option ([#164](https://github.com/welpo/tabi/issues/164)) by [@welpo](https://github.com/welpo)\n- *(cards.html)* Render markdown content ([#201](https://github.com/welpo/tabi/issues/201)) by [@donovanglover](https://github.com/donovanglover)\n- *(comments)* Allow page override of global system ([#216](https://github.com/welpo/tabi/issues/216)) by [@welpo](https://github.com/welpo)\n- *(config)* Add dynamic stylesheet loading ([#118](https://github.com/welpo/tabi/issues/118)) by [@welpo](https://github.com/welpo)\n- *(copyright)* Set copyright translations outside toml files & enhance $VARs ([#215](https://github.com/welpo/tabi/issues/215)) by [@ZzMzaw](https://github.com/ZzMzaw) and [@welpo](https://github.com/welpo)\n- *(feed)* Support hiding pages and sections ([#227](https://github.com/welpo/tabi/issues/227)) by [@welpo](https://github.com/welpo)\n- *(fonts)* Upgrade Inter to 4.0 ([012d819](https://github.com/welpo/tabi/commit/012d819f38f5b120c05830ebf0a0fed88f057c6e)) by [@welpo](https://github.com/welpo)\n- *(fonts)* Optimise font size ([#186](https://github.com/welpo/tabi/issues/186)) by [@welpo](https://github.com/welpo)\n- *(fonts)* Use woff2 & update fonts ([1c06c99](https://github.com/welpo/tabi/commit/1c06c9904772dd012c9c3b3752a537aeb8e1fc08)) by [@welpo](https://github.com/welpo)\n- *(footer)* Add configurable footer menu ([#197](https://github.com/welpo/tabi/issues/197)) by [@welpo](https://github.com/welpo)\n- *(footer)* Add configurable copyright notice ([#112](https://github.com/welpo/tabi/issues/112)) by [@welpo](https://github.com/welpo)\n- *(footer)* Add link to remote repository ([#111](https://github.com/welpo/tabi/issues/111)) by [@welpo](https://github.com/welpo)\n- *(footer)* Add link to remote repository ([99f6f2d](https://github.com/welpo/tabi/commit/99f6f2dcdc7bcab449f278fbad8cc4e9350071cd)) by [@welpo](https://github.com/welpo)\n- *(footer/copyright)* Replace \"$SEPARATOR\" to `separator` ([f1edbba](https://github.com/welpo/tabi/commit/f1edbba8bd81a9f4253c582b95af1797ab828cff)) by [@welpo](https://github.com/welpo)\n- *(footer/socials)* Add base64 encoded email protection ([427ffc7](https://github.com/welpo/tabi/commit/427ffc7241b32aba85f3dee67a2d371bb1c21420)) by [@welpo](https://github.com/welpo)\n- *(header)* Cachebust custom font subset ([130aa51](https://github.com/welpo/tabi/commit/130aa51144f6c8bf84b06e4d6cb9daf75dacc4a2)) by [@welpo](https://github.com/welpo)\n- *(header)* Add support for canonical URLs ([#168](https://github.com/welpo/tabi/issues/168)) by [@welpo](https://github.com/welpo)\n- *(header)* Support multilingual meta tags in sections ([465bc55](https://github.com/welpo/tabi/commit/465bc55b9ab8263f55e0f83b9468a669c81b6ce8)) by [@welpo](https://github.com/welpo)\n- *(header)* Add img_alt attribute for customisable alt text ([#153](https://github.com/welpo/tabi/issues/153)) by [@welpo](https://github.com/welpo)\n- *(i18n)* Translate quick navigation buttons' titles ([c0a07cb](https://github.com/welpo/tabi/commit/c0a07cb4678c5b23bf1a049966f725c6d2c4b88a)) by [@welpo](https://github.com/welpo)\n- *(i18n)* Overhaul translation system & add languages ([#145](https://github.com/welpo/tabi/issues/145)) by [@welpo](https://github.com/welpo)\n- *(image shortcodes)* Add lazy loading ([#116](https://github.com/welpo/tabi/issues/116)) by [@welpo](https://github.com/welpo)\n- *(info-page)* Enable arbitrary ToC location ([05ab241](https://github.com/welpo/tabi/commit/05ab241002be39dcc92b19b179329ada1579e6f4)) by [@welpo](https://github.com/welpo)\n- *(list_posts.html)* Display draft status in post list ([#203](https://github.com/welpo/tabi/issues/203)) by [@xvello](https://github.com/xvello) and [@welpo](https://github.com/welpo)\n- *(meta)* Allow hiding the read time indication on pages ([#124](https://github.com/welpo/tabi/issues/124)) by [@stevenroose](https://github.com/stevenroose) and [@welpo](https://github.com/welpo)\n- *(multilingual-quote)* Allow missing author ([3cf9e3a](https://github.com/welpo/tabi/commit/3cf9e3ad8d193319e5313fff3a77a83aa1254e3b)) by [@welpo](https://github.com/welpo)\n- *(pre-commit)* Improve robustness checking commands ([f98d23e](https://github.com/welpo/tabi/commit/f98d23ec937d07ef220ff043c500ac824a009367)) by [@welpo](https://github.com/welpo)\n- *(pre-commit)* Sync config.toml and theme.toml [extra] ([1601fbf](https://github.com/welpo/tabi/commit/1601fbf30c66b90592acbae9e96ae80450d2d692)) by [@welpo](https://github.com/welpo)\n- *(pre-commit)* Show which JS compressor is best ([cffe06f](https://github.com/welpo/tabi/commit/cffe06f44e695afc7c5e154ff8ec0084875240a6)) by [@welpo](https://github.com/welpo)\n- *(pre-commit)* Check for JS minification ([1027c4f](https://github.com/welpo/tabi/commit/1027c4fd02bfde8d836cc69b869523f4053ee6e0)) by [@welpo](https://github.com/welpo)\n- *(quick-navigation)* Add ToC button ([#115](https://github.com/welpo/tabi/issues/115)) by [@welpo](https://github.com/welpo)\n- *(settings)* Add hierarchy-based setting overrides ([#128](https://github.com/welpo/tabi/issues/128)) by [@welpo](https://github.com/welpo)\n- *(shortcode)* Add spoiler shortcode ([#113](https://github.com/welpo/tabi/issues/113)) by [@welpo](https://github.com/welpo)\n- *(shortcodes)* Update image path retrieval and dimensions in shortcodes ([b841969](https://github.com/welpo/tabi/commit/b841969a8fc4a783a49bd98c6d690fc06f87dc08)) by [@welpo](https://github.com/welpo)\n- *(sitemap)* Mention baseurl in header & title ([381f41d](https://github.com/welpo/tabi/commit/381f41dd0dc4cee56fd13b05f05333085e668b23)) by [@welpo](https://github.com/welpo)\n- *(sitemap)* Add XML styling ([#221](https://github.com/welpo/tabi/issues/221)) by [@welpo](https://github.com/welpo)\n- *(social-icons)* Add Matrix icon ([28fb525](https://github.com/welpo/tabi/commit/28fb52581fc6bd20b0a2cce084bea858ce2a8b63)) by [@welpo](https://github.com/welpo)\n- *(social-icons)* Add X icon ([b797879](https://github.com/welpo/tabi/commit/b797879403406063560864899b717785b1fcd359)) by [@welpo](https://github.com/welpo)\n- *(social-media-cards)* Support relative paths ([#163](https://github.com/welpo/tabi/issues/163)) by [@welpo](https://github.com/welpo)\n- *(social-media-cards)* Add support for twitter:image ([22579a5](https://github.com/welpo/tabi/commit/22579a59743c5f1e0531de025c8402d4379736eb)) by [@welpo](https://github.com/welpo)\n- *(socials)* Add Nostr icon ([79af568](https://github.com/welpo/tabi/commit/79af5680784fe0d16644d8102863725d5bb909e5)) by [@welpo](https://github.com/welpo)\n- *(socials)* Add Google Scholar icon ([dd76c18](https://github.com/welpo/tabi/commit/dd76c18faa8633713832fafe3dcf1a8b05ed0335)) by [@welpo](https://github.com/welpo)\n- *(templates)* Introduce info-page template ([#147](https://github.com/welpo/tabi/issues/147)) by [@welpo](https://github.com/welpo)\n- *(theme-switcher)* [**‼️BREAKING‼️**] Respect `theme_default` when JS is enabled ([#224](https://github.com/welpo/tabi/issues/224)) by [@welpo](https://github.com/welpo)\n- *(theme-switcher)* Add theme reset button ([#198](https://github.com/welpo/tabi/issues/198)) by [@welpo](https://github.com/welpo)\n- *(theme-switcher)* Hide button if JS is disabled ([113a7f4](https://github.com/welpo/tabi/commit/113a7f4e5d4efd87646f8c26f81b7754ea7f41e2)) by [@welpo](https://github.com/welpo)\n- *(title)* Add `invert_title_order` config option ([#142](https://github.com/welpo/tabi/issues/142)) by [@welpo](https://github.com/welpo)\n- Allow relative paths for image shortcodes ([#222](https://github.com/welpo/tabi/issues/222)) by [@welpo](https://github.com/welpo)\n- Improve clickability cue by adding pointer cursor to details summary ([1e33b9b](https://github.com/welpo/tabi/commit/1e33b9bc1f3cc995a1ed7e265a106d194c8dfcef)) by [@welpo](https://github.com/welpo)\n- Support privacy-respecting analytics ([#193](https://github.com/welpo/tabi/issues/193)) by [@welpo](https://github.com/welpo)\n- Allow internal project links ([#157](https://github.com/welpo/tabi/issues/157)) by [@welpo](https://github.com/welpo)\n- Enhance SEO and accessibility with meta tags ([#154](https://github.com/welpo/tabi/issues/154)) by [@welpo](https://github.com/welpo)\n- Add `browser_theme_color` config option ([#143](https://github.com/welpo/tabi/issues/143)) by [@welpo](https://github.com/welpo)\n- Add support for social media cards ([#130](https://github.com/welpo/tabi/issues/130)) by [@welpo](https://github.com/welpo)\n- Add base64 encoded email protection ([#110](https://github.com/welpo/tabi/issues/110)) by [@welpo](https://github.com/welpo)\n- Avoid commiting drafts ([6cb2327](https://github.com/welpo/tabi/commit/6cb23279075211380e748bdb535fe38853ff6199)) by [@welpo](https://github.com/welpo)\n- Add pre-commit hook ([22d8ae6](https://github.com/welpo/tabi/commit/22d8ae6b8aecac19500b70488f3d8f269b7f0e96)) by [@welpo](https://github.com/welpo)\n- Add support for remote codeberg repositories ([b5fae83](https://github.com/welpo/tabi/commit/b5fae83f28800ad6ef5a34180d84f05c8fe72849)) by [@welpo](https://github.com/welpo)\n- Add link to commit history on updated articles ([#109](https://github.com/welpo/tabi/issues/109)) by [@welpo](https://github.com/welpo)\n- Overhaul Table of Contents ([#108](https://github.com/welpo/tabi/issues/108)) by [@welpo](https://github.com/welpo)\n- Enhance Table of Contents ([07ceddc](https://github.com/welpo/tabi/commit/07ceddcc007b948e697da8a2dcf62edc37d91c6e)) by [@welpo](https://github.com/welpo)\n- Add `toc_levels` to control ToC depth ([965af4b](https://github.com/welpo/tabi/commit/965af4b3dd35d1c5f24b4aa0ba847804e0d3c3da)) by [@welpo](https://github.com/welpo)\n- Add wide container shortcode ([#107](https://github.com/welpo/tabi/issues/107)) by [@welpo](https://github.com/welpo)\n- Add wide container shortcode ([fb0dea0](https://github.com/welpo/tabi/commit/fb0dea01067c91cf9d91eeb14ac8bf41842645e8)) by [@welpo](https://github.com/welpo)\n- Add `rel=me` attribute to social links ([4952ce8](https://github.com/welpo/tabi/commit/4952ce88f14d5d39d583926eaac7460b9783bff3)) by [@welpo](https://github.com/welpo)\n- Introduce skins to customise the theme ([#105](https://github.com/welpo/tabi/issues/105)) by [@welpo](https://github.com/welpo)\n- Add pointer cursor to image toggler ([d2780f4](https://github.com/welpo/tabi/commit/d2780f42a3b5caff5c920be4580e01f966e30362)) by [@welpo](https://github.com/welpo)\n- Add `cachebust` for `main.css` ([2f83ef8](https://github.com/welpo/tabi/commit/2f83ef80621245d8a497c0a2ccfd7d4c375ce2b2)) by [@welpo](https://github.com/welpo)\n- Add 2 interactive image shortcodes by [@welpo](https://github.com/welpo)\n- Add 2 interactive image shortcodes ([7ccd717](https://github.com/welpo/tabi/commit/7ccd71736eb8c3e7ee1114b271e8b698bdf5df18)) by [@welpo](https://github.com/welpo)\n- Allow setting default theme by [@welpo](https://github.com/welpo)\n- Allow setting the default theme ([c78c7f5](https://github.com/welpo/tabi/commit/c78c7f5bd1d1f5f5d8d37daade0ff27a7a59b479)) by [@welpo](https://github.com/welpo)\n- Global/individual toggle for KaTeX and copy button ([edcfa35](https://github.com/welpo/tabi/commit/edcfa359ef1abab43d5bedf77a24c3fab27f0297)) by [@welpo](https://github.com/welpo)\n- Add backlinks to footnotes ([#101](https://github.com/welpo/tabi/issues/101)) by [@welpo](https://github.com/welpo)\n- Optional trailing slash in menu links ([f9c18b0](https://github.com/welpo/tabi/commit/f9c18b03987a7077301992f8369f4de75e270a2a)) by [@welpo](https://github.com/welpo)\n- Add JavaScript notice for comments ([09e5bc1](https://github.com/welpo/tabi/commit/09e5bc1aa3f09ccf83c126cd8dc1b0ef6edcb4dd)) by [@welpo](https://github.com/welpo)\n- Add support for Isso comments by [@welpo](https://github.com/welpo)\n- Add support for Isso comments ([9a98789](https://github.com/welpo/tabi/commit/9a98789922ce7dee96e0fab6ebbc9ba0c5499e80)) by [@welpo](https://github.com/welpo)\n- Add support for Hyvor Talk comments by [@welpo](https://github.com/welpo)\n- Add support for Hyvor Talk comments ([7924e82](https://github.com/welpo/tabi/commit/7924e8206c17423c9043453fb37bc394f8676345)) by [@welpo](https://github.com/welpo)\n- Add quick navigation buttons by [@welpo](https://github.com/welpo)\n- Add quick navigation buttons ([05a6d15](https://github.com/welpo/tabi/commit/05a6d15455a0ecc41f071ce03fbed5aaf16d38b5)) by [@welpo](https://github.com/welpo)\n- Add support for giscus & utterances comments by [@welpo](https://github.com/welpo)\n- Add support for giscus and utterances ([5993e95](https://github.com/welpo/tabi/commit/5993e950c7a6084f069403e3a98c85aa643654f7)) by [@welpo](https://github.com/welpo)\n- Add support for giscus & utterances comments ([19d1200](https://github.com/welpo/tabi/commit/19d120019b6f8c7041965895dad2764dc2aa4f5d)) by [@welpo](https://github.com/welpo)\n- Add utterances support ([93c8b57](https://github.com/welpo/tabi/commit/93c8b577b87aaa928a13b5696d7b3ebc91aa6cce)) by [@welpo](https://github.com/welpo)\n- Add copy button to code blocks by [@welpo](https://github.com/welpo)\n- Add copy button to code blocks ([2dec139](https://github.com/welpo/tabi/commit/2dec139e41ffe558b7d8fdd785ed5f1889d6723b)) by [@welpo](https://github.com/welpo)\n- Add basic copy button to codeblocks ([8e1473b](https://github.com/welpo/tabi/commit/8e1473bba916523f4d1855d4c631e78d603443ba)) by [@welpo](https://github.com/welpo)\n- Add full-width image option to shortcodes  by [@welpo](https://github.com/welpo)\n- Add full-width image option to shortcodes ([fd50204](https://github.com/welpo/tabi/commit/fd50204e32aee4402835263f9e420adbc2f6cb8e)) by [@welpo](https://github.com/welpo)\n- Custom feed content and style by [@welpo](https://github.com/welpo)\n- Style xml feed ([1b60be8](https://github.com/welpo/tabi/commit/1b60be8c9d4e28645e713672eeb1b9dbdf4e1789)) by [@welpo](https://github.com/welpo)\n- Use page description as summary ([54e88be](https://github.com/welpo/tabi/commit/54e88beca21976a350ce6babd9fe10b07d658cf4)) by [@welpo](https://github.com/welpo)\n- Add custom quotation marks for multilingual quotes ([dd73e52](https://github.com/welpo/tabi/commit/dd73e527298c1c4357e48589314bfd9302323c27)) by [@welpo](https://github.com/welpo)\n- Add multilingual quote shortcode ([ab29511](https://github.com/welpo/tabi/commit/ab29511d5e7f44ba5cc07e1a29c0d678c24b3cd4)) by [@welpo](https://github.com/welpo)\n- Allow using emojis as favicon ([fb061a1](https://github.com/welpo/tabi/commit/fb061a14697d45700996f32b568826b4f02764c7)) by [@welpo](https://github.com/welpo)\n- Include layout features in font subset ([534969d](https://github.com/welpo/tabi/commit/534969de11930ec957d6ea4eea658324a3d29497)) by [@welpo](https://github.com/welpo)\n- Add multi-language support ([fa92293](https://github.com/welpo/tabi/commit/fa9229377d03f229b4deb68bad7cf63b2f88f9fe)) by [@welpo](https://github.com/welpo)\n- Set width and height of project images ([8b5726c](https://github.com/welpo/tabi/commit/8b5726c34e04238b0371fb3dbd11cafaaff580f0)) by [@welpo](https://github.com/welpo)\n- Add KaTeX support ([f9335b3](https://github.com/welpo/tabi/commit/f9335b3fa45f27788cdb9c8d26ddfe4b714070c4)) by [@welpo](https://github.com/welpo)\n- Make JS light/dark toggle optional ([1803987](https://github.com/welpo/tabi/commit/1803987555cbbe0aea1ab673907192e734a55425)) by [@welpo](https://github.com/welpo)\n- Use github pages for live preview ([a36823d](https://github.com/welpo/tabi/commit/a36823d672e3894ac428642d433c5506cdbbe728)) by [@welpo](https://github.com/welpo)\n- Allow dual date-format (short/long) by [@welpo](https://github.com/welpo)\n- Allow dual date-format (short/long) ([9887fb0](https://github.com/welpo/tabi/commit/9887fb0a6f9f0e94d9d7faca57c62fb3549bdb83)) by [@welpo](https://github.com/welpo)\n- Add 'Last updated' extra tag by [@welpo](https://github.com/welpo)\n- Add 'Last updated' extra tag ([ed47f25](https://github.com/welpo/tabi/commit/ed47f2580ba4f8f15188c67f3b788b7472f4d78b)) by [@welpo](https://github.com/welpo)\n- Add dual image support for switching themes by [@welpo](https://github.com/welpo)\n- Add dual image support for switching themes ([b3dc539](https://github.com/welpo/tabi/commit/b3dc5396b4c429537f82afb8cd63a67146b5b9ac)) by [@welpo](https://github.com/welpo)\n- Use custom separator from config in meta & title ([aa00640](https://github.com/welpo/tabi/commit/aa00640e96a6e0934e7792f384f09ec476f3bdb7)) by [@welpo](https://github.com/welpo)\n- Allow customizable secure headers (CSP) ([d7caa7a](https://github.com/welpo/tabi/commit/d7caa7af5fc5d0096fbb402b0b8ce53930f3d56f)) by [@welpo](https://github.com/welpo)\n- Display word count when hovering over read time ([d9cd64e](https://github.com/welpo/tabi/commit/d9cd64e5a40c154c89d91b8b649348bcabfafb71)) by [@welpo](https://github.com/welpo)\n- Responsive (and centered) images ([9242fa4](https://github.com/welpo/tabi/commit/9242fa4389cd2ce418c2074aa00afc8107e753d0)) by [@welpo](https://github.com/welpo)\n- Clean js code ([2f80b0b](https://github.com/welpo/tabi/commit/2f80b0b5ad481d53d5621f84bffd3415fc078672)) by [@serginogal](https://github.com/serginogal)\n- Change default theme to OS setting ([26ad170](https://github.com/welpo/tabi/commit/26ad170d9fee316fb3eb131cc664cd1eed9e1cfe)) by [@welpo](https://github.com/welpo)\n- Add dimmable-image shortcode for dark theme ([174a37e](https://github.com/welpo/tabi/commit/174a37e221dc0b38eb1160bbf320c7b03b73d901)) by [@welpo](https://github.com/welpo)\n- Showcase invertable image ([0bae051](https://github.com/welpo/tabi/commit/0bae0515ec4579e26a827dba907f890198df01eb)) by [@welpo](https://github.com/welpo)\n- Showcase invertable image ([2f4695d](https://github.com/welpo/tabi/commit/2f4695dd0bcec55357f1c8829fbdf5aee8d76f9a)) by [@welpo](https://github.com/welpo)\n- Add Archive page to features. ([999861b](https://github.com/welpo/tabi/commit/999861ba583706b63e73fb7f675bedacaaf5bce4)) by [@welpo](https://github.com/welpo)\n- Remove Archive from TODO ([559c97d](https://github.com/welpo/tabi/commit/559c97d1f3a6f2e1da760ccb6adb5a20b5fe5b44)) by [@welpo](https://github.com/welpo)\n- Add archive screen ([3434cbc](https://github.com/welpo/tabi/commit/3434cbc92630d7a60963f51990216facd586e546)) by [@serginogal](https://github.com/serginogal)\n- Change markdown description style ([c29697c](https://github.com/welpo/tabi/commit/c29697c4212d59588b75013c473790cd97445164)) by [@serginogal](https://github.com/serginogal)\n- Add markdown support to index page ([c6ec7c7](https://github.com/welpo/tabi/commit/c6ec7c750588562c3a14c1ee0f2a7d00f467bbd6)) by [@serginogal](https://github.com/serginogal)\n- Set Table of Contents per page, not globally ([32d4a93](https://github.com/welpo/tabi/commit/32d4a9358ce14c9dce687520e7c58cb93d07a7b6)) by [@welpo](https://github.com/welpo)\n\n### 🐛 Bug fixes\n\n- *(SEO)* Set proper social image cards ([b26902d](https://github.com/welpo/tabi/commit/b26902da1667ecfaf5649dc9637360658c135faf)) by [@welpo](https://github.com/welpo)\n- *(a11y)* Update heading tags ([#140](https://github.com/welpo/tabi/issues/140)) by [@welpo](https://github.com/welpo)\n- *(analytics)* Make goatcounter use https ([#195](https://github.com/welpo/tabi/issues/195)) by [@nyadiia](https://github.com/nyadiia)\n- *(atom)* Remove safe filter; allow content & summary ([#165](https://github.com/welpo/tabi/issues/165)) by [@SeaDve](https://github.com/SeaDve)\n- *(atom)* Escape HTML in feed summary ([4610073](https://github.com/welpo/tabi/commit/46100730fa59d6f14d3565e87b48fad5d69afab5)) by [@welpo](https://github.com/welpo)\n- *(comments)* Fix malformed URLs in Isso mails ([#211](https://github.com/welpo/tabi/issues/211)) by [@welpo](https://github.com/welpo)\n- *(favicon)* Use `get_url` for favicon ([ab4b523](https://github.com/welpo/tabi/commit/ab4b523f9c7efa431dee4780c5b701c7496faa2d)) by [@welpo](https://github.com/welpo)\n- *(fonts)* Fix extension woff -> woff2 ([5aece98](https://github.com/welpo/tabi/commit/5aece9856a23f79143e90e91747e9f852903810b)) by [@welpo](https://github.com/welpo)\n- *(footer)* Fix external link redirection in footer ([#219](https://github.com/welpo/tabi/issues/219)) by [@joberthrogers18](https://github.com/joberthrogers18) and [@welpo](https://github.com/welpo)\n- *(footer)* Render footer again ([3344f80](https://github.com/welpo/tabi/commit/3344f80539d837df10523160083d91c55b04f58f)) by [@welpo](https://github.com/welpo)\n- *(footer)* Remove extra space between \"tabi\" and separator ([12b9184](https://github.com/welpo/tabi/commit/12b9184f15edc26acfd442923dd6d9f3bcfa9615)) by [@welpo](https://github.com/welpo)\n- *(format_date)* Actually use `short_date_format` ([#119](https://github.com/welpo/tabi/issues/119)) by [@stevenroose](https://github.com/stevenroose)\n- *(header)* Add quotes to custom subset links ([1221eee](https://github.com/welpo/tabi/commit/1221eeeb41c17556784c27d98d18a0d75cdbfc52)) by [@welpo](https://github.com/welpo)\n- *(header)* Correct self-closing script tag ([94f7a36](https://github.com/welpo/tabi/commit/94f7a367075b0bfeacccad44742d52f7732ff3c3)) by [@welpo](https://github.com/welpo)\n- *(header)* Check if translations exist while building tags ([#159](https://github.com/welpo/tabi/issues/159)) by [@welpo](https://github.com/welpo)\n- *(header)* Don't fail if custom_subset absent ([#132](https://github.com/welpo/tabi/issues/132)) by [@xvello](https://github.com/xvello)\n- *(home-banner)* Allow missing image & correct spacing ([#123](https://github.com/welpo/tabi/issues/123)) by [@welpo](https://github.com/welpo)\n- *(html)* Remove unused self-closing tags ([#176](https://github.com/welpo/tabi/issues/176)) by [@donovanglover](https://github.com/donovanglover)\n- *(hyvor-talk)* Switch from ID to class selector for initialisation ([#196](https://github.com/welpo/tabi/issues/196)) by [@welpo](https://github.com/welpo)\n- *(i18n)* Use English by default in atom.xml ([#167](https://github.com/welpo/tabi/issues/167)) by [@donovanglover](https://github.com/donovanglover)\n- *(i18n-es)* Correct 'of' translation ([0b6cb31](https://github.com/welpo/tabi/commit/0b6cb31bba927315c975dfe5bcf12b691a6e2573)) by [@welpo](https://github.com/welpo)\n- *(isso-comments)* Actually set maxCommentsNested ([47c493b](https://github.com/welpo/tabi/commit/47c493be89cfb58ab3945ff30da618767330fe1a)) by [@welpo](https://github.com/welpo)\n- *(multilingual-quote)* Open <p> regardless of attribution ([01d5322](https://github.com/welpo/tabi/commit/01d5322ef355af0b4bff9bbaf072305b45ba9f0b)) by [@welpo](https://github.com/welpo)\n- *(quick-navigation)* Limit width of toc ([#204](https://github.com/welpo/tabi/issues/204)) by [@donovanglover](https://github.com/donovanglover)\n- *(quick-navigation-buttons)* Fix overlapping background change ([f4c8826](https://github.com/welpo/tabi/commit/f4c88266e7160010ee739273c16e7b66c95f5e4a)) by [@welpo](https://github.com/welpo)\n- *(sitemap)* Use the base url for JS/CSS loading ([c8a1705](https://github.com/welpo/tabi/commit/c8a1705b51984666af676181bf24eb0352c684c1)) by [@welpo](https://github.com/welpo)\n- *(skins)* Load `teal` if set in config ([3e30557](https://github.com/welpo/tabi/commit/3e30557dca2b1d378e959eb0af41fa96fadfaf9c)) by [@welpo](https://github.com/welpo)\n- *(style)* Proper nested lists spacing ([#218](https://github.com/welpo/tabi/issues/218)) by [@xvello](https://github.com/xvello)\n- *(style)* Add line-height to .section-title ([#205](https://github.com/welpo/tabi/issues/205)) by [@donovanglover](https://github.com/donovanglover)\n- *(style)* Prevent videos from overflowing container ([#185](https://github.com/welpo/tabi/issues/185)) by [@donovanglover](https://github.com/donovanglover)\n- *(styled feed)* Url to \"Visit Website\" ([68e37f4](https://github.com/welpo/tabi/commit/68e37f4574d9bf0183d76286a45c39e824790e68)) by [@welpo](https://github.com/welpo)\n- *(stylised-feed)* Set proper date class ([#207](https://github.com/welpo/tabi/issues/207)) by [@welpo](https://github.com/welpo)\n- *(theme-switcher)* Match functionality of minified and non-minified files ([ee2b171](https://github.com/welpo/tabi/commit/ee2b171fbd0c79b7374dda807fe223c72f81e9d5)) by [@welpo](https://github.com/welpo)\n- *(theme-switcher)* Properly translate aria-label ([#192](https://github.com/welpo/tabi/issues/192)) by [@Almost-Senseless-Coder](https://github.com/Almost-Senseless-Coder), [@Tim](https://github.com/Tim) Böttcher and [@welpo](https://github.com/welpo)\n- Prevent background overflow on link hover ([#226](https://github.com/welpo/tabi/issues/226)) by [@welpo](https://github.com/welpo)\n- Remove unnecessary description filters ([6673a06](https://github.com/welpo/tabi/commit/6673a063b8968d39b8396bc83242f779615a14c7)) by [@welpo](https://github.com/welpo)\n- Improve accessibility and i18n ([#183](https://github.com/welpo/tabi/issues/183)) by [@welpo](https://github.com/welpo)\n- Allow live reload when zola is in `serve` mode ([#144](https://github.com/welpo/tabi/issues/144)) by [@welpo](https://github.com/welpo)\n- `is_draft` returning opposite value ([adb702f](https://github.com/welpo/tabi/commit/adb702f3c36ee19b43805b7cb27a5b97d9653ecc)) by [@welpo](https://github.com/welpo)\n- Check proper `toc_ignore_pattern` ([fe872a7](https://github.com/welpo/tabi/commit/fe872a7b54f8c24c3bb4d678c72d4f17b6f6d396)) by [@welpo](https://github.com/welpo)\n- Move `default_language` to proper section ([52d0ac9](https://github.com/welpo/tabi/commit/52d0ac944794d8590db5cda81015b1181ea83971)) by [@welpo](https://github.com/welpo)\n- Update bug report link format ([6178e8e](https://github.com/welpo/tabi/commit/6178e8e642fce10fcb2be378dc4440b754fe804d)) by [@welpo](https://github.com/welpo)\n- Revert flex display (b1b98eb) ([8cfab98](https://github.com/welpo/tabi/commit/8cfab98e721d9dccc05b2d5f187ddeb2248930b7)) by [@welpo](https://github.com/welpo)\n- Remove flex display on full-width elements ([56a53dc](https://github.com/welpo/tabi/commit/56a53dce0cf131c73474d33612a429a80ab1cfdc)) by [@welpo](https://github.com/welpo)\n- Improve contrast on `code` within links on hover ([f8891db](https://github.com/welpo/tabi/commit/f8891dba2cc0d1a8619bf9b01bb06e4c7560b7ec)) by [@welpo](https://github.com/welpo)\n- Draft margin ([8949db4](https://github.com/welpo/tabi/commit/8949db43e136e2646bbf7ff3da4a833e838796e4)) by [@welpo](https://github.com/welpo)\n- Add missing images for new shortcodes ([9ad0883](https://github.com/welpo/tabi/commit/9ad08838c78d51400bcb059781a708ccff0cdaf6)) by [@welpo](https://github.com/welpo)\n- Set proper color-scheme meta ([638d75c](https://github.com/welpo/tabi/commit/638d75c819ab00f5f4812109bc87c9a9918b20c1)) by [@welpo](https://github.com/welpo)\n- Remove extra newlines in final html ([ab879e3](https://github.com/welpo/tabi/commit/ab879e372705ff73b1bb4316f2ceca6b92f7cdf3)) by [@welpo](https://github.com/welpo)\n- Align text to left codeblocks with line numbers ([672772a](https://github.com/welpo/tabi/commit/672772a700de4d4b23e5b18b0ae4fa1cd3aca797)) by [@welpo](https://github.com/welpo)\n- Correct CSS path for GitHub pages ([b2ca636](https://github.com/welpo/tabi/commit/b2ca636eb98ff7e2361a1db5da3b349bf566846d)) by [@welpo](https://github.com/welpo)\n- Path to main.css ([86ab1c5](https://github.com/welpo/tabi/commit/86ab1c5f06238ae868b06b139852b0638dba5d7d)) by [@welpo](https://github.com/welpo)\n- Use relative path to main.css ([e7bd988](https://github.com/welpo/tabi/commit/e7bd988a52dc3db988e0715f8f2948aec2220360)) by [@welpo](https://github.com/welpo)\n- Multilingual issues in feeds and 'nav.html' by [@welpo](https://github.com/welpo)\n- Load English stylesheet as fallback ([f3a0dd7](https://github.com/welpo/tabi/commit/f3a0dd7f96dbb8fbd1073f66288133db348917e8)) by [@welpo](https://github.com/welpo)\n- Multilingual issues in feeds and 'nav.html' ([e7dc5eb](https://github.com/welpo/tabi/commit/e7dc5eb44e5c7f6bd00c3d2337438b8481d8101b)) by [@Smtbook](https://github.com/Smtbook)\n- Properly initialise `currentTheme` ([e1dfd2e](https://github.com/welpo/tabi/commit/e1dfd2ea0767ac7327244961c920031a8291c249)) by [@welpo](https://github.com/welpo)\n- Replace \"y\" with \"i\" ([f5e9ad8](https://github.com/welpo/tabi/commit/f5e9ad8c8a689a593af88d0ca62b11d74bbb76ce)) by [@welpo](https://github.com/welpo)\n- Use proper meta descriptions ([3e215e1](https://github.com/welpo/tabi/commit/3e215e18e29ec42761a9cd52c2f81612e3311acd)) by [@welpo](https://github.com/welpo)\n- Ensure codeblock header visibility on iOS ([abf32b8](https://github.com/welpo/tabi/commit/abf32b8147046b03270b92b5c9cf9ea69428b863)) by [@welpo](https://github.com/welpo)\n- Properly check for `theme_switcher` setting ([f710b62](https://github.com/welpo/tabi/commit/f710b62631ab2b3cb791bccd624c40b249c57fb0)) by [@welpo](https://github.com/welpo)\n- Remove leading slash from css href ([a6dfd4d](https://github.com/welpo/tabi/commit/a6dfd4d1ba5e781b4d26047a37f925e307de0e6c)) by [@welpo](https://github.com/welpo)\n- Remove leading slash from xsl href ([19bebde](https://github.com/welpo/tabi/commit/19bebde14819d8285e50e7ec3cdd1045c66de2a4)) by [@welpo](https://github.com/welpo)\n- Set svg max-height only for social icons ([4df7cdc](https://github.com/welpo/tabi/commit/4df7cdcb7a06ce6ace7463d727e7719a4d9d8c13)) by [@welpo](https://github.com/welpo)\n- Generate random id for multilingual quotes ([0e1a514](https://github.com/welpo/tabi/commit/0e1a5143c0bba6eee11de0b461400b6a85709281)) by [@welpo](https://github.com/welpo)\n- Proper URL for multilingual rss/atom feeds ([56a3061](https://github.com/welpo/tabi/commit/56a3061573de3e4e7feb044d51091c7f2671cc01)) by [@welpo](https://github.com/welpo)\n- Mention translations in 404 if there's more than 1 language ([58d0dd6](https://github.com/welpo/tabi/commit/58d0dd6f2f918cc0ce8eb649d521d8a385e94bd0)) by [@welpo](https://github.com/welpo)\n- Show language switcher if there's more than 1 language ([5ac69fe](https://github.com/welpo/tabi/commit/5ac69fe21d3b588dea38c84f3b38d8ec5e69d5ce)) by [@welpo](https://github.com/welpo)\n- Avoid empty dates when long_date_format is empty ([eded835](https://github.com/welpo/tabi/commit/eded835d3fcfce781393dd4f448ef907fd916316)) by [@welpo](https://github.com/welpo)\n- Remove extra backtick ([358e78d](https://github.com/welpo/tabi/commit/358e78d25a6afa7010933bfbb18ce89845c3a4dc)) by [@welpo](https://github.com/welpo)\n- Remove flashing from header text (Firefox) ([a5981e6](https://github.com/welpo/tabi/commit/a5981e6fdd5528d0c0461806c48d6ce573839621)) by [@welpo](https://github.com/welpo)\n- Remove flash when navigating in dark-mode ([1efb033](https://github.com/welpo/tabi/commit/1efb0330e3926235cde79bdaec9bf157dc6d1743)) by [@welpo](https://github.com/welpo)\n- Only load script when themeswitcher is enabled ([556570d](https://github.com/welpo/tabi/commit/556570d799362716c6a0b174378639206646242e)) by [@welpo](https://github.com/welpo)\n- Handle missing image metadata gracefully ([b5b50cd](https://github.com/welpo/tabi/commit/b5b50cdefbf1864778fbb5bbb3a83869811b39ac)) by [@welpo](https://github.com/welpo)\n- Properly end macro ([7aa6142](https://github.com/welpo/tabi/commit/7aa61422a8f642384a0be8a5e92e1e26c334e6a5)) by [@welpo](https://github.com/welpo)\n- Set proper scope to transparent hover for anchor ([050f546](https://github.com/welpo/tabi/commit/050f546e27f5bcfc1e5500b33279f82ef45c0caa)) by [@welpo](https://github.com/welpo)\n- Remove double equals in updated field ([dcf1af5](https://github.com/welpo/tabi/commit/dcf1af5ec5211bb8e4d97e4cafd21b08d2287ade)) by [@welpo](https://github.com/welpo)\n- Fix screenshot path ([110c872](https://github.com/welpo/tabi/commit/110c8723c39dcae23baeff6f34c501c78fbc8643)) by [@welpo](https://github.com/welpo)\n- Clarify requirements for A+ Observatory score ([4d18be3](https://github.com/welpo/tabi/commit/4d18be3a83557b6268f6d8b7de9f69c34ceacd27)) by [@welpo](https://github.com/welpo)\n- Clarify requirements for A+ Observatory score ([65a3fce](https://github.com/welpo/tabi/commit/65a3fcebba7c66209c9ac7e779c5bfe116c5eb1a)) by [@welpo](https://github.com/welpo)\n- Enable github pages ([1b5ed15](https://github.com/welpo/tabi/commit/1b5ed1546ae25bf0e5cc7e74cb922340668cf987)) by [@welpo](https://github.com/welpo)\n- Use personal access token ([fded2bd](https://github.com/welpo/tabi/commit/fded2bd53ead80954c253570aad413c19cd00763)) by [@welpo](https://github.com/welpo)\n- Update token ([91de9ad](https://github.com/welpo/tabi/commit/91de9ad036597aaec3ea24c29f81531bdebed523)) by [@welpo](https://github.com/welpo)\n- Use latest zola-deploy-action ([19bd85f](https://github.com/welpo/tabi/commit/19bd85f8db1bb5709c85907e86f47fca8402a5f6)) by [@welpo](https://github.com/welpo)\n- Remove header link background change on hover ([884cc90](https://github.com/welpo/tabi/commit/884cc9043969ac619519924f88b5b80cbc6a6b36)) by [@welpo](https://github.com/welpo)\n- Remove horizontall scroll on mobile ([e93b33e](https://github.com/welpo/tabi/commit/e93b33e6b88c2016741b220dd9e0dac3ccd0dd01)) by [@welpo](https://github.com/welpo)\n- Remove newlines around `<title>` from source code ([ac0495e](https://github.com/welpo/tabi/commit/ac0495ea6990f14a060cc909138b0fce32699bfd)) by [@welpo](https://github.com/welpo)\n- Remove h1 permalinkbackground on hover ([667f15a](https://github.com/welpo/tabi/commit/667f15abb2ba77cbf7250c71209c3eb8d6faa125)) by [@welpo](https://github.com/welpo)\n- Improve tap target sizing for tags ([9560103](https://github.com/welpo/tabi/commit/9560103ce7d35d633046f6c10d4117040a66aa47)) by [@welpo](https://github.com/welpo)\n- Improve accessibility by [@welpo](https://github.com/welpo)\n- Improve accessibility in lists & taps ([50b8d5f](https://github.com/welpo/tabi/commit/50b8d5fc7b022ace56ccf81dcf8766aa4963d6b8)) by [@welpo](https://github.com/welpo)\n- Update year to current year ([abfe2f9](https://github.com/welpo/tabi/commit/abfe2f9dc88ce6ec05f63feacd3998db2e9696f6)) by [@welpo](https://github.com/welpo)\n- Prevent malformed CSP header ([54ae781](https://github.com/welpo/tabi/commit/54ae7813529008822e4938724ecab83679db3d95)) by [@welpo](https://github.com/welpo)\n- Minor typo ([799b31d](https://github.com/welpo/tabi/commit/799b31d403a45abafddd270c471b8898972c2f06)) by [@welpo](https://github.com/welpo)\n- Rename sun icon reference ([2f4d065](https://github.com/welpo/tabi/commit/2f4d0658d8908785c7627362e5b2a7eaa96e48e2)) by [@welpo](https://github.com/welpo)\n- Set main image path relative to base_url ([f555d34](https://github.com/welpo/tabi/commit/f555d3420778db4c27a56c89520dfd1d421b0d58)) by [@welpo](https://github.com/welpo)\n- Override font-size for home subtitle <p> ([2732f49](https://github.com/welpo/tabi/commit/2732f494cfdf3314b45153f621acff6fdb4042a1)) by [@welpo](https://github.com/welpo)\n- Set remote Zola url ([1845649](https://github.com/welpo/tabi/commit/1845649613482823dbf9a597452a073425aed128)) by [@welpo](https://github.com/welpo)\n- Use serif as fallback ([c65eea1](https://github.com/welpo/tabi/commit/c65eea156ce391942daa61188b2f9393013af40e)) by [@welpo](https://github.com/welpo)\n- Close div class\"tags-item\" ([5c48acb](https://github.com/welpo/tabi/commit/5c48acba582158662c13760744e8e6a4d81646dd)) by [@welpo](https://github.com/welpo)\n- Atom feed alt description ([d41c9b5](https://github.com/welpo/tabi/commit/d41c9b5c86eafe34f2d3befb4bfd81a383e1ee66)) by [@welpo](https://github.com/welpo)\n- Remove double bottom-boder/div on Archive ([6e47145](https://github.com/welpo/tabi/commit/6e471450c6fa5608d40e870ccb08fa4a9697c9b9)) by [@welpo](https://github.com/welpo)\n- Add end-of-file newline ([099f094](https://github.com/welpo/tabi/commit/099f094f317b84b5748a9a83796fabfb29dafc7f)) by [@welpo](https://github.com/welpo)\n- _archive import typo ([55d928a](https://github.com/welpo/tabi/commit/55d928a3d59279eb5c54c861c293de3714342c48)) by [@serginogal](https://github.com/serginogal)\n- Capitalise Archive title ([31f70ff](https://github.com/welpo/tabi/commit/31f70fffafd0a0adc0cae41368bbfbd25fb27880)) by [@welpo](https://github.com/welpo)\n- Update theme screenshot in /projects ([32ca1d3](https://github.com/welpo/tabi/commit/32ca1d3e9e8823ec05a629addbdcead5b54e2a18)) by [@welpo](https://github.com/welpo)\n- Set aspect-ratio on banner-home-ig ([c6c8379](https://github.com/welpo/tabi/commit/c6c83797bff349ee0d12972264b98de0f48ea1f8)) by [@welpo](https://github.com/welpo)\n- Make navbar width and margin same as content ([1c1e124](https://github.com/welpo/tabi/commit/1c1e124115a1188ee9669451e5400d2873eee558)) by [@serginogal](https://github.com/serginogal)\n- Use Sentence case in Projects title ([cc93f9f](https://github.com/welpo/tabi/commit/cc93f9fb7fe44bb4fecb860669e368da63e57efd)) by [@welpo](https://github.com/welpo)\n- Add end-of-file newline ([2cef2e5](https://github.com/welpo/tabi/commit/2cef2e5502c871d5bafde2296882a2de96bc8ca6)) by [@welpo](https://github.com/welpo)\n- Add end-of-file newline ([095b9d0](https://github.com/welpo/tabi/commit/095b9d0d07603ec0c9fad09d9f0293f75bf11cba)) by [@welpo](https://github.com/welpo)\n- Remove duplicate .bloglist-title definition ([53d28ad](https://github.com/welpo/tabi/commit/53d28ad522d15e9446ca331715e6418501f8423b)) by [@welpo](https://github.com/welpo)\n- Consistent date format & remove invalid <time> tags ([1242b90](https://github.com/welpo/tabi/commit/1242b909a28250459b6f89d4670454ce934b0b24)) by [@welpo](https://github.com/welpo)\n- Remove duplicated meta description ([4cef13a](https://github.com/welpo/tabi/commit/4cef13a7dd473e91a6875a0037b3e282f5460571)) by [@welpo](https://github.com/welpo)\n- Remove space in atom url ([3f4259d](https://github.com/welpo/tabi/commit/3f4259d3f4669de110dea2f5b5d42b8341e72b57)) by [@welpo](https://github.com/welpo)\n- Set consistent padding between nav and main content ([6a4620a](https://github.com/welpo/tabi/commit/6a4620a33b1c468e4e0bbbe7fc4eca140decdc62)) by [@welpo](https://github.com/welpo)\n- Improve responsiveness of home banner ([1f11e84](https://github.com/welpo/tabi/commit/1f11e8480ef33427f5e3f74a8660a584817fa695)) by [@welpo](https://github.com/welpo)\n- Ensure text remains visible during webfont load ([4d57e09](https://github.com/welpo/tabi/commit/4d57e097d1b84ad5099fd9cdf39f905458a67d17)) by [@welpo](https://github.com/welpo)\n- Working live url ([898fcc8](https://github.com/welpo/tabi/commit/898fcc8975fd63e9407180460c527e628fbc3ee0)) by [@welpo](https://github.com/welpo)\n- Alt text for moon/sun icons (theme switcher) ([4edbcf0](https://github.com/welpo/tabi/commit/4edbcf08b3cc6869aed6b1fcea5b979ca1473dda)) by [@welpo](https://github.com/welpo)\n- Trailing space in front of nav-links ([dcaa705](https://github.com/welpo/tabi/commit/dcaa705b5636dc17d6aa28edde6c3c5ac4a2e720)) by [@serginogal](https://github.com/serginogal)\n- Nav-links hover style ([b044e30](https://github.com/welpo/tabi/commit/b044e30b1ae907b7f6283dde51834bd52e8e6f4a)) by [@serginogal](https://github.com/serginogal)\n- Syntax highlighting ([64ed828](https://github.com/welpo/tabi/commit/64ed8287ff89ede1ff437a149cf5230a79a2b26e)) by [@serginogal](https://github.com/serginogal)\n\n### 💄 Styling\n\n- *(404)* Split error statement from solution ([#213](https://github.com/welpo/tabi/issues/213)) by [@ZzMzaw](https://github.com/ZzMzaw)\n- *(code blocks)* Harmonise highlight colour ([327545f](https://github.com/welpo/tabi/commit/327545f2d53899673d2446e4ee3bec370e7e584a)) by [@welpo](https://github.com/welpo)\n- *(code blocks)* Change syntax highlighting theme & text size ([#131](https://github.com/welpo/tabi/issues/131)) by [@welpo](https://github.com/welpo)\n- *(code tags)* Set border radius to inline code ([0a61bd3](https://github.com/welpo/tabi/commit/0a61bd340597b93a5ce26ed6852ca17e9fa99f06)) by [@welpo](https://github.com/welpo)\n- *(codeblocks)* Unhide scrollbars ([ce7ac7c](https://github.com/welpo/tabi/commit/ce7ac7cf2c540d61a683c78c12427bc32fc10402)) by [@welpo](https://github.com/welpo)\n- *(favicon)* Reduce emoji size for compatibility ([08ef628](https://github.com/welpo/tabi/commit/08ef628abe9ab21b58811cc74364bc088ec4d762)) by [@welpo](https://github.com/welpo)\n- *(footer)* Set paragraph margin to 0 ([c7bd9b3](https://github.com/welpo/tabi/commit/c7bd9b36ce6e8db1f23eee4ce50b03090c09a0cd)) by [@welpo](https://github.com/welpo)\n- *(isso)* Align buttons and input & cleanup ([9f74868](https://github.com/welpo/tabi/commit/9f74868c729bbd9430ce8b0e830159de31bb7ff2)) by [@welpo](https://github.com/welpo)\n- *(isso)* Clean isso styles & use CSS variables ([85fe38f](https://github.com/welpo/tabi/commit/85fe38f8154e635f33fcc0f4e7d7ba5f847695e0)) by [@welpo](https://github.com/welpo)\n- *(language-switcher)* Update icon to be lighter ([#173](https://github.com/welpo/tabi/issues/173)) by [@Jieiku](https://github.com/Jieiku) and [@welpo](https://github.com/welpo)\n- *(meta)* Add \"…\" when using summary with `<!-- more -->` ([0c87f65](https://github.com/welpo/tabi/commit/0c87f6513bca41acba6e3cd5c55ab31e8a30ec28)) by [@welpo](https://github.com/welpo)\n- *(projects)* Add border-radius and cleaner shadow ([e78451d](https://github.com/welpo/tabi/commit/e78451dbba410323ad2ee6b070bb269828fcc4fc)) by [@welpo](https://github.com/welpo)\n- *(projects)* Cleaner shadow ([8b091ba](https://github.com/welpo/tabi/commit/8b091ba7e9d9d30347b4ce98adf626154f9c4771)) by [@welpo](https://github.com/welpo)\n- *(quick-navigation)* Reduce max-height for ToC ([718f477](https://github.com/welpo/tabi/commit/718f477949e20204cb1bc0158f4ebd2c6c38a179)) by [@welpo](https://github.com/welpo)\n- *(sans-serif)* Slightly increase bold weight ([355162e](https://github.com/welpo/tabi/commit/355162ebe1d470373bcc40ea0f18c00cfdfb52d8)) by [@welpo](https://github.com/welpo)\n- Hide navigation buttons on print view ([e20e1c4](https://github.com/welpo/tabi/commit/e20e1c467cf90fd4acf71736df34c8181f2a7ff5)) by [@welpo](https://github.com/welpo)\n- Fix whitespace issues ([5458b1c](https://github.com/welpo/tabi/commit/5458b1c533e6a2c66537c439c8bdd0319b8a7352)) by [@welpo](https://github.com/welpo)\n- Reduce bold text font weight on sans-serif ([bcc3f83](https://github.com/welpo/tabi/commit/bcc3f83d858bda53b95c53393de0339fb527c94c)) by [@welpo](https://github.com/welpo)\n- Add \"…\" when using summary with `<!-- more -->` ([e72a8da](https://github.com/welpo/tabi/commit/e72a8da596ad7f44cee59141dc9214736a165bb7)) by [@welpo](https://github.com/welpo)\n- Use divider-color for <hr> & remove border-color ([1adf242](https://github.com/welpo/tabi/commit/1adf2423f30cf9fdcf3f3da354623d8ff7b0d662)) by [@welpo](https://github.com/welpo)\n- Stylise scrollbar based on site's color scheme ([1be3fa8](https://github.com/welpo/tabi/commit/1be3fa886cb0bdbe2d8f893b16e8b3542fe8bd4b)) by [@welpo](https://github.com/welpo)\n- Standardise posts' summary & description ([f03a9b3](https://github.com/welpo/tabi/commit/f03a9b3c4f734ac67f0356c89ac49ef121a55d1a)) by [@welpo](https://github.com/welpo)\n- Set transparent bg for code in table headers ([a6046cd](https://github.com/welpo/tabi/commit/a6046cd9d7b2ee72dd91064d9854e7f903d1b197)) by [@welpo](https://github.com/welpo)\n- Adjust font styles for <details> and <summary> ([275950f](https://github.com/welpo/tabi/commit/275950f97db8809fa37cda424b3aff4d80971073)) by [@welpo](https://github.com/welpo)\n- Keep top/bottom element's margin in `full-width` ([2ebf72c](https://github.com/welpo/tabi/commit/2ebf72c83f62a391bc42ac4ff7796bae74b06a55)) by [@welpo](https://github.com/welpo)\n- Retain element's spacing within `full-width` ([b1b98eb](https://github.com/welpo/tabi/commit/b1b98eb2b13ed2334bfd975b06b6f07f36796bbf)) by [@welpo](https://github.com/welpo)\n- Reset margin & padding for children of `.full-width` ([4b38d53](https://github.com/welpo/tabi/commit/4b38d536933c288fcfab2003ff555d679f508b7a)) by [@welpo](https://github.com/welpo)\n- Use theme variable for table borders ([fffaece](https://github.com/welpo/tabi/commit/fffaece88497c84b989618ca0d1329edb8333e8f)) by [@welpo](https://github.com/welpo)\n- Remove dark primary colour variable ([c53ecdc](https://github.com/welpo/tabi/commit/c53ecdc472f6fdeba7c920a0d46ac054f190bcd0)) by [@welpo](https://github.com/welpo)\n- Redesign block quotes ([e3ee95d](https://github.com/welpo/tabi/commit/e3ee95d71872cf23f0cf9c44b42f473c04a9673a)) by [@welpo](https://github.com/welpo)\n- Improve footnotes and `<hr>` styling ([a1ceb74](https://github.com/welpo/tabi/commit/a1ceb747857365f5785a079efdf3d5f0c72a3c98)) by [@welpo](https://github.com/welpo)\n- Center all table text ([85ccc08](https://github.com/welpo/tabi/commit/85ccc081e4865487fa1f13459a991cb46f0fb017)) by [@welpo](https://github.com/welpo)\n- Remove figure & img max height ([3dc3e49](https://github.com/welpo/tabi/commit/3dc3e491319cfc9f08bed691e6b744a7a597b342)) by [@welpo](https://github.com/welpo)\n- Increase body left/right margins ([b76f9bb](https://github.com/welpo/tabi/commit/b76f9bbf44228c046e0221580033a8c388fa4a07)) by [@welpo](https://github.com/welpo)\n- Hide horizontal scrollbar in codeblocks ([d7b0191](https://github.com/welpo/tabi/commit/d7b01914f843b5904b06572d7921f8ebaebd77d9)) by [@welpo](https://github.com/welpo)\n- Darken codeblock highlight colour ([fcc2caf](https://github.com/welpo/tabi/commit/fcc2cafcdcfd5eb14df2208829c33404feb9ae7a)) by [@welpo](https://github.com/welpo)\n- Reduce bold font-weight ([f5603ac](https://github.com/welpo/tabi/commit/f5603acbdb8968126deb3818af2cef018ed79b75)) by [@welpo](https://github.com/welpo)\n- Add codeblock background colours ([8415535](https://github.com/welpo/tabi/commit/8415535e1337a00c9b23e524ade3734d94fc0965)) by [@welpo](https://github.com/welpo)\n- Round tables borders ([7c3463c](https://github.com/welpo/tabi/commit/7c3463c0fe1539a56202f1b36da951156a931552)) by [@welpo](https://github.com/welpo)\n- Redesign code block header ([700037a](https://github.com/welpo/tabi/commit/700037afe5e3d7e74faef4776ef158f3af34fc02)) by [@welpo](https://github.com/welpo)\n- Increase spacing in archive ([d1e51f5](https://github.com/welpo/tabi/commit/d1e51f5866086426890c687c6135a3bf1d7bbcde)) by [@welpo](https://github.com/welpo)\n- Remove newlines around summary ([67173f2](https://github.com/welpo/tabi/commit/67173f27cdfe440126a47a4fced93e01f92d53c3)) by [@welpo](https://github.com/welpo)\n- Smaller title and \"Recent Posts\" text ([eb8f630](https://github.com/welpo/tabi/commit/eb8f630fde1fecc04c067832283f7a16c945184a)) by [@welpo](https://github.com/welpo)\n- Add styling to infobox ([0494a54](https://github.com/welpo/tabi/commit/0494a54f574b7983d4fe4bef073f730bb899787e)) by [@welpo](https://github.com/welpo)\n- Add title suffix & reduce RSS icon size ([fc591f5](https://github.com/welpo/tabi/commit/fc591f549c9f5ebbee20f9c6273da27bed603fd2)) by [@welpo](https://github.com/welpo)\n- Add `padding-top` class ([36930dc](https://github.com/welpo/tabi/commit/36930dc8174966e2dd718f039307f3412454d486)) by [@welpo](https://github.com/welpo)\n- Rename \"atom/rss feed\" to \"feed\" ([0b02e72](https://github.com/welpo/tabi/commit/0b02e727e0ed21c1e1a8ac9ee45229a78339d361)) by [@welpo](https://github.com/welpo)\n- Standarise margin in projects and archive ([da38822](https://github.com/welpo/tabi/commit/da3882294c6b6a04e8d2f8f91c6412ead4496d3d)) by [@welpo](https://github.com/welpo)\n- Set custom header anchor ([5bdb804](https://github.com/welpo/tabi/commit/5bdb8045a6a517cd1ce5be945d555faea0b84922)) by [@welpo](https://github.com/welpo)\n- Add dash to toggle identifier ([c47198f](https://github.com/welpo/tabi/commit/c47198facf72abf7c2a1b84ac4a9a8adaa4c1272)) by [@welpo](https://github.com/welpo)\n- Widen article body; less reading saccades ([cfef1f6](https://github.com/welpo/tabi/commit/cfef1f65d79c41ba4be26cbd8683be53c874ff6f)) by [@welpo](https://github.com/welpo)\n- Center wrapped menu items ([5d91a0d](https://github.com/welpo/tabi/commit/5d91a0dcacd0eca6f7ce6d24e421f9c40b85bdf2)) by [@welpo](https://github.com/welpo)\n- Widen article body; less reading saccades ([00c7d15](https://github.com/welpo/tabi/commit/00c7d153c61b18c6d3a11b6f8958464b44747e54)) by [@welpo](https://github.com/welpo)\n- Round down code font-size ([4af487e](https://github.com/welpo/tabi/commit/4af487ecedb71f98cb0ec1e2b38c11def7029973)) by [@welpo](https://github.com/welpo)\n- Standarise font rendering across browsers ([67aad3e](https://github.com/welpo/tabi/commit/67aad3ef74472b40f878f7475a782e411cb75fdf)) by [@welpo](https://github.com/welpo)\n- Reduce header-content margin in mobile ([3275e8a](https://github.com/welpo/tabi/commit/3275e8a5092cfd46b424f6e071c4d5412d66ff34)) by [@welpo](https://github.com/welpo)\n- Increase margin between posts and year ([69989cc](https://github.com/welpo/tabi/commit/69989ccc94e656ee78bfdabf88995408b5f2c765)) by [@welpo](https://github.com/welpo)\n- Move padding from post-time to listing-item ([5907091](https://github.com/welpo/tabi/commit/590709189331d264cd5fd470500723c5ab2b3473)) by [@welpo](https://github.com/welpo)\n- Restore meta font in article to sans-serif ([25d3a4d](https://github.com/welpo/tabi/commit/25d3a4d109a698884ac1c6da081f852f7293fda2)) by [@welpo](https://github.com/welpo)\n- Improve mobile date display and title spacing ([6a5670d](https://github.com/welpo/tabi/commit/6a5670d5a0b86c339a68f6bc744769c96e5081b0)) by [@welpo](https://github.com/welpo)\n- Standarise font rendering across browsers ([ea00c9b](https://github.com/welpo/tabi/commit/ea00c9b9df6d8b7546900585e65acd57d955b842)) by [@welpo](https://github.com/welpo)\n- Prevent header anchor selection ([1499704](https://github.com/welpo/tabi/commit/14997045d801d89706c1c76345d963768fcdb24b)) by [@welpo](https://github.com/welpo)\n- Update project images ([18cab7d](https://github.com/welpo/tabi/commit/18cab7d37c49e9e4fa67f73210e4055aa8a2feca)) by [@welpo](https://github.com/welpo)\n- Update project images ([77dc2be](https://github.com/welpo/tabi/commit/77dc2be14bf8cd05079bc357b6f255ea2f7fcf31)) by [@welpo](https://github.com/welpo)\n- Greater resolution ([698d4eb](https://github.com/welpo/tabi/commit/698d4ebf7430e766119022ecad92851fe9f933a6)) by [@welpo](https://github.com/welpo)\n- Add link to project images ([369fa97](https://github.com/welpo/tabi/commit/369fa9700abcd24c2a644be471b8ab112e66f3e7)) by [@welpo](https://github.com/welpo)\n- Use shadow instead of border for cards ([99bb932](https://github.com/welpo/tabi/commit/99bb93219d7b28dc14c1e01be8695dbf1ead5446)) by [@welpo](https://github.com/welpo)\n- Set a minimum value for paragraph spacing ([a89fc02](https://github.com/welpo/tabi/commit/a89fc02a96e911135f4e4b60c2e8bd252c317a8c)) by [@welpo](https://github.com/welpo)\n- Use theme's primary colour for table headers ([1c07128](https://github.com/welpo/tabi/commit/1c07128aeb30695239611ed5b11695eb84a000b9)) by [@welpo](https://github.com/welpo)\n- Narrower article content for blog posts ([a39306f](https://github.com/welpo/tabi/commit/a39306f6ef98ae8b1c3358535c6fd845ec02d6db)) by [@welpo](https://github.com/welpo)\n- Change default long date format ([61e288d](https://github.com/welpo/tabi/commit/61e288d8d5df2d4c87dbf239af2dddc87e9e1dcf)) by [@welpo](https://github.com/welpo)\n- Reduce text/background contrast ([15f2629](https://github.com/welpo/tabi/commit/15f26293b54757cd06f619afba8dc9cbbea2e835)) by [@welpo](https://github.com/welpo)\n- Darken background for inline code (dark theme) ([7fbb4d7](https://github.com/welpo/tabi/commit/7fbb4d7acdf63e6e3535ced090441d54e6a15058)) by [@welpo](https://github.com/welpo)\n- Update anchor links by [@welpo](https://github.com/welpo)\n- Hide anchor links on small screens ([15c1280](https://github.com/welpo/tabi/commit/15c12807063d84be34bc28bd10be00421356fa69)) by [@welpo](https://github.com/welpo)\n- Promise I will never go blonde like Kanye ([38a38fa](https://github.com/welpo/tabi/commit/38a38fa6063ac92a559d8e36d138deae165d5e8b)) by [@welpo](https://github.com/welpo)\n- Increase content bottom margin in small screens ([845a2f1](https://github.com/welpo/tabi/commit/845a2f151a03424c210ed864bd19c8c31e39192c)) by [@welpo](https://github.com/welpo)\n- Smaller socials; add Zola & tabi mention by [@welpo](https://github.com/welpo)\n- Smaller socials; add Zola & tabi mention ([b7c8975](https://github.com/welpo/tabi/commit/b7c89753ecfef9e611164148b8a297a364214ade)) by [@welpo](https://github.com/welpo)\n- Remove round borders on tables for consistency ([0f07a6e](https://github.com/welpo/tabi/commit/0f07a6e98b4b3fc7c0561f67c27cee3871d9346f)) by [@welpo](https://github.com/welpo)\n- Fix invisible url on hover in projects ([77298d8](https://github.com/welpo/tabi/commit/77298d82b08f3567391e9d02682e4750aea6f843)) by [@welpo](https://github.com/welpo)\n- Use day-month format ([28d5704](https://github.com/welpo/tabi/commit/28d5704afd71c2e58df2cac5a303aa1c3923970b)) by [@welpo](https://github.com/welpo)\n- Increase contrast for link hover ([e51f531](https://github.com/welpo/tabi/commit/e51f5318d691182b1f7a626683d18f841d96b310)) by [@welpo](https://github.com/welpo)\n- Decrease margin between date and post title ([283f3d3](https://github.com/welpo/tabi/commit/283f3d39344b3b02d35df55739a41000984e8fdb)) by [@welpo](https://github.com/welpo)\n- Wrap post description in `<p>` ([dc64e25](https://github.com/welpo/tabi/commit/dc64e2589dc4c7fb4eaad4fb8eca43e49209c224)) by [@welpo](https://github.com/welpo)\n- Major redesign by [@welpo](https://github.com/welpo)\n- Increase margin between anchor and header ([1f4f3a9](https://github.com/welpo/tabi/commit/1f4f3a906fc4fe4de3a53a4be0870cfeaa2d1e2f)) by [@welpo](https://github.com/welpo)\n- Redesign post listing and other minor changes ([cd2ab35](https://github.com/welpo/tabi/commit/cd2ab356bef6817122aa6fd390e1735106a99458)) by [@welpo](https://github.com/welpo)\n- Remove transitions ([cec746c](https://github.com/welpo/tabi/commit/cec746c1ce58fd92275103534287e81648f1b9c3)) by [@welpo](https://github.com/welpo)\n- Redesign tag pages ([8ac1894](https://github.com/welpo/tabi/commit/8ac1894a66c52e370900b2d3f1a7de43207a38de)) by [@welpo](https://github.com/welpo)\n- Increase date opacity ([f5834a2](https://github.com/welpo/tabi/commit/f5834a2e39f8f18a90ac6fe99245b34fe4bad0d5)) by [@welpo](https://github.com/welpo)\n- Remove \"home\" from navigation ([80ec217](https://github.com/welpo/tabi/commit/80ec217607efeaebe892ae53286afeb8c0fdd8ae)) by [@welpo](https://github.com/welpo)\n- List tags redesign by [@welpo](https://github.com/welpo)\n- Use cards/boxes for tag list ([b33a942](https://github.com/welpo/tabi/commit/b33a942766ce80e15f0a0b457d5562fb084563f2)) by [@welpo](https://github.com/welpo)\n- List all posts per tag ([a761fd0](https://github.com/welpo/tabi/commit/a761fd00eebf03eaefe5647701208648e2973aed)) by [@welpo](https://github.com/welpo)\n- Set style for figure & figure captions ([1496f0c](https://github.com/welpo/tabi/commit/1496f0cf73c9a7fcdf13bb752101278dd3660658)) by [@welpo](https://github.com/welpo)\n- Increase iframe margins ([c2d9107](https://github.com/welpo/tabi/commit/c2d9107cd39c19be544b76c272e7c1e7c9bc581c)) by [@welpo](https://github.com/welpo)\n- Center and enlarge iframes (embeds) ([372d696](https://github.com/welpo/tabi/commit/372d696eec07d9171a116a85d5e46ec3c7218a05)) by [@welpo](https://github.com/welpo)\n- Update footnote styling ([f4cd6e6](https://github.com/welpo/tabi/commit/f4cd6e6a92eb4b7b5076fd2de4694316d1d328e2)) by [@welpo](https://github.com/welpo)\n- Resize headers ([e4c6705](https://github.com/welpo/tabi/commit/e4c670584388c1d33e250039abafad79c09c43fe)) by [@welpo](https://github.com/welpo)\n- Increase margins before content & after ToC ([83b7bb9](https://github.com/welpo/tabi/commit/83b7bb9d3a3afefab47dc5055a4167c0133f1b1d)) by [@welpo](https://github.com/welpo)\n- Reduce font size for meta in cards ([0ecc656](https://github.com/welpo/tabi/commit/0ecc656b91fd3f7ce55fbaa13f65aaf610117b32)) by [@welpo](https://github.com/welpo)\n- Reduce font-size and increase line-height in article-title ([367d6a8](https://github.com/welpo/tabi/commit/367d6a8be4b3b37a7c3a2f5b733417a55af6ec01)) by [@welpo](https://github.com/welpo)\n- Revert background back to pure white ([0e4d274](https://github.com/welpo/tabi/commit/0e4d274de9300a104de3cb4c0dfbab31c1b206a0)) by [@welpo](https://github.com/welpo)\n- Change colourscheme. improve cohesion and accesibility ([b7da112](https://github.com/welpo/tabi/commit/b7da11222e2899c85e3def883ddec58ec8b479f9)) by [@welpo](https://github.com/welpo)\n- Mention all table colours change, not just header ([2b861ea](https://github.com/welpo/tabi/commit/2b861ea94068a553ebd849b5fd407b5ccefb4464)) by [@welpo](https://github.com/welpo)\n- Showcase tables & invertable images ([394479d](https://github.com/welpo/tabi/commit/394479d777bcf0c03052521fbe4c9fafe437cf6c)) by [@welpo](https://github.com/welpo)\n- Center tables; colour headers; round borders ([5b45351](https://github.com/welpo/tabi/commit/5b4535197014bd3dc87eb6f59ce88b963efed037)) by [@welpo](https://github.com/welpo)\n- Set tab to 4 spaces; remove newlines ([f2cd044](https://github.com/welpo/tabi/commit/f2cd0440962d97d4c6f27cd37d50200627e81d42)) by [@welpo](https://github.com/welpo)\n- Use Cascadia Code as code font ([9db00d1](https://github.com/welpo/tabi/commit/9db00d1f9d382c5c6d2f5383ab8c145d3b522696)) by [@welpo](https://github.com/welpo)\n- Move quote author to quote block ([9f41220](https://github.com/welpo/tabi/commit/9f412205b38de9cf42cc6a1fb3baea8d3de9b2ae)) by [@welpo](https://github.com/welpo)\n- Default light mode ([723b3b6](https://github.com/welpo/tabi/commit/723b3b61395ec387b422475334f98ae79e2f1474)) by [@welpo](https://github.com/welpo)\n- Move main image to static path ([6783ff6](https://github.com/welpo/tabi/commit/6783ff6e49157b7149f54d8d18b962c2ecdd5ee9)) by [@welpo](https://github.com/welpo)\n- Remove border from images ([2b8a053](https://github.com/welpo/tabi/commit/2b8a053f510f36958edaf8a39d9d8dde51274463)) by [@welpo](https://github.com/welpo)\n- Default light mode ([722d493](https://github.com/welpo/tabi/commit/722d493a9a0c47b9c791e33226c8bf1d3a354f8d)) by [@welpo](https://github.com/welpo)\n- Increase contrast and saturation ([fd7a9da](https://github.com/welpo/tabi/commit/fd7a9da61cb10827be3d0421987e2be338f0e600)) by [@welpo](https://github.com/welpo)\n- Smaller home image; different circle crop ([c71065d](https://github.com/welpo/tabi/commit/c71065d75691a832cd9a60b8872513b1f1e61715)) by [@welpo](https://github.com/welpo)\n- Set local main image; no border-radius ([e2edab5](https://github.com/welpo/tabi/commit/e2edab51b798b7100ed46511217f7aa27d5e7a45)) by [@welpo](https://github.com/welpo)\n- Reduce headers' size ([978af11](https://github.com/welpo/tabi/commit/978af11495face1127dc30946989ecac78257207)) by [@welpo](https://github.com/welpo)\n- Change posts' dates to show off Archive ([fcc7a31](https://github.com/welpo/tabi/commit/fcc7a3127030ebcab12524647886530aacf3dbac)) by [@welpo](https://github.com/welpo)\n- Larger section titles ([4fe8643](https://github.com/welpo/tabi/commit/4fe8643a1a54837fd0aff85f0092ea6b4c1bd430)) by [@welpo](https://github.com/welpo)\n- Add styles for archive screen ([19caf3f](https://github.com/welpo/tabi/commit/19caf3fab117121e6d7e168cb4f189e44ae1f038)) by [@serginogal](https://github.com/serginogal)\n- Change switcher margin and width/height to rem ([9637e05](https://github.com/welpo/tabi/commit/9637e053c3779db7f24ea66d2738892bdd00d012)) by [@welpo](https://github.com/welpo)\n- Increase padding between navs ([973e0f7](https://github.com/welpo/tabi/commit/973e0f76f8ae0bb23a40e8099f72efd1e1c16115)) by [@welpo](https://github.com/welpo)\n- Set consistent underline in headers ([fa781db](https://github.com/welpo/tabi/commit/fa781db0e4f10d05117f35218b74d36b4605187f)) by [@welpo](https://github.com/welpo)\n- Set equal padding on social icons ([b8e372c](https://github.com/welpo/tabi/commit/b8e372c733a1052f41b90c9a115ba5bc44f3b8b8)) by [@welpo](https://github.com/welpo)\n- Increase padding between navs ([fcb8db4](https://github.com/welpo/tabi/commit/fcb8db43a60fd485927211e7c747186db5a7324d)) by [@welpo](https://github.com/welpo)\n- Change various font sizes and date format ([06dae7a](https://github.com/welpo/tabi/commit/06dae7a71172be63fe93a65319827e9e687c6a63)) by [@welpo](https://github.com/welpo)\n- Update screenshots and minor changes ([a3ff46b](https://github.com/welpo/tabi/commit/a3ff46be97d2792cfddb826911c97257ec7f3789)) by [@welpo](https://github.com/welpo)\n\n### 📝 Documentation\n\n- *(README)* Add 'Sites Using tabi' section ([0d34b98](https://github.com/welpo/tabi/commit/0d34b98e1183c1317f289cf9d976e20a5e895151)) by [@welpo](https://github.com/welpo)\n- *(README)* Add link to Table of Contents documentation ([#121](https://github.com/welpo/tabi/issues/121)) by [@stevenroose](https://github.com/stevenroose)\n- *(ToC)* Fix broken code block ([3f2e456](https://github.com/welpo/tabi/commit/3f2e45601aa4154f5d45119017b4f53cec3e0c98)) by [@welpo](https://github.com/welpo)\n- *(config)* Clarify default short date format ([f57f313](https://github.com/welpo/tabi/commit/f57f3138c26a8187447f9bcbe9af6aa0861eb8e1)) by [@welpo](https://github.com/welpo)\n- *(faq-languages)* Improve phrasing/translation ([793b063](https://github.com/welpo/tabi/commit/793b063c773addae95617e4233d5287d8ec2337c)) by [@welpo](https://github.com/welpo)\n- *(i18n)* Fix language switcher description ([a7d4120](https://github.com/welpo/tabi/commit/a7d41203393817946d721f86d2eae4ee47e1c8b8)) by [@welpo](https://github.com/welpo)\n- *(i18n)* Mention IETF BCP 47 codes ([78fd8b8](https://github.com/welpo/tabi/commit/78fd8b8f0e1e643bc956956b04d6c2dc1b857422)) by [@welpo](https://github.com/welpo)\n- *(pre-commit)* Remove redundancy in minified error ([2833d9f](https://github.com/welpo/tabi/commit/2833d9f8c045205cf4f57499f132675433eb28d0)) by [@welpo](https://github.com/welpo)\n- *(privacy)* Use GoatCounter sessions ([741a7e7](https://github.com/welpo/tabi/commit/741a7e7c6f98d2610ab8d58da4ff6c724cfd547a)) by [@welpo](https://github.com/welpo)\n- *(projects)* Remove bullet points from tabi features ([3590468](https://github.com/welpo/tabi/commit/35904683e16f303a1c3c5523688957e433b8a88e)) by [@welpo](https://github.com/welpo)\n- *(shortcodes)* Link commit and PR for relative image paths ([c2664c8](https://github.com/welpo/tabi/commit/c2664c85cd788791f1a42aab1e0a9e2cb892225c)) by [@welpo](https://github.com/welpo)\n- Add guide to all of tabi's features ([#169](https://github.com/welpo/tabi/issues/169)) by [@welpo](https://github.com/welpo)\n- Update link to multilanguage lines in config ([0cdb83e](https://github.com/welpo/tabi/commit/0cdb83e3b2fa20a498cf52040a72d50bc03b80e9)) by [@welpo](https://github.com/welpo)\n- Update link to multilanguage lines in config ([a61911d](https://github.com/welpo/tabi/commit/a61911d7df1daf65da4510479ad64d41f7180cbb)) by [@welpo](https://github.com/welpo)\n- Add ToC and quick navigation buttons ([a47a8be](https://github.com/welpo/tabi/commit/a47a8be01169914f7467f100c6c4136fd022e8eb)) by [@welpo](https://github.com/welpo)\n- Clarify the 2 possible paths for custom skins ([41009a5](https://github.com/welpo/tabi/commit/41009a50a1f45a5ad16b044fec4e0a59393650d9)) by [@welpo](https://github.com/welpo)\n- Add link to skins showcase/documentation ([6b0cc11](https://github.com/welpo/tabi/commit/6b0cc11da4a684f21211f5e19b7fdae19aae76a1)) by [@welpo](https://github.com/welpo)\n- Fix skins docs link ([e210652](https://github.com/welpo/tabi/commit/e210652ac2ddad173cb3e51d4423da808bd6dc50)) by [@welpo](https://github.com/welpo)\n- Improve Code tags example ([521a05b](https://github.com/welpo/tabi/commit/521a05b97f6a2c2f3dfacdd63b0a1790cfbb558e)) by [@welpo](https://github.com/welpo)\n- Add Isso comments mention ([6b00aae](https://github.com/welpo/tabi/commit/6b00aae20ecc1b5359ee8ed283a83604e3a4abcd)) by [@welpo](https://github.com/welpo)\n- Update [extra] settings ([da44e79](https://github.com/welpo/tabi/commit/da44e795d698899b7d577f89f3c9e3ad0f7cc4e6)) by [@welpo](https://github.com/welpo)\n- Update link to multilanguage lines in config ([ae47f79](https://github.com/welpo/tabi/commit/ae47f79d9d34d744600d4abafbc073fb44021d49)) by [@welpo](https://github.com/welpo)\n- Suggest editing language feed stylesheet ([588defd](https://github.com/welpo/tabi/commit/588defd7f221e8914ea8c57a66ef7509200302fa)) by [@welpo](https://github.com/welpo)\n- Mention support for Isso comments ([a201c05](https://github.com/welpo/tabi/commit/a201c05823d1637e68ede5561ce9e468a2317fe7)) by [@welpo](https://github.com/welpo)\n- Add code type to codeblock ([b6f7f80](https://github.com/welpo/tabi/commit/b6f7f802b49734551d2f9afad8274debde56d1d0)) by [@welpo](https://github.com/welpo)\n- Add Hyvor Talk comments mention ([f859113](https://github.com/welpo/tabi/commit/f859113c815ada8e33b62c52d17ecdac9bd4ea73)) by [@welpo](https://github.com/welpo)\n- Support Hyvor Talk comments ([3dcc462](https://github.com/welpo/tabi/commit/3dcc462056401492d676d3893cfeed68c989f27d)) by [@welpo](https://github.com/welpo)\n- Remove `updated` date from examples ([5d99da2](https://github.com/welpo/tabi/commit/5d99da2cd0f20bd78732724026c926e36b509b45)) by [@welpo](https://github.com/welpo)\n- Update for Hyvor Talk support ([31fee59](https://github.com/welpo/tabi/commit/31fee59eb21ab1ca4d3806da80c3b36338fbf706)) by [@welpo](https://github.com/welpo)\n- Remove first header ([5214dd3](https://github.com/welpo/tabi/commit/5214dd35608e7935a399303aff87f520574b30a9)) by [@welpo](https://github.com/welpo)\n- Update link to multilanguage lines in config ([54abe36](https://github.com/welpo/tabi/commit/54abe36d442bc73c3a0e62fc0748570deabec6d7)) by [@welpo](https://github.com/welpo)\n- Fix mention of \"Load comments\" button ([1a74cb4](https://github.com/welpo/tabi/commit/1a74cb4496ffdced2b032c36704a26828748ee65)) by [@welpo](https://github.com/welpo)\n- Add code language to bash code blocks ([33cfe77](https://github.com/welpo/tabi/commit/33cfe77344c62f9e7368cfa914da4f8ff53883a3)) by [@welpo](https://github.com/welpo)\n- Remove `updated` date from example ([dcbb4c5](https://github.com/welpo/tabi/commit/dcbb4c52907c5fff30ab97841a871d8e3bbbd426)) by [@welpo](https://github.com/welpo)\n- Improve KaTeX documentation ([1cd34b5](https://github.com/welpo/tabi/commit/1cd34b528efe8a8a1e0ad6ef1ee92d804aee75a9)) by [@welpo](https://github.com/welpo)\n- Set proper `updated` date ([139ca04](https://github.com/welpo/tabi/commit/139ca042f41f8ca96b105ba8e03da56400486895)) by [@welpo](https://github.com/welpo)\n- Translate 'Usage' header ([fa5e21f](https://github.com/welpo/tabi/commit/fa5e21feeaf40cd57f5145e2ea2273c378bf3306)) by [@welpo](https://github.com/welpo)\n- Mention stylized feed ([bd2b4a3](https://github.com/welpo/tabi/commit/bd2b4a33d65bf17dfe03e6351e1d9de75eb9f8c5)) by [@welpo](https://github.com/welpo)\n- Add default author ([f9d8f36](https://github.com/welpo/tabi/commit/f9d8f3668f0a268ea1799f43ee56a6a2176380de)) by [@welpo](https://github.com/welpo)\n- Remove atom feed filter in script ([41c44d3](https://github.com/welpo/tabi/commit/41c44d3d842a2c8d9ef5b7b2dd94c831f356ed5b)) by [@welpo](https://github.com/welpo)\n- Update URL to multilanguage lines in config ([761a18d](https://github.com/welpo/tabi/commit/761a18da752f29d42e7b75465ce53601bc2b75cf)) by [@welpo](https://github.com/welpo)\n- Remove unnecessary date ([8b7f53e](https://github.com/welpo/tabi/commit/8b7f53e3ca3f20ff838a0dce321d4913b6bd43c2)) by [@welpo](https://github.com/welpo)\n- Add feature request template ([e1ec7f5](https://github.com/welpo/tabi/commit/e1ec7f54d5e14fb75813745b3067e6de08152235)) by [@welpo](https://github.com/welpo)\n- Create bug report issue template ([03cbf3b](https://github.com/welpo/tabi/commit/03cbf3bf0f346065d0c5adafa991d216e3b55809)) by [@welpo](https://github.com/welpo)\n- Update URL to multilanguage lines in config ([3a8fa41](https://github.com/welpo/tabi/commit/3a8fa41cfc42a51bcc6a3c6b7bc688ee1814fb24)) by [@welpo](https://github.com/welpo)\n- Mention Zola version requirement ([6e9bcf9](https://github.com/welpo/tabi/commit/6e9bcf92046f553f6bebdbf7d9c5291739bb1cb5)) by [@welpo](https://github.com/welpo)\n- Update Zola version requirement to 0.17.0 ([a0f4f4a](https://github.com/welpo/tabi/commit/a0f4f4a6420927bb10119baf5327da4c9cf71187)) by [@welpo](https://github.com/welpo)\n- Add quotation marks to quotes ([c7a411e](https://github.com/welpo/tabi/commit/c7a411e1572075b63719d3eb81b4c469b2e56b01)) by [@welpo](https://github.com/welpo)\n- Add config info for multilingual sites ([397b63c](https://github.com/welpo/tabi/commit/397b63cefab3ffba87b0bf2ae5bf3c9cf641d83e)) by [@welpo](https://github.com/welpo)\n- Update screenshots ([56b74ed](https://github.com/welpo/tabi/commit/56b74ed23d1f4395c8245d4c444e53a951dad78b)) by [@welpo](https://github.com/welpo)\n- Revert script changes ([f657010](https://github.com/welpo/tabi/commit/f657010be4cdd028e3d17b9967b9b97221e8ef80)) by [@welpo](https://github.com/welpo)\n- Include layout features in font subset ([44980f1](https://github.com/welpo/tabi/commit/44980f141883a462bc5a88cf958786f8c56a98ec)) by [@welpo](https://github.com/welpo)\n- Rephrase last sentence ([e316dc2](https://github.com/welpo/tabi/commit/e316dc25ab649528be19d3f4b3ef165cbd976a2b)) by [@welpo](https://github.com/welpo)\n- Rephrase description ([47caa99](https://github.com/welpo/tabi/commit/47caa99e0c438396e1ef6d5ad5ec9c3664df078a)) by [@welpo](https://github.com/welpo)\n- Close codeblock ([3389638](https://github.com/welpo/tabi/commit/3389638b06efc6b534f7ac674764a6c110d38b18)) by [@welpo](https://github.com/welpo)\n- Update URL ([60680fa](https://github.com/welpo/tabi/commit/60680fad3741735a75d66c185639ab1c057c6560)) by [@welpo](https://github.com/welpo)\n- Update inspiration ([85a32e2](https://github.com/welpo/tabi/commit/85a32e263fa9ec324275c15bbc96398bb88c2fd0)) by [@welpo](https://github.com/welpo)\n- Use full URLs for images for Zola theme page ([bc84a1f](https://github.com/welpo/tabi/commit/bc84a1f669d42b161dc07bba3424151cc17347b4)) by [@welpo](https://github.com/welpo)\n- Add `data:` to CSP to load local svg ([3b22e6b](https://github.com/welpo/tabi/commit/3b22e6be6c1ff666c767cea0af4b9cd48f353c46)) by [@welpo](https://github.com/welpo)\n- Stylise KaTeX ([79f8559](https://github.com/welpo/tabi/commit/79f8559a3aa87794af45a62d9644689c00b94c51)) by [@welpo](https://github.com/welpo)\n- Acknowledge abridge's inspiration ([c36b66b](https://github.com/welpo/tabi/commit/c36b66b7d46dc30c2dd77386421a396dd777333c)) by [@welpo](https://github.com/welpo)\n- Update theme's description ([89bfdbe](https://github.com/welpo/tabi/commit/89bfdbe6870fc05a5d8b8256d8cfb5a7b9ac8969)) by [@welpo](https://github.com/welpo)\n- Update screenshots ([3ad76de](https://github.com/welpo/tabi/commit/3ad76de2fbae694522899cf9226a8f39208e83e6)) by [@welpo](https://github.com/welpo)\n- Mention responsive design ([be8ec45](https://github.com/welpo/tabi/commit/be8ec45c359f6029fe1f2360f3c026f57345044d)) by [@welpo](https://github.com/welpo)\n- Lowercase 'tabi' ([e422c40](https://github.com/welpo/tabi/commit/e422c405cb7bcf61c148a4b63cd0fd3537e22868)) by [@welpo](https://github.com/welpo)\n- Minor changes to installation instructions ([73811bd](https://github.com/welpo/tabi/commit/73811bdb7aa251620850537fc8126d8a483462e9)) by [@welpo](https://github.com/welpo)\n- Clarify installation instructions ([538cb07](https://github.com/welpo/tabi/commit/538cb07a5f79c87d3e623b35475cb8ac0d0f7cc9)) by [@welpo](https://github.com/welpo)\n- Mention perfect Lighthouse score ([7543f03](https://github.com/welpo/tabi/commit/7543f03e1291a03723c3ac28e71016fbffa32c5e)) by [@welpo](https://github.com/welpo)\n- Update `nani` image ([35cea72](https://github.com/welpo/tabi/commit/35cea7222c0fdb504a8fecf0b33c126f12f8b2a9)) by [@welpo](https://github.com/welpo)\n- Update projects ([a2ea8c3](https://github.com/welpo/tabi/commit/a2ea8c3f885ddce0bf2facbd8c166e1c21577af8)) by [@welpo](https://github.com/welpo)\n- Update description ([b5bc0d4](https://github.com/welpo/tabi/commit/b5bc0d44db80f8afe78bbd135a702933e6a9a0e1)) by [@welpo](https://github.com/welpo)\n- Update description, demo URL, author info, and extra ([3ce928a](https://github.com/welpo/tabi/commit/3ce928a067c211a4da9fd36a523dc0613e388984)) by [@welpo](https://github.com/welpo)\n- Update screenshot ([d98c0d0](https://github.com/welpo/tabi/commit/d98c0d0bb8bbe8f26e2f35a33569c8babb5f5d59)) by [@welpo](https://github.com/welpo)\n- Replace \"last lines\" to \"last directive\" ([2a681f0](https://github.com/welpo/tabi/commit/2a681f005f3d62a49c2bda293f7289824ad73e7d)) by [@welpo](https://github.com/welpo)\n- Use hierarchical headings ([bfda1df](https://github.com/welpo/tabi/commit/bfda1df9f3b640b61f0f0dfb16216b78effac4d6)) by [@welpo](https://github.com/welpo)\n- Update inspiration ([4ac3207](https://github.com/welpo/tabi/commit/4ac32077eb76a9ec65321351271eec52d1eec19f)) by [@welpo](https://github.com/welpo)\n- Set consistent paths for imgs ([9b680b0](https://github.com/welpo/tabi/commit/9b680b0da72b2984bff4275d2238196f3fd2cefb)) by [@welpo](https://github.com/welpo)\n- Change 'Usage' section styling ([eb60527](https://github.com/welpo/tabi/commit/eb60527bfb250d15b92b300eda86993404d247d2)) by [@welpo](https://github.com/welpo)\n- Add closing shortcode tag to references() ([2c4b644](https://github.com/welpo/tabi/commit/2c4b64481ca0ff05cac74b427ed031a6ee36496a)) by [@welpo](https://github.com/welpo)\n- Improve punctuation ([611790a](https://github.com/welpo/tabi/commit/611790ab2b2d49bf16ed192168cabb26c9d1a8a2)) by [@welpo](https://github.com/welpo)\n- Add Contributing guidelines ([5a2738f](https://github.com/welpo/tabi/commit/5a2738f66529ccb19b27de9c9914efc6a27d9e10)) by [@welpo](https://github.com/welpo)\n- Add code of conduct ([5cbc33a](https://github.com/welpo/tabi/commit/5cbc33a83812091fdd3393a365cfe3878e00eb94)) by [@welpo](https://github.com/welpo)\n- Replace `test` tag with `showcase` ([69d4985](https://github.com/welpo/tabi/commit/69d4985fc55fbdc37f92d07a71e1823cd107ebd5)) by [@welpo](https://github.com/welpo)\n- Fix typos ([73a5638](https://github.com/welpo/tabi/commit/73a563872e8fa94dfb186e459255f4e257249429)) by [@welpo](https://github.com/welpo)\n- Use markdown to showcase theme's capabilities ([2f07137](https://github.com/welpo/tabi/commit/2f07137cca2eadd73a4aaf71bd1361c07194923f)) by [@welpo](https://github.com/welpo)\n- Add internetVin's blog as inspiration ([e052371](https://github.com/welpo/tabi/commit/e05237186581b8b783dfb34620d53d02f9554754)) by [@welpo](https://github.com/welpo)\n- Js size has grown :( ([83e211a](https://github.com/welpo/tabi/commit/83e211a18d058e7a13da8b6a9fde752fcd194b52)) by [@welpo](https://github.com/welpo)\n- Change the codeblock example ([f6cd7d3](https://github.com/welpo/tabi/commit/f6cd7d3cf20ae3d2cb72cdd4eb3011ab58720ba8)) by [@welpo](https://github.com/welpo)\n- Add license ([1cb6003](https://github.com/welpo/tabi/commit/1cb60032c6d1b6b7ee048e46224c9d187c0b546f)) by [@welpo](https://github.com/welpo)\n\n### ♻️ Refactor\n\n- *(404)* Remove mention of translations for single language sites ([8ecd23b](https://github.com/welpo/tabi/commit/8ecd23b940b3e3079be9c248535b933ea6120030)) by [@welpo](https://github.com/welpo)\n- *(HTML)* Remove X-UA-Compatible ([#177](https://github.com/welpo/tabi/issues/177)) by [@donovanglover](https://github.com/donovanglover)\n- *(HTML)* Use whitespace control ([7d8ed20](https://github.com/welpo/tabi/commit/7d8ed20c2437ca285cc748359c4b2ff500a3968c)) by [@welpo](https://github.com/welpo)\n- *(giscus)* Move comment box to bottom ([3822f00](https://github.com/welpo/tabi/commit/3822f00d3e3d77c8b585b10a788986d967e2011f)) by [@welpo](https://github.com/welpo)\n- *(header)* Replace single %22 by `\"` ([a57eb98](https://github.com/welpo/tabi/commit/a57eb98a6912889e07e923f289001975d9c3d495)) by [@welpo](https://github.com/welpo)\n- *(tags)* Remove unnecessary id attribute ([b88e737](https://github.com/welpo/tabi/commit/b88e7376e4814b2d174f5ab1c8579d2a67f842df)) by [@welpo](https://github.com/welpo)\n- *(translation)* Introduce translation macro ([#129](https://github.com/welpo/tabi/issues/129)) by [@welpo](https://github.com/welpo)\n- *(utterances)* Further minify JS ([7251013](https://github.com/welpo/tabi/commit/725101300e77773362c2bec55d082798465aec0b)) by [@welpo](https://github.com/welpo)\n- Co-locate images ([#200](https://github.com/welpo/tabi/issues/200)) by [@welpo](https://github.com/welpo)\n- Turn macros into partials & sort CSS ([#146](https://github.com/welpo/tabi/issues/146)) by [@welpo](https://github.com/welpo)\n- Use `get_url` for social icons ([d0babd5](https://github.com/welpo/tabi/commit/d0babd5db1717efb3822c0741612b5100e04bca9)) by [@welpo](https://github.com/welpo)\n- Switch from `[@import](https://github.com/import)` to `[@use](https://github.com/use)` ([a799732](https://github.com/welpo/tabi/commit/a79973239892122f744d7c7e231a53abea46f780)) by [@welpo](https://github.com/welpo)\n- Further minify theme initialization ([92e6af1](https://github.com/welpo/tabi/commit/92e6af1d5865c0e3bca8bd89082a0a0bb26b5062)) by [@welpo](https://github.com/welpo)\n- Improve stylised feed translations & consolidate XSL/CSS loading ([#98](https://github.com/welpo/tabi/issues/98)) by [@welpo](https://github.com/welpo)\n- Translate xsl stylesheets ([7ed295d](https://github.com/welpo/tabi/commit/7ed295db3aa003595a14a21b6c24c03bdbfeea0f)) by [@welpo](https://github.com/welpo)\n- Combine and reorder media queries ([af81f88](https://github.com/welpo/tabi/commit/af81f88b925454291d4bc968bed30d0963410a17)) by [@welpo](https://github.com/welpo)\n- Unify table styling ([787243c](https://github.com/welpo/tabi/commit/787243c3c5f32735e0476359901b0df79d382c11)) by [@welpo](https://github.com/welpo)\n- Remove unused code ([8c5f305](https://github.com/welpo/tabi/commit/8c5f30556a9ecb412f52487eae0a9aed745b8e74)) by [@welpo](https://github.com/welpo)\n- Remove Yahoo Media RSS reference ([16f1aa5](https://github.com/welpo/tabi/commit/16f1aa57fa908a44861990cf9ff40ecc5e8ffb4e)) by [@welpo](https://github.com/welpo)\n- Restructure template ([e3bd75e](https://github.com/welpo/tabi/commit/e3bd75e9e384d03b070bb8bdab3c42aef645ad23)) by [@welpo](https://github.com/welpo)\n- Simplify theme switching logic ([526c8a1](https://github.com/welpo/tabi/commit/526c8a1e67d71e1554f8c57706ae3adcf9229b6f)) by [@welpo](https://github.com/welpo)\n- Simplify sans-serif-font declarations ([87a5c9a](https://github.com/welpo/tabi/commit/87a5c9a3a2c83d475a7e6083af8dc8fdeb37c0f8)) by [@welpo](https://github.com/welpo)\n- Rename katex minified css ([0c8e0d2](https://github.com/welpo/tabi/commit/0c8e0d228ca73dec066dca8b94741807981ee976)) by [@welpo](https://github.com/welpo)\n- Use `const` in theme initialization ([9512bbb](https://github.com/welpo/tabi/commit/9512bbb19400be6b770735ceb6daca4e71429630)) by [@welpo](https://github.com/welpo)\n- Rename minified KaTeX css ([eb63718](https://github.com/welpo/tabi/commit/eb63718bbffce75f5eab584fab37396ac93123a0)) by [@welpo](https://github.com/welpo)\n- Remove metadata in project files ([10b8525](https://github.com/welpo/tabi/commit/10b85257cd70e427a0beebadcb89f87565adbdef)) by [@welpo](https://github.com/welpo)\n- Improve code format; set tabs = 4 spaces ([7a1bb1d](https://github.com/welpo/tabi/commit/7a1bb1d04c022cbefb9ab9c71ec52772522899e9)) by [@welpo](https://github.com/welpo)\n- Move .last-updated to main.scss ([0f14171](https://github.com/welpo/tabi/commit/0f14171b6a7dfbf49d72c47280e7c5acbe3784b6)) by [@welpo](https://github.com/welpo)\n- Standarise left/right margins ([76db25b](https://github.com/welpo/tabi/commit/76db25b44aa4d33c1f8b20cbfe3387aa8a84ed81)) by [@welpo](https://github.com/welpo)\n- Remove useless css ([603ac09](https://github.com/welpo/tabi/commit/603ac09baa14b386e91047b576dcb386a53a5547)) by [@welpo](https://github.com/welpo)\n- Remove useless lines ([f83c75e](https://github.com/welpo/tabi/commit/f83c75ed044834882c06d9259bb77983eeee492c)) by [@welpo](https://github.com/welpo)\n- Remove unused `section_title` variable ([27dc4b9](https://github.com/welpo/tabi/commit/27dc4b9a9606772b22b944a16b5b8f9be7befe0d)) by [@welpo](https://github.com/welpo)\n- Improve spacing and indentation ([dceceff](https://github.com/welpo/tabi/commit/dceceff5fda7ec791bb2e0311663bb034ec0d582)) by [@welpo](https://github.com/welpo)\n- Move blog posts to `blog` section ([14156b5](https://github.com/welpo/tabi/commit/14156b590adbf274e527ecb4f9dd1be5c86a5c15)) by [@welpo](https://github.com/welpo)\n- Use consistent pagination ([a8e49cf](https://github.com/welpo/tabi/commit/a8e49cfedfeba14b702db8913b984c87ce20e595)) by [@welpo](https://github.com/welpo)\n- Remove 403 page ([6938ae4](https://github.com/welpo/tabi/commit/6938ae42f90be2faadae87cd9b463ed67243f306)) by [@welpo](https://github.com/welpo)\n- Move blog posts `blog` section ([64449d5](https://github.com/welpo/tabi/commit/64449d5d9ffe22ed7025f07912a8137b4745c900)) by [@welpo](https://github.com/welpo)\n- Rename extra.desc to extra.header ([de46a24](https://github.com/welpo/tabi/commit/de46a24d37eb47bafa379f7883b563303e67b520)) by [@welpo](https://github.com/welpo)\n- Use rem instead of px ([b1728fc](https://github.com/welpo/tabi/commit/b1728fc441e47119741acf23f1781f6734417599)) by [@welpo](https://github.com/welpo)\n- Rename 'Categories' to 'Tags' ([0648bc8](https://github.com/welpo/tabi/commit/0648bc80e22d05b2b13b8987a33c0e3d92a9695c)) by [@welpo](https://github.com/welpo)\n- Remove extra last_updated date ([bd72fb8](https://github.com/welpo/tabi/commit/bd72fb84be73af5e303889e49df332ada970d37e)) by [@welpo](https://github.com/welpo)\n- Use native `updated` field ([31ea93b](https://github.com/welpo/tabi/commit/31ea93bbbe8aba1a9fe7c3a6dc6b08b5c9cf7354)) by [@welpo](https://github.com/welpo)\n- Remove unused set_title argument ([9ac1504](https://github.com/welpo/tabi/commit/9ac150423b92784041e4a84be32b36ddbc4449e0)) by [@welpo](https://github.com/welpo)\n- Set font types as variables ([d6c198e](https://github.com/welpo/tabi/commit/d6c198e379abd84c877d194e052a72bc5ba0e071)) by [@welpo](https://github.com/welpo)\n- Show word count as reading time tooltip ([9ed4034](https://github.com/welpo/tabi/commit/9ed40342605f079afe4cf668ed4fbb7ab87c4664)) by [@welpo](https://github.com/welpo)\n- Improve table of contents accessibility ([1c93df0](https://github.com/welpo/tabi/commit/1c93df0a6fe9cc564678b5aba971f20d78c18166)) by [@welpo](https://github.com/welpo)\n- Set datetime format as a variable ([988034b](https://github.com/welpo/tabi/commit/988034ba4ebf622d3e704cfee7a2bd681abcf306)) by [@welpo](https://github.com/welpo)\n\n### 👥 New contributors\n\n🫶 [@welpo](https://github.com/welpo) made their first contribution\n\n🫶 [@ZzMzaw](https://github.com/ZzMzaw) made their first contribution in [#215](https://github.com/welpo/tabi/pull/215)\n\n🫶 [@joberthrogers18](https://github.com/joberthrogers18) made their first contribution in [#219](https://github.com/welpo/tabi/pull/219)\n\n🫶 [@xvello](https://github.com/xvello) made their first contribution in [#218](https://github.com/welpo/tabi/pull/218)\n\n🫶 [@donovanglover](https://github.com/donovanglover) made their first contribution in [#205](https://github.com/welpo/tabi/pull/205)\n\n🫶 [@nyadiia](https://github.com/nyadiia) made their first contribution in [#195](https://github.com/welpo/tabi/pull/195)\n\n🫶 [@Almost-Senseless-Coder](https://github.com/Almost-Senseless-Coder) made their first contribution in [#192](https://github.com/welpo/tabi/pull/192)\n\n🫶 [@Jieiku](https://github.com/Jieiku) made their first contribution in [#173](https://github.com/welpo/tabi/pull/173)\n\n🫶 [@sandman](https://github.com/sandman) made their first contribution in [#170](https://github.com/welpo/tabi/pull/170)\n\n🫶 [@Smtbook](https://github.com/Smtbook) made their first contribution in [#166](https://github.com/welpo/tabi/pull/166)\n\n🫶 [@SeaDve](https://github.com/SeaDve) made their first contribution in [#165](https://github.com/welpo/tabi/pull/165)\n\n🫶 [@stevenroose](https://github.com/stevenroose) made their first contribution in [#124](https://github.com/welpo/tabi/pull/124)\n\n🫶 [@serginogal](https://github.com/serginogal) made their first contribution\n\n<!-- generated by git-cliff -->\n"
  },
  {
    "path": "CODE_OF_CONDUCT.md",
    "content": "# Contributor Covenant Code of Conduct\n\n## Our Pledge\n\nWe as members, contributors, and leaders pledge to make participation in our\ncommunity a harassment-free experience for everyone, regardless of age, body\nsize, visible or invisible disability, ethnicity, sex characteristics, gender\nidentity and expression, level of experience, education, socio-economic status,\nnationality, personal appearance, race, religion, or sexual identity\nand orientation.\n\nWe pledge to act and interact in ways that contribute to an open, welcoming,\ndiverse, inclusive, and healthy community.\n\n## Our Standards\n\nExamples of behavior that contributes to a positive environment for our\ncommunity include:\n\n* Demonstrating empathy and kindness toward other people\n* Being respectful of differing opinions, viewpoints, and experiences\n* Giving and gracefully accepting constructive feedback\n* Accepting responsibility and apologizing to those affected by our mistakes,\n  and learning from the experience\n* Focusing on what is best not just for us as individuals, but for the\n  overall community\n\nExamples of unacceptable behavior include:\n\n* The use of sexualized language or imagery, and sexual attention or\n  advances of any kind\n* Trolling, insulting or derogatory comments, and personal or political attacks\n* Public or private harassment\n* Publishing others' private information, such as a physical or email\n  address, without their explicit permission\n* Other conduct which could reasonably be considered inappropriate in a\n  professional setting\n\n## Enforcement Responsibilities\n\nCommunity leaders are responsible for clarifying and enforcing our standards of\nacceptable behavior and will take appropriate and fair corrective action in\nresponse to any behavior that they deem inappropriate, threatening, offensive,\nor harmful.\n\nCommunity leaders have the right and responsibility to remove, edit, or reject\ncomments, commits, code, wiki edits, issues, and other contributions that are\nnot aligned to this Code of Conduct, and will communicate reasons for moderation\ndecisions when appropriate.\n\n## Scope\n\nThis Code of Conduct applies within all community spaces, and also applies when\nan individual is officially representing the community in public spaces.\nExamples of representing our community include using an official e-mail address,\nposting via an official social media account, or acting as an appointed\nrepresentative at an online or offline event.\n\n## Enforcement\n\nInstances of abusive, harassing, or otherwise unacceptable behavior may be\nreported to the community leaders responsible for enforcement at\ntabi@osc.garden.\nAll complaints will be reviewed and investigated promptly and fairly.\n\nAll community leaders are obligated to respect the privacy and security of the\nreporter of any incident.\n\n## Enforcement Guidelines\n\nCommunity leaders will follow these Community Impact Guidelines in determining\nthe consequences for any action they deem in violation of this Code of Conduct:\n\n### 1. Correction\n\n**Community Impact**: Use of inappropriate language or other behavior deemed\nunprofessional or unwelcome in the community.\n\n**Consequence**: A private, written warning from community leaders, providing\nclarity around the nature of the violation and an explanation of why the\nbehavior was inappropriate. A public apology may be requested.\n\n### 2. Warning\n\n**Community Impact**: A violation through a single incident or series\nof actions.\n\n**Consequence**: A warning with consequences for continued behavior. No\ninteraction with the people involved, including unsolicited interaction with\nthose enforcing the Code of Conduct, for a specified period of time. This\nincludes avoiding interactions in community spaces as well as external channels\nlike social media. Violating these terms may lead to a temporary or\npermanent ban.\n\n### 3. Temporary Ban\n\n**Community Impact**: A serious violation of community standards, including\nsustained inappropriate behavior.\n\n**Consequence**: A temporary ban from any sort of interaction or public\ncommunication with the community for a specified period of time. No public or\nprivate interaction with the people involved, including unsolicited interaction\nwith those enforcing the Code of Conduct, is allowed during this period.\nViolating these terms may lead to a permanent ban.\n\n### 4. Permanent Ban\n\n**Community Impact**: Demonstrating a pattern of violation of community\nstandards, including sustained inappropriate behavior,  harassment of an\nindividual, or aggression toward or disparagement of classes of individuals.\n\n**Consequence**: A permanent ban from any sort of public interaction within\nthe community.\n\n## Attribution\n\nThis Code of Conduct is adapted from the [Contributor Covenant][homepage],\nversion 2.0, available at\nhttps://www.contributor-covenant.org/version/2/0/code_of_conduct.html.\n\nCommunity Impact Guidelines were inspired by [Mozilla's code of conduct\nenforcement ladder](https://github.com/mozilla/diversity).\n\n[homepage]: https://www.contributor-covenant.org\n\nFor answers to common questions about this code of conduct, see the FAQ at\nhttps://www.contributor-covenant.org/faq. Translations are available at\nhttps://www.contributor-covenant.org/translations.\n"
  },
  {
    "path": "CONTRIBUTING.md",
    "content": "# Contributing to tabi\n\nHalló!\n\nThanks for contributing to [tabi](https://github.com/welpo/tabi). Before implementing new features and changes, please [submit an issue](https://github.com/welpo/tabi/issues/new) so that we can discuss it.\n\nWe welcome contributions in many forms, including:\n\n- Bug reports;\n- New translations;\n- Improvements to existing translations;\n- Feature requests;\n- Code patches;\n- Documentation improvements;\n- UI/UX suggestions.\n\nIf you're not sure how to contribute or need help with something, please don't hesitate to reach out via the [issue tracker](https://github.com/welpo/tabi/issues) or [mail](mailto:tabi@osc.garden?subject=[GitHub]%20tabi).\n\n## Pull Requests\n\nWorking on your first Pull Request? You can learn how from this free video series:\n\n[**How to Contribute to an Open Source Project on GitHub**](https://egghead.io/courses/how-to-contribute-to-an-open-source-project-on-github)\n\nPlease make sure the following is done when submitting a pull request:\n\n1. **Keep your PR small**. Small pull requests are much easier to review and more likely to get merged. Make sure the PR does only one thing, otherwise please split it.\n2. **Use descriptive titles**. It is recommended to follow this [commit message style](#conventional-commit-messages-with-gitmoji).\n3. **Test your changes**. Make sure to test different configurations that might affect your changes.\n4. **Fill the PR template**. The template will guide you through the process of submitting a PR.\n\n### Conventional Commit Messages with Gitmoji\n\nSee how a minor change to your commit message style can make you a better programmer.\n\nFormat: `<gitmoji> <type>(<scope>): <description>`\n\n`<gitmoji>` is an emoji from the [gitmoji](https://gitmoji.dev/) list. It makes it easier to visually scan the commit history and quickly grasp the purpose of changes.\n\n`<scope>` is optional. If your change affects a specific part of the codebase, consider adding the scope. Scopes should be brief but recognizable, e.g. `config`, `feed`, or `search`.\n\nThe various types of commits:\n\n- `feat`: a new API or behaviour **for the end user**.\n- `fix`: a bug fix **for the end user**.\n- `style`: changes to the visual appearance of the theme, e.g. CSS, fonts, images…\n- `docs`: a change to the website or other Markdown documents.\n- `refactor`: a change to code that doesn't change behaviour, e.g. splitting files, renaming internal variables, improving code style…\n- `chore`: upgrading dependencies, releasing new versions… Chores that are **regularly done** for maintenance purposes.\n- `misc`: anything else that doesn't change production code, yet is not `chore`. e.g. updating GitHub actions workflow.\n\nThe commits within your PR don't need to follow this convention (we'll [squash them](https://docs.github.com/en/repositories/configuring-branches-and-merges-in-your-repository/configuring-pull-request-merges/configuring-commit-squashing-for-pull-requests)). However, the PR title should be in this format. If you're not sure about the title, don't worry, we'll help you fix it. Your code is more important than conventions!\n\nExample:\n\n```text\n🐛 fix(i18n): localise date in single taxonomy listing\n^  ^-^^----^  ^--------------------------------------^\n|  |  |       |\n|  |  |       +-> Description in imperative and lowercase.\n|  |  |\n|  |  +-> The scope of the change.\n|  |\n|  +-------> Type: see above for the list we use.\n|\n+----------> A valid gitmoji.\n```\n\n## Coding Style Guidelines\n\nWhile we don't enforce a strict coding style, we appreciate it when contributions follow the existing code style of the project (e.g. indenting with 4 spaces). This makes the codebase easier to read and maintain. If you are making significant changes or additions, please try to maintain consistency with the current coding patterns and idioms.\n\nFor JavaScript files, we use [Prettier](https://prettier.io/) to format the code.\n\nThe CSS properties are sorted following [Concentric-CSS](https://github.com/brandon-rhodes/Concentric-CSS). If you use VSCode, the [Sort CSS](https://marketplace.visualstudio.com/items?itemName=piyushsarkar.sort-css-properties) extension makes this super easy.\n\n## Pre-commit Githook\n\n### Introduction\n\nWe use a pre-commit githook to maintain code and file quality. [This script](https://github.com/welpo/tabi/blob/main/.githooks/pre-commit) performs a series of checks and updates before a commit is made.\n\n### Setting Up\n\nTo use the pre-commit githook, run the following command from the root of the repository to configure the git hooks path and make the script executable:\n\n```bash\ngit config core.hooksPath .githooks\nchmod +x .githooks/pre-commit\n```\n\n### Features\n\nThe pre-commit githook performs the following actions:\n\n#### Automatic Updates\n\n- **Front Matter in Markdown Files**: Automatically updates the \"updated\" field in the front matter of `.md` files.\n- **PNG Compression**: Compresses PNG files using either [`oxipng`](https://github.com/shssoichiro/oxipng) or [`optipng`](https://optipng.sourceforge.net/), whichever is available on your system.\n- **Font Subsetting**: Runs `subset_font` if `config.toml` has been modified.\n\n#### Commit Checks\n\nThe script prevents you from committing if:\n\n- The `.md` file is marked as a draft.\n- Any file contains a \"TODO\".\n- A JavaScript file is being committed without a corresponding minified version.\n- A minified JavaScript file is not as small as it could be. Requires installing [UglifyJS](https://github.com/mishoo/UglifyJS) and/or [terser](https://github.com/terser/terser).\n- `config.toml` and `theme.toml` have different numbers of lines in their `[extra]` sections.\n\nThese checks are in place to ensure code quality and consistency throughout the project.\n\n## Code of Conduct\n\nWe expect all contributors to follow our [Code of Conduct](./CODE_OF_CONDUCT.md). Please be respectful and professional when interacting with other contributors.\n\n## License\n\nThe code is available under the [MIT license](./LICENSE).\n\nThank you for your contributions!\n"
  },
  {
    "path": "LICENSE",
    "content": "MIT License\n\nCopyright (c) 2023 Óscar Fernández\n\nPermission is hereby granted, free of charge, to any person obtaining a copy\nof this software and associated documentation files (the \"Software\"), to deal\nin the Software without restriction, including without limitation the rights\nto use, copy, modify, merge, publish, distribute, sublicense, and/or sell\ncopies of the Software, and to permit persons to whom the Software is\nfurnished to do so, subject to the following conditions:\n\nThe above copyright notice and this permission notice shall be included in all\ncopies or substantial portions of the Software.\n\nTHE SOFTWARE IS PROVIDED \"AS IS\", WITHOUT WARRANTY OF ANY KIND, EXPRESS OR\nIMPLIED, INCLUDING BUT NOT LIMITED TO THE WARRANTIES OF MERCHANTABILITY,\nFITNESS FOR A PARTICULAR PURPOSE AND NONINFRINGEMENT. IN NO EVENT SHALL THE\nAUTHORS OR COPYRIGHT HOLDERS BE LIABLE FOR ANY CLAIM, DAMAGES OR OTHER\nLIABILITY, WHETHER IN AN ACTION OF CONTRACT, TORT OR OTHERWISE, ARISING FROM,\nOUT OF OR IN CONNECTION WITH THE SOFTWARE OR THE USE OR OTHER DEALINGS IN THE\nSOFTWARE.\n"
  },
  {
    "path": "README.md",
    "content": "<p align=\"center\">\n    <a href=\"CONTRIBUTING.md#pull-requests\">\n        <img src=\"https://img.shields.io/badge/PRs-welcome-0?style=flat-square&labelColor=202b2d&color=087e96\" alt=\"PRs welcome\"></a>\n    <a href=\"https://github.com/welpo/tabi/graphs/contributors\">\n        <img src=\"https://img.shields.io/github/contributors/welpo/tabi?style=flat-square&labelColor=202b2d&color=087e96\" alt=\"Contributors\"></a>\n    <a href=\"https://github.com/welpo/tabi/forks\">\n        <img src=\"https://img.shields.io/github/forks/welpo/tabi?style=flat-square&labelColor=202b2d&color=087e96\" alt=\"Forks\"></a>\n    <a href=\"https://github.com/welpo/tabi/commits/main/\">\n        <img src=\"https://img.shields.io/github/last-commit/welpo/tabi?style=flat-square&labelColor=202b2d&color=087e96\" alt=\"Last commit\"></a>\n    <br>\n    <a href=\"https://github.com/welpo/tabi/releases\">\n        <img src=\"https://img.shields.io/github/v/release/welpo/tabi?style=flat-square&labelColor=202b2d&color=087e96\" alt=\"Latest release\"></a>\n    <a href=\"https://welpo.github.io/tabi/blog/mastering-tabi-settings/\">\n        <img src=\"https://img.shields.io/website?url=https%3A%2F%2Fwelpo.github.io%2Ftabi&style=flat-square&label=docs&labelColor=202b2d\" alt=\"Documentation\"></a>\n    <a href=\"https://github.com/welpo/tabi/blob/main/LICENSE\">\n        <img src=\"https://img.shields.io/github/license/welpo/tabi?style=flat-square&labelColor=202b2d&color=087e96\" alt=\"License\"></a>\n    <a href=\"https://github.com/welpo/git-sumi\">\n        <img src=\"https://img.shields.io/badge/clean_commits-git--sumi-0?style=flat-square&labelColor=202b2d&color=087e96\" alt=\"Clean commits\"></a>\n    <a href=\"https://isitmaintained.com/project/welpo/tabi\">\n        <img src=\"https://isitmaintained.com/badge/resolution/welpo/tabi.svg\" alt=\"Average time to resolve an issue\"></a>\n    <a href=\"https://isitmaintained.com/project/welpo/tabi\">\n        <img src=\"https://isitmaintained.com/badge/open/welpo/tabi.svg\" alt=\"Percentage of issues still open\"></a>\n</p>\n\n# 🌱 tabi\n\nAn accessible [Zola](https://www.getzola.org) theme with [search](https://welpo.github.io/tabi/blog/mastering-tabi-settings/#search), [multi-language support](https://welpo.github.io/tabi/blog/faq-languages/), [optional JavaScript](https://welpo.github.io/tabi/blog/javascript/), a perfect Lighthouse score, and [comprehensive documentation](https://welpo.github.io/tabi). Crafted for personal websites and blogs.\n\n> [!TIP]\n> Want to start blogging right away? Use the [tabi-start template](https://github.com/welpo/tabi-start) to get a complete site up and running in minutes.\n\nSee a live preview (and the theme's documentation) [here](https://welpo.github.io/tabi).\n\nExplore the [Sites Using tabi section](#sites-using-tabi) to see real-world applications.\n\n> tabi (旅, /<span title=\"/t/: 't' in 'sty'\">t</span><span title=\"/ɐ/: a sound between 'a' in 'sofa' and 'u' in 'nut'\">ɐ</span><span title=\"/ˈ/: primary stress mark, indicating that the following syllable is pronounced with greater emphasis\">ˈ</span><span title=\"/b/: 'b' in 'cab'\">b</span><span title=\"/i/: 'i' in 'fleece'\">i</span>/): Journey.\n\n![tabi](https://github.com/welpo/tabi/raw/main/light_dark_screenshot.png)\n\ntabi has a perfect score on Google's Lighthouse audit:\n\n![lighthouse](https://raw.githubusercontent.com/welpo/tabi/main/lighthouse_score.png)\n\n## Features\n\n- [X] [Set any language as default](https://welpo.github.io/tabi/blog/faq-languages/#how-do-i-set-a-default-language-for-my-site). Set your base site to Chinese, Spanish, French, Hindi… or any [other supported language](/i18n). The theme's interface will be translated accordingly.\n- [X] [Integration with remote repositories](https://welpo.github.io/tabi/blog/mastering-tabi-settings/#git-repository-integration) on GitHub, GitLab, Gitea & Codeberg for commit history and showing the site source.\n- [X] [Series support](https://welpo.github.io/tabi/blog/series/) for creating sequential content like tutorials, courses, and multi-part stories.\n- [X] Dark and light themes. Defaults to the OS setting, with a switcher in the navigation bar.\n- [X] Thorough documentation. See [Mastering tabi Settings: A Comprehensive Guide](https://welpo.github.io/tabi/blog/mastering-tabi-settings/).\n- [X] Perfect Lighthouse score (Performance, Accessibility, Best Practices and SEO).\n- [X] [Comprehensive multi-language support](https://welpo.github.io/tabi/blog/faq-languages/#how-does-tabi-handle-multilingual-support). Add as many languages as you wish.\n- [X] Support for [comments using giscus, utterances, Hyvor Talk, or Isso](https://welpo.github.io/tabi/blog/comments/).\n- [X] [Indieweb](https://indieweb.org/) ready with microformats, [hcard](https://welpo.github.io/tabi/blog/mastering-tabi-settings/#representative-h-card) and [webmentions](https://welpo.github.io/tabi/blog/mastering-tabi-settings/#webmentions) support.\n- [X] Code syntax highlighting with colours based on [Catppuccin](https://github.com/catppuccin/catppuccin) Frappé.\n- [X] [Mermaid support](https://welpo.github.io/tabi/blog/shortcodes/#mermaid-diagrams) to create diagrams and charts with text.\n- [X] [Local search](https://welpo.github.io/tabi/blog/mastering-tabi-settings/#search) with an accessible, multi-lingual interface.\n- [X] [Custom Twitter card](https://welpo.github.io/tabi/blog/mastering-tabi-settings/#social-media-cards) and automatic Open Graph tags.\n- [X] Anonymous [like buttons](https://welpo.github.io/tabi/blog/mastering-tabi-settings/#iine) powered by [iine](https://iine.to).\n- [X] [KaTeX](https://katex.org/) support for mathematical notation.\n- [X] [Stylized and human readable Atom feed](https://welpo.github.io/tabi/atom.xml).\n- [X] [Stylized and human readable sitemap](https://welpo.github.io/tabi/sitemap.xml).\n- [X] [Mail encoding](https://welpo.github.io/tabi/blog/mastering-tabi-settings/#encoded-email) for spam protection.\n- [X] All JavaScript can be [fully disabled](https://welpo.github.io/tabi/blog/javascript/).\n- [X] [Customizable Table of Contents](https://welpo.github.io/tabi/blog/toc/).\n- [X] [Customizable secure headers](https://welpo.github.io/tabi/blog/security/).\n- [X] [Copy button for code blocks](https://welpo.github.io/tabi/blog/mastering-tabi-settings/#copy-button-on-code-blocks).\n- [X] [Quick navigation buttons](https://welpo.github.io/tabi/blog/mastering-tabi-settings/#quick-navigation-buttons).\n- [X] [Custom copyright notice](https://welpo.github.io/tabi/blog/mastering-tabi-settings/#copyright).\n- [X] [Custom canonical URLs](https://welpo.github.io/tabi/blog/mastering-tabi-settings/#canonical-url).\n- [X] [Custom shortcodes](https://welpo.github.io/tabi/blog/shortcodes/).\n- [X] [Customizable skins](https://welpo.github.io/tabi/blog/customise-tabi/).\n- [X] [Social media cards](https://welpo.github.io/tabi/blog/mastering-tabi-settings/#social-media-cards).\n- [X] Responsive design.\n- [X] [Projects page](https://welpo.github.io/tabi/projects/).\n- [X] [Archive page](https://welpo.github.io/tabi/archive/).\n- [X] [Pinned posts](https://welpo.github.io/tabi/blog/mastering-tabi-settings/#pinning-posts).\n- [X] [Social links](https://welpo.github.io/tabi/blog/mastering-tabi-settings/#social-media-icons).\n- [X] [Tags](https://welpo.github.io/tabi/blog/mastering-tabi-settings/#tags).\n\n## Installation\n\n> [!NOTE]\n> The fastest way to create a new site is to use the [tabi-start template](https://github.com/welpo/tabi-start). This gives you a complete blog setup with all the essential configuration ready to go.\n\n### Manual installation\n\nTo add tabi to you existing Zola site:\n\n0. Initialize a Git repository in your project directory (if you haven't already):\n\n```\ngit init\n```\n\n1. Add the theme as a git submodule:\n\n```\ngit submodule add https://github.com/welpo/tabi.git themes/tabi\n```\n\nOr clone the theme into your themes directory:\n\n```\ngit clone https://github.com/welpo/tabi.git themes/tabi\n```\n\n### Required configuration\n\n2. Enable the theme in your `config.toml`:\n\n```\ntheme = \"tabi\"\n```\n\n3. Set a `title` in your `config.toml`:\n\n```\ntitle = \"Your Site Title\"\n```\n\n4. Configure code block highlighting in your `config.toml`:\n\n```toml\n[markdown.highlighting]\ntheme = \"catppuccin-frappe\"\nstyle = \"class\"\n```\n\n5. Create a `content/_index.md` file. This file controls how your home page looks and behaves. Choose one of the following options:\n\n   **Option A: Serve posts from `/`**:\n\n   ```\n   +++\n   title = \"Home\"\n   paginate_by = 5  # Show 5 posts per page.\n   +++\n   ```\n\n   - This will display posts in `content/` with pagination.\n\n   **Option B: Serve posts from a different path (e.g., `blog/`)**:\n\n   ```\n   +++\n   title = \"Home\"\n   # Note we're not setting `paginate_by` here.\n\n   [extra]\n   section_path = \"blog/_index.md\"  # Where to find your posts.\n   max_posts = 5  # Show 5 posts on the home page.\n   +++\n   ```\n\n    - This will display the latest 5 posts from the `blog/` section.\n    - Do not set `paginate_by` if you choose this option.\n    - Use the full path to the section's `_index.md` file. Using `section_path = \"blog/\"` will not work.\n\n> [!WARNING]\n> Do not set both `paginate_by` and `section_path` in `content/_index.md`.\n>\n> These settings are mutually exclusive and using both may result in no posts being displayed.\n\n6. If you want an introduction section (see screenshot above), add these lines to `content/_index.md`:\n\n```\n[extra]\nheader = {title = \"Hello! I'm tabi~\", img = \"img/main.webp\", img_alt = \"Your Name\" }\n```\n\nThe content outside the front matter will be rendered between the header title and the posts listing. In the screenshot above, it's the text that reads \"tabi is a fast, lightweight, and modern Zola theme…\".\n\n7. If you want a multilingual site, you will need to set up each language. In `config.toml`, set the title and taxonomies for each language, like:\n\n```toml\n[languages.es]\ntitle = \"~/tabi\"\ntaxonomies = [{name = \"tags\", feed = true}]\n```\n\nYou will need an `_index.{language_code}.md` per language for each section (e.g. /blog or /projects) that you want to enable in that language.\n\nThe same is true for individual posts, which should have the exact same name as the default language, with an extra `.{code}` before the extension (e.g. the Spanish version of `security.md` would be `security.es.md`).\n\nThis configuration allows the language switcher to take the user to the translation of the current URL. If a translation doesn't exist, the 404 page will be displayed, with an explanation in each language set in the config.\n\nTo learn more about multilingual support, see the [Frequently Asked Questions](https://welpo.github.io/tabi/blog/faq-languages/).\n\n### Updating tabi\n\nIf you added the theme as a git submodule, run:\n\n```bash\ngit submodule update --remote themes/tabi\n```\n\nIf you cloned it:\n\n```bash\ncd themes/tabi\ngit pull\n```\n\n## Sites using tabi\n\n| Website | Creator | Description  | Site Source   |\n| - | - | - | - |\n| [osc.garden](https://osc.garden) | Óscar Fernández ([welpo](https://github.com/welpo)) | Data science, psychology, and Zola | [Source](https://github.com/welpo/osc.garden) |\n| [seadve.github.io](https://seadve.github.io/) | Dave Patrick Caberto ([SeaDve](https://github.com/SeaDve/)) | Personal blog and portfolio with custom CSS | [Source](https://github.com/SeaDve/seadve.github.io) |\n| [mikufan.page](https://mikufan.page) | [Nadia](https://github.com/nyadiia) | Personal blog | [Source](https://github.com/nyadiia/mikufan.page) |\n| [tim-boettcher.online](https://tim-boettcher.online/) | [Tim Böttcher](https://codeberg.org/Tim-Boettcher/) | Insights and ramblings of a deafblind programmer | [Source](https://codeberg.org/Tim-Boettcher/tim-boettcher-online/) |\n| [www.richtman.au](https://www.richtman.au) | [Ariel Richtman](https://github.com/arichtman) | Personal tech blog | [Source](https://github.com/arichtman/www.richtman.au) |\n| [Ponderosa Games](https://ponderosagames.com/) | John Burak ([JVimes](https://github.com/jvimes)) | A friendly indie game company | &mdash; |\n| [jmbhughes.com](https://jmbhughes.com/) | Marcus Hughes ([jmbhughes](https://github.com/jmbhughes)) | Personal blog | [Source](https://github.com/jmbhughes/jmbhughes.github.io) |\n| [szabolcs.me](https://szabolcs.me) | Szabolcs Fazekas ([szabolcsf](https://github.com/szabolcsf)) | Personal blog | [Source](https://github.com/szabolcsf/szabolcs.me) |\n| [Nizzlay](https://nizzlay.com) | Niels Gouman ([Nizzlay](https://github.com/Nizzlay)) | Personal blog | [Source](https://github.com/Nizzlay/nizzlay.com) |\n| [ZzMzaw's blog](https://zzmzaw.github.io/) | ZzMzaw ([ZzMzaw](https://github.com/ZzMzaw)) | Personal blog | [Source](https://github.com/ZzMzaw/zzmzaw.github.io) |\n| [idle-ti.me](https://idle-ti.me/) | Jérôme Ramette ([be-next](https://github.com/be-next)) | Personal blog | [Source](https://github.com/be-next/idle-ti.me) |\n| [tzinm.me](https://tzinm.me/) | [Tzinm](https://github.com/tzinm) | Personal blog | [Source](https://codeberg.org/tzinm/blog) |\n| [b1n.io](https://b1n.io) | [b1nhack](https://github.com/b1nhack) | Linux kernel vulnerability researcher | [Source](https://github.com/b1nhack/blog) |\n| [posixlycorrect.com](https://posixlycorrect.com/) | [Fabian Montero](https://git.posixlycorrect.com/fabian) | Personal homepage | [Source](https://git.posixlycorrect.com/fabian/homepage) |\n| [Coded Chords](https://codedchords.dev/) | [yostos](https://github.com/yostos) | Personal tech blog | [Source](https://github.com/yostos/blog-yostos) |\n\nUsing tabi? Feel free to create a PR and add your site to this list.\n\n## Inspiration\n\nThis theme was inspired by:\n\n- [shadharon](https://github.com/syedzayyan/shadharon) — tabi started as a fork of [syedzayyan](https://github.com/syedzayyan)'s theme\n- [tailwind-nextjs-starter-blog](https://github.com/timlrx/tailwind-nextjs-starter-blog)\n- [abridge](https://github.com/Jieiku/abridge)\n\n## Support\n\nSomething not working? Have an idea? Let us know!\n\n- Questions? → [Start a discussion](https://github.com/welpo/tabi/discussions)\n- Found a bug? → [Report it here](https://github.com/welpo/tabi/issues/new?&labels=bug&template=2_bug_report.yml)\n- Feature request? → [Tell us more!](https://github.com/welpo/tabi/issues/new?&labels=feature&template=3_feature_request.yml)\n\n## Contributing\n\nPlease do! We appreciate bug reports, improvements to translations or documentation (however minor), feature requests…\n\nTake a look at the [Contributing Guidelines](/CONTRIBUTING.md) to learn more.\n\n## License\n\nThe code is available under the [MIT license](./LICENSE).\n"
  },
  {
    "path": "cliff.toml",
    "content": "# git-cliff ~ default configuration file\n# https://git-cliff.org/docs/configuration\n#\n# Lines starting with \"#\" are comments.\n# Configuration options are organized into tables and keys.\n# See documentation for more information on available options.\n\n[remote.github]\nowner = \"welpo\"\nrepo = \"tabi\"\n\n[changelog]\n# changelog header\nheader = \"\"\"\n# Changelog\n\nWelcome to the changelog for tabi. This document aims to provide a comprehensive list of all notable changes made to the project, organised chronologically by release version.\n\nWe use Semantic Versioning (SemVer) for our version numbers, formatted as MAJOR.MINOR.PATCH. Major version changes involve significant (breaking) changes, minor versions introduce features and improvements in a backward compatible manner, and patch versions are for bug fixes and minor tweaks.\\n\n\"\"\"\n# template for the changelog body\n# https://keats.github.io/tera/docs/#introduction\nbody = \"\"\"\n{%- macro remote_url() -%}\n  https://github.com/{{ remote.github.owner }}/{{ remote.github.repo }}\n{%- endmacro -%}\n{% if version %}\\\n    {% if previous.version %}\\\n        ## [{{ version | trim_start_matches(pat=\"v\") }}]({{ self::remote_url() }}/compare/{{ previous.version }}..{{ version }}) - {{ timestamp | date(format=\"%Y-%m-%d\") }}\n    {% else %}\\\n        ## {{ version | trim_start_matches(pat=\"v\") }} - {{ timestamp | date(format=\"%Y-%m-%d\") }}\n    {% endif %}\\\n{% else %}\\\n    ## unreleased\n{% endif %}\\\n\n{% macro commit(commit, in_breaking_section=false) -%}\n    - {% if commit.scope %}*({{ commit.scope }})* {% endif %}{% if commit.breaking and not in_breaking_section %}[**‼️BREAKING‼️**] {% endif %}\\\n        {{ commit.message | upper_first }}\\\n        {% if not commit.remote.pr_number %} ([{{ commit.id | truncate(length=7, end=\"\") }}]({{ self::remote_url() }}/commit/{{ commit.id }})){%- endif -%}{% if commit.remote.username %} by @{{ commit.remote.username }} \\\n        {%- set co_authors = commit.footers | filter(attribute=\"token\", value=\"Co-authored-by\") | map(attribute=\"value\") -%}\n        {%- set number_of_co_authors = co_authors | length -%}\n        {%- for co_author in co_authors -%}\n            {%- if not loop.last %}, {% else %} and {% endif %}@{{ co_author | split(pat=\" <\") | first | trim }}\n        {%- endfor -%}\n        {%- endif -%}\n{% endmacro -%}\n\n{%- set breaking_header_shown = false -%}\n{% for commit in commits -%}\n    {% if commit.breaking and not breaking_header_shown -%}\n        {% raw %}\\n### 💥 BREAKING CHANGES 💥\\n{% endraw %}\n        {%- set_global breaking_header_shown = true -%}\n    {%- endif -%}\n    {%- if commit.breaking %}\n        {{ self::commit(commit=commit, in_breaking_section=true) -}}\n    {% endif -%}\n{%- endfor -%}\n{%- if breaking_header_shown == true -%}\n    {% raw %}\\n{% endraw %}\\\n{%- endif -%}\n\n{% for group, commits in commits | group_by(attribute=\"group\") %}\n    ### {{ group | striptags | trim | upper_first }}\n    {% for commit in commits\n    | filter(attribute=\"scope\")\n    | sort(attribute=\"scope\") %}\n        {{ self::commit(commit=commit) }}\n    {%- endfor -%}\n    {% raw %}\\n{% endraw %}\\\n    {%- for commit in commits %}\n        {%- if not commit.scope -%}\n            {{ self::commit(commit=commit) }}\n        {% endif -%}\n    {% endfor -%}\n{% endfor %}\n\n{%- if github.contributors | filter(attribute=\"is_first_time\", value=true) | length != 0 -%}\n  {% raw %}\\n{% endraw %}### 👥 New contributors\n  {% raw -%}\\n{% endraw -%}\n{% for contributor in github.contributors | filter(attribute=\"is_first_time\", value=true) %}\n  🫶 @{{ contributor.username }} made their first contribution\n    {%- if contributor.pr_number %} in \\\n      [#{{ contributor.pr_number }}]({{ self::remote_url() }}/pull/{{ contributor.pr_number }}) \\\n    {%- endif %}\n{% endfor %}\n{%- endif %}\n{% raw -%}\\n{% endraw -%}\n\"\"\"\n\n# remove the leading and trailing whitespace from the template\ntrim = true\n# changelog footer\nfooter = \"\"\"\n<!-- generated by git-cliff -->\n\"\"\"\n# postprocessors\npostprocessors = [\n    # { pattern = \" @([a-zA-Z0-9](?:[a-zA-Z0-9]+-?)*[a-zA-Z0-9])\", replace = \" [@$1](https://github.com/$1)\"}, # add link to GitHub usernames (done in release script instead)\n]\n\n[git]\n# parse the commits based on https://www.conventionalcommits.org\nconventional_commits = true\n# filter out the commits that are not conventional\nfilter_unconventional = true\n# process each line of a commit as an individual commit\nsplit_commits = false\n# regex for preprocessing the commit messages\ncommit_preprocessors = [\n    # Replace the issue number with the link.\n    { pattern = \"\\\\(#([0-9]+)\\\\)\", replace = \"([#${1}](https://github.com/welpo/tabi/issues/${1}))\" },\n    # Remove trailing whitespace.\n    { pattern = ' +$', replace = \"\" },\n    # Replace multiple spaces with a single space.\n    { pattern = ' +', replace = \" \" },\n    # Remove gitmoji, both actual UTF emoji and :emoji:\n    { pattern = ' *(:\\w+:|[\\p{Emoji_Presentation}\\p{Extended_Pictographic}](?:\\u{FE0F})?\\u{200D}?) *', replace = \"\" },\n]\n# regex for parsing and grouping commits\ncommit_parsers = [\n    { message = \"^feat\", group = \"<!-- 0 -->✨ Features\" },\n    { message = \"^fix\", group = \"<!-- 1 -->🐛 Bug fixes\" },\n    { message = \"^style\", group = \"<!-- 2 -->💄 Styling\" },\n    { message = \"^doc\", group = \"<!-- 3 -->📝 Documentation\" },\n    { message = \"^refactor\", group = \"<!-- 4 -->♻️ Refactor\" },\n    { message = \"^test\", group = \"<!-- 5 -->✅ Testing\" },\n    { message = \"^misc\", group = \"<!-- 6 -->🔧 Miscellaneous tasks\" },\n    { message = \"^deprecate\", group = \"<!-- 7 -->🗑️️ Deprecations\" },\n    { message = \"^chore\", skip = true },\n]\n# protect breaking changes from being skipped due to matching a skipping commit_parser\nprotect_breaking_commits = true\n# filter out the commits that are not matched by commit parsers\nfilter_commits = true\n# regex for matching git tags\ntag_pattern = \"v[0-9].*\"\n\n# regex for skipping tags\nskip_tags = \"v0.1.0-beta.1\"\n# regex for ignoring tags\nignore_tags = \"\"\n# sort the tags topologically\ntopo_order = false\n# sort the commits inside sections by oldest/newest order\nsort_commits = \"newest\"\n# limit the number of commits included in the changelog.\n# limit_commits = 42\n"
  },
  {
    "path": "config.toml",
    "content": "base_url = \"https://welpo.github.io/tabi\"\ntitle = \"~/tabi\"\ndescription = \"tabi is an accessible Zola theme with search, multi-language support, optional JavaScript, a perfect Lighthouse score, and comprehensive documentation. Crafted for personal websites and blogs.\"\nauthor = \"welpo\"\ngenerate_feeds = true\ncompile_sass = true\nminify_html = true\nbuild_search_index = true\n\n# To translate the entire theme, there must be a file with the same language code\n# in the `i18n` folder of your site or the tabi theme.\n# For example, \"i18n/fr.toml\" for French or \"i18n/zh-Hans.toml\" for Simplified Chinese.\n# Otherwise the theme will be in English.\n# See https://welpo.github.io/tabi/blog/faq-languages/ for more details.\ndefault_language = \"en\"\n\ntaxonomies = [{name = \"tags\", feed = true}]\n\n[search]\n# Whether to include the title of the page/section in the index.\ninclude_title = true\n# Whether to include the description of the page/section in the index.\ninclude_description = true\n# Whether to include the path of the page/section in the index.\ninclude_path = true\n# Whether to include the rendered content of the page/section in the index.\ninclude_content = true\n# At which character to truncate the content to. Useful if you have a lot of pages and the index would\n# become too big to load on the site. Defaults to not being set.\n# truncate_content_length = 100\n# Whether to produce the search index as a javascript file or as a JSON file.\n# Accepted value \"elasticlunr_javascript\" or \"elasticlunr_json\".\nindex_format = \"elasticlunr_json\"\n\n[markdown]\nbottom_footnotes = true\nsmart_punctuation = true\n# Set to 'external' to add an indicator next to external links.\nexternal_links_class = \"external\"\n\n[markdown.highlighting]\ntheme = \"catppuccin-frappe\"\nstyle = \"class\"\nerror_on_missing_language = true\n\n[link_checker]\ninternal_level = \"warn\"\nskip_prefixes = [\n    \"https://www.vultr.com/\",\n]\nskip_anchor_prefixes = [\n    \"https://github.com/\",\n]\n\n[languages.es]\ntitle = \"~/tabi\"\ndescription = \"tabi es un tema accesible para Zola con búsqueda, soporte multilingüe, JavaScript opcional, una puntuación perfecta en Lighthouse y documentación exhaustiva. Diseñado para sitios web y blogs personales.\"\ngenerate_feeds = true\ntaxonomies = [{name = \"tags\", feed = true}]\nbuild_search_index = true\n\n[languages.ca]\ntitle = \"~/tabi\"\ndescription = \"tabi és un tema accessible per a Zola amb cerca, suport multilingüe, JavaScript opcional, una puntuació perfecta a Lighthouse i documentació exhaustiva. Dissenyat per a llocs web i blogs personals.\"\ngenerate_feeds = true\ntaxonomies = [{name = \"tags\", feed = true}]\n\n[languages.ar]\ntitle = \"~/تابي\"\ndescription = \"تابي هو قالب Zola سريع وحديث مع دعم متعدد اللغات و JavaScript اختياري ودرجة Lighthouse مثالية.\"\ngenerate_feeds = true\ntaxonomies = [{name = \"tags\", feed = true}]\nbuild_search_index = false\n\n[extra]\n# Check out the documentation (or the comments below) to learn how to customise tabi:\n# https://welpo.github.io/tabi/blog/mastering-tabi-settings/\n\n# Search index format.\n# For Zola 0.17.X users only.\n# This MUST MATCH the setting in [search].index_format.\n# Example: If [search].index_format is \"elasticlunr_javascript\", set:\n# index_format = \"elasticlunr_javascript\"\n# index_format = \"\"\n\n# Use sans-serif font everywhere.\n# By default, the serif font is only used in articles.\noverride_serif_with_sans = false\n\n# Enable JavaScript theme toggler to allow users to switch between dark/light mode.\n# If disabled, your site will use the theme specified in the `default_theme` variable.\ntheme_switcher = true\n\n# This setting determines the default theme on load (\"light\" or \"dark\").\n# To follow the user's OS theme, leave it empty or unset.\ndefault_theme = \"\"\n\n# Choose the colourscheme (skin) for the theme. Default is \"teal\".\n# Skin available: blue, lavender, mint, red, sakura, teal, monochrome, lowcontrast_orange, lowcontrast_peach, lowcontrast_pink, indigo_ingot, evangelion\n# See them live and learn how to create your own: https://welpo.github.io/tabi/blog/customise-tabi/#skins\n# WARNING! \"lowcontrast\" skins, while aesthetically pleasing, may not provide optimal\n# contrast (in light theme) for readability and might not be suitable for all users.\n# Furthermore, low contrasting elements will affect your Google Lighthouse rating.\n# All other skins have optimal contrast.\nskin = \"\"\n\n# Set browser theme colour. Can be a single colour or [light, dark].\n# Note: Bright colors may be ignored in dark mode.\n# More details: https://developer.mozilla.org/en-US/docs/Web/HTML/Element/meta/name/theme-color\nbrowser_theme_color = \"#087e96\"\n# browser_theme_color = [\"#ffffff\", \"#000000\"]  # Example of light/dark colours.\n\n# For multilingual sites: show current language code on the language switcher.\nshow_selected_language_code_in_language_switcher = false\n\n# List additional stylesheets to load site-wide.\n# These stylesheets should be located in your site's `static` directory.\n# Example: stylesheets = [\"extra1.css\", \"path/extra2.css\"]\n# You can load a stylesheet for a single post by adding it to the [extra] section of the post's front matter, following this same format.\nstylesheets = []\n\n# Sets the default canonical URL for all pages.\n# Individual pages can override this in the [extra] section using canonical_url.\n# Example: \"$base_url/blog/post1\" will get the canonical URL \"https://example.com/blog/post1\".\n# Note: To ensure accuracy in terms of matching content, consider setting 'canonical_url' individually per page.\n# base_canonical_url = \"https://example.com\"\n\n# Remote repository for your Zola site.\n# Used for `show_remote_changes` and `show_remote_source` (see below).\n# Supports GitHub, GitLab, Gitea, and Codeberg.\nremote_repository_url = \"https://github.com/welpo/tabi\"\n# Set this to \"auto\" to try and auto-detect the platform based on the repository URL.\n# Accepted values are \"github\", \"gitlab\", \"gitea\", and \"codeberg\".\nremote_repository_git_platform = \"auto\"  # Defaults to \"auto\".\n# Branch in the repo hosting the Zola site.\nremote_repository_branch = \"main\"  # Defaults to \"main\".\n# Show a link to the commit history of updated posts, right next to the last updated date.\n# Can be set at page or section levels, following the hierarchy: page > section > config. See: https://welpo.github.io/tabi/blog/mastering-tabi-settings/#settings-hierarchy\nshow_remote_changes = true  # Defaults to true.\n# Show a link to the repository of the site, right next to the \"Powered by Zola & tabi\" text.\nshow_remote_source = true  # Defaults to true.\n\n# Add a \"copy\" button to code blocks (loads ~700 bytes of JavaScript).\n# Can be set at page or section levels, following the hierarchy: page > section > config. See: https://welpo.github.io/tabi/blog/mastering-tabi-settings/#settings-hierarchy\ncopy_button = true\n\n# Make code block names clickable if they are URLs (loads ~400 bytes of JavaScript).\n# Can be set at page or section levels, following the hierarchy: page > section > config. See: https://welpo.github.io/tabi/blog/mastering-tabi-settings/#settings-hierarchy\ncode_block_name_links = false\n\n# Force left-to-right (LTR) direction for code blocks.\n# Set to false to allow code to follow the document's natural direction.\n# Can be set at page or section levels. See: https://welpo.github.io/tabi/blog/mastering-tabi-settings/#settings-hierarchy\nforce_codeblock_ltr = true\n\n# Show the author(s) of a page.\n# Can be set at page or section levels, following the hierarchy: page > section > config. See: https://welpo.github.io/tabi/blog/mastering-tabi-settings/#settings-hierarchy\nshow_author = false\n\n# Show the reading time of a page.\n# Can be set at page or section levels, following the hierarchy: page > section > config. See: https://welpo.github.io/tabi/blog/mastering-tabi-settings/#settings-hierarchy\nshow_reading_time = true\n\n# Show the date of a page below its title.\n# Can be set at page or section levels, following the hierarchy: page > section > config. See: https://welpo.github.io/tabi/blog/mastering-tabi-settings/#settings-hierarchy\nshow_date = true\n\n# Determines how dates are displayed in the post listing (e.g. front page or /blog). Options:\n# \"date\" - Show only the original date of the post (default if unset).\n# \"updated\" - Show only the last updated date of the post. If there is no last updated date, it shows the original date.\n# \"both\" - Show both the original date and the last updated date.\npost_listing_date = \"date\"\n\n# Enable iine like buttons on all posts: https://iine.to/\n# Can be set at page or section levels, following the hierarchy: page > section > config. See: https://welpo.github.io/tabi/blog/mastering-tabi-settings/#settings-hierarchy\niine = true\niine_icon = \"thumbs_up\"  # See https://iine.to/#customise\n# Unify like counts across all language versions of the same page.\n# When enabled, likes on /es/blog/hello/ will count towards /blog/hello/ (default language).\niine_unified_languages = true\n\n# Show \"Jump to posts\" link next to series' title.\n# By default, the link appears automatically when a series description exceeds 2000 characters.\n# Can be set at page or section levels, following the hierarchy: page > section > config. See: https://welpo.github.io/tabi/blog/mastering-tabi-settings/#settings-hierarchy\n# show_jump_to_posts = true\n\n# Determines if indexes should be increasing (false) or decreasing (true) in series' posts list.\n# It has only effect if the section uses indexes metadata (which is only the case for series as of now).\n# Can be set at section levels, following the hierarchy: section > config. See: https://welpo.github.io/tabi/blog/mastering-tabi-settings/#settings-hierarchy\npost_listing_index_reversed = false  # Defaults to false.\n\n# DEPRECATED!\n# Use Zola's built-in `bottom_footnotes = true` in the [markdown] section instead. (Available since v0.19.0)\n# Adds backlinks to footnotes (loads ~500 bytes of JavaScripts).\n# Can be set at page or section levels, following the hierarchy: page > section > config. See: https://welpo.github.io/tabi/blog/mastering-tabi-settings/#settings-hierarchy\nfootnote_backlinks = false\n\n# Enable KaTeX for all posts.\n# Can be set at page or section levels, following the hierarchy: page > section > config. See: https://welpo.github.io/tabi/blog/mastering-tabi-settings/#settings-hierarchy\nkatex = false\n\n# Enable Mermaid diagrams for all posts.\n# Loads ~2.5MB of JavaScript.\n# Can be set at page or section levels, following the hierarchy: page > section > config. See: https://welpo.github.io/tabi/blog/mastering-tabi-settings/#settings-hierarchy\nmermaid = false\n\n# Serve Mermaid JavaScript locally. Version bundled with tabi.\n# If set to false, it will load the latest version from JSDelivr.\n# Only relevant when `mermaid = true`.\nserve_local_mermaid = true\n\n# Show links to previous and next articles at the bottom of posts.\n# Can be set at page or section levels, following the hierarchy: page > section > config. See: https://welpo.github.io/tabi/blog/mastering-tabi-settings/#settings-hierarchy\nshow_previous_next_article_links = false\n\n# Invert order of the links to previous and next articles at the bottom of posts.\n# By default, next articles are on the left side of the page and previous articles are on the right side.\n# To reverse the order (next articles on the right and previous articles on the left), set it to true.\n# Can be set at page or section levels, following the hierarchy: page > section > config. See: https://welpo.github.io/tabi/blog/mastering-tabi-settings/#settings-hierarchy\ninvert_previous_next_article_links = false\n\n# Whether the navigation for previous/next article should match the full width of the site (same as the navigation bar at the top) or the article width.\n# To match the navigation bar at the top, set it to true.\nprevious_next_article_links_full_width = true\n\n# Quick navigation buttons.\n# Adds \"go up\" and \"go to comments\" buttons on the bottom right (hidden for mobile).\n# Can be set at page or section levels, following the hierarchy: page > section > config. See: https://welpo.github.io/tabi/blog/mastering-tabi-settings/#settings-hierarchy\nquick_navigation_buttons = false\n\n# Add a Table of Contents to posts, right below the title and metadata.\n# Can be set at page or section levels, following the hierarchy: page > section > config. See: https://welpo.github.io/tabi/blog/mastering-tabi-settings/#settings-hierarchy\ntoc = false\n\n# Date format used when listing posts (main page, /blog section, tag posts list…)\n# Default is \"6th July 2049\" in English and \"%d %B %Y\" in other languages.\nlong_date_format = \"%d %B %Y\"\n\n# Date format used for blog posts.\n# Default is \"6th July 2049\" in English and \"%-d %B %Y\" in other languages.\nshort_date_format = \"\"\n\n# Date format used for the archive page.\n# Default is \"06 July\" in English and \"%d %b\" in other languages.\narchive_date_format = \"\"\n\n# Per-language date format overrides.\n# Examples: Spanish uses \"3 de febrero de 2024\", German uses \"3. Februar 2024\"\ndate_formats = [\n    { lang = \"es\", long = \"%d de %B de %Y\", short = \"%-d %b %Y\" },\n    { lang = \"de\", long = \"%d. %B %Y\", short = \"%d.%m.%Y\", archive = \"%d. %b\" },\n]\n\n# Custom separator used in title tag and posts metadata (between date, time to read, and tags).\nseparator = \"•\"\n\n# Use a shorter layout for All tags listing.\n# Default: tag_name – n post[s]\n# Compact: tag_name^n (superscript number)\ncompact_tags = false\n\n# How tags are sorted in a Tags listing based on templates/tags/list.html.\n# \"name\" for alphabetical, \"frequency\" for descending count of posts.\n# Default: \"name\".\ntag_sorting = \"name\"\n\n# Show clickable tags above cards.html template (e.g. projects/) to filter the displayed items.\n# Loads JS to filter. If JS is disabled, the buttons are links to the tag's page.\n# Can be set at the section or config.toml level, following the hierarchy: section > config. See: https://welpo.github.io/tabi/blog/mastering-tabi-settings/#settings-hierarchy\n# Default: true\nenable_cards_tag_filtering = true\n\n# Invert the order of the site title and page title in the browser tab.\n# Example: true => \"Blog • ~/tabi\", false => \"~/tabi • Blog\"\ninvert_title_order = false\n\n# Full path after the base URL required. So if you were to place it in \"static\" it would be \"/favicon.png\"\nfavicon = \"img/seedling.png\"\n\n# Add an emoji here to use it as favicon.\n# Compatibility: https://caniuse.com/link-icon-svg\nfavicon_emoji = \"🌱\"\n\n# Path to the fallback image for social media cards (the preview image shown when sharing a link on WhatsApp, LinkedIn…).\n# Can be set at page or section levels, following the hierarchy: page > section > config. See: https://welpo.github.io/tabi/blog/mastering-tabi-settings/#settings-hierarchy\n# Learn how to create these images in batch and automatically:\n# https://osc.garden/blog/automating-social-media-cards-zola/\nsocial_media_card = \"index.jpg\"\n\nmenu = [\n    { name = \"blog\", url = \"blog\", trailing_slash = true },\n    { name = \"archive\", url = \"archive\", trailing_slash = true },\n    { name = \"tags\", url = \"tags\", trailing_slash = true },\n    { name = \"projects\", url = \"projects\", trailing_slash = true },\n]\n\n# The RSS icon will be shown if (1) it's enabled and (2) the following variable is set.\n# Set to true to use the default \"rss\" icon, or specify a custom icon name (e.g. \"square-rss\").\n# The icon must exist in static/social_icons/ (without the .svg extension).\n# Note for Zola 0.19.X users: when `feed_filenames` has two filenames, only the first one will be linked in the footer.\nfeed_icon = true\n\n# Show the full post content in the Atom feed.\n# If it's set to false, only the description or summary will be shown.\nfull_content_in_feed = false\n\n# Email address for footer's social section.\n# Protect against spambots:\n# 1. Use base64 for email (convert at https://www.base64encode.org/ or `printf 'your@email.com' | base64`).\n# 2. Or, set 'encode_plaintext_email' to true for auto-encoding (only protects on site, not in public repos).\nemail = \"dGFiaUBvc2MuZ2FyZGVu\"\n# Decoding requires ~400 bytes of JavaScript. If JS is disabled, the email won't be displayed.\nencode_plaintext_email = true  # Setting is ignored if email is already encoded.\n\n# Social media links for the footer.\n# Built-in icons: https://github.com/welpo/tabi/tree/main/static/social_icons\n# To use a custom icon, add it to your site's `static/social_icons` directory.\nsocials = [\n    { name = \"github\", url = \"https://github.com/welpo/\", icon = \"github\" },\n    { name = \"soundcloud\", url = \"https://soundcloud.com/oskerwyld\", icon = \"soundcloud\" },\n    { name = \"instagram\", url = \"https://instagram.com/oskerwyld\", icon = \"instagram\" },\n    { name = \"youtube\", url = \"https://youtube.com/@oskerwyld\", icon = \"youtube\" },\n    { name = \"spotify\", url = \"https://open.spotify.com/artist/5Hv2bYBhMp1lUHFri06xkE\", icon = \"spotify\" },\n]\n\n# Fediverse profile.\n# Adds metadata to feature the author's profile in Mastodon link previews.\n# Example: for @username@example.com, use:\n# fediverse_creator = { handle = \"username\", domain = \"example.com\" }\n\n# Extra menu to show on the footer, below socials section.\nfooter_menu = [\n    {url = \"about\", name = \"about\", trailing_slash = true},\n    {url = \"privacy\", name = \"privacy\", trailing_slash = true},\n    {url = \"https://tabi-stats.osc.garden\", name = \"site_statistics\", trailing_slash = true},\n    {url = \"sitemap.xml\", name = \"sitemap\", trailing_slash = false},\n]\n\n# Enable a copyright notice for the footer, shown between socials and the \"Powered by\" text.\n# $TITLE will be replaced by the website's title.\n# $CURRENT_YEAR will be replaced by the current year.\n# $AUTHOR will be replaced by the `author` variable.\n# $SEPARATOR will be replaced by the `separator` variable.\n# Markdown is supported (links, emphasis, etc).\n# copyright = \"$TITLE © $CURRENT_YEAR $AUTHOR $SEPARATOR Unless otherwise noted, the content in this website is available under the [CC BY-SA 4.0](https://creativecommons.org/licenses/by-sa/4.0/) license.\"\n\n# For multi-language sites, you can set a different copyright for each language.\n# The old way of setting `translated_copyright = true` and using i18n files is deprecated.\n# If a translation is missing for language, the `copyright` value will be used.\n# copyright_translations.es = \"$TITLE © $CURRENT_YEAR $AUTHOR $SEPARATOR A menos que se indique lo contrario, el contenido de esta web está disponible bajo la licencia [CC BY-SA 4.0](https://creativecommons.org/licenses/by-sa/4.0/deed.es).\"\n\n# Custom security headers. What urls should your website be able to connect to?\n# You need to specify the CSP and the URLs associated with the directive.\n# Useful if you want to load remote content safely (embed YouTube videos, which needs frame-src, for example).\n# Default directive is self.\n# Default config, allows for https remote images and embedding YouTube and Vimeo content.\n# This configuration (along with the right webserver settings) gets an A+ in Mozilla's Observatory: https://observatory.mozilla.org\n# Note: to use a Zola built-in syntax highlighting theme, allow unsafe-inline for style-src.\nallowed_domains = [\n    { directive = \"font-src\", domains = [\"'self'\", \"data:\"] },\n    { directive = \"img-src\", domains = [\"'self'\", \"https://*\", \"data:\"] },\n    { directive = \"media-src\", domains = [\"'self'\", \"https://cdn.jsdelivr.net/\"] },\n    { directive = \"script-src\", domains = [\"'self'\"] },\n    { directive = \"style-src\", domains = [\"'self'\"] },\n    { directive = \"frame-src\", domains = [\"player.vimeo.com\", \"https://www.youtube-nocookie.com\"] },\n]\n\n# Enable the CSP directives configured (or default).\n# Can be set at page or section levels, following the hierarchy: page > section > config. See: https://welpo.github.io/tabi/blog/mastering-tabi-settings/#settings-hierarchy\nenable_csp = true\n\n# Font subsetting configuration.\n# This feature helps prevent text flashing in Firefox when using custom fonts.\n# See: https://welpo.github.io/tabi/blog/custom-font-subset/\n# Enable or disable font subsetting completely, both built-in and custom subsets.\nenable_subset = true\n# Use a custom subset of characters for the header.\n# If true, tabi will load the `static/custom_subset.css` file.\n# If false, tabi will use the default language-specific subset (English or Spanish).\ncustom_subset = true\n\n[extra.analytics]\n# Specify which analytics service you want to use.\n# Supported options: [\"goatcounter\", \"umami\", \"plausible\"]\nservice = \"goatcounter\"\n\n# Unique identifier for tracking.\n# For GoatCounter, this is the code you choose during signup.\n# For Umami, this is the website ID.\n# For Plausible, this is the random script name without the extension (e.g. \"pa-XXXXXX\") or the domain name (legacy mode, e.g. \"example.com\").\n# Note: Leave this field empty if you're self-hosting GoatCounter.\n# id = \"yourID\"\n\n# Optional: Specify the URL for self-hosted analytics instances.\n# For GoatCounter: Base URL like \"https://stats.example.com\"\n# For Umami: Base URL like \"https://umami.example.com\"\n# For Plausible: Base URL like \"https://plausible.example.com\"\n# Leave this field empty if you're using the service's default hosting.\nself_hosted_url = \"https://tabi-stats.osc.garden\"\n\n# Optional: For Umami, enable this option to respect users' Do Not Track (DNT) settings. The default is true.\ndo_not_track = true\n\n# giscus support for comments. https://giscus.app\n# Setup instructions: https://welpo.github.io/tabi/blog/comments/#setup\n[extra.giscus]\nenabled_for_all_posts = false  # Enables giscus on all posts. It can be enabled on individual posts by setting `giscus = true` in the [extra] section of a post's front matter.\nautomatic_loading = true  # If set to false, a \"Load comments\" button will be shown.\nrepo = \"welpo/tabi-comments\"\nrepo_id = \"R_kgDOJ59Urw\"  # Find this value in https://giscus.app/\ncategory = \"Announcements\"\ncategory_id = \"DIC_kwDOJ59Ur84CX0QG\"  # Find this value in https://giscus.app/\nmapping = \"slug\"  # Available: pathname; url; title; slug. \"slug\" will use the post's filename (slug); this is the only way to share comments between languages.\nstrict_title_matching = 1  # 1 to enable, 0 to disable. https://github.com/giscus/giscus/blob/main/ADVANCED-USAGE.md#data-strict\nenable_reactions = 1  # 1 to enable, 0 to disable.\ncomment_box_above_comments = false\nlight_theme = \"noborder_light\"\ndark_theme = \"noborder_dark\"\nlang = \"\"  # Leave blank to match the page's language.\nlazy_loading = true\n\n# utterances support for comments. https://utteranc.es\n# Setup instructions: https://welpo.github.io/tabi/blog/comments/#setup\n[extra.utterances]\nenabled_for_all_posts = false  # Enables utterances on all posts. It can be enabled on individual posts by setting `utterances = true` in the [extra] section of a post's front matter.\nautomatic_loading = true  # If set to false, a \"Load comments\" button will be shown.\nrepo = \"yourGithubUsername/yourRepo\"  # https://utteranc.es/#heading-repository\nissue_term = \"slug\"  # Available: pathname; url; title; slug. \"slug\" will use the post's filename (slug); this is the only way to share comments between languages. https://utteranc.es/#heading-mapping\nlabel = \"💬\"  # https://utteranc.es/#heading-issue-label\nlight_theme = \"github-light\"  # https://utteranc.es/#heading-theme\ndark_theme = \"photon-dark\"  # https://utteranc.es/#heading-theme\nlazy_loading = true\n\n# Hyvor Talk support for comments. https://talk.hyvor.com\n[extra.hyvortalk]\nenabled_for_all_posts = false  # Enables hyvortalk on all posts. It can be enabled on individual posts by setting `hyvortalk = true` in the [extra] section of a post's front matter.\nautomatic_loading = true  # If set to false, a \"Load comments\" button will be shown.\nwebsite_id = \"1234\"\npage_id_is_slug = true  # If true, it will use the post's filename (slug) as id; this is the only way to share comments between languages. If false, it will use the entire url as id.\nlang = \"\"  # Leave blank to match the page's language.\npage_author = \"\"  # Email (or base64 encoded email) of the author.\nlazy_loading = true\n\n# Isso support for comments. https://isso-comments.de/\n# You need to self-host the backend first: https://blog.phusion.nl/isso-simple-self-hosted-commenting-system/\n# More info on some settings: https://isso-comments.de/docs/reference/client-config/\n[extra.isso]\nenabled_for_all_posts = false  # Enables Isso on all posts. It can be enabled on individual posts by setting `isso = true` in the [extra] section of a post's front matter.\nautomatic_loading = true  # If set to false, a \"Load comments\" button will be shown.\nendpoint_url = \"\"  # Accepts relative paths like \"/comments/\" or \"/isso/\", as well as full urls like \"https://example.com/comments/\". Include the trailing slash.\npage_id_is_slug = true  # If true, it will use the relative path for the default language as id; this is the only way to share comments between languages. If false, it will use the entire url as id.\nlang = \"\"  # Leave blank to match the page's language.\nmax_comments_top = \"inf\"  # Number of top level comments to show by default. If some comments are not shown, an “X Hidden” link is shown.\nmax_comments_nested = \"5\"  # Number of nested comments to show by default. If some comments are not shown, an “X Hidden” link is shown.\navatar = true\nvoting = true\npage_author_hashes = \"\"  # hash (or list of hashes) of the author.\nlazy_loading = true  # Loads when the comments are in the viewport (using the Intersection Observer API).\n\n[extra.webmentions]\n# To disable for a specific section or page, set webmentions = false in that page/section's front matter's [extra] section.\nenable = false\n# Specify the domain registered with webmention.io.\ndomain = \"\"\n\n# The HTML ID for the object to fill in with the webmention data.\n# Defaults to \"webmentions\"\n# id = \"webmentions\"\n\n# data configuration for the webmention.min.js script\n# The base URL to use for this page. Defaults to window.location\n# page_url =\n\n# Additional URLs to check, separated by |s\n# add_urls\n\n# The maximum number of words to render in reply mentions.\n# wordcount = 20\n\n# The maximum number of mentions to retrieve. Defaults to 30.\n# max_webmentions = 30\n\n# By default, Webmentions render using the mf2 'url' element, which plays\n# nicely with webmention bridges (such as brid.gy and telegraph)\n# but allows certain spoofing attacks. If you would like to prevent\n# spoofing, set this to a non-empty string (e.g. \"true\").\n# prevent_spoofing\n\n# What to order the responses by; defaults to 'published'. See\n# https://github.com/aaronpk/webmention.io#api\n# sort_by\n\n# The order to sort the responses by; defaults to 'up' (i.e. oldest\n# first). See https://github.com/aaronpk/webmention.io#api\n# sort_dir\n\n# If set to a non-empty string (e.g. \"true\"), will display comment-type responses\n# (replies/mentions/etc.) as being part of the reactions\n# (favorites/bookmarks/etc.) instead of in a separate comment list.\n# comments_are_reactions = \"true\"\n\n# h-card configuration\n# Will identify you on the indieweb (see https://microformats.org/wiki/h-card)\n[extra.hcard]\n# Enable home page h-card.\n# enable = true\n# Add your email to the card if extra.email is set and not encoded.\n# with_mail = true\n# Add your social links ('socials' config) to the card.\n# with_social_links = true\n# Homepage url. Defaults to the value of 'base_url'.\n# homepage = \"https://myhomepage.net\"\n# avatar = \"img/profile.webp\"\n# Display name, default to the value of 'author'.\n# full_name = \"John Doe\"\n# Small bio, as shown on social media profiles.\n# biography = \"Fond of the indieweb\"\n#\n# You can add any property from https://microformats.org/wiki/h-card#Properties\n# Make sure to replace all '-' characters by '_'\n# Examples:\n# p_nickname = \"nickname\"\n# p_locality = \"Bordeaux\"\n# p_country_name = \"France\"\n"
  },
  {
    "path": "content/_index.ar.md",
    "content": "+++\ntitle = \"أخر التدوينات\"\nsort_by = \"date\"\ntemplate = \"section.html\"\n\n[extra]\nheader = {title = \"اهلاً انا تابي~\", img = \"img/main.webp\", img_alt = \"أوسكار فرنانديز, كاتب السمة\" }\nsection_path = \"blog/_index.ar.md\"\nmax_posts = 4\nprojects_path = \"projects/_index.ar.md\"\nmax_projects = 3\nshow_projects_first = false\nsocial_media_card = \"ar.jpg\"\n+++\n\nتابي هو قالب Zola سريع وعصري. يهدف ليكون صفحة ومدونة شخصية. يتميز بتقييم مثالي في Lighthouse، وتصميم متجاوب، وسمات داكنة وفاتحة، وشِفرات قصيرة مخصصة، والعديد من المميزات الأخرى.\n\n> ملاحظة: هذه الصفحة هي عرض توضيحي لدعم اللغة العربية في تابي. للحصول على التوثيق الكامل، يرجى الرجوع إلى [النسخة الإنجليزية](@/_index.md).\n"
  },
  {
    "path": "content/_index.ca.md",
    "content": "+++\ntitle = \"Publicacions recents\"\nsort_by = \"date\"\n\n[extra]\nheader = {title = \"Hola! Soc tabi~\", img = \"img/main.webp\", img_alt = \"Óscar Fernández, l'autor de tabi\" }\nsection_path = \"blog/_index.ca.md\"\nmax_posts = 4\nprojects_path = \"projects/_index.ca.md\"\nmax_projects = 3\nshow_projects_first = false\nsocial_media_card = \"ca.jpg\"\n+++\n\ntabi és un tema accessible per a Zola amb [cerca](@/blog/mastering-tabi-settings/index.ca.md#cerca), [suport multilingüe](@/blog/faq-languages/index.ca.md), [JavaScript opcional](@/blog/javascript/index.ca.md), una puntuació perfecta a Lighthouse i documentació exhaustiva. Dissenyat per a llocs web i blogs personals.\n"
  },
  {
    "path": "content/_index.es.md",
    "content": "+++\ntitle = \"Publicaciones recientes\"\nsort_by = \"date\"\n\n[extra]\nheader = {title = \"¡Hola! Soy tabi~\", img = \"img/main.webp\", img_alt = \"Óscar Fernández, el autor de tabi\" }\nsection_path = \"blog/_index.es.md\"\nmax_posts = 4\nprojects_path = \"projects/_index.es.md\"\nmax_projects = 3\nshow_projects_first = false\nsocial_media_card = \"es.jpg\"\n+++\n\ntabi es un tema accesible para [Zola](https://www.getzola.org) con [búsqueda](@/blog/mastering-tabi-settings/index.es.md#busqueda), [soporte multilingüe](@/blog/faq-languages/index.es.md), [JavaScript opcional](@/blog/javascript/index.es.md), una puntuación perfecta en Lighthouse y documentación exhaustiva. Diseñado para sitios web y blogs personales.\n"
  },
  {
    "path": "content/_index.md",
    "content": "+++\ntitle = \"Latest posts\"\nsort_by = \"date\"\n\n[extra]\nheader = {title = \"Hello! I'm tabi~\", img = \"img/main.webp\", img_alt = \"Óscar Fernández, the theme's author\" }\nsection_path = \"blog/_index.md\"\nmax_posts = 4\nprojects_path = \"projects/_index.md\"\nmax_projects = 3\nshow_projects_first = false\nsocial_media_card = \"index.jpg\"\n+++\n\ntabi is an accessible [Zola](https://www.getzola.org) theme with [search](@/blog/mastering-tabi-settings/index.md#search), [multi-language support](@/blog/faq-languages/index.md), [optional JavaScript](@/blog/javascript/index.md), a perfect Lighthouse score, and comprehensive documentation. Crafted for personal websites and blogs.\n"
  },
  {
    "path": "content/archive/_index.ar.md",
    "content": "+++\ntitle = \"الأرشيف\"\ntemplate = \"archive.html\"\n+++\n"
  },
  {
    "path": "content/archive/_index.ca.md",
    "content": "+++\ntitle = \"Arxiu\"\ntemplate = \"archive.html\"\n\n[extra]\nsocial_media_card = \"ca_archive.jpg\"\n+++\n"
  },
  {
    "path": "content/archive/_index.es.md",
    "content": "+++\ntitle = \"Archivo\"\ntemplate = \"archive.html\"\n\n[extra]\nsocial_media_card = \"es_archive.jpg\"\n+++\n"
  },
  {
    "path": "content/archive/_index.md",
    "content": "+++\ntitle = \"Archive\"\ntemplate = \"archive.html\"\n\n[extra]\nsocial_media_card = \"archive.jpg\"\n+++\n"
  },
  {
    "path": "content/blog/_index.ar.md",
    "content": "+++\npaginate_by = 5\ntitle = \"التدوينات\"\nsort_by = \"date\"\ntemplate = \"section.html\"\ninsert_anchor_links = \"right\"\n\n[extra]\nshow_previous_next_article_links = true\n+++\n"
  },
  {
    "path": "content/blog/_index.ca.md",
    "content": "+++\npaginate_by = 5\ntitle = \"Blog\"\nsort_by = \"date\"\ninsert_anchor_links = \"left\"\n\n[extra]\nsocial_media_card = \"ca_blog.jpg\"\nshow_previous_next_article_links = true\n+++\n"
  },
  {
    "path": "content/blog/_index.es.md",
    "content": "+++\npaginate_by = 5\ntitle = \"Blog\"\nsort_by = \"date\"\ninsert_anchor_links = \"left\"\n\n[extra]\nsocial_media_card = \"es_blog.jpg\"\nshow_previous_next_article_links = true\n+++\n"
  },
  {
    "path": "content/blog/_index.md",
    "content": "+++\npaginate_by = 5\ntitle = \"Blog\"\nsort_by = \"date\"\ninsert_anchor_links = \"left\"\n\n[extra]\nsocial_media_card = \"blog.jpg\"\nshow_previous_next_article_links = true\n+++\n"
  },
  {
    "path": "content/blog/comments/index.ca.md",
    "content": "+++\ntitle = \"Afegeix comentaris a les teves publicacions amb aquestes 4 plataformes\"\ndate = 2023-07-14\nupdated = 2025-12-12\ndescription = \"Descobreix com habilitar una secció de comentaris a les teves publicacions utilitzant giscus, utterances, Hyvor Talk, o Isso, permetent la interacció i feedback dels lectors.\"\n\n[taxonomies]\ntags = [\"funcionalitat\", \"tutorial\"]\n\n[extra]\ngiscus = true\nquick_navigation_buttons = true\ntoc = true\nsocial_media_card = \"social_cards/ca_blog_comments.jpg\"\n+++\n\ntabi actualment suporta quatre sistemes de comentaris: [giscus](https://giscus.app/ca) i [utterances](https://utteranc.es/), [Hyvor Talk](https://talk.hyvor.com/) i [Isso](https://isso-comments.de/).\n\ngiscus i utterances són projectes de codi obert que et permeten afegir una secció de comentaris al teu lloc web utilitzant les «issues» (utterances) o «discussions» (giscus) de GitHub. Són perfectes per a generadors de llocs estàtics com Zola, ja que permeten als teus lectors interactuar i deixar comentaris a les teves publicacions sense requerir un backend tradicional o una base de dades.\n\nCom que tots dos es basen en GitHub, giscus i utterances requereixen que els usuaris tinguin un compte a GitHub i autoritzin l'aplicació respectiva. Alternativament, els visitants també poden comentar directament en la discussió o «issue» corresponent a GitHub.\n\nAmbdues són excel·lents eines per afegir comentaris al teu blog, però giscus té alguns avantatges:\n- Més temes.\n- Suport per a reaccions.\n- Respostes a comentaris i vista de conversa.\n- Més segur: utterances requereix habilitar estils en línia no segurs («unsafe inline styles») per establir l'altura del frame; giscus no.\n- Suport multilingüe: utterances només està disponible en anglès; giscus suporta més de 20 idiomes.\n- Desenvolupament més actiu: l'últim commit de giscus, en el moment d'aquesta publicació, va ser fa dos dies. L'últim commit d'utterances es va fer fa més d'un any.\n\nHyvor Talk és una plataforma de comentaris de pagament centrada en la privadesa. Ofereix tots els avantatges del giscus i alguns més, com la moderació i la detecció de correu brossa.\n\nIsso és un sistema de comentaris de codi obert autoallotjat que emmagatzema els comentaris a la seva pròpia base de dades. Un dels seus principals avantatges és la privacitat; no comparteix les dades dels usuaris amb tercers. També té una interfície lleugera i neta, facilitant als teus visitants deixar comentaris. Isso també permet comentaris anònims, potencialment augmentant la participació dels usuaris a la teva pàgina web.\n\n## Configuració\n\n### Sistemes basats en GitHub\n\ngiscus y utterances requereixen una configuració similar. Primer, visita el lloc web del sistema que vulguis habilitar: [giscus.app](https://giscus.app/ca) o [utteranc.es](https://utteranc.es/).\n\nSegueix les instruccions de la secció **Configuració** del lloc web, i tria les opcions que prefereixis. Finalment, estableix els valors que es mostren a la secció **Habilitar giscus/utterances** (el bloc de codi `script`) en la secció corresponent del teu `config.toml`: `[extra.giscus]` o `[extra.utterances]`.\n\n#### giscus\n\ngiscus té més opcions que utterances:\n\n```toml\n[extra.giscus]\nenabled_for_all_posts = false\nautomatic_loading = true\nrepo = \"elTeuNomDUsuariDeGithub/elTeuRepositori\"\nrepo_id = \"LaTevaIDdeRepositori\"\ncategory = \"Anuncis\"\ncategory_id = \"LaTevaIDdeCategoria\"\nmapping = \"slug\"\nstrict_title_matching = 1  # 1 per habilitar, 0 per deshabilitar.\nenable_reactions = 1  # 1 per habilitar, 0 per deshabilitar.\ncomment_box_above_comments = true\nlight_theme = \"noborder_light\"\ndark_theme = \"noborder_dark\"\nlang = \"\"  # Deixa en blanc perquè coincideixi amb l'idioma de la pàgina.\nlazy_loading = true\n```\n#### utterances\n\n```\n[extra.utterances]\nenabled_for_all_posts = false\nautomatic_loading = true\nrepo = \"elTeuNomDUsuariDeGithub/elTeuRepositori\"\nissue_term = \"slug\"\nlabel = \"💬\"\nlight_theme = \"github-light\"\ndark_theme = \"photon-dark\"\nlazy_loading = true\n```\n\n### Hyvor Talk\n\nConfigura el teu lloc web des de la [consola Hyvor Talk](https://talk.hyvor.com/console) i completa la configuració a `config.toml`:\n\n```toml\n[extra.hyvortalk]\nenabled_for_all_posts = false\nautomatic_loading = true\nwebsite_id = \"1234\"\npage_id_is_slug = true\nlang = \"\"\npage_author = \"\"  # Correu (o correu codificat en base64) de l'autor.\nlazy_loading = true\n```\n\n### Isso\n\nPer habilitar Isso, primer hauràs d'instal·lar i executar un servidor Isso ([aquí tens una guia útil](https://blog.phusion.nl/isso-simple-self-hosted-commenting-system/#1installingisso)). Després, completa aquestes configuracions a `config.toml`:\n\n```toml\n[extra.isso]\nenabled_for_all_posts = false\nautomatic_loading = true\nendpoint_url = \"https://example.com/comments/\"  # URL a Isso.\npage_id_is_slug = true\nlang = \"\"\nmax_comments_top = \"inf\"\nmax_comments_nested = \"5\"\navatar = true\nvoting = true\npage_author_hashes = \"\"\nlazy_loading = true\n```\n\n### Configuracions comunes\n\nLa opció `enabled_for_all_posts = true` habilita globalment el sistema de comentaris corresponent.\n\nAlternativament, pots habilitar els comentaris a publicacions concretes afegint el nom del sistema (`utterances`, `giscus`, `hyvortalk` o `isso`) ` = true`. Per exemple, així és com habilitaries giscus:\n\n```toml,hl_lines=09-10\ntitle = \"L'art de l'entremaliadura segons Shin-Chan\ndate = 1990-02-14\ndescription = \"Descobreix com les travessures poden canviar la teva perspectiva de vida.\"\n\n[taxonomies]\ntags = [\"personal\", \"travessures\"]\n\n[extra]\ngiscus = true\n```\n\nSi accidentalment habilites més d'un sistema, Zola mostrarà un error.\n\nSi el teu lloc web té múltiples idiomes amb publicacions coincidents (com aquesta demo), i t'agradaria compartir comentaris entre idiomes, has d'utilitzar `issue_term = \"slug\"` (per giscus y utterances) o `page_id_is_slug = true` (per Hyvor Talk o Isso). Això utilitzarà el nom de l'arxiu Markdown (sense l'etiqueta d'idioma) com a identificador. Totes les altres opcions crearan diferents seccions de comentaris per a cada idioma.\n\n## Exemple en viu\n\nA continuació trobaràs el widget de giscus amb la configuració mostrada [a dalt](#giscus).\n"
  },
  {
    "path": "content/blog/comments/index.es.md",
    "content": "+++\ntitle = \"Añade comentarios a tus publicaciones con estas 4 plataformas\"\ndate = 2023-07-14\nupdated = 2025-12-12\ndescription = \"Descubre cómo habilitar una sección de comentarios en tus publicaciones usando giscus, utterances, Hyvor Talk, o Isso, permitiendo la interacción y feedback de los lectores.\"\n\n[taxonomies]\ntags = [\"funcionalidad\", \"tutorial\"]\n\n[extra]\ngiscus = true\nquick_navigation_buttons = true\ntoc = true\nsocial_media_card = \"social_cards/es_blog_comments.jpg\"\n+++\n\ntabi actualmente soporta cuatro sistemas de comentarios: [giscus](https://giscus.app/es) y [utterances](https://utteranc.es/), [Hyvor Talk](https://talk.hyvor.com/) e [Isso](https://isso-comments.de/).\n\ngiscus y utterances son proyectos de código abierto que te permiten añadir una sección de comentarios a tu sitio web usando las «issues» (utterances) o «discussions» (giscus) de GitHub. Son perfectos para generadores de sitios estáticos como Zola, ya que permiten a tus lectores interactuar y dejar comentarios en tus publicaciones sin requerir un backend tradicional ni una base de datos.\n\nAl estar basados en GitHub, giscus y utterances requieren que los usuarios tengan una cuenta en dicha plataforma y autoricen la respectiva aplicación. Alternativamente, los visitantes también pueden comentar directamente en la discusión o «issue» correspondiente de GitHub.\n\nAmbas son excelentes herramientas para agregar comentarios a tu blog, pero giscus tiene algunas ventajas:\n- Más temas.\n- Soporte para reacciones.\n- Respuestas a comentarios y vista de conversación.\n- Más seguro: utterances requiere habilitar estilos en línea inseguros («unsafe inline styles») para ajustar la altura del frame; giscus no.\n- Soporte multilingüe: utterances solo está disponible en inglés; giscus soporta más de 20 idiomas.\n- Desarrollo más activo: el último commit de giscus, a fecha de esta publicación, fue hace una dos días. El último commit de utterances fue hace más de un año.\n\nHyvor Talk es una plataforma de comentarios de pago centrada en la privacidad. Ofrece todas las ventajas de giscus y algunas más, como moderación y detección de spam.\n\nIsso es un sistema de comentarios de código abierto y autoalojado que almacena los comentarios en su propia base de datos. Una de sus principales ventajas es la privacidad; no comparte los datos de los usuarios con terceros. También tiene una interfaz ligera y limpia, lo que facilita que tus visitantes dejen comentarios. Isso también permite comentarios anónimos, lo que podría aumentar la participación de los usuarios en tu sitio web.\n\n## Configuración\n\n### Sistemas basados en GitHub\n\ngiscus y utterances requieren una configuración similar. Primero, visita el sitio web del sistema que quieras habilitar: [giscus.app](https://giscus.app/es) o [utteranc.es](https://utteranc.es/).\n\nSigue las instrucciones de la sección **Configuración** del sitio web, y elige las opciones que prefieras. Luego, establece los valores que se muestran en la sección **Habilitar giscus/utterances** (el bloque de código `script`) en la sección correspondiente de tu `config.toml`: `[extra.giscus]` o `[extra.utterances]`.\n\n#### giscus\n\ngiscus tiene algunos ajustes más que utterances:\n\n```toml\n[extra.giscus]\nenabled_for_all_posts = false\nautomatic_loading = true\nrepo = \"tuNombreDeUsuarioDeGithub/tuRepositorio\"\nrepo_id = \"TuIDdeRepositorio\"\ncategory = \"Anuncios\"\ncategory_id = \"TuIDdeCategoría\"\nmapping = \"slug\"\nstrict_title_matching = 1  # 1 para habilitar, 0 para deshabilitar.\nenable_reactions = 1  # 1 para habilitar, 0 para deshabilitar.\ncomment_box_above_comments = true\nlight_theme = \"noborder_light\"\ndark_theme = \"noborder_dark\"\nlang = \"\"  # Deja en blanco para que coincida con el idioma de la página.\nlazy_loading = true\n```\n\n#### utterances\n\n```toml\n[extra.utterances]\nenabled_for_all_posts = false\nautomatic_loading = true\nrepo = \"tuNombreDeUsuarioDeGithub/tuRepositorio\"\nissue_term = \"slug\"\nlabel = \"💬\"\nlight_theme = \"github-light\"\ndark_theme = \"photon-dark\"\nlazy_loading = true\n```\n\n### Hyvor Talk\n\nConfigura tu web desde la [consola de Hyvor Talk](https://talk.hyvor.com/console) y rellena las opciones en `config.toml`:\n\n```toml\n[extra.hyvortalk]\nenabled_for_all_posts = false\nautomatic_loading = true\nwebsite_id = \"1234\"\npage_id_is_slug = true\nlang = \"\"\npage_author = \"\"  # Correo (o correo codificado en base64) del autor.\nlazy_loading = true\n```\n\n### Isso\n\nPara habilitar Isso, primero necesitarás instalar y ejecutar un servidor Isso ([aquí tienes una guía útil](https://blog.phusion.nl/isso-simple-self-hosted-commenting-system/#1installingisso)). Luego, completa estas configuraciones en `config.toml`:\n\n```toml\n[extra.isso]\nenabled_for_all_posts = false\nautomatic_loading = true\nendpoint_url = \"https://example.com/comments/\"  # URL a Isso.\npage_id_is_slug = true\nlang = \"\"\nmax_comments_top = \"inf\"\nmax_comments_nested = \"5\"\navatar = true\nvoting = true\npage_author_hashes = \"\"\nlazy_loading = true\n```\n\n### Ajustes comunes\n\nLa opción `enabled_for_all_posts = true` habilitará globalmente el sistema de comentarios correspondiente.\n\nAlternativamente, puedes habilitar los comentarios en publicaciones concretas añadiendo el nombre del sistema (`utterances`, `giscus`, `hyvortalk` o `isso`) `= true`. Por ejemplo, así habilitarías giscus:\n\n```toml,hl_lines=09-10\ntitle = \"Los molinos de viento de mi vida: reflexiones de un escudero\"\ndate = 1605-01-16\ndescription = \"Mi viaje junto a Don Quijote, enfrentándome a gigantes imaginarios y descubriendo las verdaderas batallas de la vida.\"\n\n[taxonomies]\ntags = [\"personal\", \"reflexiones\"]\n\n[extra]\ngiscus = true\n```\n\nSi accidentalmente habilitas más de un sistema, Zola mostrará un error.\n\nSi tu web tiene múltiples idiomas con publicaciones coincidentes (como esta demo), y te gustaría compartir comentarios entre idiomas, debes usar `issue_term = \"slug\"` (en el caso de giscus y utterances) o `page_id_is_slug = true` (para Hyvor Talk e Isso). Esto usará el nombre del archivo Markdown (sin la etiqueta de idioma) como identificador. Todas las demás opciones crearán diferentes secciones de comentarios para cada idioma.\n\n\n## Ejemplo en vivo\n\nAl final de esta publicación encontrarás el widget de giscus usando los ajustes mostrados [arriba](#giscus).\n"
  },
  {
    "path": "content/blog/comments/index.md",
    "content": "+++\ntitle = \"Add comments to your posts with these 4 comment systems\"\ndate = 2023-07-14\nupdated = 2025-12-12\ndescription = \"Discover how to enable a comments section on your posts using giscus, utterances, Hyvor Talk, or Isso, enabling reader interaction and feedback.\"\n\n[taxonomies]\ntags = [\"showcase\", \"tutorial\"]\n\n[extra]\ngiscus = true\nquick_navigation_buttons = true\ntoc = true\nsocial_media_card = \"social_cards/blog_comments.jpg\"\n+++\n\ntabi currently supports four comment systems: [giscus](https://giscus.app/), [utterances](https://utteranc.es/), [Hyvor Talk](https://talk.hyvor.com/), and [Isso](https://isso-comments.de/).\n\ngiscus and utterances are open-source projects that let you add a comments section to your website using GitHub issues (utterances) or discussions (giscus). They are perfect for static site generators like Zola, since they enable your readers to interact and leave comments on your posts without requiring a traditional backend or database.\n\nAs they are based on GitHub, giscus and utterances require users to have a GitHub account and authorize the respective app. Alternatively, visitors can also comment directly on the corresponding GitHub discussion or issue.\n\nBoth are great tools for adding comments to your blog, but giscus has a few advantages:\n- More themes.\n- Support for reactions.\n- Comment replies and conversation view.\n- Safer: utterances requires enabling unsafe inline styles to set the height of the frame; giscus doesn't.\n- Multilanguage support: utterances is only available in English; giscus supports over 20 languages.\n- More active development: giscus' last commit, as of this post, was a two days ago. utterances' last commit was over a year ago.\n\nHyvor Talk is a paid privacy-focused commenting platform. It offers all of the giscus' advantages, and a few more, like moderation and spam detection.\n\nIsso is an open-source self-hosted commenting system that stores comments in its own database. One of its main advantages is privacy; it does not share user data with third parties. It also has a lightweight and clean interface, making it easy for your visitors to leave comments. Isso also allows anonymous comments, potentially increasing user engagement on your website.\n\n## Setup\n\n### GitHub based systems\n\nThe configuration of both giscus and utterances is quite similar. First, visit the website of the system you want to enable: [giscus.app](https://giscus.app/) or [utteranc.es](https://utteranc.es/).\n\nFollow the instructions on the **Configuration** section of the website, and set it up it to your liking. Then, set the values shown on the **Enable giscus/utterances** section (the `script` codeblock) on the proper section of your `config.toml`: `[extra.giscus]` or `[extra.utterances]`.\n\n#### giscus\n\ngiscus has a few more settings than utterances:\n\n```toml\n[extra.giscus]\nenabled_for_all_posts = false\nautomatic_loading = true\nrepo = \"yourGithubUsername/yourRepo\"\nrepo_id = \"YourRepoID\"\ncategory = \"Announcements\"\ncategory_id = \"YourCategoryID\"\nmapping = \"slug\"\nstrict_title_matching = 1  # 1 to enable, 0 to disable.\nenable_reactions = 1  # 1 to enable, 0 to disable.\ncomment_box_above_comments = true\nlight_theme = \"noborder_light\"\ndark_theme = \"noborder_dark\"\nlang = \"\"  # Leave blank to match the page's language.\nlazy_loading = true\n```\n\n#### utterances\n\n```toml\n[extra.utterances]\nenabled_for_all_posts = false\nautomatic_loading = true\nrepo = \"yourgithubuser/yourrepo\"\nissue_term = \"slug\"\nlabel = \"💬\"\nlight_theme = \"github-light\"\ndark_theme = \"photon-dark\"\nlazy_loading = true\n```\n\n### Hyvor Talk\n\nSet up your website from the [Hyvor Talk console](https://talk.hyvor.com/console) and fill in the settings in `config.toml`:\n\n```toml\n[extra.hyvortalk]\nenabled_for_all_posts = false\nautomatic_loading = true\nwebsite_id = \"1234\"\npage_id_is_slug = true\nlang = \"\"\npage_author = \"\"  # Email (or base64 encoded email) of the author.\nlazy_loading = true\n```\n\n### Isso\n\nTo enable Isso, you'll first need to install and run an Isso server ([here's a useful guide](https://blog.phusion.nl/isso-simple-self-hosted-commenting-system/#1installingisso)). Then, complete these settings in `config.toml`:\n\n```toml\n[extra.isso]\nenabled_for_all_posts = false\nautomatic_loading = true\nendpoint_url = \"https://example.com/comments/\"  # URL to Isso host.\npage_id_is_slug = true\nlang = \"\"\nmax_comments_top = \"inf\"\nmax_comments_nested = \"5\"\navatar = true\nvoting = true\npage_author_hashes = \"\"\nlazy_loading = true\n```\n\n### Common settings\n\nSetting `enabled_for_all_posts = true` for a comment system will enable it globally.\n\nAlternatively, enable comments on an individual post's front matter by adding the name of the system (`utterances`, `giscus`, `hyvortalk`, or `isso`) `= true`. For example, this is how you would enable giscus:\n\n```toml,hl_lines=09-10\ntitle = \"Bears, Beets, Battlestar Galactica: The Dwight Schrute Guide to Life\"\ndate = 2007-04-26\ndescription = \"Lessons learned from beet farming and paper sales.\"\n\n[taxonomies]\ntags = [\"personal\", \"beets\"]\n\n[extra]\ngiscus = true\n```\n\nIf you accidentally enable more than one system, your site will fail to build with an error.\n\nIf your site has multiple languages with matching posts (like this demo), and you'd like to share comments between languages, you must use `issue_term = \"slug\"` (for giscus and utterances) or `page_id_is_slug = true` (for Hyvor Talk or Isso). This will use the name of the Markdown file (sans the language tag) as the identifier. All other options will create different comment sections for each language.\n\n## Live example\n\nBelow you'll find the giscus widget using the settings shown [above](#giscus).\n"
  },
  {
    "path": "content/blog/custom-font-subset/index.ca.md",
    "content": "+++\ntitle = \"Optimitza la càrrega amb un subconjunt de font personalitzat\"\ndate = 2023-04-29\nupdated = 2025-01-12\ndescription = \"Aprèn com crear un subconjunt personalitzat que només inclogui els glifs necessaris.\"\n\n[taxonomies]\ntags = [\"funcionalitat\", \"tutorial\"]\n\n[extra]\nsocial_media_card = \"social_cards/ca_blog_custom_font_subset.jpg\"\n+++\n\n## El problema\n\nLes fonts personalitzades causen parpelleig de text a Firefox. Per veure un gif i més detalls, mira [aquesta issue](https://github.com/welpo/tabi/issues/75).\n\n## La solució\n\nPer solucionar això, tabi carrega un subconjunt de glifs per a l'encapçalament. Donat que això augmenta lleugerament el temps de càrrega inicial, és una bona idea intentar minimitzar la mida d'aquest subconjunt.\n\nPer defecte, tabi inclou fitxers de subconjunts per a caràcters en anglès i espanyol (amb alguns símbols). Aquests fitxers es carreguen quan la pàgina o el lloc web de Zola està en aquest idioma.\n\n{% admonition(type=\"tip\") %}\nSi estàs fent servir una font personalitzada, pots crear el teu propi subconjunt (segueix llegint) o desactivar completament els subconjunts predeterminats amb `enable_subset = false` a `config.toml`.\n{% end %}\n\nPer a una optimització addicional, a continuació t'expliquem com crear un subconjunt de fonts personalitzat que només inclogui els caràcters utilitzats en el teu encapçalament.\n\n## Requisits\n\nInstal·la aquestes eines:\n\n- [fonttools](https://github.com/fonttools/fonttools)\n\n- [brotli](https://github.com/google/brotli)\n\nExecuta `pip install fonttools brotli` per instal·lar totes dues.\n\n## L'script\n\nEl següent script pren un fitxer `config.toml` i un fitxer de font com a entrada, extreu els caràcters necessaris, crea un subconjunt de la font i genera un fitxer CSS que conté el subconjunt codificat en base64.\n\n```bash\n#!/usr/bin/env bash\n\nusage() {\n    echo \"Usage: $0 [--config | -c CONFIG_FILE] [--font | -f FONT_FILE] [--output | -o OUTPUT_PATH]\"\n    echo\n    echo \"Options:\"\n    echo \"  --config, -c   Path to the config.toml file.\"\n    echo \"  --font, -f     Path to the font file.\"\n    echo \"  --output, -o   Output path for the generated custom_subset.css file (default: current directory)\"\n    echo \"  --help, -h     Show this help message and exit\"\n}\n\n# La sortida per defecte és el directori actual.\noutput_path=\".\"\n\n# Opcions de la línia de comandes.\nwhile [ \"$#\" -gt 0 ]; do\n    case \"$1\" in\n        --config|-c)\n            config_file=\"$2\"\n            shift 2\n            ;;\n        --font|-f)\n            font_file=\"$2\"\n            shift 2\n            ;;\n        --output|-o)\n            output_path=\"$2\"\n            shift 2\n            ;;\n        --help|-h)\n            usage\n            exit 0\n            ;;\n        *)\n            echo \"Unknown option: $1\"\n            usage\n            exit 1\n            ;;\n    esac\ndone\n\n# Comprova si s'han proporcionat les opcions -c i -f.\nif [ -z \"$config_file\" ]; then\n    echo \"Error: --config|-c option is required.\"\n    usage\n    exit 1\nfi\n\nif [ -z \"$font_file\" ]; then\n    echo \"Error: --font|-f option is required.\"\n    usage\n    exit 1\nfi\n\n# Comprova si els fitxers de configuració i de font existeixen.\nif [ ! -f \"$config_file\" ]; then\n    echo \"Error: Config file '$config_file' not found.\"\n    exit 1\nfi\n\nif [ ! -f \"$font_file\" ]; then\n    echo \"Error: Font file '$font_file' not found.\"\n    exit 1\nfi\n\n# Extreu el títol i els noms del menú del fitxer de configuració.\ntitle=$(awk -F' = ' '/^title/{print $2}' \"$config_file\" | tr -d '\"')\nmenu_names=$(awk -F' = ' '/^menu/{f=1;next} /socials/{f=0} f && /name/{print $2}' \"$config_file\" | cut -d',' -f1 | tr -d '\"' )\nlanguage_names=$(awk -F' = ' '/^language_name\\./{print $2}' \"$config_file\" | tr -d '\"' )\n\n# Si el lloc web és multilingüe, obté les traduccions del menú.\nif [ -n \"$language_names\" ]; then\n    for menu_name in $menu_names; do\n        # Find the line with the menu name inside a [languages.*.translations] section and get the translated menus.\n        menu_translation=$(awk -F' = ' \"/\\\\[languages.*\\\\.translations\\\\]/{f=1;next} /^\\\\[/ {f=0} f && /$menu_name =/{print \\$2}\" \"$config_file\" | tr -d '\"' )\n        # Add the found menu value to the translations string\n        menu_names+=\"$menu_translation\"\n    done\nfi\n\n# Combina les cadenes extretes.\ncombined=\"$title$menu_names$language_names\"\n\n# Obté els caràcters únics.\nunique_chars=$(echo \"$combined\" | grep -o . | sort -u | tr -d '\\n')\n\n# Crea un fitxer temporal per a subset.woff2.\ntemp_subset=$(mktemp)\n\n# Crea el subconjunto.\npyftsubset \"$font_file\" \\\n    --text=\"$unique_chars\" \\\n    --layout-features=\"*\" --flavor=\"woff2\" --output-file=\"$temp_subset\" --with-zopfli\n\n# Codifica en base64 el fitxer temporal subset.woff2 i crea el fitxer CSS.\nbase64_encoded_font=$(base64 -i \"$temp_subset\")\necho \"@font-face{font-family:\\\"Inter Subset\\\";src:url(data:application/font-woff2;base64,$base64_encoded_font);}\" > \"$output_path/custom_subset.css\"\n\n# Elimina el fitxer temporal subset.woff2.\nrm \"$temp_subset\"\n```\n\n## Ús\n\nGuarda l'script a algun lloc com `~/bin/subset_font`. Fes-lo executable amb `chmod +x ~/bin/subset_font`.\n\nAra pots executar-lo amb les opcions requerides `--config` i `--font`:\n\n```bash\n~/bin/subset_font --config path/to/config.toml --font path/to/font.woff2\n```\n\nDe forma predeterminada, això generarà un fitxer `custom_subset.css` al directori actual. Utilitza `-o` o `--output` per especificar una ruta diferent:\n\n```bash\n~/bin/subset_font -c path/to/config.toml -f path/to/font.woff2 -o path/to/output\n```\n\nCol·loca aquest fitxer `custom_subset.css` dins del directori `static/` del teu projecte de Zola.\n\n## Automatització amb un Pre-commit Hook\n\nÉs possible que canviïs el títol o les opcions del menú del teu lloc web, la qual cosa faria que el subconjunt personalitzat deixés de ser útil.\n\nPer automatitzar el procés de creació d'aquest fitxer, pots integrar l'script en un ganxo (hook) pre-commit de Git que s'activi en detectar canvis al fitxer `config.toml`, executi l'script i guardi el fitxer CSS resultant al directori `static/` del teu lloc web.\n\n1. Crea un fitxer `.git/hooks/pre-commit` al teu projecte de Git, si encara no existeix.\n\n2. Fes-lo executable amb `chmod +x .git/hooks/pre-commit`.\n\n3. Afegeix el següent codi al fitxer:\n\n```bash\n# Comprova si config.toml s'ha modificat.\nif git diff --cached --name-only | grep -q \"config.toml\"; then\n    echo \"config.toml modified. Running subset_font…\"\n\n    # Executa l'script subset_font.\n    ~/bin/subset_font -c config.toml -f static/fonts/Inter4.woff2 -o static/\n\n    # Afegeix el fitxer subset.css generat al commit.\n    git add static/custom_subset.css\nfi\n```\n\nAsegura't de modificar l'script perquè coincideixi amb la ruta on has guardat l'script `subset_font`. Les rutes de configuració i font haurien de funcionar correctament amb la configuració predeterminada de tabi.\n\nAra, cada vegada que facis canvis al teu projecte de Git i facis commit, el ganxo pre-commit verificarà les modificacions al fitxer `config.toml` i executarà automàticament l'script `subset_font` per actualitzar el fitxer `custom_subset.css`.\n\nPer cert, si t'interessa una forma d'actualitzar automàticament la data de les teves publicacions a Zola o comprimir automàticament els teus fitxers PNG, fes un cop d'ull a [aquesta publicació](https://osc.garden/ca/blog/zola-date-git-hook/).\n\nSi desitges utilitzar tots els scripts alhora (compressió de fitxers PNG, actualització de la data i creació del subconjunt de fonts), combina el seu codi en un sol fitxer `.git/hooks/pre-commit`.\n"
  },
  {
    "path": "content/blog/custom-font-subset/index.es.md",
    "content": "+++\ntitle = \"Optimiza la carga con un subconjunto de fuente personalizado\"\ndate = 2023-04-29\nupdated = 2025-01-12\ndescription = \"Aprende cómo crear un subconjunto personalizado que solo incluya los glifos necesarios.\"\n\n[taxonomies]\ntags = [\"funcionalidad\", \"tutorial\"]\n\n[extra]\nsocial_media_card = \"social_cards/es_blog_custom_font_subset.jpg\"\n+++\n\n## El problema\n\nLas fuentes personalizadas causan parpadeo de texto en Firefox. Para ver un gif y más detalles, mira [esta issue](https://github.com/welpo/tabi/issues/75).\n\n## La solución\n\nPara solucionar esto, tabi carga un subconjunto de glifos para el encabezado. Dado que esto aumenta ligeramente el tiempo de carga inicial, es una buena idea tratar de minimizar el tamaño de este subconjunto.\n\nPor defecto, tabi incluye archivos de subconjuntos para caracteres en inglés y español (con algunos símbolos). Estos archivos se cargan cuando la página o el sitio de Zola está en ese idioma.\n\n{% admonition(type=\"tip\") %}\nSi estás usando una fuente personalizada, puedes crear tu propio subconjunto (ver más abajo) o desactivar completamente los subconjuntos predeterminados con `enable_subset = false` en tu `config.toml`.\n{% end %}\n\nPara una optimización adicional, a continuación verás cómo crear un subconjunto de fuentes personalizado que solo incluya los caracteres utilizados en tu encabezado.\n\n## Requisitos\n\nInstala estas herramientas:\n\n- [fonttools](https://github.com/fonttools/fonttools)\n\n- [brotli](https://github.com/google/brotli)\n\nEjecuta `pip install fonttools brotli` para instalar ambas.\n\n## El script\n\nEl script que sigue toma un archivo `config.toml` y un archivo de fuente como entrada, extrae los caracteres necesarios, crea un subconjunto de la fuente y genera un archivo CSS que contiene el subconjunto codificado en base64.\n\n```bash\n#!/usr/bin/env bash\n\nusage() {\n    echo \"Usage: $0 [--config | -c CONFIG_FILE] [--font | -f FONT_FILE] [--output | -o OUTPUT_PATH]\"\n    echo\n    echo \"Options:\"\n    echo \"  --config, -c   Path to the config.toml file.\"\n    echo \"  --font, -f     Path to the font file.\"\n    echo \"  --output, -o   Output path for the generated custom_subset.css file (default: current directory)\"\n    echo \"  --help, -h     Show this help message and exit\"\n}\n\n# La salida predeterminada es el directorio actual.\noutput_path=\".\"\n\n# Analiza las opciones de la línea de comandos.\nwhile [ \"$#\" -gt 0 ]; do\n    case \"$1\" in\n        --config|-c)\n            config_file=\"$2\"\n            shift 2\n            ;;\n        --font|-f)\n            font_file=\"$2\"\n            shift 2\n            ;;\n        --output|-o)\n            output_path=\"$2\"\n            shift 2\n            ;;\n        --help|-h)\n            usage\n            exit 0\n            ;;\n        *)\n            echo \"Unknown option: $1\"\n            usage\n            exit 1\n            ;;\n    esac\ndone\n\n# Comprueba si se proporcionan las opciones -c y -f.\nif [ -z \"$config_file\" ]; then\n    echo \"Error: --config|-c option is required.\"\n    usage\n    exit 1\nfi\n\nif [ -z \"$font_file\" ]; then\n    echo \"Error: --font|-f option is required.\"\n    usage\n    exit 1\nfi\n\n# Comprueba si existen los archivos de configuración y de fuentes.\nif [ ! -f \"$config_file\" ]; then\n    echo \"Error: Config file '$config_file' not found.\"\n    exit 1\nfi\n\nif [ ! -f \"$font_file\" ]; then\n    echo \"Error: Font file '$font_file' not found.\"\n    exit 1\nfi\n\n# Extrae el título y los nombres de los menús del archivo de configuración.\ntitle=$(awk -F' = ' '/^title/{print $2}' \"$config_file\" | tr -d '\"')\nmenu_names=$(awk -F' = ' '/^menu/{f=1;next} /socials/{f=0} f && /name/{print $2}' \"$config_file\" | cut -d',' -f1 | tr -d '\"' )\nlanguage_names=$(awk -F' = ' '/^language_name\\./{print $2}' \"$config_file\" | tr -d '\"' )\n\n# Si el sitio es multilingüe, obtiene las traducciones de los menús.\nif [ -n \"$language_names\" ]; then\n    for menu_name in $menu_names; do\n        # Find the line with the menu name inside a [languages.*.translations] section and get the translated menus.\n        menu_translation=$(awk -F' = ' \"/\\\\[languages.*\\\\.translations\\\\]/{f=1;next} /^\\\\[/ {f=0} f && /$menu_name =/{print \\$2}\" \"$config_file\" | tr -d '\"' )\n        # Add the found menu value to the translations string\n        menu_names+=\"$menu_translation\"\n    done\nfi\n\n# Combina las cadenas extraídas.\ncombined=\"$title$menu_names$language_names\"\n\n# Obtiene los caracteres únicos.\nunique_chars=$(echo \"$combined\" | grep -o . | sort -u | tr -d '\\n')\n\n# Crea un archivo temporal para subset.woff2.\ntemp_subset=$(mktemp)\n\n# Crea el subconjunto.\npyftsubset \"$font_file\" \\\n    --text=\"$unique_chars\" \\\n    --layout-features=\"*\" --flavor=\"woff2\" --output-file=\"$temp_subset\" --with-zopfli\n\n# Codifica en Base64 el archivo temporal subset.woff2 y crea el archivo CSS.\nbase64_encoded_font=$(base64 -i \"$temp_subset\")\necho \"@font-face{font-family:\\\"Inter Subset\\\";src:url(data:application/font-woff2;base64,$base64_encoded_font);}\" > \"$output_path/custom_subset.css\"\n\n# Elimina el archivo temporal subset.woff2.\nrm \"$temp_subset\"\n```\n\n## Uso\n\nGuarda el script en algún lugar como `~/bin/subset_font`. Hazlo ejecutable con `chmod +x ~/bin/subset_font`.\n\nAhora puedes ejecutarlo con las opciones requeridas `--config` y `--font`:\n\n```bash\n~/bin/subset_font --config path/to/config.toml --font path/to/font.woff2\n```\n\nDe forma predeterminada, esto generará un archivo `custom_subset.css` en el directorio actual. Usa `-o` o `--output` para especificar una ruta diferente:\n\n```bash\n~/bin/subset_font -c path/to/config.toml -f path/to/font.woff2 -o path/to/output\n```\n\nColoca este archivo `custom_subset.css` dentro del directorio `static/`.\n\n\n## Automatización con un Pre-commit Hook\n\nEs posible que cambies el título o las opciones del menú de tu sitio, lo que haría que el subconjunto personalizado deje de ser útil.\n\nPara automatizar el proceso de creación de este archivo, puedes integrar el script en un gancho (hook) pre-commit de Git que se active al detectar cambios en el archivo `config.toml`, ejecute el script y guarde el archivo CSS resultante en el directorio `static/` de tu sitio.\n\n1. Crea un archivo `.git/hooks/pre-commit` en tu proyecto de Git, si aún no existe.\n\n2. Hazlo ejecutable con `chmod +x .git/hooks/pre-commit`.\n\n3. Agrega el siguiente código al archivo:\n\n```bash\n# Comprueba si config.toml se ha modificado.\nif git diff --cached --name-only | grep -q \"config.toml\"; then\n    echo \"config.toml modified. Running subset_font…\"\n\n    # Ejecuta el script subset_font.\n    ~/bin/subset_font -c config.toml -f static/fonts/Inter4.woff2 -o static/\n\n    # Añadie el subset.css recién generado al commit.\n    git add static/custom_subset.css\nfi\n```\n\nAsegúrate de modificar el script para que coincida con la ruta donde has guardado el script `subset_font`. Las rutas de configuración y fuente deberían funcionar correctamente con la configuración predeterminada de tabi.\n\nAhora, cada vez que hagas cambios en tu proyecto de Git y los confirmes, el gancho pre-commit verificará las modificaciones en el archivo `config.toml` y ejecutará automáticamente el script `subset_font` para actualizar el archivo `custom_subset.css`.\n\nPor cierto, si te interesa una forma de actualizar automáticamente la fecha de tus publicaciones en Zola o comprimir automáticamente tus archivos PNG, echa un vistazo a [esta publicación](https://osc.garden/es/blog/zola-date-git-hook/).\n\nSi deseas utilizar todos los scripts a la vez (compresión de archivos PNG, actualización de la fecha y creación del subconjunto de fuentes), combina su código en un solo archivo `.git/hooks/pre-commit`.\n"
  },
  {
    "path": "content/blog/custom-font-subset/index.md",
    "content": "+++\ntitle = \"Optimise loading times with a custom font subset\"\ndate = 2023-04-29\nupdated = 2025-01-12\ndescription = \"Learn how to create a custom subset that only includes the necessary glyphs.\"\n\n[taxonomies]\ntags = [\"showcase\", \"tutorial\"]\n\n[extra]\nsocial_media_card = \"social_cards/blog_custom_font_subset.jpg\"\n+++\n\n## The problem\n\nCustom fonts cause flashing text in Firefox. For a gif and more details, see [this issue](https://github.com/welpo/tabi/issues/75).\n\n## The solution\n\nTo fix this, tabi loads a subset of glyphs for the header. Since this (slightly) increases the initial load time, it's a good idea to try and minimise the size of this subset.\n\nBy default, there are subset files for English and Spanish characters (with a few symbols). These files are loaded when the Zola page/site is set to that language.\n\n{% admonition(type=\"tip\") %}\nIf you're using a custom font, either create your custom subset (see below) or disable the built-in subsets completely with `enable_subset = false` in your `config.toml`.\n{% end %}\n\nHere's how you can create a custom font subset that only includes the characters used in your header, for maximum efficiency.\n\n## Requirements\n\nInstall these tools:\n\n- [fonttools](https://github.com/fonttools/fonttools)\n\n- [brotli](https://github.com/google/brotli)\n\nRun `pip install fonttools brotli` to install both.\n\n## The script\n\nThe script below takes a `config.toml` file and a font file as input, extracts the necessary characters, creates a subset of the font, and generates a CSS file containing the base64 encoded subset.\n\n```bash\n#!/usr/bin/env bash\n\nusage() {\n    echo \"Usage: $0 [--config | -c CONFIG_FILE] [--font | -f FONT_FILE] [--output | -o OUTPUT_PATH]\"\n    echo\n    echo \"Options:\"\n    echo \"  --config, -c   Path to the config.toml file.\"\n    echo \"  --font, -f     Path to the font file.\"\n    echo \"  --output, -o   Output path for the generated subset.css file (default: current directory)\"\n    echo \"  --help, -h     Show this help message and exit\"\n}\n\n# Default output is current directory.\noutput_path=\".\"\n\n# Parse command line options\nwhile [ \"$#\" -gt 0 ]; do\n    case \"$1\" in\n        --config|-c)\n            config_file=\"$2\"\n            shift 2\n            ;;\n        --font|-f)\n            font_file=\"$2\"\n            shift 2\n            ;;\n        --output|-o)\n            output_path=\"$2\"\n            shift 2\n            ;;\n        --help|-h)\n            usage\n            exit 0\n            ;;\n        *)\n            echo \"Unknown option: $1\"\n            usage\n            exit 1\n            ;;\n    esac\ndone\n\n# Check if -c and -f options are provided\nif [ -z \"$config_file\" ]; then\n    echo \"Error: --config|-c option is required.\"\n    usage\n    exit 1\nfi\n\nif [ -z \"$font_file\" ]; then\n    echo \"Error: --font|-f option is required.\"\n    usage\n    exit 1\nfi\n\n# Check if config and font files exist.\nif [ ! -f \"$config_file\" ]; then\n    echo \"Error: Config file '$config_file' not found.\"\n    exit 1\nfi\n\nif [ ! -f \"$font_file\" ]; then\n    echo \"Error: Font file '$font_file' not found.\"\n    exit 1\nfi\n\n# Extract the title and menu names from the config file.\ntitle=$(awk -F' = ' '/^title/{print $2}' \"$config_file\" | tr -d '\"')\nmenu_names=$(awk -F' = ' '/^menu/{f=1;next} /socials/{f=0} f && /name/{print $2}' \"$config_file\" | cut -d',' -f1 | tr -d '\"' )\nlanguage_names=$(awk -F' = ' '/^language_name\\./{print $2}' \"$config_file\" | tr -d '\"' )\n\n# If the site is multilingual, get the menu translations.\nif [ -n \"$language_names\" ]; then\n    for menu_name in $menu_names; do\n        # Find the line with the menu name inside a [languages.*.translations] section and get the translated menus.\n        menu_translation=$(awk -F' = ' \"/\\\\[languages.*\\\\.translations\\\\]/{f=1;next} /^\\\\[/ {f=0} f && /$menu_name =/{print \\$2}\" \"$config_file\" | tr -d '\"' )\n        # Add the found menu value to the translations string\n        menu_names+=\"$menu_translation\"\n    done\nfi\n\n# Combine the extracted strings.\ncombined=\"$title$menu_names$language_names\"\n\n# Get unique characters.\nunique_chars=$(echo \"$combined\" | grep -o . | sort -u | tr -d '\\n')\n\n# Create a temporary file for subset.woff2.\ntemp_subset=$(mktemp)\n\n# Create the subset.\npyftsubset \"$font_file\" \\\n    --text=\"$unique_chars\" \\\n    --layout-features=\"*\" --flavor=\"woff2\" --output-file=\"$temp_subset\" --with-zopfli\n\n# Remove trailing slash from output path, if present.\noutput_path=${output_path%/}\n\n# Base64 encode the temporary subset.woff2 file and create the CSS file.\nbase64_encoded_font=$(base64 -i \"$temp_subset\")\necho \"@font-face{font-family:\\\"Inter Subset\\\";src:url(data:application/font-woff2;base64,$base64_encoded_font);}\" > \"$output_path/custom_subset.css\"\n\n# Remove the temporary subset.woff2 file.\nrm \"$temp_subset\"\n```\n\n## Usage\n\nSave the script somewhere like `~/bin/subset_font`. Make it executable with `chmod +x ~/bin/subset_font`.\n\nNow you can run it with the required `--config` and `--font` options:\n\n```bash\n~/bin/subset_font --config path/to/config.toml --font path/to/font.woff2\n```\nBy default, this generates a `custom_subset.css` file in the current directory. Use `-o` or `--output` to specify a different path:\n\n```bash\n~/bin/subset_font -c path/to/config.toml -f path/to/font.woff2 -o path/to/output\n```\n\nYou should place this `custom_subset.css` file inside the `static/` directory.\n\n\n## Automating with Pre-commit Hook\n\nYou might change the title or menu options of your site, making the custom subset no longer useful.\n\nTo automate the process of creating this file, you can integrate the script into a Git pre-commit hook that checks for changes in the `config.toml` file, runs the script, and stores the resulting CSS file in the `static/` directory of your site.\n\n1. Create a `.git/hooks/pre-commit` file in your Git project, if it doesn't already exist.\n\n2. Make it executable with `chmod +x .git/hooks/pre-commit`.\n\n3. Add the following code to the file:\n\n```bash\n# Check if config.toml has been modified.\nif git diff --cached --name-only | grep -q \"config.toml\"; then\n    echo \"config.toml modified. Running subset_font…\"\n\n    # Call the subset_font script.\n    ~/bin/subset_font -c config.toml -f static/fonts/Inter4.woff2 -o static/\n\n    # Add the generated subset.css file to the commit.\n    git add static/custom_subset.css\nfi\n```\n\nMake sure to modify the script to match the path where you stored the `subset_font` script. The config and font paths should work fine with tabi's default setup.\n\nNow, every time you commit changes to your Git project, the pre-commit hook will check for modifications in the `config.toml` file and automatically run the `subset_font` script to update the `custom_subset.css` file.\n\nBy the way, if you're interested in a way to automatically update the date of your Zola posts or compress your PNG files, check out [this post](https://osc.garden/blog/zola-date-git-hook/).\n\nIf you want to use all scripts at once (compressing PNG files, updating the date, and creating the font subset), combine their code into a single `.git/hooks/pre-commit` file.\n"
  },
  {
    "path": "content/blog/customise-tabi/index.ca.md",
    "content": "+++\ntitle = \"Personalitza el color de tabi i el tema per defecte\"\ndate = 2023-08-09\nupdated = 2024-09-12\ndescription = \"Aprèn a personalitzar tabi fent servir skins i establint un tema per defecte, aconseguint un aspecte únic.\"\n\n[taxonomies]\ntags = [\"funcionalitat\", \"tutorial\"]\n\n[extra]\ntoc = true\nquick_navigation_buttons = true\nsocial_media_card = \"social_cards/ca_blog_customise_tabi.jpg\"\n+++\n\ntabi pot ser personalitzat de dues maneres: establint el tema per defecte (fosc o clar) i triant el color principal per al tema (\"skin\").\n\n## Tema per defecte\n\nUtilitza `default_theme = \"dark\"` per establir el tema fosc com a predeterminat, o `default_theme = \"light\"` per establir el tema clar com a predeterminat.\n\nEstablir `default_theme = \"\"` (o comentar la variable) seguirà la preferència del sistema de l'usuari (mode clar o fosc).\n\nPer configurar permanentment el teu lloc en el tema fosc o clar, necessites desactivar el `theme_switcher` a `config.toml` i establir el teu tema preferit (`light` o `dark`) a `default_theme`.\n\nPer exemple, per tenir un tema fosc permanent:\n\n```toml\n[extra]\ntheme_switcher = false\ndefault_theme = \"dark\"\n```\n\n## Skins\n\nNo t'agrada l'aiguamarina? Cap problema! tabi té 12 skins per triar. Si cap d'aquestes t'agrada, pots [crear la teva pròpia skin](#crea-la-teva-propia-skin).\n\nUna skin és un arxiu CSS amb dues variables: el color principal per al tema clar i el color principal per al tema fosc.\n\nActivar una skin és tan fàcil com establir la variable `skin` a la teva `config.toml` amb el nom de la skin. Per exemple:\n\n```toml\n[extra]\nskin = \"sakura\"\n```\n\nFes una ullada a les skins disponibles a continuació.\n\n**Fes clic a les imatges** per canviar entre els temes fosc i clar.\n\n<hr>\n\n### Aiguamarina\n\nLa skin per defecte. Si la variable `skin` no està configurada (o és igual a `\"teal\"`), aquest és l'aspecte de tabi:\n\n{{ image_toggler(default_src=\"blog/customise-tabi/skins/teal_light.webp\", toggled_src=\"blog/customise-tabi/skins/teal_dark.webp\", default_alt=\"teal skin in light mode\", toggled_alt=\"teal skin in dark mode\", full_width=true) }}\n\n<hr>\n\n### Lavanda\n\n{{ image_toggler(default_src=\"blog/customise-tabi/skins/lavender_light.webp\", toggled_src=\"blog/customise-tabi/skins/lavender_dark.webp\", default_alt=\"skin lavanda en mode clar\", toggled_alt=\"skin lavanda en mode fosc\", full_width=true) }}\n\nPer aplicar-la, utilitza `skin = \"lavender\"`.\n\n<hr>\n\n### Vermell\n\n{{ image_toggler(default_src=\"blog/customise-tabi/skins/red_light.webp\", toggled_src=\"blog/customise-tabi/skins/red_dark.webp\", default_alt=\"skin vermell en mode clar\", toggled_alt=\"skin vermell en mode fosc\", full_width=true) }}\n\nCanvia a aquesta skin establint `skin = \"red\"`.\n\n<hr>\n\n### Menta\n\nUna skin dissenyada per 🅿️.\n\n{{ image_toggler(default_src=\"blog/customise-tabi/skins/mint_light.webp\", toggled_src=\"blog/customise-tabi/skins/mint_dark.webp\", default_alt=\"skin menta amb tema clar\", toggled_alt=\"skin menta amb tema fosc\", full_width=true) }}\n\nActiva-la amb `skin = \"mint\"`.\n\n<hr>\n\n### Sakura\n\nInspirat per la temporada de floració dels cirerers al Japó.\n\n{{ image_toggler(default_src=\"blog/customise-tabi/skins/sakura_light.webp\", toggled_src=\"blog/customise-tabi/skins/sakura_dark.webp\", default_alt=\"skin sakura en mode clar\", toggled_alt=\"skin sakura en mode fosc\", full_width=true) }}\n\nPer habilitar aquesta skin, ajusta `skin = \"sakura\"`.\n\n<hr>\n\n### Blau\n\n{{ image_toggler(default_src=\"blog/customise-tabi/skins/blue_light.webp\", toggled_src=\"blog/customise-tabi/skins/blue_dark.webp\", default_alt=\"skin blau en mode clar\", toggled_alt=\"skin blau en mode fosc\", full_width=true) }}\n\nPer activar aquesta aparença, estableix `skin = \"blue\"`.\n\n<hr>\n\n### Lingot indigo\n\n*Indigo* pel blau (en el tema clar) i *lingot* pel daurat (en el tema fosc).\n\n{{ image_toggler(default_src=\"blog/customise-tabi/skins/indigo_ingot_light.webp\", toggled_src=\"blog/customise-tabi/skins/indigo_ingot_dark.webp\", default_alt=\"skin lingot indigo en mode clar\", toggled_alt=\"skin lingot indigo en mode fosc\", full_width=true) }}\n\nPer activar aquest tema, utilitza `skin = \"indigo_ingot\"`.\n\n<hr>\n\n### Evangelion\n\nInspirat pels colors de la Unitat Evangelion-01 (en el tema fosc) i la Unitat-02 (en el tema clar).\n\n{{ image_toggler(default_src=\"blog/customise-tabi/skins/evangelion_light.webp\", toggled_src=\"blog/customise-tabi/skins/evangelion_dark.webp\", default_alt=\"skin evangelion en mode clar\", toggled_alt=\"skin evangelion en mode fosc\", full_width=true) }}\n\n<hr>\n\n### Monocromàtic\n\n{{ image_toggler(default_src=\"blog/customise-tabi/skins/monochrome_light.webp\", toggled_src=\"blog/customise-tabi/skins/monochrome_dark.webp\", default_alt=\"skin monocromàtic en mode clar\", toggled_alt=\"skin monocromàtic en mode fosc\", full_width=true) }}\n\nPer aconseguir aquesta aparença, estableix `skin = \"monochrome\"`.\n\n<hr>\n\n### Taronja (baix contrast)\n\n**AVÍS!** Aquesta skin en mode clar pot tenir [baix contrast](https://www.w3.org/WAI/WCAG21/Understanding/contrast-minimum.html), afectant l'accessibilitat i la qualificació Lighthouse. (El mode fosc té bon contrast.)\n\n{{ image_toggler(default_src=\"blog/customise-tabi/skins/lowcontrast_orange_light.webp\", toggled_src=\"blog/customise-tabi/skins/lowcontrast_orange_dark.webp\", default_alt=\"skin taronja de baix contrast en mode clar\", toggled_alt=\"skin taronja de baix contrast en mode fosc\", full_width=true) }}\n\nPer utilitzar-la, estableix `skin = \"lowcontrast_orange\"`.\n\n<hr>\n\n### Préssec (baix contrast)\n\n**AVÍS!** Aquesta skin en mode clar pot tenir [baix contrast](https://www.w3.org/WAI/WCAG21/Understanding/contrast-minimum.html), afectant l'accessibilitat i la qualificació Lighthouse. (El mode fosc té bon contrast.)\n\n{{ image_toggler(default_src=\"blog/customise-tabi/skins/lowcontrast_peach_light.webp\", toggled_src=\"blog/customise-tabi/skins/lowcontrast_peach_dark.webp\", default_alt=\"skin préssec de baix contrast en mode clar\", toggled_alt=\"skin préssec de baix contrast en mode fosc\", full_width=true) }}\n\nEspecifica `skin = \"lowcontrast_peach\"` per utilitzar aquesta skin.\n\n<hr>\n\n### Rosa (baix contrast)\n\n**AVÍS!** Aquesta skin en mode clar pot tenir [baix contrast](https://www.w3.org/WAI/WCAG21/Understanding/contrast-minimum.html), afectant l'accessibilitat i la qualificació Lighthouse. (El mode fosc té bon contrast.)\n\n{{ image_toggler(default_src=\"blog/customise-tabi/skins/lowcontrast_pink_light.webp\", toggled_src=\"blog/customise-tabi/skins/lowcontrast_pink_dark.webp\", default_alt=\"skin rosa de baix contrast en tema clar\", toggled_alt=\"skin rosa de baix contrast en tema fosc\", full_width=true) }}\n\nPer utilitzar aquests colors, assigna `skin = \"lowcontrast_pink\"`.\n\n<hr>\n\n### Crea la teva pròpia skin\n\nNo estàs limitat a les skins predefinides. Per què no crees un disseny únic que et representi?\n\nPots guardar la teva nova skin en qualsevol d'aquests dos directoris:\n1. Dins del directori del tema: `themes/tabi/sass/skins`\n2. Dins del directori principal del teu lloc: `sass/skins` (necessitaràs crear aquesta carpeta)\n\nCrea un nou arxiu `.scss` (per exemple, `la_teva_skin.scss`) a la ubicació que prefereixis. Aquest arxiu ha de contenir aquestes dues variables (aquesta és la skin predeterminada, \"teal\"):\n\n```scss\n// This defines theme-specific variables.\n@mixin theme-variables($theme) {\n    @if $theme =='light' {\n        // Light theme colours.\n        --primary-color: #087e96; // Contrast ratio: 4.73:1\n    }\n    @else if $theme == 'dark' {\n        // Dark theme colours.\n        --primary-color: #91e0ee;  // Contrast ratio: 11.06:1\n    }\n}\n\n// Apply light theme variables by default.\n:root {\n    @include theme-variables('light');\n}\n\n[data-theme='dark'] {\n    @include theme-variables('dark');\n}\n\n// Apply dark theme variables when user's system prefers dark mode\n// and the theme is not explicitly set to light.\n@media (prefers-color-scheme: dark) {\n    :root:not([data-theme='light']) {\n        @include theme-variables('dark');\n    }\n}\n```\n\nModifica els colors al teu gust. Una vegada estiguis satisfet, actualitza la variable `skin` perquè coincideixi amb el nom del teu arxiu.\n\nRecorda tenir en compte l'accesibilitat dels colors que triis. Aquí tens un enllaç que et pot ajudar: [WebAIM: Contrast Checker](https://webaim.org/resources/contrastchecker/). El fondo del tema clar és `#fff`, i el del tema fosc `#1f1f1f`.\n"
  },
  {
    "path": "content/blog/customise-tabi/index.es.md",
    "content": "+++\ntitle = \"Personaliza el color de tabi y el tema predeterminado\"\ndate = 2023-08-09\nupdated = 2024-09-12\ndescription = \"Aprende a personalizar tabi usando skins y estableciendo un tema predeterminado, haciendo que tu sitio sea único.\"\n\n[taxonomies]\ntags = [\"funcionalidad\", \"tutorial\"]\n\n[extra]\ntoc = true\nquick_navigation_buttons = true\nsocial_media_card = \"social_cards/es_blog_customise_tabi.jpg\"\n+++\n\ntabi puede ser personalizado de dos maneras: estableciendo el tema predeterminado (oscuro o claro) y eligiendo el color principal para el tema (\"skin\").\n\n## Tema predeterminado\n\nUsa `default_theme = \"dark\"` para establecer el tema oscuro como predeterminado, o `default_theme = \"light\"` para establecer el tema claro como predeterminado.\n\nEstablecer `default_theme = \"\"` (o no especificar la variable) seguirá las preferencias del sistema del usuario (modo claro u oscuro).\n\nPara configurar permanentemente tu sitio en el tema oscuro o claro, necesitas desactivar el `theme_switcher` en `config.toml` y establecer tu tema preferido (`light` o `dark`) como el `default_theme`.\n\nPor ejemplo, para tener un tema oscuro permanente:\n\n```toml\n[extra]\ntheme_switcher = false\ndefault_theme = \"dark\"\n```\n\n## Skins\n\n¿No te gusta el aguamarina? ¡No hay problema! tabi tiene 12 skins (pieles) para elegir. Si ninguna de estas te convence, puedes [crear tu propia skin](#crea-tu-propia-skin).\n\nUna skin es un archivo CSS con dos variables: el color principal para el tema claro y el color principal para el tema oscuro.\n\nActivar una skin es tan fácil como establecer la variable `skin` en tu `config.toml` con el nombre de la skin. Por ejemplo:\n\n```toml\n[extra]\nskin = \"sakura\"\n```\n\nEcha un vistazo a las pieles disponibles a continuación.\n\n**Haz clic en las imágenes** para cambiar entre los temas oscuro y claro.\n\n<hr>\n\n### Aguamarina\n\nLa skin predeterminada. Si la variable `skin` no está configurada (o es igual a `\"teal\"`), este es el aspecto de tabi:\n\n{{ image_toggler(default_src=\"blog/customise-tabi/skins/teal_light.webp\", toggled_src=\"blog/customise-tabi/skins/teal_dark.webp\", default_alt=\"skin aguamarina en tema claro\", toggled_alt=\"skin aguamarina en tema oscuro\", full_width=true) }}\n\n<hr>\n\n### Lavanda\n\n{{ image_toggler(default_src=\"blog/customise-tabi/skins/lavender_light.webp\", toggled_src=\"blog/customise-tabi/skins/lavender_dark.webp\", default_alt=\"skin lavanda en tema claro\", toggled_alt=\"skin lavanda en tema oscuro\", full_width=true) }}\n\nAplica esta skin con `skin = \"lavender\"`.\n\n<hr>\n\n### Rojo\n\n{{ image_toggler(default_src=\"blog/customise-tabi/skins/red_light.webp\", toggled_src=\"blog/customise-tabi/skins/red_dark.webp\", default_alt=\"skin rojo en tema claro\", toggled_alt=\"skin rojo en tema oscuro\", full_width=true) }}\n\nCambia a esta skin con la configuración `skin = \"red\"`.\n\n<hr>\n\n### Menta\n\nUna skin hecha por 🅿️.\n\n{{ image_toggler(default_src=\"blog/customise-tabi/skins/mint_light.webp\", toggled_src=\"blog/customise-tabi/skins/mint_dark.webp\", default_alt=\"skin menta en tema claro\", toggled_alt=\"skin menta en tema oscuro\", full_width=true) }}\n\nActívala con `skin = \"mint\"`.\n\n<hr>\n\n### Sakura\n\nInspirada en la temporada de florecimiento de los cerezos en Japón.\n\n{{ image_toggler(default_src=\"blog/customise-tabi/skins/sakura_light.webp\", toggled_src=\"blog/customise-tabi/skins/sakura_dark.webp\", default_alt=\"skin sakura en tema claro\", toggled_alt=\"skin sakura en tema oscuro\", full_width=true) }}\n\nPara activar esta skin, ajusta `skin = \"sakura\"`.\n\n<hr>\n\n### Azul\n\n{{ image_toggler(default_src=\"blog/customise-tabi/skins/blue_light.webp\", toggled_src=\"blog/customise-tabi/skins/blue_dark.webp\", default_alt=\"skin azul en tema claro\", toggled_alt=\"skin azul en tema oscuro\", full_width=true) }}\n\nPara lograr esta apariencia, establece `skin = \"blue\"`.\n\n<hr>\n\n### Lingote índigo\n\n*Índigo* por el azul (en el tema claro) y *lingote* por el oro (en el tema oscuro).\n\n{{ image_toggler(default_src=\"blog/customise-tabi/skins/indigo_ingot_light.webp\", toggled_src=\"blog/customise-tabi/skins/indigo_ingot_dark.webp\", default_alt=\"skin lingote índigo en tema claro\", toggled_alt=\"skin lingote índigo en tema oscuro\", full_width=true) }}\n\nPara activar esta skin, usa `skin = \"indigo_ingot\"`.\n\n<hr>\n\n### Evangelion\n\nInspirada en los colores de la Unidad-01 de Evangelion (en el tema oscuro) y el EVA-02 (en el tema claro).\n\n{{ image_toggler(default_src=\"blog/customise-tabi/skins/evangelion_light.webp\", toggled_src=\"blog/customise-tabi/skins/evangelion_dark.webp\", default_alt=\"skin evangelion en tema claro\", toggled_alt=\"skin evangelion en tema oscuro\", full_width=true) }}\n\nActívala con `skin = \"evangelion\"`.\n\n<hr>\n\n### Monocromático\n\n{{ image_toggler(default_src=\"blog/customise-tabi/skins/monochrome_light.webp\", toggled_src=\"blog/customise-tabi/skins/monochrome_dark.webp\", default_alt=\"skin monocromático en tema claro\", toggled_alt=\"skin monocromático en tema oscuro\", full_width=true) }}\n\nSi te gusta este look, usa `skin = \"monochrome\"`.\n\n<hr>\n\n### Naranja (bajo contraste)\n\n**¡ADVERTENCIA!** El tema claro de esta skin podría tener [poco contraste](https://www.w3.org/WAI/WCAG21/Understanding/contrast-minimum.html), afectando la accesibilidad y la calificación de Lighthouse. (El tema oscuro tiene buen contraste.)\n\n{{ image_toggler(default_src=\"blog/customise-tabi/skins/lowcontrast_orange_light.webp\", toggled_src=\"blog/customise-tabi/skins/lowcontrast_orange_dark.webp\", default_alt=\"skin naranja de bajo contraste en tema claro\", toggled_alt=\"skin naranja de bajo contraste en tema oscuro\", full_width=true) }}\n\nPara activarla, configura `skin = \"lowcontrast_orange\"`.\n\n<hr>\n\n### Melocotón (bajo contraste)\n\n**¡ADVERTENCIA!** El tema claro de esta skin podría tener [poco contraste](https://www.w3.org/WAI/WCAG21/Understanding/contrast-minimum.html), afectando la accesibilidad y la calificación de Lighthouse. (El tema oscuro tiene buen contraste.)\n\n{{ image_toggler(default_src=\"blog/customise-tabi/skins/lowcontrast_peach_light.webp\", toggled_src=\"blog/customise-tabi/skins/lowcontrast_peach_dark.webp\", default_alt=\"skin melocotón de bajo contraste en tema claro\", toggled_alt=\"skin melocotón de bajo contraste en tema oscuro\", full_width=true) }}\n\nEspecifica `skin = \"lowcontrast_peach\"` para usar esta skin.\n\n<hr>\n\n### Rosa (bajo contraste)\n\n**¡ADVERTENCIA!** El tema claro de esta skin podría tener [poco contraste](https://www.w3.org/WAI/WCAG21/Understanding/contrast-minimum.html), afectando la accesibilidad y la calificación de Lighthouse. (El tema oscuro tiene buen contraste.)\n\n{{ image_toggler(default_src=\"blog/customise-tabi/skins/lowcontrast_pink_light.webp\", toggled_src=\"blog/customise-tabi/skins/lowcontrast_pink_dark.webp\", default_alt=\"skin rosa de bajo contraste en tema claro\", toggled_alt=\"skin rosa de bajo contraste en tema oscuro\", full_width=true) }}\n\nPara usar estos colores, asigna `skin = \"lowcontrast_pink\"`.\n\n<hr>\n\n### Crea tu propia skin\n\nNo estás limitado a las skins predefinidas. ¿Por qué no diseñas un aspecto único que te represente?\n\nPuedes guardar tu nueva skin en cualquiera de estos dos directorios:\n1. Dentro del directorio del tema: `themes/tabi/sass/skins`\n2. Dentro del directorio principal de tu sitio: `sass/skins` (necesitarás crear esta carpeta)\n\nCrea un nuevo archivo `.scss` (por ejemplo, `tu_skin.scss`) en la ubicación que prefieras. Este archivo debe contener estas dos variables (esta es la skin predeterminada, \"teal\"):\n\n```scss\n// This defines theme-specific variables.\n@mixin theme-variables($theme) {\n    @if $theme =='light' {\n        // Light theme colours.\n        --primary-color: #087e96; // Contrast ratio: 4.73:1\n    }\n    @else if $theme == 'dark' {\n        // Dark theme colours.\n        --primary-color: #91e0ee;  // Contrast ratio: 11.06:1\n    }\n}\n\n// Apply light theme variables by default.\n:root {\n    @include theme-variables('light');\n}\n\n// Apply dark theme variables when dark theme is explicitly set.\n[data-theme='dark'] {\n    @include theme-variables('dark');\n}\n\n// Apply dark theme variables when user's system prefers dark mode\n// and the theme is not explicitly set to light.\n@media (prefers-color-scheme: dark) {\n    :root:not([data-theme='light']) {\n        @include theme-variables('dark');\n    }\n}\n```\n\nModifica los colores a tu gusto. Una vez que estés satisfecho, actualiza la variable `skin` para que coincida con el nombre de tu archivo.\n\nRecuerda tener en cuenta la accesibilidad de los colores que elijas. Aquí tienes un enlace que te puede ayudar: [WebAIM: Contrast Checker](https://webaim.org/resources/contrastchecker/). El fondo del tema claro es `#fff`, y el del tema oscuro `#1f1f1f`.\n"
  },
  {
    "path": "content/blog/customise-tabi/index.md",
    "content": "+++\ntitle = \"Customise tabi with skins and a default theme\"\ndate = 2023-08-09\nupdated = 2024-09-12\ndescription = \"Learn how to customise tabi using skins and setting a default theme, making your site uniquely yours.\"\n\n[taxonomies]\ntags = [\"showcase\", \"tutorial\"]\n\n[extra]\ntoc = true\nquick_navigation_buttons = true\nsocial_media_card = \"social_cards/blog_customise_tabi.jpg\"\n+++\n\ntabi can be customised in two ways: by setting the default theme (dark or light) and by choosing the main colour for the theme (skins).\n\n## Default theme\n\nUse `default_theme = \"dark\"` to set the dark theme as the default, or `default_theme = \"light\"` to set the light theme as the default.\n\nSetting `default_theme = \"\"` (or commenting out the variable) will follow the user's system preference (light or dark mode).\n\nTo permanently set your site to either the dark or light theme, you need to disable the theme switcher in `config.toml` and set your preferred theme as the `default_theme`.\n\nFor example, to have a permanent dark theme:\n\n```toml\n[extra]\ntheme_switcher = false\ndefault_theme = \"dark\"\n```\n\n## Skins\n\nNot a fan of teal? No problem! tabi has 12 skins for you to choose from. If none of these work for you, you can [create your own](#create-your-own-skin).\n\nA skin is a CSS file with two variables: the primary colour for the light theme, and the primary colour for the dark theme.\n\nEnabling a skin is as easy as setting the `skin` variable in your `config.toml` with the name of the skin. For example:\n\n```toml\n[extra]\nskin = \"sakura\"\n```\n\nTake a look below at the available skins below.\n\n**Click on the images** to switch between dark and light themes.\n\n<hr>\n\n### Teal\n\nThe default skin. If the `skin` variable is unset (or set to `\"teal\"`), this is what tabi looks like:\n\n{{ image_toggler(default_src=\"blog/customise-tabi/skins/teal_light.webp\", toggled_src=\"blog/customise-tabi/skins/teal_dark.webp\", default_alt=\"teal skin in light mode\", toggled_alt=\"teal skin in dark mode\", full_width=true) }}\n\n<hr>\n\n### Lavender\n\n{{ image_toggler(default_src=\"blog/customise-tabi/skins/lavender_light.webp\", toggled_src=\"blog/customise-tabi/skins/lavender_dark.webp\", default_alt=\"lavender skin in light mode\", toggled_alt=\"lavender skin in dark mode\", full_width=true) }}\n\nTo apply, use `skin = \"lavender\"`.\n\n\n<hr>\n\n### Red\n\n{{ image_toggler(default_src=\"blog/customise-tabi/skins/red_light.webp\", toggled_src=\"blog/customise-tabi/skins/red_dark.webp\", default_alt=\"red skin in light mode\", toggled_alt=\"red skin in dark mode\", full_width=true) }}\n\nSwitch to this by setting `skin = \"red\"`.\n\n\n<hr>\n\n### Mint\n\nA skin designed by 🅿️.\n\n{{ image_toggler(default_src=\"blog/customise-tabi/skins/mint_light.webp\", toggled_src=\"blog/customise-tabi/skins/mint_dark.webp\", default_alt=\"mint skin in light mode\", toggled_alt=\"mint skin in dark mode\", full_width=true) }}\n\nActivate it with `skin = \"mint\"`.\n\n\n<hr>\n\n### Sakura\n\nInspired by the Japanese cherry blossom season.\n\n{{ image_toggler(default_src=\"blog/customise-tabi/skins/sakura_light.webp\", toggled_src=\"blog/customise-tabi/skins/sakura_dark.webp\", default_alt=\"sakura skin in light mode\", toggled_alt=\"sakura skin in dark mode\", full_width=true) }}\n\nTo enable this skin, adjust `skin = \"sakura\"`.\n\n\n<hr>\n\n### Blue\n\n{{ image_toggler(default_src=\"blog/customise-tabi/skins/blue_light.webp\", toggled_src=\"blog/customise-tabi/skins/blue_dark.webp\", default_alt=\"blue skin in light mode\", toggled_alt=\"blue skin in dark mode\", full_width=true) }}\n\nFor this appearance, set `skin = \"blue\"`.\n\n\n<hr>\n\n### Indigo Ingot\n\n*Indigo* for blue (in light theme) and *ingot* for gold (in dark theme).\n\n{{ image_toggler(default_src=\"blog/customise-tabi/skins/indigo_ingot_light.webp\", toggled_src=\"blog/customise-tabi/skins/indigo_ingot_dark.webp\", default_alt=\"indigo ingot skin in light mode\", toggled_alt=\"indigo ingot skin in dark mode\", full_width=true) }}\n\nTo activate this skin, use `skin = \"indigo_ingot\"`.\n\n\n<hr>\n\n### Evangelion\n\nInspired by the colours of Evangelion Unit-01 (in dark theme) and Unit-02 (in light theme).\n\n{{ image_toggler(default_src=\"blog/customise-tabi/skins/evangelion_light.webp\", toggled_src=\"blog/customise-tabi/skins/evangelion_dark.webp\", default_alt=\"evangelion skin in light mode\", toggled_alt=\"evangelion skin in dark mode\", full_width=true) }}\n\n\n<hr>\n\n### Monochrome\n\n{{ image_toggler(default_src=\"blog/customise-tabi/skins/monochrome_light.webp\", toggled_src=\"blog/customise-tabi/skins/monochrome_dark.webp\", default_alt=\"monochrome skin in light mode\", toggled_alt=\"monochrome skin in dark mode\", full_width=true) }}\n\nTo achieve this look, set `skin = \"monochrome\"`.\n\n\n<hr>\n\n### Low contrast orange\n\n**WARNING!** This skin's light theme may have [low contrast](https://www.w3.org/WAI/WCAG21/Understanding/contrast-minimum.html), affecting accessibility and Lighthouse rating. (Dark theme is fine.)\n\n{{ image_toggler(default_src=\"blog/customise-tabi/skins/lowcontrast_orange_light.webp\", toggled_src=\"blog/customise-tabi/skins/lowcontrast_orange_dark.webp\", default_alt=\"low contrast orange skin in light mode\", toggled_alt=\"low contrast orange skin in dark mode\", full_width=true) }}\n\nTo use, set `skin = \"lowcontrast_orange\"`.\n\n\n<hr>\n\n### Low contrast peach\n\n**WARNING!** This skin's light theme may have [low contrast](https://www.w3.org/WAI/WCAG21/Understanding/contrast-minimum.html), affecting accessibility and Lighthouse rating. (Dark theme is fine.)\n\n{{ image_toggler(default_src=\"blog/customise-tabi/skins/lowcontrast_peach_light.webp\", toggled_src=\"blog/customise-tabi/skins/lowcontrast_peach_dark.webp\", default_alt=\"low contrast peach skin in light mode\", toggled_alt=\"low contrast peach skin in dark mode\", full_width=true) }}\n\nTo enable it, specify `skin = \"lowcontrast_peach\"`.\n\n\n<hr>\n\n### Low contrast pink\n\n**WARNING!** This skin's light theme may have [low contrast](https://www.w3.org/WAI/WCAG21/Understanding/contrast-minimum.html), affecting accessibility and Lighthouse rating. (Dark theme is fine.)\n\n{{ image_toggler(default_src=\"blog/customise-tabi/skins/lowcontrast_pink_light.webp\", toggled_src=\"blog/customise-tabi/skins/lowcontrast_pink_dark.webp\", default_alt=\"low contrast pink skin in light mode\", toggled_alt=\"low contrast pink skin in dark mode\", full_width=true) }}\n\nFor this colourscheme, choose `skin = \"lowcontrast_pink\"`.\n\n\n<hr>\n\n### Create your own skin\n\nYou're not just limited to predefined skins. Why not create a look that's distinctively tailored to your preferences?\n\nYou can save your new skin it in either of these two directories:\n1. Inside the theme's directory: `themes/tabi/sass/skins`\n2. Inside your main site's directory: `sass/skins` (you'll need to create this folder)\n\nCreate a new `.scss` file (for example, `your_skin.scss`) in your preferred location. This file needs to have these two variables (this is the default skin, \"teal\"):\n\n```scss\n// This defines theme-specific variables.\n@mixin theme-variables($theme) {\n    @if $theme =='light' {\n        // Light theme colours.\n        --primary-color: #087e96; // Contrast ratio: 4.73:1\n    }\n    @else if $theme == 'dark' {\n        // Dark theme colours.\n        --primary-color: #91e0ee;  // Contrast ratio: 11.06:1\n    }\n}\n\n// Apply light theme variables by default.\n:root {\n    @include theme-variables('light');\n}\n\n// Apply dark theme variables when dark theme is explicitly set.\n[data-theme='dark'] {\n    @include theme-variables('dark');\n}\n\n// Apply dark theme variables when user's system prefers dark mode\n// and the theme is not explicitly set to light.\n@media (prefers-color-scheme: dark) {\n    :root:not([data-theme='light']) {\n        @include theme-variables('dark');\n    }\n}\n```\n\nModify the colours to your taste. Once you're satisfied, update the `skin` variable to match your filename.\n\nRemember to consider the accessibility of the colours you choose. Here's a link that can help you: [WebAIM: Contrast Checker](https://webaim.org/resources/contrastchecker/). The background of the light theme is `#fff`, and the dark one is `#1f1f1f`.\n"
  },
  {
    "path": "content/blog/faq-languages/index.ca.md",
    "content": "+++\ntitle = \"Lost in Translation? Explora les capacitats multilingües de tabi\"\ndate = 2023-09-12\nupdated = 2026-01-03\ndescription = \"Descobreix com tabi t'ajuda a connectar amb una audiència global gràcies a les seves funcions multilingües. Aprèn a canviar la llengua per defecte, afegir més llengües i aportar les teves pròpies traduccions.\"\n\n[taxonomies]\ntags = [\"funcionalitat\", \"tutorial\", \"Preguntes Freqüents\"]\n\n[extra]\nquick_navigation_buttons = true\ntoc_ignore_pattern = \"^(Preguntes Freqüents)\"\nsocial_media_card = \"social_cards/ca_blog_faq_languages.jpg\"\n+++\n\ntabi simplifica el procés de creació de llocs web multilingües perquè puguis connectar amb una audiència global. En aquesta guia, t'explicarem tot el que necessites saber, des de com configurar la llengua principal en el teu lloc fins a com contribuir amb les teves pròpies traduccions. Comencem!\n\n### Preguntes Freqüents\n\n<!-- toc -->\n\n## Quines llengües admet tabi?\n\ntabi admet les següents llengües:\n\n- Alemany\n- Àrab\n- Anglès\n- Català\n- Coreà\n- Espanyol\n- Estonià\n- Finès\n- Francès\n- Hindi\n- Italià\n- Japonès\n- Odia\n- Persa\n- Portuguès (Europeu)\n- Portuguès (Brasil)\n- Rus\n- Ucraïnès\n- Xinès (Simplificat)\n\nPer a una llista sempre actualitzada de llengües suportades, consulta la [carpeta `i18n`](https://github.com/welpo/tabi/tree/main/i18n) en el repositori de tabi.\n\n## Com estableixo la llengua predeterminada del meu lloc?\n\nPots definir la llengua principal del teu lloc configurant la variable `default_language` a `config.toml`.\n\nPer exemple, si vols que la llengua principal sigui el Xinès, simplement afegeix aquesta línia a l'arxiu `config.toml`:\n\n```toml, hl_lines=03\nbase_url = \"https://welpo.github.io/tabi\"\ntitle = \"~/tabi\"\ndefault_language = \"zh\"\n```\n\nSi el valor de `default_language` coincideix amb el nom d'un fitxer TOML al [directori `i18n`](https://github.com/welpo/tabi/tree/main/i18n), tots els textos de tabi es traduiran a aquest idioma.\n\n## Com gestiona tabi el suport multilingüe?\n\nZola genera automàticament URLs per a cada llengua que no sigui la predeterminada de la següent manera: `{base_url}/{codi_idioma}/{post}`.\n\ntabi facilita la navegació entre llengües afegint un commutador de llengua en la barra de navegació (que només es mostra quan hi ha més d'una llengua habilitada).\n\nSi [pujes](#) a la barra de navegació, veuràs el commutador de llengua. Si cliques sobre ell, es mostrarà un desplegable amb les llengües disponibles. Si fas clic en el nom d'una llengua, et portarà a la mateixa pàgina en aquesta llengua.\n\nSi una pàgina específica no està disponible en una llengua, tabi mostrarà una pàgina 404 amb el text:\n\n> La pàgina que has sol·licitat sembla que no existeix o encara no s'ha traduït al teu idioma. Comprova l'URL per detectar errors o torna a la pàgina d'inici.\n\nAquest text es mostrarà una vegada per cada llengua activada en el teu lloc. Pots veure aquesta pàgina en acció [aquí](https://welpo.github.io/tabi/404.html).\n\n## Com activo el suport multilingüe?\n\nPer habilitar el suport per a diverses llengües, necessites configurar la variable `languages` a `config.toml`. Per exemple, si vols un lloc amb anglès com a llengua principal que també admeti hindi i espanyol, pots configurar el teu `config.toml` de la següent manera:\n\n```toml\nbase_url = \"https://example.com\"\ntitle = \"My Site\"\ndefault_language = \"en\"\n\n[languages.hi]\ntitle = \"मेरी वेबसाइट\"\n\n[languages.es]\ntitle = \"El meu web\"\n```\n\nEn cada secció de llengua pots establir altres variables com `taxonomies`, `description`… Consulta la [documentació de suport multilingüe de Zola](https://www.getzola.org/documentation/content/multilingual/) per a més informació.\n\n## Què són aquests codis de dues lletres?\n\nEls codis de dues lletres són [codis d'idioma ISO 639-1](https://localizely.com/iso-639-1-list/) (o [IETF BCP 47](https://ca.wikipedia.org/wiki/Codi_de_llengua_IETF), quan cal), que serveixen per identificar idiomes d'una manera estandarditzada.\n\ntabi utilitza aquests codis per permetre la navegació entre idiomes i traduir el tema.\n\n## Com personalitzo o reemplaço una cadena de text específica al meu lloc web?\n\ntabi cerca els fitxers de cadenes en el següent ordre. `$base_directory` és on resideix el teu lloc Zola (allà on està `config.toml`):\n\n1. `$base_directory + \"i18n\"`\n2. `$base_directory + \"themes/tabi/i18n\"`\n\nPer tant, si crees `i18n/ca.toml` al teu directori base, tabi llegirà les cadenes de text d'aquest fitxer en lloc de les cadenes predeterminades en català. Pots fer això per a qualsevol idioma, suportat o no.\n\nAssegura't de copiar tot el fitxer per a aquest idioma primer, o el tema utilitzarà l'anglès per les claus que faltin.\n\n## Com personalitzo els formats de data per a diferents idiomes?\n\nPots establir formats de data específics per idioma al teu `config.toml` utilitzant la matriu `date_formats`:\n\n```toml\ndate_formats = [\n    { lang = \"es\", long = \"%d de %B de %Y\", short = \"%-d %b %Y\", archive = \"%d de %b\" },\n    { lang = \"de\", long = \"%d. %B %Y\", short = \"%d.%m.%Y\", archive = \"%d. %b\" },\n]\n```\n\nAixò permet que cada idioma mostri les dates segons les convencions locals. Per exemple, l'espanyol mostrarà «3 de febrero de 2024» mentre que l'alemany mostrarà «3. Februar 2024». Si no es defineix un format específic per a un idioma, tabi utilitzarà la configuració global `long_date_format`, `short_date_format` i `archive_date_format`.\n\n## Què passa si falta una traducció o està incompleta?\n\nSi una cadena no es troba en el fitxer d'idioma, tabi utilitzarà a la cadena predeterminada en anglès.\n\n## El meu idioma no està suportat. Puc contribuir amb una traducció?\n\nÉs clar! Sempre estem buscant afegir suport per a més idiomes. Pots contribuir amb una traducció creant una Pull Request al [repositori de tabi](https://github.com/welpo/tabi).\n\nPots utilitzar el [fitxer en català](https://github.com/welpo/tabi/blob/main/i18n/ca.toml) com a base per traduir les cadenes al teu idioma. Assegura't de mantenir la mateixa estructura.\n\nEl fitxer ha de portar el nom del codi de dues lletres del teu idioma i ha de ser un fitxer TOML. Per exemple, si vols afegir suport per al suahili, pots crear un fitxer anomenat `sw.toml` al directori `i18n`.\n\nNota: quan provis la teva traducció, és possible que necessitis reiniciar `zola serve` per veure els canvis, ja que Zola no sempre detecta canvis en els fitxers TOML.\n\n## He trobat un error en una traducció. Com el corregeixo?\n\nSi trobes un error en una traducció, pots crear una issue o una Pull Request al [repositori de tabi](https://github.com/welpo/tabi).\n\n## Com actualitzo les traduccions després d'una actualització del tema?\n\nSi no vas personalitzar les traduccions, simplement actualitza el tema.\n\nSi ho vas fer, hauràs d'actualitzar manualment les traduccions. Pots fer-ho copiant les noves cadenes dels fitxers corresponents i enganxant-les al teu fitxer personalitzat.\n\n## tabi tradueix el meu contingut?\n\nNo. tabi només tradueix les cadenes de text del tema. Hauràs de traduir el teu contingut tu mateix.\n\n## Com puc mostrar el codi de l'idioma actual al commutador d'idioma?\n\nAfegeix `show_selected_language_code_in_language_switcher = true` a la secció `[extra]` del teu `config.toml`.\n"
  },
  {
    "path": "content/blog/faq-languages/index.es.md",
    "content": "+++\ntitle = \"¿Lost in Translation? Explora las capacidades multilingües de tabi\"\ndate = 2023-09-12\nupdated = 2026-01-03\ndescription = \"Descubre cómo tabi te ayuda a conectar con una audiencia global gracias a sus funciones multilingües. Aprende a cambiar el idioma por defecto, añadir más idiomas y aportar tus propias traducciones.\"\n\n[taxonomies]\ntags = [\"funcionalidad\", \"tutorial\", \"Preguntas Frecuentes\"]\n\n[extra]\nquick_navigation_buttons = true\ntoc_ignore_pattern = \"^(Preguntas Frecuentes)\"\nsocial_media_card = \"social_cards/es_blog_faq_languages.jpg\"\n+++\n\ntabi simplifica el proceso de crear sitios web multilingües para que puedas conectar con una audiencia global. En esta guía, te explicaremos todo lo que necesitas saber, desde cómo configurar el idioma principal en tu sitio hasta cómo contribuir con tus propias traducciones. ¡Empecemos!\n\n### Preguntas Frecuentes\n\n<!-- toc -->\n\n## ¿Qué idiomas admite tabi?\n\ntabi admite los siguientes idiomas:\n\n- Alemán\n- Árabe\n- Catalán\n- Chino (Simplificado)\n- Coreano\n- Español\n- Estonio\n- Finlandés\n- Francés\n- Hindi\n- Inglés\n- Italiano\n- Japonés\n- Odia\n- Persa\n- Portugués (Europeo)\n- Portugués (Brasil)\n- Ruso\n- Ucraniano\n\nPara una lista siempre actualizada de idiomas soportados, consulta la [carpeta `i18n`](https://github.com/welpo/tabi/tree/main/i18n) en el repositorio de tabi.\n\n## ¿Cómo establezco el idioma predeterminado de mi sitio?\n\nPuedes definir el idioma principal de tu sitio configurando la variable `default_language` en tu archivo `config.toml`.\n\nPor ejemplo, si deseas que el idioma principal sea el Chino, simplemente añade esta línea al archivo `config.toml`:\n\n```toml, hl_lines=03\nbase_url = \"https://welpo.github.io/tabi\"\ntitle = \"~/tabi\"\ndefault_language = \"zh\"\n```\n\nSi el valor de `default_language` coincide con el nombre de un archivo TOML en el [directorio `i18n`](https://github.com/welpo/tabi/tree/main/i18n), todos los textos de tabi se traducirán a ese idioma.\n\n## ¿Cómo gestiona tabi el soporte multilingüe?\n\nZola genera automáticamente URLs para cada idioma que no sea el predeterminado de la siguiente manera: `{base_url}/{código_idioma}/{post}`.\n\ntabi facilita la navegación entre idiomas añadiendo un conmutador de idioma en la barra de navegación (que sólo se muestra cuando hay más de un idioma habilitado).\n\nSi [subes](#) a la barra de navegación, verás el conmutador de idioma. Al hacer clic sobre él, se mostrará un desplegable con los idiomas disponibles. Si haces clic en el nombre de un idioma, te llevará a la misma página en ese idioma.\n\nSi una página específica no está disponible en un idioma, tabi mostrará una página 404 con el texto:\n\n> La página que has solicitado parece no existir o aún no se ha traducido a tu idioma. Revisa la URL en busca de errores o regresa a la página de inicio.\n\nEste texto se mostrará una vez por cada idioma activado en tu sitio. Puedes ver esta página en acción [aquí](https://welpo.github.io/tabi/404.html).\n\n## ¿Cómo activo el soporte multilingüe?\n\nPara habilitar el soporte para varios idiomas, necesitas configurar la variable `languages` en `config.toml`. Por ejemplo, si quieres un sitio con inglés como idioma principal que también admita hindi y español, puedes configurar tu `config.toml` de la siguiente manera:\n\n```toml\nbase_url = \"https://example.com\"\ntitle = \"My Site\"\ndefault_language = \"en\"\n\n[languages.hi]\ntitle = \"मेरी वेबसाइट\"\n\n[languages.es]\ntitle = \"Mi web\"\n```\n\nEn cada sección de idioma puedes establecer otras variables como `taxonomies`, `description`… Consulta la [documentación de soporte multilingüe de Zola](https://www.getzola.org/documentation/content/multilingual/) para más información.\n\n## ¿Qué son estos códigos de dos letras?\n\nLos códigos de dos letras son [códigos de idioma ISO 639-1](https://localizely.com/iso-639-1-list/) (o [IETF BCP 47](https://es.wikipedia.org/wiki/C%C3%B3digo_de_idioma_IETF), si hace falta), que sirven para identificar idiomas de una manera estandarizada.\n\ntabi utiliza estos códigos para permitir la navegación entre idiomas y traducir el tema.\n\n## ¿Cómo personalizo o reemplazo una cadena de texto específica en mi sitio web?\n\ntabi busca los archivos de cadenas en el siguiente orden. `$base_directory` es donde reside tu sitio Zola (donde se guarda `config.toml`):\n\n1. `$base_directory + \"i18n\"`\n2. `$base_directory + \"themes/tabi/i18n\"`\n\nPor lo tanto, si creas `i18n/en.toml` en tu directorio base, tabi leerá las cadenas de texto de ese archivo en lugar de las cadenas predeterminadas en inglés. Puedes hacer esto para cualquier idioma, soportado o no.\n\nAsegúrate de copiar todo el archivo para ese idioma primero, o el tema usará el inglés para las claves faltantes.\n\n## ¿Cómo personalizo los formatos de fecha para diferentes idiomas?\n\nPuedes establecer formatos de fecha específicos por idioma en tu `config.toml` usando la matriz `date_formats`:\n\n```toml\ndate_formats = [\n    { lang = \"es\", long = \"%d de %B de %Y\", short = \"%-d %b %Y\", archive = \"%d de %b\" },\n    { lang = \"de\", long = \"%d. %B %Y\", short = \"%d.%m.%Y\", archive = \"%d. %b\" },\n]\n```\n\nEsto permite que cada idioma muestre las fechas según las convenciones locales. Por ejemplo, el español mostrará «3 de febrero de 2024» mientras que el alemán mostrará «3. Februar 2024». Si no se define un formato específico para un idioma, tabi usará la configuración global `long_date_format`, `short_date_format` y `archive_date_format`.\n\n## ¿Qué pasa si falta una traducción o está incompleta?\n\nSi una cadena no se encuentra en el archivo de idioma, tabi recurrirá a la cadena predeterminada en inglés.\n\n## Mi idioma no está soportado. ¿Puedo contribuir con una traducción?\n\n¡Por supuesto! Siempre estamos buscando añadir soporte para más idiomas. Puedes contribuir con una traducción creando una Pull Request en el [repositorio de tabi](https://github.com/welpo/tabi).\n\nPuedes usar el [archivo en inglés](https://github.com/welpo/tabi/blob/main/i18n/en.toml) como base para traducir las cadenas a tu idioma. Asegúrate de mantener la misma estructura.\n\nEl archivo debe llevar el nombre del código de dos letras de tu idioma y debe ser un archivo TOML. Por ejemplo, si quieres añadir soporte para el suajili, puedes crear un archivo llamado `sw.toml` en el directorio `i18n`.\n\nNota: cuando pruebes tu traducción, es posible que necesites reiniciar `zola serve` para ver los cambios, ya que Zola no siempre detecta cambios en los archivos TOML.\n\n## He encontrado un error en una traducción. ¿Cómo lo corrijo?\n\nSi encuentras un error en una traducción, puedes abrir un ticket o una Pull Request en el [repositorio de tabi](https://github.com/welpo/tabi).\n\n## ¿Cómo actualizo las traducciones después de una actualización del tema?\n\nSi no personalizaste las traducciones, basta con actualizar el tema.\n\nSi lo hiciste, tendrás que actualizar manualmente las traducciones. Puedes hacerlo copiando las nuevas cadenas de los archivos correspondientes y pegándolas en tu archivo personalizado.\n\n## ¿tabi traduce el contenido de mi sitio?\n\nNo. tabi sólo traduce el tema. Los posts deberás traducirlos tú mismo.\n\n## ¿Cómo puedo mostrar el código del idioma actual en el conmutador de idioma?\n\nAñade `show_selected_language_code_in_language_switcher = true` en la sección `[extra]` de tu `config.toml`.\n"
  },
  {
    "path": "content/blog/faq-languages/index.md",
    "content": "+++\ntitle = \"Lost in Translation? Not with tabi's Multilingual Capabilities\"\ndate = 2023-09-12\nupdated = 2026-01-03\ndescription = \"Master the art of serving a global audience through tabi's built-in multilingual features. Learn how to change the default language, add multilingual support, and contribute your own translations.\"\n\n[taxonomies]\ntags = [\"showcase\", \"tutorial\", \"FAQ\"]\n\n[extra]\nquick_navigation_buttons = true\ntoc_ignore_pattern = \"^(Frequently Asked Questions)\"\nsocial_media_card = \"social_cards/blog_faq_languages.jpg\"\n+++\n\nTo broaden your reach to a global audience, tabi streamlines the process of building multilingual websites. In this guide, we'll walk you through everything you need to know—from setting a default language for your site to contributing your own translations. Let's get started!\n\n### Frequently Asked Questions\n\n<!-- toc -->\n\n## What languages does tabi support?\n\ntabi supports the following languages:\n\n- Arabic\n- Catalan\n- Chinese (Simplified)\n- Chinese (Traditional)\n- English\n- Estonian\n- Finnish\n- French\n- German\n- Hindi\n- Italian\n- Japanese\n- Korean\n- Odia\n- Persian\n- Portuguese (European)\n- Portuguese (Brazilian)\n- Russian\n- Spanish\n- Ukranian\n\nFor an always up to date list of supported languages, refer to the [`i18n` directory](https://github.com/welpo/tabi/tree/main/i18n) in the tabi repository.\n\n## How do I set a default language for my site?\n\nYou can set the default language for your site by defining the `default_language` variable in your `config.toml` file.\n\nFor instance, if you want (Simplified) Chinese to be the primary language, simply add this line to  `config.toml`:\n\n```toml, hl_lines=03\nbase_url = \"https://welpo.github.io/tabi\"\ntitle = \"~/tabi\"\ndefault_language = \"zh-Hans\"\n```\n\nIf the value of `default_language` matches the name of a TOML file in the [`i18n` directory](https://github.com/welpo/tabi/tree/main/i18n), all of tabi's text strings will be translated to that language.\n\n## How does tabi handle multilingual support?\n\nZola automatically generates URLs for each non-default language like this: `{base_url}/{language_code}/{post}`.\n\ntabi facilitates the navigation between languages by adding a language switcher to the navigation bar (only shown when there's more than one language enabled).\n\nIf you [scroll up](#) to the navigation bar, you'll see the language switcher (the globe icon). Clicking on it will display a dropdown with the available languages. Clicking on a language's name will take you to the same page in that language.\n\nIf a specific page is not available in a language, tabi will display a 404 page with the text:\n\n> The page you've requested seems to be missing or hasn't been translated into your language yet. Check the URL for errors or go back to the homepage.\n\nThis text will be shown once for each language enabled on your site. You can see this page in action [here](https://welpo.github.io/tabi/404.html).\n\n## How do I enable multilingual support?\n\nTo enable multilingual support, you need to set the `languages` variable in your `config.toml` file. For example, if want an English-default site with support for Hindi and Spanish, you can set up your `config.toml` like so:\n\n```toml\nbase_url = \"https://example.com\"\ntitle = \"My Site\"\ndefault_language = \"en\"\n\n[languages.hi]\ntitle = \"मेरी वेबसाइट\"\n\n[languages.es]\ntitle = \"Mi web\"\n```\n\nOn each language's section, you can set other variables like `taxonomies`, `description`, whether to generate a feed… Refer to Zola's [multilingual support documentation](https://www.getzola.org/documentation/content/multilingual/) for more information.\n\n## What are these two letter codes?\n\nThe two letter codes are [ISO 639-1 language codes](https://localizely.com/iso-639-1-list/) (or [IETF BCP 47](https://en.wikipedia.org/wiki/IETF_language_tag), when necessary). They are used to identify languages in a standardised way.\n\ntabi uses these codes to allow navigation between languages and translate the theme.\n\n## How do I customise or override a specific text string on my website?\n\ntabi looks for the strings files in the following order. `$base_directory` is where your Zola site resides (where `config.toml` is stored):\n\n1. `$base_directory + \"i18n\"`\n2. `$base_directory + \"themes/tabi/i18n\"`\n\nSo if you create  `i18n/en.toml` in your base directory, tabi will read the strings from that file instead of the default English strings. You can do this for any language, supported or not.\n\nMake sure to copy the entire file for that language first, or the theme will fall back to the default English strings.\n\n## How do I customize date formats for different languages?\n\nYou can set language-specific date formats in your `config.toml` using the `date_formats` array:\n\n```toml\ndate_formats = [\n    { lang = \"es\", long = \"%d de %B de %Y\", short = \"%-d %b %Y\", archive = \"%d de %b\" },\n    { lang = \"de\", long = \"%d. %B %Y\", short = \"%d.%m.%Y\", archive = \"%d. %b\" },\n]\n```\n\nThis allows each language to display dates according to local conventions. For example, Spanish will show \"3 de febrero de 2024\" while German will show \"3. Februar 2024\". If no language-specific format is defined, tabi will use the global `long_date_format`, `short_date_format` and `archive_date_format` settings.\n\n## What happens if a translation is missing or incomplete?\n\nIf a string is not found in the language file, tabi will fall back to the default English string.\n\n## My language is not supported. Can I contribute a translation?\n\nPlease do! We are always looking to add support for more languages. You can contribute a translation by creating a pull request in the [tabi repository](https://github.com/welpo/tabi).\n\nYou can use the [English file](https://github.com/welpo/tabi/blob/main/i18n/en.toml) as a base to translate the strings to your language. Please make sure to follow the same structure.\n\nThe file should be named after the two letter code of your language, and should be a TOML file. For example, if you want to add support for Swahili, you can create a file named `sw.toml` in the `i18n` directory.\n\nNote: when testing your translation, you might need to restart `zola serve` to see the changes, as Zola doesn't always detect changes in the TOML files.\n\n## I've found an error in a translation. How do I fix it?\n\nIf you find an error in a translation, you can create an issue or a pull request in the [tabi repository](https://github.com/welpo/tabi).\n\n## How do I update the translations after a theme update?\n\nIf you didn't customise the translations, simply updating the theme will update the translations.\n\nIf you did, you will need to manually update the translations. You can do this by copying the new strings from the corresponding files, and pasting them in your custom file.\n\n## Does tabi translate my content?\n\nNo. tabi only translates the theme's text strings. You will need to translate your content yourself.\n\n# How to show current language code on the language switcher?\n\nAdd `show_selected_language_code_in_language_switcher = true` in your config extras.\n"
  },
  {
    "path": "content/blog/javascript/index.ca.md",
    "content": "+++\ntitle = \"Sense JavaScript obligatori\"\ndate = 2023-01-06\nupdated = 2025-02-21\ndescription = \"JavaScript només s'utilitza quan HTML i CSS no són suficients.\"\n\n[taxonomies]\ntags = [\"funcionalitat\", \"tutorial\"]\n\n[extra]\nsocial_media_card = \"social_cards/ca_blog_javascript.jpg\"\n+++\n\nAquest tema no requereix JavaScript obligatori. Opcionalment, pot carregar una quantitat mínima per afegir algunes característiques que són impossibles d'aconseguir amb HTML i CSS.\n\n## Opcions habilitades globalment\n\n- [**Cerca**](@/blog/mastering-tabi-settings/index.ca.md#cerca). Activada establint un idioma per defecte i `build_search_index = true` a la secció principal de `config.toml`. (~23KB de JavaScript)\n\n- L'**interruptor de mode clar/fosc** es pot habilitar configurant `theme_switcher = true` a la secció `[extra]` del teu `config.toml` (~1KB de JavaScript).\n\n- **Decodificació de correu electrònic** (~400 bytes). Per protegir contra robots de correu brossa, pots configurar `encode_plaintext_email = true`. Si el teu lloc web està en un repositori públic, considera utilitzar el teu `email` com una cadena codificada en base64[^1].\n\n## Opcions que es poden sobreescriure de forma jeràrquica\n\nLes següents opcions es poden especificar per a publicacions, seccions i globalment, seguint la jerarquia de `pàgina > secció > config.toml`:\n\n- [**Suport de KaTeX**](@/blog/markdown/index.ca.md#katex). Habilitat configurant `katex = true` (274 KB). Per renderitzar fórmules sense JS, prova [MathML](https://developer.mozilla.org/docs/Web/MathML/).\n- [**Diagrames de Mermaid**](@/blog/shortcodes/index.ca.md#diagrames-de-mermaid). Habilitat configurant `mermaid = true` (~2.5 MB).\n- [**Còpia de blocs de codi amb un sol clic**](@/blog/markdown/index.ca.md#bloc-de-codi). Habilitada configurant `copy_button = true`. (~700 bytes)\n- [**Noms de blocs de codi clicables**](@/blog/shortcodes/index.ca.md#mostrar-ruta-o-url). S'activa configurant `code_block_name_links = true`. (~400 bytes)\n- [**Filtratge per etiquetes** per a graelles de targetes](@/blog/mastering-tabi-settings/index.ca.md#filtrar-projectes) (p. ex. [projectes](@/projects/_index.ca.md)) (~2KB). S'habilita configurant `enable_cards_tag_filtering = true`.\n\nPer especificar aquestes opcions:\n\n- **Globalment**: Afegeix-les sota la secció `[extra]` al teu `config.toml`.\n- **Per a una secció**: Afegeix-les sota la secció `[extra]` al front matter de l'`_index.md` de la secció.\n- **Per a una publicació individual**: Configura les variables corresponents a la secció `[extra]` del front matter de la publicació.\n\n## Opcions que es poden habilitar globalment o per a publicacions individuals\n\n- [**Comentaris**](@/blog/comments/index.ca.md). giscus (2 KB), utterances (1 KB), Hyvor Talk (~800 bytes) o Isso (1KB) es poden habilitar globalment configurant `enabled_for_all_posts = true` a la secció apropiada del teu `config.toml` (`[extra.giscus]`, `[extra.utterances]`, `[extra.hyvortalk]` o `[extra.isso]`). Per habilitar comentaris en publicacions individuals, configura el nom del sistema `= true` (per exemple, `hyvortalk = true`) al front matter del post.\n\nA part d'això, és un tema ràpid amb HTML i CSS que funciona sense JavaScript. Just com hauria de ser (la majoria de) la web :-)\n\n---\n\n[^1]: Per codificar el teu correu en base64 pots utilitzar [eines en línia](https://www.base64encode.org/) o, al terminal, executa: `printf 'mail@example.com' | base64`.\n"
  },
  {
    "path": "content/blog/javascript/index.es.md",
    "content": "+++\ntitle = \"Sin JavaScript obligatorio\"\ndate = 2023-01-06\nupdated = 2025-02-21\ndescription = \"JavaScript solo se utiliza cuando HTML y CSS no son suficientes.\"\n\n[taxonomies]\ntags = [\"funcionalidad\", \"tutorial\"]\n\n[extra]\nsocial_media_card = \"social_cards/es_blog_javascript.jpg\"\n+++\n\nEste tema no requiere JavaScript de manera obligatoria. Opcionalmente, puede cargar una cantidad mínima de JavaScript para añadir algunas características que son imposibles de lograr con solo HTML y CSS.\n\n## Opciones habilitadas globalmente\n\n- [**Búsqueda**](@/blog/mastering-tabi-settings/index.es.md#busqueda). Habilitada estableciendo un idioma por defecto y `build_search_index = true` en la sección principal de `config.toml`. (~23KB de JavaScript)\n\n- El **interruptor de modo claro/oscuro** puede habilitarse configurando `theme_switcher = true` en la sección `[extra]` de tu `config.toml` (~1KB de JavaScript).\n\n- **Descodificación de correo electrónico** (~400 bytes). Para proteger contra bots que recopilan correos electrónicos desde tu sitio web, puedes configurar `encode_plaintext_email = true`. Si tu sitio está en un repositorio público, para mayor protección, considera configurar tu `email` como una cadena codificada en base64[^1].\n\n## Opciones que se pueden sobreescribir de forma jerárquica\n\nLas siguientes opciones pueden especificarse para publicaciones, secciones y a nivel global, siguiendo la jerarquía de `página > sección > config.toml`:\n\n- [**Soporte de KaTeX**](@/blog/markdown/index.es.md#katex). Habilitado al configurar `katex = true` (274 KB). Para renderizar fórmulas sin JS, prueba [MathML](https://developer.mozilla.org/docs/Web/MathML/).\n- [**Diagramas de Mermaid**](@/blog/shortcodes/index.es.md#diagramas-de-mermaid). Habilitado al configurar `mermaid = true` (~2.5 MB).\n- [**Copia de bloques de código con un solo clic**](@/blog/markdown/index.es.md#bloque-de-codigo). Habilitado al configurar `copy_button = true` (~700 bytes).\n- [**Nombres de bloques de código clicables**](@/blog/shortcodes/index.es.md#mostrar-ruta-o-url). Se activa configurando `code_block_name_links = true`. (~400 bytes)\n- [**Filtraje por etiquetas** para cuadrículas de tarjetas](@/blog/mastering-tabi-settings/index.es.md#filtrar-proyectos) (p. ej. [proyectos](@/projects/_index.es.md)) (~2KB). Habilitado al configurar `enable_cards_tag_filtering = true`.\n\nPara especificar estas opciones:\n\n- **Globalmente**: Añádelas en la sección `[extra]` de tu `config.toml`.\n- **Para una sección**: Añádelas en la sección `[extra]` del front matter del `_index.md` de la sección.\n- **Para una publicación individual**: Configura las variables correspondientes en la sección `[extra]` del front matter de la publicación.\n\n## Opciones que pueden habilitarse globalmente o para publicaciones individuales\n\n- [**Comentarios**](@/blog/comments/index.es.md). giscus (2 KB), utterances (1 KB), Hyvor Talk (~800 bytes) o Isso (1KB) pueden habilitarse globalmente configurando `enabled_for_all_posts = true` en la sección apropiada de tu `config.toml` (`[extra.giscus]`, `[extra.utterances]`, `[extra.hyvortalk]` o `[extra.isso]`). Para habilitar comentarios en publicaciones individuales, configura el nombre del sistema `= true` (por ejemplo, `hyvortalk = true`) en el front matter de la publicación.\n\nAparte de eso, es un tema rápido con HTML y CSS que funciona con JavaScript deshabilitado. Justo como debería ser (la mayoría de) la web :-)\n\n---\n\n[^1]: Para codificar tu correo electrónico en base64 puedes usar [herramientas en línea](https://www.base64encode.org/) o, en tu terminal, ejecutar: `printf 'mail@example.com' | base64`.\n"
  },
  {
    "path": "content/blog/javascript/index.md",
    "content": "+++\ntitle = \"No mandatory JavaScript\"\ndate = 2023-01-06\nupdated = 2025-02-21\ndescription = \"JavaScript is only used when HTML and CSS aren't enough.\"\n\n[taxonomies]\ntags = [\"showcase\", \"tutorial\"]\n\n[extra]\nsocial_media_card = \"social_cards/blog_javascript.jpg\"\n+++\n\nThis theme has no mandatory JavaScript. Optionally, it can load a minimal amount to add some features that are impossible to achieve with HTML and CSS.\n\n## Globally enabled settings\n\n- [**Search**](@/blog/mastering-tabi-settings/index.md#search). Enabled by setting a default language and `build_search_index = true` on the main section of `config.toml`. (~23KB of JavaScript)\n\n- The **light/dark mode switch** can be enabled by setting `theme_switcher = true` in the `[extra]` section of your `config.toml` (~1KB of JavaScript).\n\n- **E-mail decoding** (~400 bytes). To protect against spambots scraping your e-mail from your website, you can set `encode_plaintext_email = true`. If your site is on a public repository, for extra protection, consider setting your `email` as a base64-encoded string[^1] directly.\n\n## Settings with hierarchical override capability\n\nThe following settings can be specified for posts, sections and globally, following the hierarchy of `page > section > config.toml`:\n\n- [**KaTeX support**](@/blog/markdown/index.md#katex). Enabled by setting `katex = true` (274 KB). To render math without JS, check out [MathML](https://developer.mozilla.org/docs/Web/MathML/).\n- [**Mermaid diagrams**](@/blog/shortcodes/index.md#mermaid-diagrams). Enabled by setting `mermaid = true` (~2.5 MB).\n- [**One-click copy of code blocks**](@/blog/markdown/index.md#code-block). Enabled by setting `copy_button = true`. (~700 bytes)\n- [**Clickable code block names**](@/blog/shortcodes/index.md#show-source-or-path). Enabled by setting `code_block_name_links = true`. (~300 bytes)\n- [**Tag filtering** for card grids](@/blog/mastering-tabi-settings/index.md#filtering-projects) (e.g. [projects](@/projects/_index.md)) (~2KB). Enabled by setting `enable_cards_tag_filtering = true`.\n\nTo specify these settings:\n\n- **Globally**: Add them under the `[extra]` section in your `config.toml` file.\n- **For a section**: Add them under the `[extra]` section in the front matter of the section's `_index.md`.\n- **For an individual post**: Set the corresponding variables in the `[extra]` section of the post's front matter.\n\n## Settings that can be enabled globally or for individual posts\n\n- [**Comments**](@/blog/comments/index.md). giscus (2 KB), utterances (1 KB), Hyvor Talk (~800 bytes) or Isso (1KB) can be globally enabled by setting `enabled_for_all_posts = true` in the right section of your  `config.toml` (i.e. `[extra.giscus]`, `[extra.utterances]`, `[extra.hyvortalk]` or `[extra.isso]`). To enable comments on individual posts, set the name of the system `= true` (e.g. `hyvortalk = true`) in the post's front matter.\n\nOther than that, it's a fast theme with HTML and CSS which works with JavaScript disabled. Just the way (most of) the web should be :-)\n\n---\n\n[^1]: To encode your email in base64 you can use [online tools](https://www.base64encode.org/) or, on your terminal, run: `printf 'mail@example.com' | base64`.\n"
  },
  {
    "path": "content/blog/markdown/index.ar.md",
    "content": "+++\ntitle = \"أمثلة على ماركداون\"\ndate = 2023-01-31\nupdated = 2026-01-01\ndescription = \"تعرض هذه التدوينة بعض الأمثلة على تنسيق ماركداون، بما في ذلك الجداول، والشِفرات البرمجية والوسوم، والاقتباسات، والهوامش.\"\n\n[taxonomies]\ntags = [\"ماركداون\", \"توضيحي\"]\n\n[extra]\nkatex = true\n+++\n\n{% admonition(type=\"note\", title=\"ملاحظة عن الاتجاه\", icon=\"info\") %}\nيدعم تابي الكتابة من اليمين إلى اليسار (RTL) بشكل كامل، مما يجعله مثالياً للغة العربية.\n\nالشفرات البرمجية والمعادلات الرياضية تبقى من اليسار إلى اليمين كما هو متوقع.\n{% end %}\n\n## الرياضيات مع $\\KaTeX$\n\n{{ aside(text=\"تأتي كلمة *ماركداون* من الإنجليزية *Markdown*، وهي لغة ترميز بسيطة صممها جون غروبر وآرون سوارتز في عام 2004.\") }}\n\n[$\\KaTeX$](https://katex.org/) هي مكتبة سريعة وسهلة الاستخدام تمكن من عرض الرموز الرياضية باستخدام صيغة LaTeX.\n\nيمكنك استخدام $\\KaTeX$ **ضمن السطر** عن طريق وضع التعبير بين `$` أو بين `\\\\(` و `\\\\)`.\n\nعلى سبيل المثال، `$ \\sin(x) = \\sum_{n=0}^{\\infty} \\frac{(-1)^n}{(2n + 1)!} x^{2n + 1} $` سيظهر هكذا: $ \\sin(x) = \\sum_{n=0}^{\\infty} \\frac{(-1)^n}{(2n + 1)!} x^{2n + 1} $\n\nلعرض التعبير **في سطر منفصل ومتوسط**، ضعه بين `$$` أو بين `\\\\[` و `\\\\]`.\n\nعلى سبيل المثال، `\\\\[ r = \\frac{\\sum_{i=1}^{n}(x_i - \\bar{x})(y_i - \\bar{y})}{\\sqrt{\\sum_{i=1}^{n}(x_i - \\bar{x})^2}\\sqrt{\\sum_{i=1}^{n}(y_i - \\bar{y})^2}} \\\\]` سيظهر هكذا: \\\\[ r = \\frac{\\sum_{i=1}^{n}(x_i - \\bar{x})(y_i - \\bar{y})}{\\sqrt{\\sum_{i=1}^{n}(x_i - \\bar{x})^2}\\sqrt{\\sum_{i=1}^{n}(y_i - \\bar{y})^2}} \\\\]\n\nفعّل $\\KaTeX$ بإضافة `katex = true` في `[extra]`:\n\n```toml,hl_lines=5-6\ntitle = \"تجربة KaTeX\"\ndate = 2002-11-30\n\n[extra]\nkatex = true\n```\n\nيعمل في: مقدمة الصفحة، `_index.md` للقسم، أو عالمياً في `config.toml`.\n\nلتحسين الأداء والأمان، يتم استضافة ملفات جافا سكريبت و CSS والخطوط الخاصة بـ $\\KaTeX$ محلياً.\n\n**ملاحظة**: بعد تفعيل $\\KaTeX$، إذا أردت استخدام \\$ بدون عرض تعبير رياضي، استخدم شرطة مائلة للخلف قبلها: `\\$`.\n\n### الصيغ الكيميائية\n\nالصيغ الكيميائية مدعومة عبر [امتداد mhchem](https://mhchem.github.io/MathJax-mhchem/)، الذي يُحمّل تلقائياً عند استخدام `\\ce{}` أو `\\pu{}`.\n\n`\\ce{}` للكيمياء: $\\ce{H2O}$، $\\ce{CO2 + H2O -> H2CO3}$\n\n`\\pu{}` للوحدات: $\\pu{25 °C}$، $\\pu{1.2 mol/L}$\n\n## جدول\n\nهذا مثال على جدول[^1]. تتغير ألوانه حسب سمة التدوينة.\n\n| الرمز  | العنصر   | العدد الذري |\n|--------|----------|--------------|\n| H      | هيدروجين| 1            |\n| C      | كربون    | 6            |\n| Fe     | حديد     | 26           |\n| Au     | ذهب      | 79           |\n\n## الشِفرات البرمجية\n\n```rust\nfn main() {\n    println!(\"مرحباً يا عالم\") -> ();\n}\n```\n\n### من اليمين إلى اليسار\n{% force_text_direction(direction=\"rtl\") %}\n\n```\nنص قل_مرحباً(نص الاسم) {\n    أرجع تنسيق(\"مرحباً {الاسم}\")؛\n}\n```\n\n{% end %}\n\n## سطر برمجي\n\nفي Rust، تعلن عن متغير متغير باستخدام `let mut x = 5`، بينما في Python، تستخدم ببساطة `x = 5`. وبالمثل، لطباعة قيمة في Rust، تستخدم `println!(\"القيمة: {}\", x)`، ولكن في Python، الأمر بسيط مثل `print(f\"القيمة: {x}\")`\nوفي لغة البرمجة العربية هو `مهنة = \"صائد فئران\"`\n\n## اقتباس\n\n> وابِطُكَ قابِضِ الأَرواحِ يَرمي   ...   بِسَهمِ المَوتِ مِن تَحتِ الثِيابِ\n>\n> شَرابُكَ في السَرابِ إِذا عَطِشنا   ...   وَخُبزُكَ عِندَ مُنقَطِعِ التُرابِ\n>\n> — أبو الشمقمق، العصر العباسي\n\n[^1]: وهذا مثال على الهامش!\n"
  },
  {
    "path": "content/blog/markdown/index.ca.md",
    "content": "+++\ntitle = \"Exemples de Markdown\"\ndate = 2023-01-31\nupdated = 2026-01-01\ndescription = \"Aquesta publicació mostra alguns exemples de format en Markdown, incloent-hi una taula, blocs de codi i etiquetes, citacions, taules i notes a peu de pàgina.\"\n\n[taxonomies]\ntags = [\"markdown\", \"funcionalitat\"]\n\n[extra]\nkatex = true\nsocial_media_card = \"social_cards/ca_blog_markdown.jpg\"\n+++\n\n## $\\KaTeX$\n\n[$\\KaTeX$](https://katex.org/) és una llibreria ràpida i fàcil d'usar que permet representar notació matemàtica mitjançant la sintaxi LaTeX.\n\nPots utilitzar $\\KaTeX$ **en línia** embolcallant l'expressió entre `$` o entre `\\\\(` i `\\\\)`.\n\nPer exemple, `$ \\sin(x) = \\sum_{n=0}^{\\infty} \\frac{(-1)^n}{(2n + 1)!} x^{2n + 1} $` es renderitzarà com: $ \\sin(x) = \\sum_{n=0}^{\\infty} \\frac{(-1)^n}{(2n + 1)!} x^{2n + 1} $\n\nPer mostrar l'expressió **en una línia pròpia i centrada**, embolcalla-la amb `$$` o entre `\\\\[` i `\\\\]`.\n\nPer exemple, `\\\\[ r = \\frac{\\sum_{i=1}^{n}(x_i - \\bar{x})(y_i - \\bar{y})}{\\sqrt{\\sum_{i=1}^{n}(x_i - \\bar{x})^2}\\sqrt{\\sum_{i=1}^{n}(y_i - \\bar{y})^2}} \\\\]` es renderitzarà com: \\\\[ r = \\frac{\\sum_{i=1}^{n}(x_i - \\bar{x})(y_i - \\bar{y})}{\\sqrt{\\sum_{i=1}^{n}(x_i - \\bar{x})^2}\\sqrt{\\sum_{i=1}^{n}(y_i - \\bar{y})^2}} \\\\]\n\nActiva $\\KaTeX$ afegint `katex = true` a `[extra]`:\n\n```toml,hl_lines=5-6\ntitle = \"Provant KaTeX\"\ndate = 2002-11-30\n\n[extra]\nkatex = true\n```\n\nFunciona a: metadades de pàgina, `_index.md` de secció, o globalment a `config.toml`.\n\nPer obtenir un millor rendiment i seguretat, els fitxers JavaScript, CSS i les tipografies de $\\KaTeX$ s'allotgen localment.\n\n**Nota**: Després d'activar $\\KaTeX$, si vols utilitzar el caràcter \\$ sense renderitzar-lo com a expressió matemàtica, escapa'l amb una barra inversa: `\\$`.\n\n### Fórmules químiques\n\nLes fórmules químiques estan suportades mitjançant l'[extensió mhchem](https://mhchem.github.io/MathJax-mhchem/), que es carrega automàticament en utilitzar `\\ce{}` o `\\pu{}`.\n\n`\\ce{}` per química: $\\ce{H2O}$, $\\ce{CO2 + H2O -> H2CO3}$\n\n`\\pu{}` per unitats: $\\pu{25 °C}$, $\\pu{1.2 mol/L}$\n\n## Taula\n\nAquí tens un exemple de taula[^1]. Els seus colors canvien en funció del tema actual.\n\n| Símbol  | Element | Nombre atòmic |\n|---------|---------|---------------|\n| H       | Hidrogen| 1             |\n| C       | Carboni | 6             |\n| Fe      | Ferro   | 26            |\n| Au      | Or      | 79            |\n\n## Bloc de codi\n\n```rust\nfn main() {\n    println!(\"Hola, món!\") -> ();\n}\n```\n\n### Amb numeració de línies\n\n```rust,linenos\nuse std::collections::HashMap;\n\n#[derive(Debug)]\nstruct TwinPeaksCharacter {\n    name: String,\n    coffee_rating: f32,\n    pie_preference: String,\n}\n\nfn main() {\n    let mut black_lodge = HashMap::new();\n\n    black_lodge.insert(\"agent\", TwinPeaksCharacter {\n        name: String::from(\"Dale Cooper\"),\n        coffee_rating: 9999.99,\n        pie_preference: String::from(\"Damn Fine Cherry\"),\n    });\n\n    black_lodge.insert(\"giant\", TwinPeaksCharacter {\n        name: String::from(\"The Fireman\"),\n        coffee_rating: 42.424242,\n        pie_preference: String::from(\"Garmonbozia\"),\n    });\n\n    // Calculate total appreciation of damn fine coffee\n    let total_coffee: f32 = black_lodge.values()\n        .map(|character| character.coffee_rating)\n        .sum();\n\n    println!(\"☕ Total coffee appreciation: {:.2} cups\", total_coffee);\n}\n```\n\n## Etiquetes de codi\n\nA Rust, declares una variable mutable amb `let mut x = 5;`, mentre que a Python, simplement fas `x = 5`. De manera similar, per imprimir un valor a Rust, utilitzaries `println!(\"Valor: {}\", x);`, però a Python, és tan senzill com `print(f\"Valor: {x}\")`.\n\n## Quote\n\n> «La vida, perquè sigui vida, s'ha de viure a poc a poc…»\n>\n> — Mercè Rodoreda, La plaça del Diamant\n\n---\n\n[^1]: I aquí tens un exemple de nota a peu de pàgina!\n"
  },
  {
    "path": "content/blog/markdown/index.es.md",
    "content": "+++\ntitle = \"Ejemplos de Markdown\"\ndate = 2023-01-31\nupdated = 2026-01-01\ndescription = \"Esta publicación muestra algunos ejemplos de formato Markdown, incluyendo una tabla, bloques de código y etiquetas, citas, tablas y notas al pie de página.\"\n\n[taxonomies]\ntags = [\"markdown\", \"funcionalidad\"]\n\n[extra]\nkatex = true\nsocial_media_card = \"social_cards/es_blog_markdown.jpg\"\n+++\n\n## $\\KaTeX$\n\n[$\\KaTeX$](https://katex.org/) es una biblioteca rápida y fácil de usar que permite la representación de notación matemática utilizando sintaxis LaTeX.\n\nPuedes usar $\\KaTeX$ **en línea** al envolver la expresión entre `$` o entre `\\\\(` y `\\\\)`.\n\nPor ejemplo, `$ \\sin(x) = \\sum_{n=0}^{\\infty} \\frac{(-1)^n}{(2n + 1)!} x^{2n + 1} $` se mostraría como: $ \\sin(x) = \\sum_{n=0}^{\\infty} \\frac{(-1)^n}{(2n + 1)!} x^{2n + 1} $\n\nPara mostrar la expresión **en su propia línea y centrada**, envuélvela entre `$$` o entre `\\\\[` y `\\\\]`.\n\nPor ejemplo, `\\\\[ r = \\frac{\\sum_{i=1}^{n}(x_i - \\bar{x})(y_i - \\bar{y})}{\\sqrt{\\sum_{i=1}^{n}(x_i - \\bar{x})^2}\\sqrt{\\sum_{i=1}^{n}(y_i - \\bar{y})^2}} \\\\]` se mostraría como: \\\\[ r = \\frac{\\sum_{i=1}^{n}(x_i - \\bar{x})(y_i - \\bar{y})}{\\sqrt{\\sum_{i=1}^{n}(x_i - \\bar{x})^2}\\sqrt{\\sum_{i=1}^{n}(y_i - \\bar{y})^2}} \\\\]\n\nActiva $\\KaTeX$ añadiendo `katex = true` en `[extra]`:\n\n```toml,hl_lines=5-6\ntitle = \"Probando KaTeX\"\ndate = 2002-11-30\n\n[extra]\nkatex = true\n```\n\nFunciona en: encabezado de página, `_index.md` de sección, o globalmente en `config.toml`.\n\nPara un mejor rendimiento y seguridad, el JavaScript, CSS y las fuentes de $\\KaTeX$ se alojan localmente.\n\n**Nota**: Después de habilitar $\\KaTeX$, si deseas usar \\$ sin representar una expresión matemática, escápalo con una sola barra invertida: `\\$`.\n\n### Fórmulas químicas\n\nLas fórmulas químicas están soportadas mediante la [extensión mhchem](https://mhchem.github.io/MathJax-mhchem/), que se carga automáticamente al usar `\\ce{}` o `\\pu{}`.\n\n`\\ce{}` para química: $\\ce{H2O}$, $\\ce{CO2 + H2O -> H2CO3}$\n\n`\\pu{}` para unidades: $\\pu{25 °C}$, $\\pu{1.2 mol/L}$\n\n## Tabla\n\nAquí tienes un ejemplo de una tabla[^1]. Los colores cambian dependiendo del tema actual.\n\n| Símbolo | Elemento | Número atómico |\n|---------|----------|----------------|\n| H       | Hidrógeno| 1              |\n| C       | Carbono  | 6              |\n| Fe      | Hierro   | 26             |\n| Au      | Oro      | 79             |\n\n## Bloque de código\n\n```rust\nfn main() {\n    println!(\"¡Hola, mundo!\") -> ();\n}\n```\n\n### Con números de línea\n\n```rust,linenos\nuse std::collections::HashMap;\n\n#[derive(Debug)]\nstruct TwinPeaksCharacter {\n    name: String,\n    coffee_rating: f32,\n    pie_preference: String,\n}\n\nfn main() {\n    let mut black_lodge = HashMap::new();\n\n    black_lodge.insert(\"agent\", TwinPeaksCharacter {\n        name: String::from(\"Dale Cooper\"),\n        coffee_rating: 9999.99,\n        pie_preference: String::from(\"Damn Fine Cherry\"),\n    });\n\n    black_lodge.insert(\"giant\", TwinPeaksCharacter {\n        name: String::from(\"The Fireman\"),\n        coffee_rating: 42.424242,\n        pie_preference: String::from(\"Garmonbozia\"),\n    });\n\n    // Calculate total appreciation of damn fine coffee\n    let total_coffee: f32 = black_lodge.values()\n        .map(|character| character.coffee_rating)\n        .sum();\n\n    println!(\"☕ Total coffee appreciation: {:.2} cups\", total_coffee);\n}\n```\n\n## Etiquetas de código\n\nEn Rust, declaras una variable mutable con `let mut x = 5;`, mientras que en Python, simplemente usas `x = 5`. De manera similar, para imprimir un valor en Rust, utilizarías `println!(\"Valor: {}\", x);`, pero en Python, es tan sencillo como `print(f\"Valor: {x}\")`.\n\n## Cita\n\n> «A mí me sobra el cuerpo, Orfeo, me sobra el cuerpo porque me falta alma.»\n>\n> — Miguel de Unamuno, Niebla\n\n---\n\n[^1]: ¡Y aquí tienes un ejemplo de una nota al pie de página!\n"
  },
  {
    "path": "content/blog/markdown/index.md",
    "content": "+++\ntitle = \"Markdown examples\"\ndate = 2023-01-31\nupdated = 2026-01-01\ndescription = \"This post showcases some examples of Markdown formatting, including a table, code blocks and tags, quotes, tables, and footnotes.\"\n\n[taxonomies]\ntags = [\"markdown\", \"showcase\"]\n\n[extra]\nkatex = true\nsocial_media_card = \"social_cards/blog_markdown.jpg\"\n+++\n\n## $\\KaTeX$\n\n[$\\KaTeX$](https://katex.org/) is a fast and easy-to-use library that enables the rendering of mathematical notation, using LaTeX syntax.\n\nYou can use $\\KaTeX$ **inline** by wrapping the expression between `$` or between `\\\\(` and `\\\\)`.\n\nFor example, `$ \\sin(x) = \\sum_{n=0}^{\\infty} \\frac{(-1)^n}{(2n + 1)!} x^{2n + 1} $` would render: $ \\sin(x) = \\sum_{n=0}^{\\infty} \\frac{(-1)^n}{(2n + 1)!} x^{2n + 1} $\n\nTo display the expression **on its own line and centered**, wrap it around `$$` or between `\\\\[` and `\\\\]`.\n\nFor example, `\\\\[ r = \\frac{\\sum_{i=1}^{n}(x_i - \\bar{x})(y_i - \\bar{y})}{\\sqrt{\\sum_{i=1}^{n}(x_i - \\bar{x})^2}\\sqrt{\\sum_{i=1}^{n}(y_i - \\bar{y})^2}} \\\\]` renders: \\\\[ r = \\frac{\\sum_{i=1}^{n}(x_i - \\bar{x})(y_i - \\bar{y})}{\\sqrt{\\sum_{i=1}^{n}(x_i - \\bar{x})^2}\\sqrt{\\sum_{i=1}^{n}(y_i - \\bar{y})^2}} \\\\]\n\nEnable $\\KaTeX$ by setting `katex = true` in `[extra]`:\n\n```toml,hl_lines=5-6\ntitle = \"Testing KaTeX\"\ndate = 2002-11-30\n\n[extra]\nkatex = true\n```\n\nWorks in: page front matter, section `_index.md`, or globally in `config.toml`.\n\nFor enhanced performance and security, the $\\KaTeX$ JavaScript, CSS, and fonts are hosted locally.\n\n**Note**: After enabling $\\KaTeX$, if you want to use \\$ without rendering a mathematical expression, escape it with a single backslash: `\\$`.\n\n### Chemistry formulas\n\nChemistry formulas are supported via the [mhchem extension](https://mhchem.github.io/MathJax-mhchem/), which loads automatically when using `\\ce{}` or `\\pu{}`.\n\n`\\ce{}` for chemistry: $\\ce{H2O}$, $\\ce{CO2 + H2O -> H2CO3}$\n\n`\\pu{}` for units: $\\pu{25 °C}$, $\\pu{1.2 mol/L}$\n\n## Table\n\nHere's an example of a table[^1]. Its colours change depending on the current theme.\n\n| Symbol  | Element | Atomic Number |\n|---------|---------|---------------|\n| H       | Hydrogen| 1             |\n| C       | Carbon  | 6             |\n| Fe      | Iron    | 26            |\n| Au      | Gold    | 79            |\n\n## Code Block\n\n```rust\nfn main() {\n    println!(\"Hello, world!\") -> ();\n}\n```\n\n### With line numbers\n\n```rust,linenos\nuse std::collections::HashMap;\n\n#[derive(Debug)]\nstruct TwinPeaksCharacter {\n    name: String,\n    coffee_rating: f32,\n    pie_preference: String,\n}\n\nfn main() {\n    let mut black_lodge = HashMap::new();\n\n    black_lodge.insert(\"agent\", TwinPeaksCharacter {\n        name: String::from(\"Dale Cooper\"),\n        coffee_rating: 9999.99,\n        pie_preference: String::from(\"Damn Fine Cherry\"),\n    });\n\n    black_lodge.insert(\"giant\", TwinPeaksCharacter {\n        name: String::from(\"The Fireman\"),\n        coffee_rating: 42.424242,\n        pie_preference: String::from(\"Garmonbozia\"),\n    });\n\n    // Calculate total appreciation of damn fine coffee\n    let total_coffee: f32 = black_lodge.values()\n        .map(|character| character.coffee_rating)\n        .sum();\n\n    println!(\"☕ Total coffee appreciation: {:.2} cups\", total_coffee);\n}\n```\n\n## Code tags\n\nIn Rust, you declare a mutable variable with `let mut x = 5;`, whereas in Python, you simply use `x = 5`. Similarly, to print a value in Rust, you would use `println!(\"Value: {}\", x);`, but in Python, it's as straightforward as `print(f\"Value: {x}\")`.\n\n## Quote\n\n> \"We're all hurtling towards death. Yet here we are, for the moment, alive. Each of us knowing we're going to die. Each of us secretly believing we won't.\"\n>\n> — Charlie Kaufman, Synecdoche, New York\n\n---\n\n[^1]: And here's an example of a footnote!\n"
  },
  {
    "path": "content/blog/mastering-tabi-settings/index.ca.md",
    "content": "+++\ntitle = \"Domina la configuració de tabi: guia completa\"\ndate = 2023-09-18\nupdated = 2026-01-31\ndescription = \"Descobreix les múltiples maneres en què pots personalitzar tabi.\"\n\n[taxonomies]\ntags = [\"funcionalitat\", \"tutorial\", \"preguntes freqüents\"]\n\n[extra]\npinned = true\nquick_navigation_buttons = true\nsocial_media_card = \"social_cards/ca_blog_mastering_tabi_settings.jpg\"\n+++\n\nAquesta és la guia completa sobre la configuració a tabi. Si tens alguna pregunta, pots [obrir un issue a GitHub](https://github.com/welpo/tabi/issues/new) o [iniciar una discussió](https://github.com/welpo/tabi/discussions).\n\n<details>\n    <summary><b>Taula de continguts</b></summary>\n    <!-- toc -->\n</details>\n\n## Jerarquia de configuració\n\ntabi té una jerarquia que permet personalitzar el teu lloc a diferents nivells. La jerarquia (de menor a major prioritat) és la següent:\n\n1. **Configuracions globals**: Aquestes són les configuracions que s'apliquen a tot el teu lloc. Es configuren a `config.toml`.\n2. **Configuracions de secció**: Aquestes són les configuracions que s'apliquen a una secció del teu lloc (per exemple, `/blog` o `/projects`). Es configuren a la metainformació de l'arxiu `_index.md` de la secció.\n3. **Configuració de la pàgina «pare»**: Per a pàgines anidades (pàgines dins de pàgines), aquestes són les configuracions de la pàgina que les conté.\n4. **Configuracions de pàgina**: Aquestes són les configuracions que s'apliquen a una sola pàgina. Es configuren a la metainformació de la pàgina.\n\nEn tots els casos, les opcions de tabi es configuren a la secció `[extra]`.\n\nPer a les configuracions que segueixen aquesta jerarquia, el valor establert a una pàgina reemplaça el valor d'una secció, que al seu torn reemplaça el valor global. En resum: com més específica sigui la configuració, més prioritat tindrà, o `pàgina > pàgina pare/secció > config.toml`.\n\n---\n\n## Cerca\n\n| Pàgina | Secció | `config.toml` | Segueix la jerarquia | Requereix JavaScript |\n|:------:|:------:|:-------------:|:--------------------:|:--------------------:|\n|   ❌   |   ❌   |       ✅      |          ❌          |          ✅          |\n\ntabi permet cerca local accessible i multilingüe amb [Elasticlunr](http://elasticlunr.com/). Per activar-la, necessites:\n\n1. Establir un `default_language` a `config.toml`.\n2. Establir `build_search_index = true`.\n3. Opcionalment, configurar la secció `[search]`.\n\nPer exemple:\n\n```toml\nbase_url = \"https://example.com\"\ndefault_language = \"en\"\nbuild_search_index = true\n\n[search]\nindex_format = \"elasticlunr_json\" # O el menys eficient \"elasticlunr_javascript\".\ninclude_title = true\ninclude_description = true\ninclude_path = true\ninclude_content = true\n```\n\n**Nota**: per suport de cerca en Xinès/Japonès, necessites utilitzar una [build personalitzada de Zola](https://github.com/getzola/zola/blob/master/Cargo.toml#L54-L55). Addicionalment, actualment no hi ha suport per a la cerca en català.\n\n### Consideracions per a usuaris de Zola 0.17.X\n\nZola 0.17.X no proporciona accés a la variable `search.index_format` ([informe del bug](https://github.com/getzola/zola/issues/2165)). En utilitzar tabi, s'assumeix l'ús de l'índex JSON, que és més eficient. No obstant això, a causa d'[un altre bug](https://github.com/getzola/zola/issues/2193) solucionat en 0.18.0, l'índex JSON per a llocs multilingües no es genera correctament.\n\nEls usuaris amb versions de Zola anteriors a 0.18.0 que vulguin utilitzar l'índex JavaScript necessiten establir la variable `index_format` a dos llocs:\n\n```toml\n[search]\nindex_format = \"elasticlunr_javascript\"\n\n[extra]\nindex_format = \"elasticlunr_javascript\"\n```\n\nAixò assegura que tabi carregui els arxius correctes. Recomanem actualitzar a Zola 0.18.0 o posterior per a una funcionalitat òptima.\n\n### Detalls d'implementació\n\nPer a detalls tècnics sobre la implementació de la cerca, incloent quan es carrega l'índex, característiques d'accessibilitat i altres detalls, consulta el [Pull Request #250](https://github.com/welpo/tabi/pull/250).\n\n---\n\n## Suport multilingüe\n\ntabi ofereix suport multilingüe complet per al teu lloc Zola, des de configurar un idioma predeterminat fins a afegir tots els que vulguis. Consulta les [preguntes freqüents sobre idiomes](@/blog/faq-languages/index.ca.md) per a més informació.\n\n---\n\n## Aparença\n\n### Pàgina principal\n\nLa [pàgina principal](/) d'aquesta demo té una capçalera amb una imatge, un títol i una descripció:\n\n{{ dual_theme_image(light_src=\"blog/mastering-tabi-settings/img/header_light.webp\", dark_src=\"blog/mastering-tabi-settings/img/header_dark.webp\", alt=\"Capçalera de la pàgina principal\") }}\n\n#### Capçalera\n\nPer configurar la imatge i el títol, pots utilitzar la variable `header` al front matter de l'arxiu `_index.md` de la secció. Per exemple:\n\n```toml\n[extra]\nheader = {title = \"Hola! Soc tabi~\", img = \"img/main.webp\", img_alt = \"Óscar Fernández, l'autor del tema\" }\n```\n\nLa descripció és contingut Markdown normal, escrit fora del front matter.\n\n#### Llistant publicacions recents\n\nPer mostrar publicacions a la pàgina principal, primer has de decidir d'on es serviran: de la ruta arrel (`/`) o d'un subdirectori (per exemple, `/blog`).\n\n**Opció A: Servir publicacions des de la ruta arrel (`/`)**\n\nConfigura `paginate_by` al front matter del teu arxiu `content/_index.md`:\n\n```toml\ntitle = \"Últimes publicacions\"\nsort_by = \"date\"\npaginate_by = 5  # Mostra 5 publicacions per pàgina.\n\n[extra]\nheader = {title = \"Hola! Soc tabi~\", img = \"img/main.webp\", img_alt = \"El teu nom\" }\n```\n\n{{ admonition(type=\"note\", text=\"La configuració `paginate_by` va al front matter principal, no a la secció `[extra]`.\") }}\n\n**Opció B: Servir publicacions des d'un subdirectori (per exemple, `/blog`)**\n\nUtilitza `section_path` a la secció `[extra]` del teu arxiu `content/_index.md`:\n\n```toml\ntitle = \"Últimes publicacions\"\nsort_by = \"date\"\n# No configuris `paginate_by` aquí.\n\n[extra]\nheader = {title = \"Hola! Soc tabi~\", img = \"img/main.webp\", img_alt = \"El teu nom\" }\nsection_path = \"blog/_index.md\"  # On trobar les teves publicacions.\nmax_posts = 5  # Mostra fins a 5 publicacions a la pàgina principal.\n```\n\n{{ admonition(type=\"warning\", title=\"ALERTA\", text=\"No configuris `paginate_by` i `section_path` alhora. Aquestes configuracions són mútuament excloents i usar ambdues pot fer que no es mostrin publicacions.\") }}\n\nNotes addicionals:\n\n- El `title` al front matter estableix el títol que apareix sobre les publicacions.\n- Utilitza la ruta completa a l'arxiu `_index.md` de la secció per a `section_path`. Usar `section_path = \"blog/\"` no funcionarà.\n\n##### Fixar entrades\n\nPots fixar entrades per mantenir-les a la part superior de la pàgina principal. En aquesta demo, aquesta entrada està fixada, així que apareix primera amb una icona i etiqueta de \"fixada\":\n\n{{ dual_theme_image(light_src=\"blog/mastering-tabi-settings/img/pinned_post_light.webp\", dark_src=\"blog/mastering-tabi-settings/img/pinned_post_dark.webp\", alt=\"Entrada fixada\", full_width=true) }}\n\nLes entrades fixades es mostren primer, mantenint el seu ordre relatiu segons el `sort_by` de la secció, seguides per les entrades regulars.\n\nPer fixar una entrada, afegeix el següent al seu front matter:\n\n```toml\n[extra]\npinned = true\n```\n\n{{ admonition(type=\"info\", text=\"Aquesta configuració només afecta les pàgines principals del lloc (com `/`, `/es/`, `/fr/`). Altres seccions com `blog/`, `tags/` o `archive/` mostren les publicacions en el seu ordre habitual.\") }}\n\n{{ admonition(type=\"warning\", text='Quan s'utilitza la paginació (`paginate_by`), les entrades fixades poden aparèixer dues vegades: una vegada a la part superior de la primera pàgina, i una altra en la seva posició cronològica normal en pàgines posteriors.') }}\n\n##### Mostrar la data dels articles al llistat\nPer defecte, quan es llisten els articles, es mostra la data de creació. Pots configurar quina(es) data(es) mostrar utilitzant l'opció `post_listing_date`. Configuracions disponibles:\n\n- `date`: Mostra només la data de publicació original de l'article (opció per defecte).\n- `updated`: Mostra només la data de l'última actualització de l'article. Si no hi ha data d'actualització, es mostra la data de publicació original.\n- `both`: Mostra tant la data de publicació original com la data de l'última actualització.\n\n{% admonition(type=\"tip\") %}\nAquesta configuració segueix la jerarquia: pots establir un valor global a `config.toml` o canviar-lo per a seccions específiques al seu arxiu `_index.md`. En ambdós casos, afegeix-lo a la secció `[extra]`.\n{% end %}\n\n#### Llistat de Projectes\n\nPots mostrar una selecció de projectes a la teva pàgina principal. Per fer això, primer necessitaràs configurar el directori `projects`.\n\nUn cop fet això, configura la ruta als projectes a la secció `[extra]` del teu fitxer `_index.md`:\n\n```toml\n[extra]\nprojects_path = \"projects/_index.md\"\n```\n\nAixò mostrarà els 3 projectes de major prioritat (amb menor pes; el mateix ordre que a Projectes). Per mostrar més o menys projectes, configura `max_projects` a `[extra]`.\n\nPer defecte, la secció de projectes es mostrarà a la part inferior de la pàgina principal, sota les publicacions. Si prefereixes mostrar-la a la part superior, configura `show_projects_first = true`.\n\n### Commutador de mode clar i fosc\n\n| Pàgina | Secció | `config.toml` | Segueix la jerarquia | Requereix JavaScript |\n|:------:|:------:|:-------------:|:---------------:|:-------------------:|\n|   ❌   |   ❌    |      ✅       |        ❌        |         ✅          |\n\nEl commutador de mode clar i fosc (la icona de lluna/sol a la cantonada superior dreta) es pot habilitar configurant `theme_switcher = true` a `config.toml`.\n\n### Mode predeterminat (clar/fosc)\n\n| Pàgina | Secció | `config.toml` | Segueix la jerarquia | Requereix JavaScript |\n|:------:|:------:|:-------------:|:---------------:|:-------------------:|\n|   ❌   |   ❌    |      ✅       |        ❌        |         ❌          |\n\nEl mode predeterminat es pot especificar amb la variable `default_theme`, que accepta `\"dark\"` o `\"light\"`. Si no s'especifica, el mode predeterminat és el mode del sistema.\n\n### Skin personalitzada\n\n| Pàgina | Secció | `config.toml` | Segueix la jerarquia | Requereix JavaScript |\n|:------:|:------:|:-------------:|:---------------:|:-------------------:|\n|   ❌   |   ❌    |      ✅       |        ❌        |         ❌          |\n\nLes skins («pells») de tabi canvien el color principal del lloc web. Pots configurar la skin a `config.toml` amb `skin = \"nom_de_la_skin`. Per exemple, `skin = \"lavender\"` es veu així (clica per canviar entre mode clar i fosc):\n\n{{ image_toggler(default_src=\"blog/customise-tabi/skins/lavender_light.webp\", toggled_src=\"blog/customise-tabi/skins/lavender_dark.webp\", default_alt=\"pell lavender en mode clar\", toggled_alt=\"pell lavender en mode fosc\", full_width=true) }}\n\nExplora les skins disponibles i aprèn com crear la teva pròpia consultant [la documentació](@/blog/customise-tabi/index.ca.md#skins).\n\n### Font sans serif (pal sec)\n\n| Pàgina | Secció | `config.toml` | Segueix la jerarquia | Requereix JavaScript |\n|:------:|:------:|:-------------:|:---------------:|:-------------------:|\n|   ❌   |   ❌    |      ✅       |        ❌        |         ❌          |\n\ntabi utilitza una font serif per als paràgrafs dels articles (la que estàs veient ara). Pots canviar a una font sans-serif (la que veus als encapçalaments/menú) a tot el teu lloc configurant `override_serif_with_sans = true` a `config.toml`.\n\nFes clic a la imatge a continuació per comparar les fonts:\n\n{{ image_toggler(default_src=\"blog/mastering-tabi-settings/img/serif.webp\", toggled_src=\"blog/mastering-tabi-settings/img/sans-serif.webp\", default_alt=\"Font serif\", toggled_alt=\"Font sans-serif\", full_width=true) }}\n\n### Indicador d'enllaços externs\n\n| Pàgina | Secció | `config.toml` | Segueix Jerarquia | Requereix JavaScript |\n|:------:|:------:|:-------------:|:-----------------:|:-------------------:|\n|   ❌   |   ❌   |      ✅       |         ❌        |         ❌          |\n\n{{ admonition(type=\"info\", text=\"Requereix Zola 0.20.0 o posterior.\") }}\n\nSi vols afegir una icona als enllaços externs, configura la secció `[markdown]` (no `[extra]`) al teu `config.toml`:\n\n```toml\n[markdown]\nexternal_links_class = \"external\"\n```\n\nAixò afegirà una petita icona al costat dels enllaços externs:\n\n{{ dual_theme_image(light_src=\"blog/mastering-tabi-settings/img/external_link_light.webp\", dark_src=\"blog/mastering-tabi-settings/img/external_link_dark.webp\", alt=\"Icona d'enllaç extern\", full_width=true) }}\n\n### Estils CSS personalitzats\n\n| Pàgina | Secció | `config.toml` | Segueix la jerarquia | Requereix JavaScript |\n|:------:|:------:|:-------------:|:---------------:|:-------------------:|\n|   ✅   |   ❌    |      ✅       |        ❌        |         ❌          |\n\nPots carregar estils CSS personalitzats per a tot el lloc web o en pàgines específiques utilitzant `stylesheets`, que accepta una llista de rutes cap a arxius CSS. Per exemple:\n\n```toml\nstylesheets = [\"css/custom.css\", \"css/another.css\"]\n```\n\n### Color del tema del navegador\n\n| Pàgina | Secció | `config.toml` | Segueix la jerarquia | Requereix JavaScript |\n|:------:|:------:|:-------------:|:---------------:|:-------------------:|\n|   ❌   |   ❌    |      ✅       |        ❌        |         ❌          |\n\nEl color del tema del navegador és el color que apareix a la barra de pestanyes del navegador:\n\n{{ dual_theme_image(light_src=\"blog/mastering-tabi-settings/img/browser_theme_color_light.webp\", dark_src=\"blog/mastering-tabi-settings/img/browser_theme_color_dark.webp\" alt=\"pestanyes amb un tema de navegador de color\") }}\n\nPots establir-ho a `config.toml` com a `browser_theme_color = \"#087e96\"`. Si vols diferents colors per als modes clar/obscur, pots establir un conjunt de colors amb `browser_theme_color = [\"#ffffff\", \"#000000\"]`. El primer color és per al mode clar, el segon per al fosc.\n\nAquesta variable accepta qualsevol color CSS vàlid, així que pots utilitzar paraules clau (per exemple, `blue`), codis hexadecimals (per exemple, `#087e96`) o valors RGB/HSL (per exemple, `rgb(8, 126, 150)`).\n\n### Etiquetes compactes\n\n| Pàgina | Secció | `config.toml` | Segueix la jerarquia | Requereix JavaScript |\n|:------:|:------:|:-------------:|:-----------------:|:--------------------:|\n|   ❌   |   ❌   |      ✅       |         ❌        |          ❌          |\n\nPer defecte, la [pàgina d'etiquetes](/ca/tags) mostra les etiquetes com:\n\n[NomEtiqueta](#) — n entrada[es]\n\nEstablir `compact_tags = true` les mostrarà com:\n\n[NomEtiqueta](#) <sup>n</sup>\n\n### Ordre de les etiquetes\n\n| Pàgina | Secció | `config.toml` | Segueix la jerarquia | Requereix JavaScript |\n|:------:|:------:|:-------------:|:-----------------:|:--------------------:|\n|   ❌   |   ❌   |      ✅       |         ❌        |          ❌          |\n\nPer defecte, la [pàgina d'etiquetes](/ca/tags) ordena les etiquetes alfabèticament, donada la configuració predeterminada de `tag_sorting = \"name\"`.\nSi configures `tag_sorting = \"frequency\"`, s'ordenaran segons el nombre de publicacions (de més a menys).\n\n---\n\n## Sèries\n\nPer a una explicació detallada, consulta la [documentació de sèries](@/blog/series/index.ca.md).\n\n### Enllaç per saltar a les publicacions\n\n| Pàgina | Secció | `config.toml` | Segueix la jerarquia | Requereix JavaScript |\n|:------:|:-------:|:-------------:|:------------------:|:-------------------:|\n|   ❌   |   ✅    |      ✅       |        ✅          |         ❌          |\n\nPer defecte, apareix automàticament un enllaç \"Salta a les publicacions\" al costat del títol de la sèrie quan una sèrie té un contingut de més de 2000 caràcters:\n\n{{ dual_theme_image(light_src=\"blog/series/img/jump_to_series_posts_light.webp\", dark_src=\"blog/series/img/jump_to_series_posts_dark.webp\" alt=\"enllaç per saltar a les publicacions de la sèrie\", full_width=true) }}\n\nEstableix `show_jump_to_posts = true` per forçar l'activació de la funció i `show_jump_to_posts = false` per desactivar-la.\n\n### Indexació de pàgines de sèries\n\n| Pàgina | Secció | `config.toml` | Segueix la jerarquia | Requereix JavaScript |\n|:------:|:-------:|:-------------:|:------------------:|:-------------------:|\n|   ❌   |   ✅    |      ✅       |        ✅          |         ❌          |\n\nPer defecte, les pàgines de sèries s'indexen (usant una indexació basada en 1) segons el `sort_by` de la secció de sèries.\n\nEstableix `post_listing_index_reversed = true` per invertir aquest índex.\n\n---\n\n## Integració amb repositoris Git\n\n| Pàgina | Secció | `config.toml` | Segueix la jerarquia | Requereix JavaScript |\n|:------:|:-------:|:-------------:|:--------------------:|:--------------------:|\n|   ❓   |    ❓   |       ✅      |          ❓          |          ❌          |\n\n❓: `show_remote_source` sí que segueix [la jerarquia](#jerarquia-de-configuracio) i es pot configurar en una pàgina, secció o de manera global. La resta de les configuracions només es poden establir a `config.toml`.\n\nAquestes configuracions et permeten vincular el teu lloc web tabi amb un repositori públic de Git a GitHub, GitLab, Gitea o Codeberg. Exemples de configuració:\n\n```toml\nremote_repository_url = \"https://github.com/welpo/tabi\"\nremote_repository_git_platform = \"auto\"\nremote_repository_branch = \"main\"\nshow_remote_changes = true\nshow_remote_source = true\n```\n\nAixò habilita dues funcions:\n\n1. `show_remote_source = true` afegeix un enllaç al codi font del teu lloc web (el teu `remote_repository_url`) que es mostrarà al peu de pàgina:\n\n{{ dual_theme_image(light_src=\"blog/mastering-tabi-settings/img/site_source_light.webp\", dark_src=\"blog/mastering-tabi-settings/img/site_source_dark.webp\" alt=\"Peu de pàgina del lloc web, mostrant un enllaç 'Codi font del lloc'\") }}\n\n2. `show_remote_changes = true` afegeix un enllaç \"Veure canvis ↗\" a l'historial de commits de l'article actualitzat, al costat de la data de l'última actualització [^1]:\n\n{{ dual_theme_image(light_src=\"blog/mastering-tabi-settings/img/see_changes_light.webp\", dark_src=\"blog/mastering-tabi-settings/img/see_changes_dark.webp\" alt=\"Títol de l'article i metadades, mostrant un enllaç 'Veure canvis'\") }}\n\nEn clicar aquest enllaç, seràs dirigit a l'historial de commits de l'article, on podràs veure els canvis realitzats en ell:\n\n{{ dual_theme_image(light_src=\"blog/mastering-tabi-settings/img/commit_history_light.webp\", dark_src=\"blog/mastering-tabi-settings/img/commit_history_dark.webp\" alt=\"Historial de commits d'un article\", full_width=true) }}\n\n---\n\n## Pàgines\n\n### Projectes\n\ntabi té una plantilla integrada per a projectes. Per habilitar-la, pots crear un directori a `content/projects/`. Allà, pots crear un fitxer `_index.md` amb el següent contingut al bloc de metadades:\n\n```toml\ntitle = \"Projectes\"\nsort_by = \"weight\"\ntemplate = \"cards.html\"\ninsert_anchor_links = \"left\"\n\n[extra]\nshow_reading_time = false\nquick_navigation_buttons = true\n```\n\n- `title` és el títol de la pàgina.\n- `sort_by` determina com s'ordenen els projectes. Pots ordenar per \"date\", \"update_date\", \"title\", \"title_bytes\", \"weight\", \"slug\" o \"none\".\n- `template = \"cards.html\"` estableix la plantilla per renderitzar la pàgina de projectes.\n- `insert_anchor_links = \"left\"` afegeix enllaços àncora als encapçalaments.\n- `show_reading_time = false` amaga el temps estimat de lectura.\n- `quick_navigation_buttons = true` mostra els botons de navegació ràpida.\n\nAl costat del fitxer `_index.md`, pots crear un fitxer per a cada projecte. Per exemple, aquest és el bloc de metadades per a la pàgina del projecte [tabi](@/projects/tabi/index.ca.md):\n\n```toml\ntitle = \"tabi\"\ndescription = \"Un tema de Zola ràpid, lleuger i modern amb suport multilingüe.\"\nweight = 1\n\n[extra]\nlocal_image = \"img/tabi.webp\"\ninvertible_image = false\n```\n\n- `title` és el títol del projecte.\n- `description` és la descripció del projecte.\n- `weight` determina l'ordre en què es mostren els projectes. Com menor sigui el pes, més amunt apareixerà el projecte.\n- `local_image` és la ruta de la imatge del projecte. Aquesta imatge es mostra a la pàgina de projectes.\n- `local_image_dark` és una variant opcional per al mode fosc. Requereix que `local_image` estigui configurat.\n- `remote_image` és una URL a una imatge externa, com a alternativa a `local_image`.\n- `remote_image_dark` és una variant opcional per al mode fosc. Requereix que `remote_image` estigui configurat.\n- `invertible_image` inverteix els colors de la imatge en mode fosc. Útil per a logotips o icones en blanc i negre.\n\nQuan un usuari faci clic a la imatge o al títol d'un projecte, serà portat a la pàgina del projecte. Si prefereixes que els usuaris vagin a un enllaç extern, pots establir `link_to = \"https://example.com\"` a la secció `[extra]` del fitxer `.md` del projecte.\n\nLa pàgina del projecte individual es renderitza amb la plantilla predeterminada, tret que estableixis una altra, per exemple, `template = \"info-page.html\"`.\n\n#### Filtrar projectes\n\nSi afegeixes etiquetes als teus projectes, veuràs un filtre d'etiquetes:\n\n{{ dual_theme_image(light_src=\"blog/mastering-tabi-settings/img/projects_tag_filter_light.webp\", dark_src=\"blog/mastering-tabi-settings/img/projects_tag_filter_dark.webp\", alt=\"Pàgina de projectes amb filtre d'etiquetes\", full_width=true) }}\n\nEl sistema de filtratge d'etiquetes utilitza millora progressiva:\n\n- Sense JavaScript: Les etiquetes enllacen directament a pàgines d'etiquetes dedicades (per exemple, `/tags/nom-de-l-etiqueta`).\n- Amb JavaScript: Filtratge instantani, sincronització d'URL (#nom-etiqueta) i navegació amb el teclat.\n\nPer desactivar aquesta funció, estableix `enable_cards_tag_filtering = false` a la secció `[extra]` del fitxer `projects/_index.md` o a `config.toml`.\n\n{% admonition(type=\"tip\") %}\n\nPer filtrar projectes per etiquetes, necessites establir etiquetes a la front matter de cada projecte. Per exemple:\n\n```toml\ntitle = \"nom del projecte\"\nweight = 40\n\n[taxonomies]\ntags = [\"etiqueta\", \"etiqueta 2\", \"tercera etiqueta\"]\n```\n\n{% end %}\n\n### Arxiu\n\nAfegir una pàgina d'arxiu és similar a afegir una pàgina de projectes. Pots crear un directori a `content/archive/`. Allà, pots crear un fitxer `_index.md` amb el següent encapçalament:\n\n```toml\ntitle = \"Arxiu\"\ntemplate = \"archive.html\"\n```\n\nPer defecte, l'arxiu llistarà les publicacions situades a `blog/`. Per personalitzar això, pots modificar la secció `[extra]` de l'arxiu `_index.md`:\n\n- **Per a un sol directori**: Estableix `section_path = \"directori/\"` per llistar publicacions d'un directori específic. Assegura't d'incloure la barra inclinada al final.\n\n- **Per a múltiples directoris**: Si vols agregar publicacions de diversos directoris, especifica la llista a `section_path`. Per exemple:\n\n  ```toml\n  [extra]\n  section_path = [\"blog/\", \"notes/\", \"camí-tres/\"]\n  ```\n\nL'arxiu mostra les publicacions en ordre cronològic invers (les més recents primer). Pots invertir aquest ordre establint `archive_reverse = true` a la secció `[extra]`:\n\n```toml\n[extra]\narchive_reverse = true  # mostra les publicacions més antigues primer\n```\n\n{{ admonition(type=\"note\", title=\"nota\" text=\"La pàgina d'arxiu només llistarà publicacions que tinguin data al seu encapçalament.\") }}\n\n### Etiquetes\n\ntabi té suport integrat per a etiquetes. Per habilitar-les, simplement afegeix la taxonomia al teu `config.toml`:\n\n```toml\ntaxonomies = [{name = \"tags\", feed = true}]\n```\n\nDesprés, pots afegir etiquetes a les teves publicacions afegint-les a l'array `tags` en el bloc de metadades de la teva publicació. Per exemple:\n\n```toml,hl_lines=05-06\ntitle = \"Els molins de vent de la meva vida: reflexions d'un escuder\"\ndate = 1605-01-16\ndescription = \"El meu viatge al costat de Don Quixot, enfrontant-me a gegants imaginats i descobrint les veritables batalles de la vida.\"\n\n[taxonomies]\ntags = [\"personal\", \"reflexions\"]\n```\n\n### Pàgina sobre\n\nSi vols tenir una pàgina que no sigui un article, per exemple per a una secció \"Sobre\", \"Contacte\" o \"Drets d'autor\", pots utilitzar la plantilla `info-page.html`.\n\nPrimer, crea un directori dins de `content/` amb el nom que prefereixis. Per exemple, `content/pages/`. Després, crea un fitxer `_index.md` dins d'aquest directori. El fitxer hauria de ser així:\n\n```markdown\n+++\nrender = false\ninsert_anchor_links = \"left\"\n+++\n```\n\n- `render = false` indica a Zola que no renderitzi la secció.\n- `insert_anchor_links = \"left\"` afegeix enllaços àncora als encapçalaments. Això és opcional.\n\nDins del directori, pots crear qualsevol quantitat de fitxers `.md`.\n\nEn aquesta demo, la pàgina [Sobre mi](/ca/about/) utilitza la plantilla `info-page.html`. El bloc de metadades és el següent:\n\n```toml\ntitle = \"Sobre mi\"\ntemplate = \"info-page.html\"\npath = \"about\"\n```\n\nFixa't com s'estableix `path = \"about\"`. Zola situarà la pàgina a `$base_url/about/`. Si vols que la pàgina estigui disponible a `/contacte/`, hauries d'establir `path = \"contacte\"`.\n\nLa plantilla `info-page.html` també es pot utilitzar per crear landing pages a la ruta arrel (`\"/\"`). Per fer-ho, l'arxiu `content/_index.md` hauria de ser així:\n\n```markdown\n+++\ntitle = \"Títol de la pàgina\"\ntemplate = \"info-page.html\"\n+++\n\nContingut amb Markdown.\n```\n\n---\n\n## SEO\n\ntabi s'encarrega de la majoria de tasques de SEO per a tu (com ara les etiquetes del protocol Open Graph, descripció, paleta de colors...), però hi ha certes configuracions que pots personalitzar.\n\n### Favicon\n\n| Pàgina | Secció  | `config.toml` | Segueix la jerarquia | Requereix JavaScript |\n|:------:|:-------:|:-------------:|:--------------------:|:--------------------:|\n|   ❌   |   ❌    |      ✅       |          ❌          |          ❌           |\n\nEl favicon és la petita imatge que es mostra a la pestanya del navegador. Pots establir-la a `config.toml` amb `favicon = \"img/favicon.png\"`.\n\n### Favicon d'emoji\n\n| Pàgina | Secció  | `config.toml` | Segueix la jerarquia | Requereix JavaScript |\n|:------:|:-------:|:-------------:|:--------------------:|:--------------------:|\n|   ❌   |   ❌    |      ✅       |          ❌          |          ❌           |\n\nTambé pots establir un emoji com a favicon amb `favicon_emoji`. Per exemple, `favicon_emoji = \"👾\"`.\n\nNota: Alguns navegadors no suporten favicons d'emoji. Consulta la taula de compatibilitat a [caniuse](https://caniuse.com/link-icon-svg).\n\n### URL canònica\n\n| Pàgina | Secció  | `config.toml` | Segueix la jerarquia | Requereix JavaScript |\n|:------:|:-------:|:-------------:|:--------------------:|:--------------------:|\n|   ✅   |   ✅    |      ✅       |          ❌          |          ❌           |\n\nL'URL canònica és una manera d'indicar als motors de cerca quina és l'URL preferida per al contingut del teu lloc web. Això és útil per al SEO i per evitar problemes de contingut duplicat.\n\nPer defecte, l'URL canònica és l'URL de la pàgina on et trobes. No obstant això, pots canviar això configurant `canonical_url` al front matter de la teva pàgina o secció.\n\nSi tens un lloc amb una estructura idèntica i contingut coincident, pots configurar `base_canonical_url` al teu `config.toml`. L'URL canònica es crearà substituint el `$base_url` de l'URL actual amb el `$base_canonical_url` que establisquis.\n\nPer exemple, si configures `base_canonical_url = \"https://example.com\"`, l'URL canònica de la pàgina `$base_url/blog/post1` serà `https://example.com/blog/post1`. Això és útil si tens un lloc amb diversos dominis que comparteixen el mateix contingut.\n\n**Nota**: per assegurar-te que l'URL canònica sigui correcta, probablement serà millor configurar `canonical_url` individualment per a cada pàgina.\n\n### Targetes per a xarxes socials\n\n| Pàgina | Secció | `config.toml` | Segueix la jerarquia | Requereix JavaScript |\n|:------:|:------:|:-------------:|:---------------:|:-------------------:|\n|   ✅   |   ✅   |      ✅       |        ✅       |         ❌          |\n\nLes targetes per a xarxes socials són les imatges que es mostren quan comparteixes un enllaç a les xarxes socials:\n\n{{ dimmable_image(src=\"img/with_social_media_card.webp\", alt=\"Una captura de pantalla de WhatsApp mostrant un enllaç amb una targeta per a xarxes socials\") }}\n\nPots establir la imatge per a xarxes socials amb `social_media_card = \"img/social_media_card.png\"`.\n\nPots especificar rutes tant relatives com absolutes.\n\n- **Ruta relativa**: Posiciona la imatge a la mateixa carpeta que la teva entrada de blog i especifica el seu nom. Per exemple, `social_media_card = \"relative_image.png\"`.\n\n- **Ruta absoluta**: Posiciona la imatge en una carpeta específica i especifica la ruta des de l'arrel. Per exemple, `social_media_card = \"/img/absolute_image.png\"`.\n\nSi ambdues rutes, relativa i absoluta, són vàlides, la ruta relativa tindrà prioritat.\n\nJa que segueix la [jerarquia](#jerarquia-de-configuracio), si no està configurat en una pàgina, però sí ho està en una secció, s'utilitzarà la imatge de la secció. Si no està configurat en una pàgina o secció, però sí en `config.toml`, s'utilitzarà la imatge global.\n\n{{ admonition(type=\"tip\", title=\"CONSELL\", text=\"Automatitza la seva creació amb un [script](https://github.com/welpo/osc.garden/blob/main/static/code/social-cards-zola): [Automatitzant les vistes prèvies dels enllaços amb Zola](https://osc.garden/ca/blog/automating-social-media-cards-zola/).\") }}\n\n### Creador del fedivers\n\n| Pàgina | Secció | `config.toml` | Segueix jerarquia | Requereix JavaScript |\n|:------:|:------:|:-------------:|:-----------------:|:--------------------:|\n|   ❌   |   ❌   |      ✅       |         ❌        |          ❌          |\n\nPots mostrar el perfil del fedivers de l'autor en les previsualitzacions d'enllaços de Mastodon configurant la variable `fediverse_creator` al teu `config.toml`. Per exemple, per a @username@example.com, fes servir:\n\n```toml\nfediverse_creator = { handle = \"username\", domain = \"example.com\" }\n```\n\n---\n\n## Navegació\n\n### Barra de navegació\n\n| Pàgina | Secció | `config.toml` | Segueix la jerarquia | Requereix JavaScript |\n|:------:|:------:|:-------------:|:---------------:|:-------------------:|\n|   ❌   |   ❌   |      ✅       |        ❌       |         ❌          |\n\nLa barra de navegació és la franja a la part superior de la pàgina que conté el títol del lloc i el menú de navegació. Pots personalitzar els elements que apareixen configurant `menu` en `config.toml`.\n\nSoporta links relatius per a pàgines internes i links absoluts per a enllaços externs. Per exemple:\n\n```toml\nmenu = [\n    { name = \"blog\", url = \"blog\", trailing_slash = true },\n    { name = \"arxiu\", url = \"archive\", trailing_slash = true },\n    { name = \"etiquetes\", url = \"tags\", trailing_slash = true },\n    { name = \"projectes\", url = \"projects\", trailing_slash = true },\n    { name = \"sobre nosaltres\", url = \"about\", trailing_slash = true },\n    { name = \"github\", url = \"https://github.com/welpo/tabi\", trailing_slash = false },\n]\n```\n\n### Botons de navegació ràpida\n\n| Pàgina | Secció | `config.toml` | Segueix la jerarquia | Requereix JavaScript |\n|:------:|:------:|:-------------:|:---------------:|:-------------------:|\n|   ✅   |   ✅   |      ✅       |        ✅       |         ❌          |\n\nEls botons de navegació ràpida són els botons que apareixen a la part inferior dreta de la pantalla. Hauries de veure'ls en aquesta pàgina, si no estàs en un dispositiu mòbil. Es veuen així:\n\n{{ dual_theme_image(light_src=\"blog/mastering-tabi-settings/img/quick_navigation_buttons_light.webp\", dark_src=\"blog/mastering-tabi-settings/img/quick_navigation_buttons_dark.webp\", alt=\"Botons de navegació ràpida\") }}\n\nPer activar-los, estableix `quick_navigation_buttons = true`.\n\n### Taula de continguts\n\n| Pàgina | Secció | `config.toml` | Segueix la jerarquia | Requereix JavaScript |\n|:------:|:------:|:-------------:|:------------------:|:--------------------:|\n|   ✅   |   ✅   |      ✅       |        ✅          |          ❌          |\n\nActiva l'índex de continguts just sota del títol i metadades de l'article amb `toc = true`.\n\nPer saber més sobre com personalitzar-ho, consulta [la documentació sobre la Taula de continguts](@/blog/toc/index.ca.md).\n\n### Enllaços als articles anterior i següent\n\n| Pàgina | Secció | `config.toml` | Segueix la jerarquia | Requereix JavaScript |\n|:------:|:------:|:-------------:|:--------------------:|:--------------------:|\n|   ✅   |   ✅   |      ✅       |          ✅          |         ❌          |\n\nMostra enllaços als articles anterior i següent a la part inferior dels posts. Es veu així:\n\n{{ dual_theme_image(light_src=\"blog/mastering-tabi-settings/img/show_previous_next_article_links_light.webp\", dark_src=\"blog/mastering-tabi-settings/img/show_previous_next_article_links_dark.webp\", alt=\"Enllaços als articles anterior i següent\", full_width=true) }}\n\nPer activar aquesta funció, estableix `show_previous_next_article_links = true` i assegura't que la secció té un valor `sort_by` (per exemple `sort_by = \"date\"`).\n\nPer defecte, els articles següents estaran al costat esquerre de la pàgina i els articles anteriors al costat dret.\nPer invertir l'ordre (articles següents al costat dret i articles anteriors al costat esquerre), configura `invert_previous_next_article_links = true`.\n\nPer defecte, aquesta secció de navegació tindrà l'amplada completa del lloc (igual que la barra de navegació de la part superior). Per fer-la més estreta, coincidint amb l'amplada de l'article, configura `previous_next_article_links_full_width = false`.\n\nTotes aquestes configuracions segueixen la jerarquia.\n\n### Enllaços de retorn a les notes a peu de pàgina\n\n{{ admonition(type=\"warning\", title=\"ADVERTÈNCIA DE DEPRECACIÓ\", text=\"Zola v0.19.0 i posterior pot fer això de forma nativa. Estableix `bottom_footnotes = true` a la secció `[markdown]` de la teva configuració.\") }}\n\n| Pàgina | Secció | `config.toml` | Segueix la jerarquia | Requereix JavaScript |\n|:------:|:------:|:-------------:|:------------------:|:--------------------:|\n|   ✅   |   ✅   |      ✅       |        ✅          |          ✅          |\n\nEstablir `footnote_backlinks = true` afegirà enllaços de retorn a les notes a peu de pàgina de les teves publicacions, com aquest:\n\n{{ dual_theme_image(light_src=\"blog/mastering-tabi-settings/img/footnote_backlinks_light.webp\", dark_src=\"blog/mastering-tabi-settings/img/footnote_backlinks_dark.webp\", alt=\"Enllaços de retorn a les notes a peu de pàgina\", full_width=true) }}\n\nQuan facis clic en un enllaç de retorn (la fletxa ↩), et portarà de tornada al punt del text on es va fer referència a la nota a peu de pàgina.\n\n---\n\n## Usabilitat\n\n### Botó de copiar en blocs de codi\n\n| Pàgina | Secció | `config.toml` | Segueix la jerarquia | Requereix JavaScript |\n|:------:|:------:|:-------------:|:------------------:|:--------------------:|\n|   ✅   |   ✅   |      ✅       |        ✅          |          ✅          |\n\nEstablir `copy_button = true` afegirà un petit botó de copiar a la part superior dreta dels blocs de codi, com aquest:\n\n{{ dual_theme_image(light_src=\"blog/mastering-tabi-settings/img/copy_button_on_code_blocks_light.webp\", dark_src=\"blog/mastering-tabi-settings/img/copy_button_on_code_blocks_dark.webp\", alt=\"Botó de copiar en blocs de codi\", full_width=true) }}\n\n### Nom del bloc de codi clicable\n\n| Pàgina | Secció | `config.toml` | Segueix la jerarquia | Requereix JavaScript |\n|:------:|:-------:|:-------------:|:---------------:|:-------------------:|\n|   ✅   |   ✅    |      ✅       |        ✅       |         ✅          |\n\nEn establir `code_block_name_links = true` s'habiliten els enllaços clicables als noms dels blocs de codi. Consulta la [documentació](@/blog/shortcodes/index.ca.md#mostrar-ruta-o-url) per veure exemples i ús.\n\n### Forçar blocs de codi d'esquerra a dreta\n\n| Pàgina | Secció | `config.toml` | Segueix la jerarquia | Requereix JavaScript |\n|:------:|:------:|:-------------:|:-----------------:|:--------------------:|\n|   ✅   |   ✅   |      ✅       |         ✅        |          ❌          |\n\nPer defecte, els blocs de codi es renderitzen d'esquerra a dreta, independentment de la direcció general del text. Estableix `force_codeblock_ltr = false` per permetre que els blocs de codi segueixin la direcció del document. Útil per a idiomes de dreta a esquerra que necessiten blocs de codi de dreta a esquerra.\n\n### Suport per a KaTeX\n\n| Pàgina | Secció | `config.toml` | Segueix la jerarquia | Requereix JavaScript |\n|:------:|:------:|:-------------:|:------------------:|:--------------------:|\n|   ✅   |   ✅   |      ✅       |        ✅          |          ✅          |\n\nKaTeX és una biblioteca JavaScript ràpida i fàcil d'usar per a la representació de matemàtiques TeX a la web. Pots habilitar-ho amb `katex = true`. Mira com es veu en tabi [aquí](/ca/blog/markdown/#katex).\n\n### Suport per a Mermaid\n\n| Pàgina | Secció | `config.toml` | Segueix la jerarquia | Requereix JavaScript |\n|:------:|:------:|:-------------:|:-----------------:|:--------------------:|\n|   ✅   |   ✅   |      ✅       |         ✅        |          ✅          |\n\n[Mermaid](https://github.com/mermaid-js/mermaid) és una eina de diagramació i gràfics basada en JavaScript. Pots activar-la amb `mermaid = true`.\n\nPer defecte, la biblioteca Mermaid es serveix localment. Si prefereixes utilitzar un CDN, estableix `serve_local_mermaid = false` a `config.toml`. L'ús d'un CDN servirà la versió més recent de Mermaid; l'opció local servirà la versió inclosa amb tabi.\n\nConsulta la [documentació de Mermaid](@/blog/shortcodes/index.ca.md#diagrames-de-mermaid) per a instruccions d'ús i exemples.\n\n### Subconjunt de tipus de lletra personalitzat\n\n| Pàgina | Secció | `config.toml` | Segueix la jerarquia | Requereix JavaScript |\n|:------:|:------:|:-------------:|:------------------:|:--------------------:|\n|   ❌   |   ❌   |      ✅       |        ❌          |          ❌          |\n\nLes tipus de lletra personalitzades causen parpalleig del text en Firefox. Per resoldre això, tabi carrega un subconjunt de glifs per a la capçalera. Donat que això (lleugerament) augmenta el temps de càrrega inicial, és una bona idea intentar minimitzar la mida d'aquest subconjunt, o desactivar-lo completament si no estàs fent servir un tipus de lletra personalitzat al teu tema.\n\nPots crear un subconjunt personalitzat adaptat al teu lloc, guardar-lo com a `static/custom_subset.css`, i fer que es carregui amb `custom_subset = true`.\n\nPer desactivar el subconjunt, utilitza `enable_subset = false`.\n\nPer obtenir més informació, incloent instruccions sobre com crear un subconjunt personalitzat, consulta la [documentació](@/blog/custom-font-subset/index.ca.md).\n\n### Contingut complet al feed\n\n| Pàgina | Secció | `config.toml` | Segueix la jerarquia | Requereix JavaScript |\n|:------:|:-------:|:-------------:|:-------------------:|:-------------------:|\n|   ❌   |   ❌   |      ✅       |         ❌          |         ❌          |\n\nPer defecte, el feed Atom només conté el resum o descripció de les teves publicacions. Pots incloure el contingut complet de les publicacions establint `full_content_in_feed = true` a `config.toml`.\n\n### Amagar contingut del feed\n\n| Pàgina | Secció | `config.toml` | Segueix la jerarquia | Requereix JavaScript |\n|:----:|:-------:|:-------------:|:-----------------:|:-------------------:|\n|  ✅  |   ✅    |      ✅       |         ✅        |         ❌          |\n\nPots controlar com apareix el contingut als feeds utilitzant dues configuracions:\n\n- `hide_from_feed = true`: Amaga el contingut de tots els feeds (feed principal, feeds de secció i feeds d'etiquetes)\n- `hide_from_main_feed = true`: Amaga el contingut només del feed principal mentre el manté visible als feeds de secció i d'etiquetes\n\n### Comentaris {#afegir-comentaris}\n\n| Pàgina | Secció | `config.toml` | Segueix la jerarquia | Requereix JavaScript |\n|:------:|:-------:|:-------------:|:-------------------:|:-------------------:|\n|   ✅   |   ❌   |      ✅       |         ❌          |         ✅          |\n\nPer activar els comentaris en una pàgina, establert el nom del sistema com a `true` al front matter. Per exemple, `utterances = true`.\n\nSi vols activar els comentaris de forma global, pots fer-ho establint `enabled_for_all_posts = true` a la secció apropiada del teu `config.toml` (per exemple, a `[extra.giscus]`).\n\nSi has activat un sistema de forma global i vols desactivar-lo per a una pàgina específica, pots fer-ho establint el nom del sistema com a `false` al front matter. Per exemple, `utterances = false`.\n\nLlegeix la [documentació](@/blog/comments/index.ca.md) per a més informació sobre els sistemes disponibles i la seva configuració.\n\n### Botons d'iine {#iine}\n\n| Pàgina | Secció | `config.toml` | Segueix la jerarquia | Requereix JavaScript |\n|:------:|:------:|:-------------:|:--------------------:|:--------------------:|\n|   ✅   |   ✅   |      ✅       |          ✅          |         ❌          |\n\ntabi permet botons d'[iine](https://iine.to/) per mostrar apreciació anònima pel teu contingut. Aquests botons centrats en la privadesa funcionen sense JavaScript i no rastegen usuaris.\n\nPer activar els botons iine globalment:\n\n```toml\n[extra]\niine = true\n```\n\nPots personalitzar la icona usada als botons (segueix la jerarquia):\n\n```toml\n[extra]\niine_icon = \"thumbs_up\"  # Opcions: \"heart\", \"thumbs_up\", \"upvote\", o qualsevol emoji\n```\n\nPer a llocs multilingües, pots unificar els recomptes de likes entre versions en diferents idiomes del mateix contingut (configuració només de config; valor per defecte és `true`):\n\n```toml\n[extra]\niine_unified_languages = true  # Els likes a /ca/blog/hello/ compten cap a /blog/hello/\n```\n\n### Anàlisi web\n\n| Pàgina | Secció  | `config.toml` | Segueix la jerarquia | Requereix JavaScript |\n|:------:|:-------:|:-------------:|:-----------------:|:--------------------:|\n|   ❌   |    ❌   |       ✅       |         ❌         |          ✅          |\n\ntabi ofereix suport per a 3 sistemes d'anàlisi web que respecten la privacitat: [Plausible](https://plausible.io/), [GoatCounter](https://www.goatcounter.com/) i [Umami](https://umami.is/).\n\nPots configurar-los en la secció `[extra.analytics]` del teu arxiu `config.toml`.\n\n- `service`: el servei a utilitzar. Les opcions disponibles són `\"goatcounter\"`, `\"umami\", i \"plausible\"`.\n\n- `id`: l'identificador únic per al teu servei d'anàlisi. Això varia segons el servei:\n  - Per a GoatCounter, és el codi triat durant el registre. Instàncies auto-allotjades de GoatCounter no requereixen aquest camp.\n  - Per a Umami, és l'ID del lloc web.\n  - Per a Plausible, pot ser:\n    - **Format nou** (Plausible v3.1.0+): El nom d'script aleatori sense l'extensió (ex. `\"pa-XXXXXX\"`). Troba'l al teu panell de Plausible a Configuració → Detalls del lloc web → Nom de l'script.\n    - **Format heretat**: El teu nom de domini (ex. `\"example.com\"`). Útil si necessites enviar estadístiques a múltiples panells simultàniament; el nou format no admet aquesta funcionalitat. Consulta la [guia d'actualització d'scripts de Plausible](https://plausible.io/docs/script-update-guide) per a més detalls.\n\n- `self_hosted_url`: Opcional. Utilitza aquest camp per especificar l'URL si tens una instància auto-allotjada. L'URL base variarà segons la teva configuració particular. Alguns exemples:\n  - Per a GoatCounter: `\"https://stats.example.com\"`\n  - Per a Umami: `\"https://umami.example.com\"`\n  - Per a Plausible: `\"https://plausible.example.com\"`\n\n- `do_not_track`: (només per a Umami) opcional. Quan s'estableix com a `true`, es desactiva el seguiment per als usuaris els navegadors dels quals envien una capçalera \"Do Not Track\".\n\nUn exemple de configuració per a GoatCounter no auto-allotjada seria:\n\n```toml\n[extra.analytics]\nservice = \"goatcounter\"\nid = \"tabi\"\nself_hosted_url = \"\"\n```\n\n## Icones al peu de pàgina\n\n### Icones de xarxes socials\n\n| Pàgina | Secció | `config.toml` | Respecta jerarquia | Requereix JavaScript |\n|:------:|:-------:|:-------------:|:------------------:|:-------------------:|\n|   ❌   |   ❌   |      ✅       |         ❌         |         ❌          |\n\nPots afegir icones de xarxes socials al peu de pàgina amb `socials`, que accepta una llista d'objectes de xarxes socials. Per exemple:\n\n```toml\nsocials = [\n    { name = \"github\", url = \"https://github.com/welpo/\", icon = \"github\" },\n    { name = \"soundcloud\", url = \"https://soundcloud.com/oskerwyld\", icon = \"soundcloud\" },\n    { name = \"instagram\", url = \"https://instagram.com/oskerwyld\", icon = \"instagram\" },\n    { name = \"youtube\", url = \"https://youtube.com/@oskerwyld\", icon = \"youtube\" },\n    { name = \"spotify\", url = \"https://open.spotify.com/artist/5Hv2bYBhMp1lUHFri06xkE\", icon = \"spotify\" },\n]\n```\n\nPer veure una llista de totes les icones integrades, fes un cop d'ull al directori [`static/social_icons` a GitHub](https://github.com/welpo/tabi/tree/main/static/social_icons).\n\nTrobes a faltar algun icona? Si creus que seria una bona addició a tabi, no dubtis en [obrir un issue](https://github.com/welpo/tabi/issues/new?assignees=&labels=enhancement&projects=&template=feature_request.md&title=) o enviar una pull request ([exemple](https://github.com/welpo/tabi/pull/333)).\n\nPer utilitzar una icona personalitzada, pots afegir-la al directori `static/social_icons` del teu lloc web. Per exemple, si afegeixes `custom.svg`, la pots referenciar així:\n\n```\n{ name = \"custom\", url = \"https://example.com\", icon = \"custom\" }\n```\n\n{{ admonition(type=\"note\", title=\"NOTA\", text=\"Tots els enllaços a xarxes socials inclouen l'[atribut](https://developer.mozilla.org/docs/Web/HTML/Attributes/rel/me) `rel='me'`. Això ajuda els motors de cerca i serveis web a verificar que les comptes de xarxes socials et pertanyen.\") }}\n\n### Icona de feed\n\n| Pàgina | Secció  | `config.toml` | Segueix la jerarquia | Requereix JavaScript |\n|:------:|:-------:|:-------------:|:---------------------:|:-------------------:|\n|   ❌   |   ❌    |      ✅       |          ❌           |         ❌          |\n\nPots afegir un enllaç al teu feed RSS/Atom al peu de pàgina amb `feed_icon = true`.\n\nPer utilitzar una icona personalitzada, estableix `feed_icon` amb el nom de la icona (per exemple, `feed_icon = \"square-rss\"`). La icona ha d'existir a `static/social_icons/` (sense l'extensió `.svg`).\n\nNota pels usuaris de Zola 0.19.X: quan hi ha dos noms de fitxer a `feed_filenames`, només s'enllaçarà el primer al peu de pàgina.\n\n#### Menú de peu de pàgina\n\n| Pàgina | Secció  | `config.toml` | Segueix la jerarquia | Requereix JavaScript |\n|:------:|:-------:|:-------------:|:-----------------:|:--------------------:|\n|   ❌   |   ❌    |      ✅       |        ❌         |          ❌          |\n\nPots afegir un menú al peu de pàgina amb `footer_menu`, que accepta una llista d'elements de menú. Per exemple:\n\n```toml\nfooter_menu = [\n    {url = \"about\", name = \"about\", trailing_slash = true},\n    {url = \"privacy\", name = \"privacy\", trailing_slash = true},\n    {url = \"sitemap.xml\", name = \"sitemap\", trailing_slash = false},\n    {url = \"https://example.com\", name = \"external link\", trailing_slash = true},\n]\n```\n\n### Copyright\n\n| Pàgina | Secció  | `config.toml` | Segueix la jerarquia | Requereix JavaScript |\n|:------:|:-------:|:-------------:|:---------------------:|:-------------------:|\n|   ❌   |   ❌    |      ✅       |          ❌           |         ❌          |\n\nPer afegir una menció sobre els drets d'autor al teu lloc web, configura `copyright`:\n\n```toml\ncopyright = \"© $CURRENT_YEAR Your Name $SEPARATOR Unless otherwise noted, the content in this website is available under the [CC BY-SA 4.0](https://creativecommons.org/licenses/by-sa/4.0/) license.\"\n```\n\n- `$TITLE` serà reemplaçat per la variable `title` configurada a `config.toml`\n- `$CURRENT_YEAR` serà reemplaçat per l'any actual\n- `$AUTHOR` serà reemplaçat per la variable `author`\n- `$SEPARATOR` serà reemplaçat per la [variable `separator`](#separador-personalitzat).\n\nEl text es processarà en Markdown. Per exemple, la configuració d'adalt:\n\n{{ dual_theme_image(light_src=\"blog/mastering-tabi-settings/img/copyright_light.webp\", dark_src=\"blog/mastering-tabi-settings/img/copyright_dark.webp\" alt=\"Secció de drets d'autor\", full_width=true) }}\n\nSi tens un lloc multilingüe i vols establir diferents notificacions de drets d'autor per a diferents idiomes, afegeix la traducció corresponent a `copyright_translations.{codi_d'idioma}` per a cada idioma que vulguis donar suport. El codi de llengua ha de coincidir amb el [codi de llengua de tabi](https://welpo.github.io/tabi/ca/blog/faq-languages/#que-son-aquests-codis-de-dues-lletres). Per exemple, pel castellà:\n\n```toml\ncopyright_translations.es = \"© $CURRENT_YEAR $AUTHOR $SEPARATOR A menos que se indique lo contrario, el contenido de esta web está disponible bajo la licencia [CC BY-SA 4.0](https://creativecommons.org/licenses/by-sa/4.0/).\"\n```\n\n---\n\n## Metadades\n\n### Mostrar autoria\n\n| Pàgina | Secció  | `config.toml` | Segueix la jerarquia | Requereix JavaScript |\n|:------:|:-------:|:-------------:|:---------------------:|:-------------------:|\n|   ✅   |   ✅    |      ✅       |          ✅           |         ❌          |\n\nPer mostrar l'autoria d'un article, estableix `show_author = true`.\n\nAixò mostrarà els autors establerts a la variable `authors = []` al front matter del post. Si aquest camp no està configurat, mostrarà l'autor de `config.toml` (`author = \"\"`).\n\n### Temps de lectura\n\n| Pàgina | Secció  | `config.toml` | Segueix la jerarquia | Requereix JavaScript |\n|:------:|:-------:|:-------------:|:---------------------:|:-------------------:|\n|   ✅   |   ✅    |      ✅       |          ✅           |         ❌          |\n\nPots activar o desactivar el temps estimat de lectura d'un article amb `show_reading_time`. Si el configures com a `true`, apareixerà a les metadades de l'article, com això:\n\n{{ dual_theme_image(light_src=\"blog/mastering-tabi-settings/img/see_changes_light.webp\", dark_src=\"blog/mastering-tabi-settings/img/see_changes_dark.webp\" alt=\"Títol de l'article i metadades, mostrant un enllaç «Veure canvis»\") }}\n\nCom que segueix [la jerarquia](#jerarquia-de-configuracio), pots activar-lo o desactivar-lo per a pàgines o seccions específiques. Per exemple, aquesta demo desactiva `show_reading_time = false` a la secció [projectes](https://welpo.github.io/tabi/ca/projects/) a l'arxiu [`_index.md`](https://github.com/welpo/tabi/blob/main/content/projects/_index.es.md?plain=1), de manera que les seves publicacions individuals no mostren el temps de lectura.\n\n### Mostrar la data\n\n| Pàgina | Secció  | `config.toml` | Segueix la jerarquia | Requereix JavaScript |\n|:------:|:------:|:-------------:|:--------------------:|:-------------------:|\n|   ✅   |   ✅   |      ✅       |         ✅           |         ❌          |\n\nPer defecte, la data es mostra sota el títol de la publicació. Pots amagar-la amb `show_date = false`. Aquest ajust segueix [la jerarquia](#jerarquia-de-configuracio).\n\n### Format de data\n\n| Pàgina | Secció  | `config.toml` | Segueix la jerarquia | Requereix JavaScript |\n|:------:|:-------:|:-------------:|:---------------------:|:-------------------:|\n|   ❌   |   ❌    |      ✅       |          ❌           |         ❌          |\n\ntabi té tres formats de data: `long_date_format`, `short_date_format` i `archive_data_format`. El format curt s'utilitza a les metadades d'una publicació, mentre que el format llarg s'utilitza al llistar les publicacions (és a dir, a la [secció de blog](/ca/blog/) o a la [pàgina principal](/ca/)). El format d'arxiu s'utilitza per mostrar el dia i el mes a la pàgina d'arxiu.\n\nPer defecte és \"6th July 2049\" per als formats curt i llarg en anglès. Per a altres idiomes, el predeterminat és `\"%d %B %Y\"` per al format llarg i `\"%-d %b %Y\"` per al format curt. El format d'arxiu predeterminat universal és `\"%d %b\"`.\n\nA Zola, la sintaxi per al format de temps està inspirada en strftime. Una referència completa està disponible a la [documentació de chrono](https://docs.rs/chrono/0.4.31/chrono/format/strftime/index.html).\n\n#### Formats de data per idioma\n\nPots personalitzar els formats de data per idiomes específics utilitzant la matriu `date_formats` a `config.toml`:\n\n```toml\ndate_formats = [\n    { lang = \"es\", long = \"%d de %B de %Y\", short = \"%-d %b %Y\", archive = \"%d de %b\" },\n    { lang = \"de\", long = \"%d. %B %Y\", short = \"%d.%m.%Y\", archive = \"%d. %b\" },\n]\n```\n\nAixò permet que diferents idiomes utilitzin formats de data culturalment apropiats (per exemple, \"6. Juli 2049\" per a alemany VS \"6 de julio de 2049\" per a espanyol).\n\n### Separador personalitzat\n\n| Pàgina | Secció  | `config.toml` | Segueix la jerarquia | Requereix JavaScript |\n|:------:|:-------:|:-------------:|:---------------------:|:-------------------:|\n|   ❌   |   ❌    |      ✅       |          ❌           |         ❌          |\n\nEl separador apareix en diversos llocs: al títol del navegador, entre les metadades d'una publicació...\n\nEl separador per defecte és un punt de llista (`•`), però pots canviar-lo configurant alguna cosa com `separator = \"|\"`.\n\n### Ordre del títol\n\n| Pàgina | Secció  | `config.toml` | Segueix la jerarquia | Requereix JavaScript |\n|:------:|:-------:|:-------------:|:---------------------:|:-------------------:|\n|   ❌   |   ❌    |      ✅       |          ❌           |         ❌          |\n\nPer defecte, el títol a la pestanya del navegador és el nom del lloc seguit del títol de la pàgina. Per exemple, el títol de la secció del blog és «~/tabi • Blog».\n\nConfigurant `invert_title_order = true`, pots invertir l'ordre del títol del lloc i el títol de la pàgina a la pestanya del navegador. Per exemple, l'etiqueta del títol de la secció del blog es convertiria en «Blog • ~/tabi».\n\n---\n\nCertainly, here is a high-quality, non-literal translation of the provided text into Catalan. I've adhered to your specifications, keeping the variables and English terms unchanged.\n\n## Seguretat\n\n### Correu electrònic codificat\n\n| Pàgina | Secció | `config.toml` | Segueix la jerarquia | Requereix JavaScript |\n|:------:|:------:|:-------------:|:--------------------:|:--------------------:|\n|   ❌   |   ❌   |      ✅       |          ❌          |          ✅          |\n\nPer tal de protegir la teva adreça de correu electrònic contra els spambots, pots codificar-la al peu de pàgina. Pots fer això establint `email` a una versió codificada en base64 de la teva adreça de correu electrònic[^2]. Per exemple, `email = \"bWFpbEBleGFtcGxlLmNvbQ==\"` és la versió codificada en base64 de \"mail@example.com\".\n\nSi no vols codificar el teu correu electrònic tu mateix, tabi pot fer-ho per tu si configures `encode_plaintext_email = true`. Això et permet establir un correu electrònic en text pla en la configuració. Tingues en compte que això només protegeix la teva adreça de correu electrònic al teu lloc web, no en repositoris públics.\n\nSi el correu electrònic està codificat (ja sigui per tu o per tabi), els usuaris amb JavaScript desactivat no veuran la icona de correu electrònic.\n\n### CSP (Content Security Policy)\n\n| Pàgina | Secció | `config.toml` | Segueix la jerarquia | Requereix JavaScript |\n|:------:|:------:|:-------------:|:--------------------:|:--------------------:|\n|   ❌   |   ❌   |      ✅       |          ❌          |          ❌          |\n\nLa Content Security Policy (CSP) és una capa addicional de seguretat que ajuda a detectar i mitigar certs tipus d'atacs, inclosos atacs de Cross Site Scripting (XSS) i injecció de dades. Aquests atacs s'utilitzen per a tot, des del robatori de dades fins a la desfiguració de llocs web i la distribució de programari maliciós.\n\ntabi té una CSP predeterminada que permet imatges i vídeos remots, així com incrustacions de YouTube i Vimeo. Pots personalitzar-la amb `allowed_domains`, que accepta una llista de directrius de CSP. Aquesta és la CSP predeterminada:\n\n```toml\nallowed_domains = [\n    { directive = \"font-src\", domains = [\"'self'\", \"data:\"] },\n    { directive = \"img-src\", domains = [\"'self'\", \"https://*\", \"data:\"] },\n    { directive = \"script-src\", domains = [\"'self'\"] },\n    { directive = \"style-src\", domains = [\"'self'\"] },\n    { directive = \"frame-src\", domains = [\"player.vimeo.com\", \"https://www.youtube-nocookie.com\"] },\n]\n```\n\nAquesta opció està habilitada per defecte. Per desactivar-la per una pàgina, secció o globalment, estableix `enable_csp = false`. La configuració de `enable_csp` segueix la jerarquia.\n\nPer a més informació, consulta la [pàgina de documentació de CSP](@/blog/security/index.ca.md).\n\n---\n\n## Indieweb\n\n### Webmentions\n\n| Pàgina | Secció | `config.toml` | Segueix jerarquia | Requereix JavaScript |\n|:------:|:------:|:-------------:|:-----------------:|:--------------------:|\n|   ❓   |   ❓   |      ✅       |         ❓        |         ✅           |\n\nCom es descriu en l'estàndard W3C recomanat, [Webmention](https://www.w3.org/TR/webmention/#abstract-p-1) és una manera senzilla de notificar qualsevol URL quan la menciones al teu lloc web. Des de la perspectiva del receptor, és una manera de sol·licitar notificacions quan altres llocs web la mencionen.\n\nPer a llocs web estàtics, [webmention.io](https://webmention.io/) allotja un punt final de webmention que es pot utilitzar per rebre webmentions. Aquesta funcionalitat recupera les webmentions emmagatzemades a webmention.io i les mostra per a una pàgina. Hauràs de configurar un compte per al teu lloc web a webmention.io. Quan habilitis la funcionalitat de webmention, anunciarà el teu punt final de webmention.io i mostrarà les webmentions per a qualsevol pàgina.\n\nHabilita les webmentions per al teu lloc web afegint el següent al teu fitxer `config.toml`.\n\n```toml\n[extra.webmentions]\nenable = true\n# Especifica el domini registrat amb webmention.io.\ndomain = \"www.example.com\"\n```\n\n❓: Per desactivar les webmentions per a una secció o pàgina específica, estableix `webmentions = false` a la secció `[extra]` del front matter d'aquesta secció o pàgina.\n\nLa secció de webmentions es veu així:\n\n{{ dual_theme_image(light_src=\"blog/mastering-tabi-settings/img/webmention_light.webp\", dark_src=\"blog/mastering-tabi-settings/img/webmention_dark.webp\" alt=\"Captura de pantalla de webmentions mostrant republications, m'agrada, marcadors i comentaris\", full_width=true) }}\n\n### h-card representativa\n\n| Pàgina | Secció | `config.toml` | Segueix la jerarquia | Requereix JavaScript |\n| :--: | :-----: | :-----------: | :---------------: | :-----------------: |\n|  ❌  |   ❌    |      ✅       |        ❌         |         ❌          |\n\nPer defecte, tabi afegeix una h-card representativa [h-card](https://microformats.org/wiki/h-card) **oculta** a la pàgina d'inici. Tot i que és invisible per als visitants, està disponible per als analitzadors de microformats. Pots comprovar la validesa de la targeta amb l'eina [Indiewebify.me](https://indiewebify.me/validate-h-card/).\n\nPer desactivar l'h-card, estableix `enable = false` a la secció `[extra.hcard]` de `config.toml`.\n\nL'h-card predeterminada inclou el teu nom, l'URL del lloc web i els enllaços a les xarxes socials.\n\nPots establir una imatge de perfil i una petita biografia amb els paràmetres `avatar` i `biography`.\n\nTotes les altres [propietats h-card](https://microformats.org/wiki/h-card#Properties) es poden afegir llistant-les a la secció `[extra.hcard]` del fitxer de configuració. Simplement substitueix tots els caràcters `-` per `_`.\n\n---\n\n## Estenent elements HTML a tabi\n\nAlguns elements HTML a tabi es poden estendre per admetre casos d'ús addicionals, com ara afegir JavaScript personalitzat per a comportaments a tot el lloc al final de l'etiqueta `<body>` o incloure contingut addicional al final de l'element `<head>` que no estigui suportat per altres configuracions de tabi.\n\nConsulta la taula a continuació per veure els elements que es poden estendre:\n\n| Element | Plantilla                         |\n| :------: | :-------------------------------: |\n| `<head>` | `templates/tabi/extend_head.html` |\n| `<body>` | `templates/tabi/extend_body.html` |\n\nNo hi ha configuracions explícites que hagis d'establir per al teu lloc o pàgines. Simplement crea el fitxer de plantilla corresponent per al teu lloc, i tabi l'inclourà automàticament.\n\n---\n\n[^1]: Si estàs utilitzant un repositori Git remot, potser voldràs automatitzar el procés d'actualització del camp `updated`. Aquí tens una guia per a això: [Zola Git Hook: actualitzant les dates de les publicacions](https://osc.garden/ca/blog/zola-date-git-hook/).\n\n[^2]: Per a codificar el teu correu electrònic en base64 pots utilitzar [eines en línia](https://www.base64encode.org/) o, al teu terminal, executar: `printf 'mail@example.com' | base64`\n"
  },
  {
    "path": "content/blog/mastering-tabi-settings/index.es.md",
    "content": "+++\ntitle = \"Domina la configuración de tabi: guía completa\"\ndate = 2023-09-18\nupdated = 2026-01-31\ndescription = \"Descubre las múltiples maneras en que puedes personalizar tabi.\"\n\n[taxonomies]\ntags = [\"funcionalidad\", \"tutorial\", \"preguntas frecuentes\"]\n\n[extra]\npinned = true\nquick_navigation_buttons = true\nsocial_media_card = \"social_cards/es_blog_mastering_tabi_settings.jpg\"\n+++\n\nEsta es la guía completa sobre la configuración en tabi. Si tienes alguna pregunta, puedes [abrir un issue en GitHub](https://github.com/welpo/tabi/issues/new) o [inciar una discusión](https://github.com/welpo/tabi/discussions).\n\n<details>\n    <summary><b>Tabla de contenido</b></summary>\n    <!-- toc -->\n</details>\n\n## Jerarquía de configuración\n\ntabi tiene una jerarquía que te permite personalizar tu sitio en diferentes niveles. La jerarquía (de menor a mayor prioridad) es la siguiente:\n\n1. **Configuraciones globales**: Estas son las configuraciones que se aplican a todo tu sitio. Se establecen en `config.toml`.\n2. **Configuraciones de sección**: Estas son las configuraciones que se aplican a una sección de tu sitio (por ejemplo, `/blog` o `/projects`). Se establecen en la metainformación del archivo `_index.md` de la sección.\n3. **Configuración de la página «padre»**: Para páginas anidadas (páginas dentro de páginas), estas son las configuraciones de la página que las contiene.\n4. **Configuraciones de página**: Estas son las configuraciones que se aplican a una sola página. Se establecen en la metainformación de la página.\n\nEn todos los casos, las opciones de tabi se establecen en la sección `[extra]`.\n\nPara las configuraciones que siguen esta jerarquía, el valor establecido en una página reemplaza el valor de una sección, que a su vez reemplaza el valor global. En resumen: cuanto más específica sea la configuración, mayor prioridad tendrá, o `página > página padre/sección > config.toml`.\n\n---\n\n## Búsqueda\n\n| Página | Sección | `config.toml` | Sigue la jerarquía | Requiere JavaScript |\n|:------:|:-------:|:-------------:|:---------------:|:-------------------:|\n|   ❌   |   ❌    |       ✅       |        ❌       |          ✅          |\n\ntabi soporta búsqueda local accesible y multilingüe con [Elasticlunr](http://elasticlunr.com/). Para activarla, necesitas:\n\n1. Establecer un `default_language` en `config.toml`.\n2. Establecer `build_search_index = true`.\n3. Opcionalmente, configurar la sección `[search]`.\n\nPor ejemplo:\n\n```toml\nbase_url = \"https://example.com\"\ndefault_language = \"en\"\nbuild_search_index = true\n\n[search]\nindex_format = \"elasticlunr_json\" # O el menos eficiente \"elasticlunr_javascript\".\ninclude_title = true\ninclude_description = true\ninclude_path = true\ninclude_content = true\n```\n\n**Nota**: para soporte de búsqueda en Chino/Japonés, necesitas usar una [build personalizada de Zola](https://github.com/getzola/zola/blob/master/Cargo.toml#L54-L55).\n\n### Consideraciones para usuarios de Zola 0.17.X\n\nZola 0.17.X no proporciona acceso a la variable `search.index_format` ([reporte del bug](https://github.com/getzola/zola/issues/2165)). Al usar tabi, se asume el uso del índice JSON, que es más eficiente. Sin embargo, debido a [otro bug](https://github.com/getzola/zola/issues/2193) solucionado en 0.18.0, el índice JSON para sitios multilingües no se genera correctamente.\n\nLos usuarios con versiones de Zola anteriores a 0.18.0 que quieran usar el índice JavaScript necesitan establecer la variable `index_format` en dos lugares:\n\n```toml\n[search]\nindex_format = \"elasticlunr_javascript\"\n\n[extra]\nindex_format = \"elasticlunr_javascript\"\n```\n\nEsto asegura que tabi cargue los archivos correctos. Recomendamos actualizar a Zola 0.18.0 o posterior para una funcionalidad óptima.\n\n### Detalles de implementación\n\nPara detalles técnicos sobre la implementación de la búsqueda en tabi, incluyendo cuándo se carga el índice, características de accesibilidad y otros detalles, consulta el [Pull Request #250](https://github.com/welpo/tabi/pull/250).\n\n---\n\n## Soporte multilingüe\n\ntabi ofrece soporte multilingüe completo para tu sitio Zola, desde configurar un idioma predeterminado hasta añadir todos los que desees. Consulta la [preguntas frecuentes sobre idiomas](@/blog/faq-languages/index.es.md) para más información.\n\n---\n\n## Apariencia\n\n### Página principal\n\nLa [página principal](/) de esta demo tiene un encabezado con una imagen, un título y una descripción:\n\n{{ dual_theme_image(light_src=\"blog/mastering-tabi-settings/img/header_light.webp\", dark_src=\"blog/mastering-tabi-settings/img/header_dark.webp\", alt=\"Encabezado de la página principal\") }}\n\n#### Cabecera\n\nPara configurar la imagen y el título, puedes usar la variable `header` en el front matter del archivo `_index.md` de la sección. Por ejemplo:\n\n```toml\n[extra]\nheader = {title = \"¡Hola! Soy tabi~\", img = \"blog/mastering-tabi-settings/img/main.webp\", img_alt = \"Óscar Fernández, el autor del tema\" }\n```\n\nLa descripción es contenido Markdown normal, escrito fuera del front matter.\n\n#### Listando publicaciones recientes\n\nPara mostrar publicaciones en la página principal, primero debes decidir de dónde se servirán: de la ruta raíz (`/`) o de un subdirectorio (por ejemplo, `/blog`).\n\n**Opción A: Servir publicaciones desde la ruta raíz (`/`)**\n\nConfigura `paginate_by` en el front matter de tu archivo `content/_index.md`:\n\n```toml\ntitle = \"Últimas publicaciones\"\nsort_by = \"date\"\npaginate_by = 5  # Muestra 5 publicaciones por página.\n\n[extra]\nheader = {title = \"¡Hola! Soy tabi~\", img = \"img/main.webp\", img_alt = \"Tu nombre\" }\n```\n\n{{ admonition(type=\"note\", text=\"La configuración `paginate_by` va en el front matter principal, no en la sección `[extra]`.\") }}\n\n**Opción B: Servir publicaciones desde un subdirectorio (por ejemplo, `/blog`)**\n\nUtiliza `section_path` en la sección `[extra]` de tu archivo `content/_index.md`:\n\n```toml\ntitle = \"Últimas publicaciones\"\nsort_by = \"date\"\n# No configures `paginate_by` aquí.\n\n[extra]\nheader = {title = \"¡Hola! Soy tabi~\", img = \"img/main.webp\", img_alt = \"Tu nombre\" }\nsection_path = \"blog/_index.md\"  # Dónde encontrar tus publicaciones.\nmax_posts = 5  # Muestra hasta 5 publicaciones en la página principal.\n```\n\n{{ admonition(type=\"warning\", title=\"ALERTA\", text=\"No configures `paginate_by` y `section_path` a la vez. Estas configuraciones son mutuamente excluyentes y usarlas juntas puede resultar en que no se muestren publicaciones.\") }}\n\nNotas adicionales:\n\n- El `title` en el front matter establece el título que aparece sobre las publicaciones.\n- Usa la ruta completa al archivo `_index.md` de la sección para `section_path`. Usar `section_path = \"blog/\"` no funcionará.\n\n##### Fijar publicaciones\n\nPuedes fijar publicaciones para mantenerlas en la parte superior de la página principal. En esta demo, esta publicación está fijada, por lo que aparece primera con un icono y etiqueta de \"fijada\":\n\n{{ dual_theme_image(light_src=\"blog/mastering-tabi-settings/img/pinned_post_light.webp\", dark_src=\"blog/mastering-tabi-settings/img/pinned_post_dark.webp\", alt=\"Entrada fijada\", full_width=true) }}\n\nLas publicaciones fijadas se muestran primero, manteniendo su orden relativo según el `sort_by` de la sección, seguidas por el resto de las publicaciones.\n\nPara fijar una publicación, añade lo siguiente a su front matter:\n\n```toml\n[extra]\npinned = true\n```\n\n{{ admonition(type=\"info\", text=\"Este ajuste solo afecta a las páginas principales del sitio (como `/`, `/es/`, `/fr/`). Otras secciones como `blog/`, `tags/` o `archive/` muestran las publicaciones en su orden habitual.\") }}\n\n{{ admonition(type=\"warning\", text='Cuando se utiliza la paginación (`paginate_by`), las publicaciones destacadas pueden aparecer dos veces: una vez en la parte superior de la primera página, y otra en su posición cronológica normal en páginas posteriores.') }}\n\n##### Mostrar la fecha de los artículos en el listado\n\nPor defecto, cuando se listan los artículos, se muestra la fecha de creación. Puedes configurar qué fecha(s) mostrar usando la opción `post_listing_date`. Configuraciones disponibles:\n\n- `date`: Muestra solo la fecha de publicación original del artículo (opción por defecto).\n- `updated`: Muestra solo la fecha de la última actualización del artículo. Si no hay fecha de actualización, muestra la fecha de publicación original.\n- `both`: Muestra tanto la fecha de publicación original como la fecha de la última actualización.\n\n{% admonition(type=\"tip\") %}\nEsta configuración sigue la jerarquía: puedes establecer un valor global en `config.toml` o configurarlo para secciones específicas en su archivo `_index.md`. En ambos casos, añádelo a la sección `[extra]`.\n{% end %}\n\n#### Listado de proyectos\n\nPuedes mostrar una selección de proyectos en tu página principal. Para hacer esto, primero necesitarás configurar el directorio `projects`.\n\nUna vez hecho esto, configura la ruta a los proyectos en la sección `[extra]` de tu archivo `_index.md`:\n\n```toml\n[extra]\nprojects_path = \"projects/_index.md\"\n```\n\nEsto mostrará los 3 proyectos de mayor prioridad (con menor peso; el mismo orden que en Proyectos). Para mostrar más o menos proyectos, puedes establecer `max_projects` en `[extra]`.\n\nPor defecto, la sección de proyectos se mostrará en la parte inferior de la página principal, bajo los posts. Si prefieres que aparezca en la parte superior, establece `show_projects_first = true`.\n\n### Interruptor de modo claro y oscuro\n\n| Página | Sección | `config.toml` | Sigue la jerarquía | Requiere JavaScript |\n|:------:|:-------:|:-------------:|:---------------:|:-------------------:|\n|   ❌   |   ❌    |      ✅       |        ❌        |         ✅          |\n\nEl interruptor de modo claro y oscuro (el icono de luna/sol en la esquina superior derecha) puede habilitarse configurando `theme_switcher = true` en `config.toml`.\n\n### Modo predeterminado (claro/oscuro)\n\n| Página | Sección | `config.toml` | Sigue la jerarquía | Requiere JavaScript |\n|:------:|:-------:|:-------------:|:---------------:|:-------------------:|\n|   ❌   |   ❌    |      ✅       |        ❌        |         ❌          |\n\nEl tema predeterminado puede especificarse con la variable `default_theme`, que acepta `\"dark\"` o `\"light\"`. Si no lo especificas, el tema predeterminado será el tema del sistema.\n\n### Pieles personalizadas\n\n| Página | Sección | `config.toml` | Sigue la jerarquía | Requiere JavaScript |\n|:------:|:-------:|:-------------:|:---------------:|:-------------------:|\n|   ❌   |   ❌    |      ✅       |        ❌        |         ❌          |\n\nLas pieles de tabi cambian el color principal del sitio. Puedes configurar la piel en `config.toml` con `skin = \"nombre_de_la_piel\"`. Por ejemplo, `skin = \"lavender\"` se ve así (haz clic para cambiar entre modo claro y oscuro):\n\n{{ image_toggler(default_src=\"blog/customise-tabi/skins/lavender_light.webp\", toggled_src=\"blog/customise-tabi/skins/lavender_dark.webp\", default_alt=\"piel lavender en modo claro\", toggled_alt=\"piel lavender en modo oscuro\", full_width=true) }}\n\nExplora las pieles disponibles y aprende cómo crear la tuya propia consultando [la documentación](@/blog/customise-tabi/index.es.md#skins).\n\n### Fuente sans serif (paloseco)\n\n| Página | Sección | `config.toml` | Sigue la jerarquía | Requiere JavaScript |\n|:------:|:-------:|:-------------:|:---------------:|:-------------------:|\n|   ❌   |   ❌    |      ✅       |        ❌        |         ❌          |\n\ntabi utiliza una fuente serif para los párrafos de los artículos (la que estás viendo ahora). Puedes cambiar a una fuente sans serif (la que ves en los encabezados/menú) en todo tu sitio configurando `override_serif_with_sans = true` en `config.toml`.\n\nHaz clic en la imagen para comparar las fuentes:\n\n{{ image_toggler(default_src=\"blog/mastering-tabi-settings/img/serif.webp\", toggled_src=\"blog/mastering-tabi-settings/img/sans-serif.webp\", default_alt=\"Fuente serif\", toggled_alt=\"Fuente sans-serif\", full_width=true) }}\n\n### Indicador de enlaces externos\n\n| Página | Sección | `config.toml` | Sigue Jerarquía | Requiere JavaScript |\n|:------:|:-------:|:-------------:|:----------------:|:------------------:|\n|   ❌   |    ❌   |      ✅       |        ❌        |         ❌         |\n\n{{ admonition(type=\"info\", text=\"Requiere Zola 0.20.0 o posterior.\") }}\n\nSi deseas añadir un icono a los enlaces externos, configura la sección `[markdown]` (no `[extra]`) en tu `config.toml`:\n\n```toml\n[markdown]\nexternal_links_class = \"external\"\n```\n\nEsto añadirá un pequeño icono junto a los enlaces externos:\n\n{{ dual_theme_image(light_src=\"blog/mastering-tabi-settings/img/external_link_light.webp\", dark_src=\"blog/mastering-tabi-settings/img/external_link_dark.webp\", alt=\"Icono de enlace externo\", full_width=true) }}\n\n### Estilos CSS personalizados\n\n| Página | Sección | `config.toml` | Sigue la jerarquía | Requiere JavaScript |\n|:------:|:-------:|:-------------:|:---------------:|:-------------------:|\n|   ✅   |    ❌   |       ✅      |        ❌       |         ❌         |\n\nPuedes cargar estilos CSS personalizados para todo el sitio o en páginas específicas utilizando `stylesheets`, que acepta una lista de rutas hacia archivos CSS. Por ejemplo:\n\n```toml\nstylesheets = [\"css/custom.css\", \"css/another.css\"]\n```\n\n### Color del tema del navegador\n\n| Página | Sección | `config.toml` | Sigue la jerarquía | Requiere JavaScript |\n|:------:|:-------:|:-------------:|:---------------:|:-------------------:|\n|   ❌   |    ❌   |       ✅      |        ❌       |         ❌         |\n\nEl color del tema del navegador es el color que aparece en la barra de pestañas del navegador:\n\n{{ dual_theme_image(light_src=\"blog/mastering-tabi-settings/img/browser_theme_color_light.webp\", dark_src=\"blog/mastering-tabi-settings/img/browser_theme_color_dark.webp\" alt=\"pestañas con un tema de navegador de color\") }}\n\nPuedes establecerlo en `config.toml` como `browser_theme_color = \"#087e96\"`. Si deseas diferentes colores para los modos oscuro/claro, puedes establecer un conjunto de colores con `browser_theme_color = [\"#ffffff\", \"#000000\"]`. El primer color es para el modo claro, el segundo para el oscuro.\n\nEsta variable acepta cualquier color CSS válido, así que puedes usar palabras clave (por ejemplo, `blue`), códigos hexadecimales (por ejemplo, `#087e96`) o valores RGB/HSL (por ejemplo, `rgb(8, 126, 150)`).\n\n### Etiquetas compactas\n\n| Página | Sección | `config.toml` | Sigue la jerarquía | Requiere JavaScript |\n|:------:|:-------:|:-------------:|:---------------:|:-------------------:|\n|   ❌   |   ❌    |      ✅       |        ❌        |         ❌          |\n\nPor defecto, la [página de etiquetas](/es/tags) muestra las etiquetas así:\n\n[NombreEtiqueta](#) — n publicación[es]\n\nEstablecer `compact_tags = true` mostrará las mismas de este modo:\n\n[NombreEtiqueta](#) <sup>n</sup>\n\n### Orden de las etiquetas\n\n| Página | Sección | `config.toml` | Sigue la jerarquía | Requiere JavaScript |\n|:------:|:-------:|:-------------:|:---------------:|:-------------------:|\n|   ❌   |   ❌    |      ✅       |        ❌        |         ❌          |\n\nPor defecto, la [página de etiquetas](/es/tags) ordena las etiquetas alfabéticamente, dada la configuración predeterminada de `tag_sorting = \"name\"`.\nSi configuras `tag_sorting = \"frequency\"`, se ordenarán según el número de publicaciones (de mayor a menor).\n\n---\n\n## Series\n\nPara una explicación detallada, consulta la [documentación de series](@/blog/series/index.es.md).\n\n### Enlace para saltar a las publicaciones\n\n| Página | Sección | `config.toml` | Sigue jerarquía | Requiere JavaScript |\n|:------:|:-------:|:-------------:|:------------------:|:-------------------:|\n|   ❌   |   ✅    |      ✅       |        ✅          |         ❌          |\n\nPor defecto, aparece automáticamente un enlace \"Saltar a publicaciones\" junto al título de la serie cuando una serie tiene un contenido de más de 2000 caracteres:\n\n{{ dual_theme_image(light_src=\"blog/series/img/jump_to_series_posts_light.webp\", dark_src=\"blog/series/img/jump_to_series_posts_dark.webp\" alt=\"enlace para saltar a las publicaciones de la serie\", full_width=true) }}\n\nEstablece `show_jump_to_posts = true` para forzar la activación de la función y `show_jump_to_posts = false` para desactivarla.\n\n### Indexación de páginas de series\n\n| Página | Sección | `config.toml` | Sigue la jerarquía | Requiere JavaScript |\n|:------:|:-------:|:-------------:|:------------------:|:-------------------:|\n|   ❌   |   ✅    |      ✅       |        ✅          |         ❌          |\n\nPor defecto, las páginas de series se indexan (usando una indexación basada en 1) según el `sort_by` de la sección de series.\n\nEstablece `post_listing_index_reversed = true` para invertir el índice.\n\n---\n\n## Integración con repositorios Git\n\n| Página | Sección | `config.toml` | Sigue la jerarquía | Requiere JavaScript |\n|:------:|:-------:|:-------------:|:---------------:|:-------------------:|\n|   ❓   |    ❓   |       ✅      |        ❓       |         ❌         |\n\n❓: `show_remote_source` sí sigue [la jerarquía](#jerarquia-de-configuracion) y puede configurarse en una página, sección o globalmente. El resto de las configuraciones solo pueden establecerse en `config.toml`.\n\nEstas configuraciones te permiten vincular tu sitio web tabi con un repositorio público de Git en GitHub, GitLab, Gitea o Codeberg. Configuraciones de ejemplo:\n\n```toml\nremote_repository_url = \"https://github.com/welpo/tabi\"\nremote_repository_git_platform = \"auto\"\nremote_repository_branch = \"main\"\nshow_remote_changes = true\nshow_remote_source = true\n```\n\nEsto habilita dos funciones:\n\n1. `show_remote_source = true` añade un enlace al código fuente de tu sitio (tu `remote_repository_url`) que se mostrará en el pie de página:\n\n{{ dual_theme_image(light_src=\"blog/mastering-tabi-settings/img/site_source_light.webp\", dark_src=\"blog/mastering-tabi-settings/img/site_source_dark.webp\" alt=\"Pie de página del sitio, mostrando un enlace 'Código fuente del sitio'\") }}\n\n1. `show_remote_changes = true` añade un enlace «Ver cambios ↗» al historial de commits del artículo actualizado, al lado de la fecha de última actualización [^1]:\n\n{{ dual_theme_image(light_src=\"blog/mastering-tabi-settings/img/see_changes_light.webp\", dark_src=\"blog/mastering-tabi-settings/img/see_changes_dark.webp\" alt=\"Título del artículo y metadatos, mostrando un enlace 'Ver cambios'\") }}\n\nAl hacer clic en este enlace, serás dirigido al historial de commits del artículo, donde podrás ver los cambios realizados en él:\n\n{{ dual_theme_image(light_src=\"blog/mastering-tabi-settings/img/commit_history_light.webp\", dark_src=\"blog/mastering-tabi-settings/img/commit_history_dark.webp\" alt=\"Historial de commits de un artículo\", full_width=true) }}\n\n---\n\n## Páginas\n\n### Proyectos\n\ntabi tiene una plantilla integrada para proyectos. Para habilitarla, puedes crear un directorio en `content/projects/`. Allí, puedes crear un archivo `_index.md` con el siguiente contenido en el bloque de metadatos:\n\n```toml\ntitle = \"Proyectos\"\nsort_by = \"weight\"\ntemplate = \"cards.html\"\ninsert_anchor_links = \"left\"\n\n[extra]\nshow_reading_time = false\nquick_navigation_buttons = true\n```\n\n- `title` es el título de la página.\n- `sort_by` determina cómo se ordenan los proyectos. Puedes ordenar por «date», «update_date», «title», «title_bytes», «weight», «slug» o «none».\n- `template = \"cards.html\"` establece la plantilla para renderizar la página de proyectos.\n- `insert_anchor_links = \"left\"` añade enlaces ancla a los encabezados.\n- `show_reading_time = false` oculta el tiempo estimado de lectura.\n- `quick_navigation_buttons = true` muestra los botones de navegación rápida.\n\nJunto al archivo `_index.md`, puedes crear un archivo para cada proyecto. Por ejemplo, este es el bloque de metadatos para la página del proyecto [tabi](@/projects/tabi/index.es.md):\n\n```toml\ntitle = \"tabi\"\ndescription = \"Un tema de Zola rápido, ligero y moderno con soporte multilingüe.\"\nweight = 1\n\n[extra]\nlocal_image = \"img/tabi.webp\"\ninvertible_image = false\n```\n\n- `title` es el título del proyecto.\n- `description` es la descripción del proyecto.\n- `weight` determina el orden en el que se muestran los proyectos. Cuanto menor sea el peso, más arriba aparecerá el proyecto.\n- `local_image` es la ruta de la imagen del proyecto. Esta imagen se muestra en la página de proyectos.\n- `local_image_dark` es una variante opcional para el modo oscuro. Requiere que `local_image` esté configurado.\n- `remote_image` es una URL a una imagen externa, como alternativa a `local_image`.\n- `remote_image_dark` es una variante opcional para el modo oscuro. Requiere que `remote_image` esté configurado.\n- `invertible_image` invierte los colores de la imagen en modo oscuro. Útil para logotipos o iconos en blanco y negro.\n\nCuando un usuario haga clic en la imagen o el título de un proyecto, será llevado a la página del proyecto. Si prefieres que los usuarios vayan a un enlace externo, puedes establecer `link_to = \"https://example.com\"` en la sección `[extra]` del archivo `.md` del proyecto.\n\nLa página del proyecto individual se renderiza con la plantilla predeterminada, a menos que establezcas otra, por ejemplo, `template = \"info-page.html\"`.\n\n#### Filtrar proyectos\n\nSi agregas etiquetas a tus proyectos, verás un filtro de etiquetas:\n\n{{ dual_theme_image(light_src=\"blog/mastering-tabi-settings/img/projects_tag_filter_light.webp\", dark_src=\"blog/mastering-tabi-settings/img/projects_tag_filter_dark.webp\", alt=\"Página de proyectos con filtro de etiquetas\", full_width=true) }}\n\nEl sistema de filtrado de etiquetas utiliza mejora progresiva:\n\n- Sin JavaScript: Las etiquetas enlazan directamente a páginas de etiquetas dedicadas (por ejemplo, `/tags/nombre-etiqueta`).\n- Con JavaScript: Filtrado instantáneo, sincronización de URL (#nombre-etiqueta) y navegación por teclado.\n\nPara desactivar esta función, establece `enable_cards_tag_filtering = false` en la sección `[extra]` del archivo `projects/_index.md` o en `config.toml`.\n\n{% admonition(type=\"tip\") %}\n\nPara filtrar proyectos por etiquetas, necesitas establecer etiquetas en el front matter de cada proyecto. Por ejemplo:\n\n```toml\ntitle = \"nombre del proyecto\"\nweight = 40\n\n[taxonomies]\ntags = [\"etiqueta uno\", \"etiqueta 2\", \"tercera etiqueta\"]\n```\n\n{% end %}\n\n### Archivo\n\nAgregar una página de archivo es similar a agregar una página de proyectos. Puedes crear un directorio en `content/archive/`. Allí, puedes crear un archivo `_index.md` con el siguiente encabezado:\n\n```toml\ntitle = \"Archivo\"\ntemplate = \"archive.html\"\n```\n\nPor defecto, el archivo mostrará las publicaciones ubicadas en `blog/`. Para personalizar esto, puedes modificar la sección `[extra]` del archivo `_index.md`:\n\n- **Para una sola ruta**: Establece `section_path = \"tu-ruta/\"` para listar publicaciones de un directorio específico. Asegúrate de incluir la barra inclinada al final.\n\n- **Para múltiples rutas**: Si deseas agregar publicaciones de varios directorios, `section_path` puede especificarse como una lista de rutas. Por ejemplo:\n\n  ```toml\n  [extra]\n  section_path = [\"blog/\", \"notas/\", \"ruta-tres/\"]\n  ```\n\nEl archivo muestra las publicaciones en orden cronológico inverso (las más recientes primero). Puedes invertir este orden estableciendo `archive_reverse = true` en la sección `[extra]`:\n\n```toml\n[extra]\narchive_reverse = true  # muestra las publicaciones más antiguas primero\n```\n\n{{ admonition(type=\"note\", title=\"nota\" text=\"La página de Archivo sólo listará publicaciones que tengan fecha en su encabezado.\") }}\n\n### Etiquetas\n\ntabi tiene soporte integrado para etiquetas. Para habilitarlas, simplemente añade la taxonomía a tu `config.toml`:\n\n```toml\ntaxonomies = [{name = \"tags\", feed = true}]\n```\n\nLuego, puedes añadir etiquetas a tus publicaciones agregándolas al array `tags` en el bloque de metadatos de tu publicación. Por ejemplo:\n\n```toml,hl_lines=05-06\ntitle = \"Los molinos de viento de mi vida: reflexiones de un escudero\"\ndate = 1605-01-16\ndescription = \"Mi viaje junto a Don Quijote, enfrentándome a gigantes imaginarios y descubriendo las verdaderas batallas de la vida.\"\n\n[taxonomies]\ntags = [\"personal\", \"reflexiones\"]\n```\n\n### Página acerca de\n\nSi deseas tener una página que no sea un artículo, por ejemplo para un apartado \"Acerca de\", \"Contacto\" o \"Derechos de autor\", puedes usar la plantilla `info-page.html`.\n\nPrimero, crea un directorio dentro de `content/` con el nombre que prefieras. Por ejemplo, `content/pages/`. Luego, crea un archivo `_index.md` dentro de ese directorio. El archivo debería verse así:\n\n```markdown\n+++\nrender = false\ninsert_anchor_links = \"left\"\n+++\n```\n\n- `render = false` indica a Zola que no renderice la sección.\n- `insert_anchor_links = \"left\"` añade enlaces ancla a los encabezados. Esto es opcional.\n\nDentro del directorio, puedes crear cualquier cantidad de archivos `.md`.\n\nEn esta demo, la página [Sobre mí](/es/about/) utiliza la plantilla `info-page.html`. El bloque de metadatos es el siguiente:\n\n```toml\ntitle = \"Sobre mí\"\ntemplate = \"info-page.html\"\npath = \"about\"\n```\n\nFíjate cómo se establece `path = \"about\"`. Zola colocará la página en `$base_url/about/`. Si deseas que la página esté disponible en `/contacto/`, tendrías que establecer `path = \"contacto\"`.\n\nLa plantilla `info-page.html` también se puede utilizar para crear lading pages en la ruta raíz (`\"/\"`). Para hacerlo, el archivo `content/_index.md` debería verse así:\n\n```markdown\n+++\ntitle = \"Título de la página\"\ntemplate = \"info-page.html\"\n+++\n\nContenido con Markdown.\n```\n\n---\n\n## SEO\n\ntabi se encarga de la mayoría de las tareas de SEO por ti (como etiquetas del protocolo Open Graph, descripción, esquema de colores…), pero hay ciertas configuraciones que puedes personalizar.\n\n### Favicon\n\n| Página | Sección | `config.toml` | Sigue la jerarquía | Requiere JavaScript |\n|:------:|:-------:|:-------------:|:---------------:|:-------------------:|\n|   ❌   |   ❌    |      ✅       |        ❌       |         ❌          |\n\nEl favicon es el pequeño icono que aparece en la pestaña del navegador. Puedes establecerlo en `config.toml` con `favicon = \"img/favicon.png\"`.\n\n### Favicon de emoji\n\n| Página | Sección | `config.toml` | Sigue la jerarquía | Requiere JavaScript |\n|:------:|:-------:|:-------------:|:---------------:|:-------------------:|\n|   ❌   |   ❌    |      ✅       |        ❌       |         ❌          |\n\nTambién puedes establecer un emoji como tu favicon con `favicon_emoji`. Por ejemplo, `favicon_emoji = \"👾\"`.\n\nNota: Algunos navegadores no admiten favicons de emoji. Consulta la tabla de compatibilidad en [caniuse](https://caniuse.com/link-icon-svg).\n\n### URL canónica\n\n| Página | Sección | `config.toml` | Sigue la jerarquía | Requiere JavaScript |\n|:------:|:-------:|:-------------:|:---------------:|:-------------------:|\n|   ✅   |   ✅    |      ✅       |        ❌       |         ❌          |\n\nLa URL canónica es una manera de indicar a los motores de búsqueda cuál es la URL preferida para el contenido de tu sitio web. Esto es útil para el SEO y para evitar problemas de contenido duplicado.\n\nPor defecto, la URL canónica es la URL de la página en la que te encuentras. Sin embargo, puedes cambiar esto configurando `canonical_url` en el front matter de tu página o sección.\n\nSi tienes un sitio con una estructura idéntica y contenido coincidente, puedes configurar `base_canonical_url` en tu `config.toml`. La URL canónica se creará reemplazando el `$base_url` de la URL actual con el `$base_canonical_url` que establezcas.\n\nPor ejemplo, si configuras `base_canonical_url = \"https://example.com\"`, la URL canónica de la página `$base_url/blog/post1` será `https://example.com/blog/post1`. Esto es útil si tienes un sitio con varios dominios que comparten el mismo contenido.\n\n**Nota**: para asegurarte de que la URL canónica sea correcta, probablemente sea mejor configurar `canonical_url` individualmente para cada página.\n\n### Tarjetas para redes sociales\n\n| Página | Sección | `config.toml` | Sigue la jerarquía | Requiere JavaScript |\n|:------:|:-------:|:-------------:|:---------------:|:-------------------:|\n|   ✅   |   ✅    |      ✅       |        ✅       |         ❌          |\n\nLas tarjetas para redes sociales son las imágenes que se muestran cuando compartes un enlace en redes sociales:\n\n{{ dimmable_image(src=\"img/with_social_media_card.webp\", alt=\"Una captura de pantalla de WhatsApp mostrando un enlace con una tarjeta para redes sociales\") }}\n\nPuedes establecer la imagen para redes sociales con `social_media_card = \"img/social_media_card.png\"`.\n\nPuedes especificar rutas tanto relativas como absolutas.\n\n- **Ruta relativa**: Coloca la imagen en la misma carpeta que tu entrada de blog y especifica su nombre. Por ejemplo, `social_media_card = \"relative_image.png\"`.\n\n- **Ruta absoluta**: Coloca la imagen en una carpeta específica y especifica la ruta desde la raíz. Por ejemplo, `social_media_card = \"img/absolute_image.png\"`.\n\nSi ambas rutas, relativa y absoluta, son válidas, la ruta relativa tendrá prioridad.\n\nDado que sigue la [jerarquía](#jerarquia-de-configuracion), si no está configurado en una página, pero sí lo está en una sección, se utilizará la imagen de la sección. Si no está configurado en una página o sección, pero sí en `config.toml`, se usará la imagen global.\n\n{{ admonition(type=\"tip\", title=\"CONSEJO\", text=\"Automatiza su creación con un [script](https://github.com/welpo/osc.garden/blob/main/static/code/social-cards-zola): [Automatizando las vistas previas de los enlaces con Zola](https://osc.garden/es/blog/automating-social-media-cards-zola/).\") }}\n\n### Creador del fediverso\n\n| Página | Sección | `config.toml` | Sigue jerarquía | Requiere JavaScript |\n|:------:|:-------:|:-------------:|:----------------:|:-------------------:|\n|   ❌   |    ❌   |      ✅       |        ❌        |         ❌          |\n\nPuedes mostrar tu perfil del fediverso en las vistas previas de enlaces de Mastodon configurando la variable `fediverse_creator` en tu `config.toml`. Por ejemplo, para @username@example.com, usa:\n\n```toml\nfediverse_creator = { handle = \"username\", domain = \"example.com\" }\n```\n\n---\n\n## Navegación\n\n### Barra de navegación\n\n| Página | Sección | `config.toml` | Sigue la jerarquía | Requiere JavaScript |\n|:------:|:-------:|:-------------:|:---------------:|:-------------------:|\n|   ❌   |   ❌    |      ✅       |        ❌       |         ❌          |\n\nLa barra de navegación es la barra en la parte superior de la página que contiene el título del sitio y el menú de navegación. Puedes personalizar los elementos que aparecen configurando `menu` en `config.toml`.\n\nSoporta links relativos para páginas internas y links absolutos para enlaces externos. Por ejemplo:\n\n```toml\nmenu = [\n    { name = \"blog\", url = \"blog\", trailing_slash = true },\n    { name = \"archivo\", url = \"archive\", trailing_slash = true },\n    { name = \"etiquetas\", url = \"tags\", trailing_slash = true },\n    { name = \"proyectos\", url = \"projects\", trailing_slash = true },\n    { name = \"acerca de\", url = \"about\", trailing_slash = true },\n    { name = \"github\", url = \"https://github.com/welpo/tabi\", trailing_slash = false },\n]\n```\n\n### Botones de navegación rápida\n\n| Página | Sección | `config.toml` | Sigue la jerarquía | Requiere JavaScript |\n|:------:|:-------:|:-------------:|:---------------:|:-------------------:|\n|   ✅   |   ✅    |      ✅       |        ✅       |         ❌          |\n\nLos botones de navegación rápida son los botones que aparecen en la parte inferior derecha de la pantalla. Deberías verlos en esta página, si no estás en un dispositivo móvil. Se ven así:\n\n{{ dual_theme_image(light_src=\"blog/mastering-tabi-settings/img/quick_navigation_buttons_light.webp\", dark_src=\"blog/mastering-tabi-settings/img/quick_navigation_buttons_dark.webp\", alt=\"Botones de navegación rápida\") }}\n\nPara activarlos, establece `quick_navigation_buttons = true`.\n\n### Table de contenido\n\n| Página | Sección | `config.toml` | Sigue la jerarquía | Requiere JavaScript |\n|:------:|:-------:|:-------------:|:---------------:|:-------------------:|\n|   ✅   |   ✅    |      ✅       |        ✅       |         ❌          |\n\nHabilita el índice de contenidos justo debajo del título y metadatos del artículo con `toc = true`.\n\nPara saber más sobre cómo personalizarlo, consulta [la documentación sobre la Tabla de contenido](@/blog/toc/index.es.md).\n\n### Enlace a los artículos anterior y siguiente\n\n| Página | Sección | `config.toml` | Sigue la jerarquía | Requiere JavaScript |\n|:------:|:-------:|:-------------:|:------------------:|:-------------------:|\n|   ✅   |   ✅   |      ✅       |         ✅         |         ❌         |\n\nMuestra enlaces a los artículos anterior y siguiente en la parte inferior de los posts. Se ve así:\n\n{{ dual_theme_image(light_src=\"blog/mastering-tabi-settings/img/show_previous_next_article_links_light.webp\", dark_src=\"blog/mastering-tabi-settings/img/show_previous_next_article_links_dark.webp\", alt=\"Enlaces a los artículos anterior y siguiente\", full_width=true) }}\n\nPara activar esta función, configura `show_previous_next_article_links = true` y asegúrate de que tu sección tiene `sort_by` (por ejemplo, `sort_by = \"date\"`).\n\nPor defecto, los artículos siguientes estarán en el lado izquierdo de la página y los artículos anteriores en el lado derecho.\nPara invertir el orden (artículos siguientes en el lado derecho y artículos anteriores en el lado izquierdo), establece `invert_previous_next_article_links = true`.\n\nPor defecto, esta sección de navegación tendrá el ancho completo del sitio (igual que la barra de navegación de la parte superior). Para hacerla más estrecha, coincidiendo con el ancho del artículo, establece `previous_next_article_links_full_width = false`.\n\nTodas estas configuraciones siguen la jerarquía.\n\n### Enlaces de retorno en notas al pie\n\n{{ admonition(type=\"warning\", title=\"ADVERTENCIA DE DEPRECACIÓN\", text=\"Zola v0.19.0 y posterior puede hacer esto de forma nativa. Establece `bottom_footnotes = true` en la sección `[markdown]` de tu configuración.\") }}\n\n| Página | Sección | `config.toml` | Sigue la jerarquía | Requiere JavaScript |\n|:------:|:-------:|:-------------:|:---------------:|:-------------------:|\n|   ✅   |   ✅    |      ✅       |        ✅       |         ✅          |\n\nEstablecer `footnote_backlinks = true` añadirá enlaces de retorno a las notas al pie de tus publicaciones, como este:\n\n{{ dual_theme_image(light_src=\"blog/mastering-tabi-settings/img/footnote_backlinks_light.webp\", dark_src=\"blog/mastering-tabi-settings/img/footnote_backlinks_dark.webp\", alt=\"Enlaces de retorno en notas al pie\", full_width=true) }}\n\nCuando hagas clic en un enlace de retorno (la flecha ↩), te llevará de vuelta al punto del texto donde se hizo referencia a la nota al pie.\n\n---\n\n## Usabilidad\n\n### Botón de copiar en bloques de código\n\n| Página | Sección | `config.toml` | Sigue la jerarquía | Requiere JavaScript |\n|:------:|:-------:|:-------------:|:---------------:|:-------------------:|\n|   ✅   |   ✅    |      ✅       |        ✅       |         ✅          |\n\nEstablecer `copy_button = true` añadirá un pequeño botón de copiar en la parte superior derecha de los bloques de código, como este:\n\n{{ dual_theme_image(light_src=\"blog/mastering-tabi-settings/img/copy_button_on_code_blocks_light.webp\", dark_src=\"blog/mastering-tabi-settings/img/copy_button_on_code_blocks_dark.webp\", alt=\"Botón de copiar en bloques de código\", full_width=true) }}\n\n### Nombres de bloques de código clicables\n\n| Página | Sección | `config.toml` | Sigue la jerarquía | Requiere JavaScript |\n|:------:|:-------:|:-------------:|:---------------:|:-------------------:|\n|   ✅   |   ✅    |      ✅       |        ✅       |         ✅          |\n\nEstablece `code_block_name_links = true` para habilitan los enlaces clickables en los nombres de bloques de código. Consulta la [documentación](@/blog/shortcodes/index.es.md#mostrar-ruta-o-url) para ver ejemplos y uso.\n\n### Forzar bloques de código de izquierda a derecha\n\n| Página | Sección | `config.toml` | Sigue la jerarquía | Requiere JavaScript |\n|:------:|:-------:|:-------------:|:----------------:|:-------------------:|\n|   ✅   |   ✅    |      ✅       |        ✅        |         ❌         |\n\nPor defecto, los bloques de código se renderizan de izquierda a derecha, independientemente de la dirección general del texto. Establece `force_codeblock_ltr = false` para permitir que los bloques de código sigan la dirección del documento. Útil para idiomas de derecha a izquierda que necesitan bloques de código de derecha a izquierda.\n\n### Soporte para KaTeX\n\n| Página | Sección | `config.toml` | Sigue la jerarquía | Requiere JavaScript |\n|:------:|:-------:|:-------------:|:---------------:|:-------------------:|\n|   ✅   |   ✅    |      ✅       |        ✅       |         ✅          |\n\nKaTeX es una biblioteca JavaScript rápida y fácil de usar para la representación de matemáticas TeX en la web. Puedes habilitarlo con `katex = true`. Mira cómo se ve en tabi [aquí](/es/blog/markdown/#katex).\n\n### Soporte para Mermaid\n\n| Página | Sección | `config.toml` | Sigue la jerarquía | Requiere JavaScript |\n|:------:|:-------:|:-------------:|:----------------:|:-------------------:|\n|   ✅   |    ✅   |      ✅       |        ✅        |         ✅          |\n\n[Mermaid](https://github.com/mermaid-js/mermaid) es una herramienta de diagramación y gráficos basada en JavaScript. Puedes activarla con `mermaid = true`.\n\nPor defecto, la biblioteca Mermaid se sirve localmente. Si prefieres usar un CDN, establece `serve_local_mermaid = false` en `config.toml`. El uso de un CDN servirá la versión más reciente de Mermaid; la opción local servirá la versión incluida con tabi.\n\nConsulta la [documentación de Mermaid](@/blog/shortcodes/index.es.md#diagramas-de-mermaid) para instrucciones de uso y ejemplos.\n\n### Subconjunto de fuente personalizada\n\n| Página | Sección | `config.toml` | Sigue la jerarquía | Requiere JavaScript |\n|:------:|:-------:|:-------------:|:---------------:|:-------------------:|\n|   ❌   |   ❌    |      ✅       |        ❌       |         ❌          |\n\nLas fuentes personalizadas causan parpadeo del texto en Firefox. Para solucionar esto, tabi carga un subconjunto de glifos para el encabezado. Dado que esto (ligeramente) aumenta el tiempo de carga inicial, es una buena idea intentar minimizar el tamaño de este subconjunto, o desactivarlo por completo si no estás usando una fuente personalizada en tu tema.\n\nPuedes crear un subconjunto personalizado adaptado a tu sitio, guardarlo como `static/custom_subset.css`, y hacer que se cargue con `custom_subset = true`.\n\nPara desactivar el subconjunto, puedes usar `enable_subset = false`.\n\nPara obtener más información, incluyendo instrucciones sobre cómo crear un subconjunto personalizado, consulta la [documentación](@/blog/custom-font-subset/index.es.md).\n\n### Contenido completo en el feed\n\n| Página | Sección | `config.toml` | Sigue la jerarquía | Requiere JavaScript |\n|:------:|:-------:|:-------------:|:---------------:|:-------------------:|\n|   ❌   |   ❌    |      ✅       |        ❌       |         ❌          |\n\nPor defecto, el feed Atom solo contiene el resumen/descripción de tus publicaciones. Puedes incluir el contenido completo de las publicaciones estableciendo `full_content_in_feed = true` en `config.toml`.\n\n### Ocultar contenido del feed\n\n| Página | Sección | `config.toml` | Sigue la jerarquía | Requiere JavaScript |\n|:----:|:-------:|:-------------:|:-----------------:|:-------------------:|\n|  ✅  |   ✅    |      ✅       |         ✅        |         ❌          |\n\nPuedes controlar cómo aparece el contenido en los feeds usando dos configuraciones:\n\n- `hide_from_feed = true`: Oculta el contenido de todos los feeds (feed principal, feeds de sección y feeds de etiquetas)\n- `hide_from_main_feed = true`: Oculta el contenido solo del feed principal mientras lo mantiene visible en los feeds de sección y de etiquetas\n\n### Comentarios {#añadir-comentarios}\n\n| Página | Sección | `config.toml` | Sigue la jerarquía | Requiere JavaScript |\n|:------:|:-------:|:-------------:|:---------------:|:-------------------:|\n|   ✅   |  ❌    |      ✅        |        ❌       |         ✅          |\n\nPara activar los comentarios en una página, establece el nombre del sistema como `true` en el front matter. Por ejemplo, `utterances = true`.\n\nSi quieres activar los comentarios de forma global, puedes hacerlo estableciendo `enabled_for_all_posts = true` en la sección apropiada de tu `config.toml` (por ejemplo, en `[extra.giscus]`).\n\nSi has activado un sistema globalmente, pero quieres desactivarlo en una página específica, puedes hacerlo estableciendo el nombre del sistema como `false` en el front matter. Por ejemplo, `utterances = false`.\n\nLee la [documentación](@/blog/comments/index.es.md) para obtener más información sobre los sistemas disponibles y su configuración.\n\n### Botones de iine {#iine}\n\n| Página | Sección | `config.toml` | Sigue la jerarquía | Requiere JavaScript |\n|:------:|:-------:|:-------------:|:-------------------:|:-------------------:|\n|   ✅   |   ✅    |      ✅       |          ✅         |         ❌          |\n\ntabi soporta botones de [iine](https://iine.to/) para mostrar apreciación anónima por tu contenido. Estos botones centrados en la privacidad funcionan sin JavaScript y no rastrean usuarios.\n\nPara activar los botones iine globalmente:\n\n```toml\n[extra]\niine = true\n```\n\nPuedes personalizar el icono usado en los botones (esta configuración sigue la jerarquía):\n\n```toml\n[extra]\niine_icon = \"thumbs_up\"  # Opciones: \"heart\", \"thumbs_up\", \"upvote\", o cualquier emoji\n```\n\nPara sitios multilingües, puedes unificar los conteos de likes entre versiones en diferentes idiomas del mismo contenido (configuración solo de config; valor predeterminado: `true`):\n\n```toml\n[extra]\niine_unified_languages = true  # Los likes en /es/blog/hello/ cuentan hacia /blog/hello/\n```\n\nTambién puedes activar los botones iine en páginas o secciones individuales estableciendo `iine = true` en su front matter, o personalizar el icono con `iine_icon = \"🚀\"`.\n\n### Análisis web\n\n| Página | Sección  | `config.toml` | Sigue Jerarquía | Requiere JavaScript |\n|:------:|:--------:|:-------------:|:----------------:|:-------------------:|\n|   ❌   |    ❌    |       ✅      |        ❌        |          ✅         |\n\ntabi ofrece soporte para 3 sistemas de análisis web que respetan la privacidad: [Plausible](https://plausible.io/), [GoatCounter](https://www.goatcounter.com/) y [Umami](https://umami.is/).\n\nPuedes configurarlos en la sección `[extra.analytics]` de tu archivo `config.toml`.\n\n- `service`: el servicio a utilizar. Las opciones disponibles son `\"goatcounter\"`, `\"umami\"`, y `\"plausible\"`.\n\n- `id`: el identificador único para tu servicio de análisis. Esto varía según el servicio:\n  - Para GoatCounter, es el código elegido durante el registro. Instancias auto-alojadas de GoatCounter no requieren este campo.\n  - Para Umami, es la ID del sitio web.\n  - Para Plausible, puede ser:\n    - **Formato nuevo** (Plausible v3.1.0+): El nombre de script aleatorio sin la extensión (ej. `\"pa-XXXXXX\"`). Encuéntralo en tu panel de Plausible en Ajustes → Detalles del sitio web → Nombre del script.\n    - **Formato heredado**: Tu nombre de dominio (ej. `\"example.com\"`). Útil si necesitas enviar estadísticas a múltiples paneles simultáneamente; el nuevo formato no admite esta funcionalidad. Consulta la [guía de actualización de scripts de Plausible](https://plausible.io/docs/script-update-guide) para más detalles.\n\n- `self_hosted_url`. Opcional. Utiliza este campo para especificar la URL si tienes una instancia auto-alojada. La URL base variará según tu configuración particular. Algunos ejemplos:\n  - Para GoatCounter: `\"https://stats.example.com\"`\n  - Para Umami: `\"https://umami.example.com\"`\n  - Para Plausible: `\"https://plausible.example.com\"`\n\n- `do_not_track`: (sólo para Umami) opcional. Cuando se establece en `true`, se desactiva el seguimiento para los usuarios cuyos navegadores envían un encabezado \"Do Not Track\".\n\nUn ejemplo de configuración para GoatCounter no auto-alojada sería:\n\n```toml\n[extra.analytics]\nservice = \"goatcounter\"\nid = \"tabi\"\nself_hosted_url = \"\"\n```\n\n---\n\n## Pie de página\n\n### Iconos de redes sociales\n\n| Página | Sección | `config.toml` | Respeta jerarquía | Requiere JavaScript |\n|:------:|:-------:|:-------------:|:-----------------:|:-------------------:|\n|   ❌   |   ❌    |      ✅       |         ❌        |         ❌          |\n\nPuedes añadir iconos de redes sociales al pie de página con `socials`, que acepta una lista de objetos de redes sociales. Por ejemplo:\n\n```toml\nsocials = [\n    { name = \"github\", url = \"https://github.com/welpo/\", icon = \"github\" },\n    { name = \"soundcloud\", url = \"https://soundcloud.com/oskerwyld\", icon = \"soundcloud\" },\n    { name = \"instagram\", url = \"https://instagram.com/oskerwyld\", icon = \"instagram\" },\n    { name = \"youtube\", url = \"https://youtube.com/@oskerwyld\", icon = \"youtube\" },\n    { name = \"spotify\", url = \"https://open.spotify.com/artist/5Hv2bYBhMp1lUHFri06xkE\", icon = \"spotify\" },\n]\n```\n\nPara ver una lista de todos los iconos integrados, echa un vistazo al directorio [`static/social_icons` en GitHub](https://github.com/welpo/tabi/tree/main/static/social_icons).\n\n¿Echas en falta algún icono? Si crees que sería una buena adición a tabi, no dudes en [abrir un issue](https://github.com/welpo/tabi/issues/new?assignees=&labels=enhancement&projects=&template=feature_request.md&title=) o enviar un pull request ([ejemplo](https://github.com/welpo/tabi/pull/333)).\n\nPara usar un icono personalizado, puedes añadirlo al directorio `static/social_icons` de tu sitio. Por ejemplo, si añades `custom.svg`, puedes referenciarlo así:\n\n```\n{ name = \"custom\", url = \"https://example.com\", icon = \"custom\" }\n```\n\n{{ admonition(type=\"note\", title=\"NOTA\", text=\"Todos los enlaces sociales incluyen el [atributo](https://developer.mozilla.org/docs/Web/HTML/Attributes/rel/me) `rel='me'`. Esto ayuda a los motores de búsqueda y servicios web a verificar que las cuentas de redes sociales te pertenecen.\") }}\n\n### Icono de feed\n\n| Página | Sección | `config.toml` | Respeta jerarquía | Requiere JavaScript |\n|:------:|:-------:|:-------------:|:-----------------:|:-------------------:|\n|   ❌   |   ❌    |      ✅       |         ❌        |         ❌          |\n\nPuedes añadir un enlace a tu feed RSS/Atom en el pie de página con `feed_icon = true`.\n\nPara usar un icono personalizado, establece `feed_icon` con el nombre del icono (por ejemplo, `feed_icon = \"square-rss\"`). El icono debe existir en `static/social_icons/` (sin la extensión `.svg`).\n\nNota para usuarios de Zola 0.19.X: cuando hay dos nombres de archivo en `feed_filenames`, solo se enlazará el primero en el pie de página.\n\n### Menú de pie de página\n\n| Página | Sección | `config.toml` | Respeta jerarquía | Requiere JavaScript |\n|:------:|:-------:|:-------------:|:---------------:|:------------------:|\n|   ❌   |   ❌    |      ✅       |        ❌       |        ❌          |\n\nPuedes añadir un menú al pie de página con `footer_menu`, que acepta una lista de elementos de menú. Por ejemplo:\n\n```toml\nfooter_menu = [\n    {url = \"about\", name = \"about\", trailing_slash = true},\n    {url = \"privacy\", name = \"privacy\", trailing_slash = true},\n    {url = \"sitemap.xml\", name = \"sitemap\", trailing_slash = false},\n    {url = \"https://example.com\", name = \"external link\", trailing_slash = true},\n]\n```\n\n### Copyright\n\n| Página | Sección | `config.toml` | Respeta jerarquía | Requiere JavaScript |\n|:------:|:-------:|:-------------:|:-----------------:|:-------------------:|\n|   ❌   |   ❌    |      ✅       |         ❌        |         ❌          |\n\nPara añadir una mención sobre los derechos de autor a tu sitio web, configura `copyright`:\n\n```toml\ncopyright = \"© $CURRENT_YEAR Your Name $SEPARATOR Unless otherwise noted, the content in this website is available under the [CC BY-SA 4.0](https://creativecommons.org/licenses/by-sa/4.0/) license.\"\n```\n\n- `$TITLE` será reemplazado por la variable `title` configurada en `config.toml`\n- `$CURRENT_YEAR` será reemplazado por el año actual\n- `$AUTHOR` será reemplazado por la variable `author`\n- `$SEPARATOR` será reemplazado por la [variable `separator`](#separador-personalizado).\n\nSe procesará el texto en Markdown. Por ejemplo, la configuració de arriba:\n\n{{ dual_theme_image(light_src=\"blog/mastering-tabi-settings/img/copyright_light.webp\", dark_src=\"blog/mastering-tabi-settings/img/copyright_dark.webp\" alt=\"Sección de derechos de autor\", full_width=true) }}\n\nSi tienes un sitio multilingüe y deseas establecer diferentes notificaciones de derechos de autor para diferentes idiomas, añade la traducción correspondiente a `copyright_translations.{código_de_idioma}` para cada idioma que quieras dar soporte. El código de idioma debe coincidir con el [código de idioma de tabi](https://welpo.github.io/tabi/es/blog/faq-languages/#que-son-estos-codigos-de-dos-letras). Por ejemplo:\n\n```toml\ncopyright_translations.es = \"© $CURRENT_YEAR $AUTHOR $SEPARATOR A menos que se indique lo contrario, el contenido de esta web está disponible bajo la licencia [CC BY-SA 4.0](https://creativecommons.org/licenses/by-sa/4.0/).\"\n```\n\n---\n\n## Metadatos\n\n### Mostrar autoría\n\n| Página | Sección | `config.toml` | Respeta jerarquía | Requiere JavaScript |\n|:------:|:-------:|:-------------:|:-----------------:|:-------------------:|\n|   ✅   |   ✅    |      ✅       |         ✅        |         ❌          |\n\nPara mostrar la autoría de un artículo, usa `show_author = true`.\n\nEsto mostrará lxs autorxs establecidxs en la variable `authors = []` en el front matter del artículo. Si esto no está disponible, se usará `author = \"\"` en `config.toml`.\n\n### Tiempo de lectura\n\n| Página | Sección | `config.toml` | Respeta jerarquía | Requiere JavaScript |\n|:------:|:-------:|:-------------:|:-----------------:|:-------------------:|\n|   ✅   |   ✅    |      ✅       |         ✅        |         ❌          |\n\nPuedes activar o desactivar el tiempo estimado de lectura de un artículo con `show_reading_time`. Si lo estableces en `true`, se mostrará en los metadatos del artículo, así:\n\n{{ dual_theme_image(light_src=\"blog/mastering-tabi-settings/img/see_changes_light.webp\", dark_src=\"blog/mastering-tabi-settings/img/see_changes_dark.webp\" alt=\"Título del artículo y metadatos, mostrando un enlace «Ver cambios»\") }}\n\nDado que sigue [la jerarquía](#jerarquia-de-configuracion), puedes activarlo o desactivarlo para páginas o secciones específicas. Por ejemplo, esta demo desactiva `show_reading_time = false` en la sección [proyectos](https://welpo.github.io/tabi/es/projects/) en el archivo [`_index.md`](https://github.com/welpo/tabi/blob/main/content/projects/_index.es.md?plain=1), por lo que sus publicaciones individuales no muestran el tiempo de lectura.\n\n### Mostrar la fecha\n\n| Página | Sección | `config.toml` | Sigue la jerarquía | Requiere JavaScript |\n|:------:|:-------:|:-------------:|:------------------:|:-------------------:|\n|   ✅   |   ✅    |      ✅       |         ✅          |         ❌          |\n\nPor defecto, la fecha se muestra debajo del título de la publicación. Puedes ocultarla con `show_date = false`. Esta configuración sigue [la jerarquía](#jerarquia-de-configuracion).\n\n### Formato de fecha\n\n| Página | Sección | `config.toml` | Sigue la jerarquía | Requiere JavaScript |\n|:------:|:-------:|:-------------:|:------------------:|:-------------------:|\n|   ❌   |   ❌    |      ✅       |         ❌         |         ❌          |\n\ntabi tiene tres formatos de fecha: `long_date_format`, `short_date_format` y `archive_date_format`. El formato corto se utiliza en los metadatos de una publicación, mientras que el formato largo se utiliza al listar las publicaciones (es decir, en la [sección de blog](/es/blog/) o en la [página principal](/es/)). El formato de archivo se usa para mostrar el día y el mes en la página de archivo.\n\nPor defecto es \"6th July 2049\" para los formatos corto y largo en inglés. Para otros idiomas, el predeterminado es `\"%d %B %Y\"` para el formato largo y `\"%-d %b %Y\"` para el formato corto. El formato de archivo predeterminado universal es `\"%d %b\"`.\n\nEn Zola, la sintaxis para el formateo de tiempo está inspirada en strftime. Una referencia completa está disponible en la [documentación de chrono](https://docs.rs/chrono/0.4.31/chrono/format/strftime/index.html).\n\n#### Formatos de fecha por idioma\n\nPuedes personalizar los formatos de fecha para idiomas específicos usando la matriz `date_formats` en `config.toml`:\n\n```toml\ndate_formats = [\n    { lang = \"es\", long = \"%d de %B de %Y\", short = \"%-d %b %Y\", archive = \"%d de %b\" },\n    { lang = \"de\", long = \"%d. %B %Y\", short = \"%d.%m.%Y\", archive = \"%d. %b\" },\n]\n```\n\nEsto permite que diferentes idiomas usen formatos de fecha culturalmente apropiados (por ejemplo, \"6 de julio de 2049\" en español o \"6. Juli 2049\" en alemán).\n\n### Separador personalizado\n\n| Página | Sección | `config.toml` | Sigue la jerarquía | Requiere JavaScript |\n|:------:|:-------:|:-------------:|:------------------:|:-------------------:|\n|   ❌   |   ❌    |      ✅       |         ❌         |         ❌          |\n\nEl separador aparece en varios lugares: en el título del navegador, entre los metadatos de una publicación…\n\nEl separador por defecto es un punto de viñeta (`•`), pero puedes cambiarlo configurando algo como `separator = \"|\"`.\n\n### Orden del título\n\n| Página | Sección | `config.toml` | Sigue la jerarquía | Requiere JavaScript |\n|:------:|:-------:|:-------------:|:------------------:|:-------------------:|\n|   ❌   |   ❌    |      ✅       |         ❌         |         ❌          |\n\nPor defecto, el título en la pestaña del navegador es el nombre del sitio seguido del título de la página. Por ejemplo, el título de la sección del blog es «~/tabi • Blog».\n\nAl configurar `invert_title_order = true`, puedes invertir el orden del título del sitio y el título de la página en la pestaña del navegador. Por ejemplo, la etiqueta del título de la sección del blog se convertiría en «Blog • ~/tabi».\n\n---\n\n## Seguridad\n\n### Correo electrónico codificado\n\n| Página | Sección | `config.toml` | Sigue la jerarquía | Requiere JavaScript |\n|:------:|:-------:|:-------------:|:------------------:|:-------------------:|\n|   ❌   |   ❌    |      ✅       |         ❌         |         ✅          |\n\nPara proteger tu dirección de correo electrónico de los spambots, puedes codificarla en el pie de página. Puedes hacer esto estableciendo `email` en una versión codificada en base64 de tu dirección de correo electrónico[^2]. Por ejemplo, `email = \"bWFpbEBleGFtcGxlLmNvbQ==\"` es la versión codificada en base64 de \"mail@example.com\".\n\nSi no quieres codificar tu correo electrónico tú mismo, tabi puede hacerlo por ti si configuras `encode_plaintext_email = true`. Esto te permite establecer un correo electrónico en texto plano en la configuración. Ten en cuenta que esto sólo protege tu dirección de correo electrónico en tu sitio, no en repositorios públicos.\n\nSi el correo electrónico está codificado (ya sea por ti o por tabi), los usuarios con JavaScript desactivado no verán el icono de correo electrónico.\n\n### CSP (Content Security Policy)\n\n| Página | Sección | `config.toml` | Sigue la jerarquía | Requiere JavaScript |\n|:------:|:-------:|:-------------:|:------------------:|:-------------------:|\n|   ❌   |   ❌    |      ✅       |         ❌         |         ❌          |\n\nLa Content Security Policy (CSP) es una capa adicional de seguridad que ayuda a detectar y mitigar ciertos tipos de ataques, incluidos ataques de Cross Site Scripting (XSS) e inyección de datos. Estos ataques se utilizan para todo, desde robo de datos hasta desfiguración de sitios y distribución de malware.\n\ntabi tiene una CSP predeterminada que permite imágenes y vídeos remotos, así como incrustaciones de YouTube y Vimeo. Puedes personalizarla con `allowed_domains`, que toma una lista de directivas de CSP. Esta es la CSP predeterminada:\n\n```toml\nallowed_domains = [\n    { directive = \"font-src\", domains = [\"'self'\", \"data:\"] },\n    { directive = \"img-src\", domains = [\"'self'\", \"https://*\", \"data:\"] },\n    { directive = \"script-src\", domains = [\"'self'\"] },\n    { directive = \"style-src\", domains = [\"'self'\"] },\n    { directive = \"frame-src\", domains = [\"player.vimeo.com\", \"https://www.youtube-nocookie.com\"] },\n]\n```\n\nEsta función está habilitada por defecto. Para deshabilitarla (y permitir todo), configura `enable_csp = false` en una página, sección o globalmente. La opción `enable_csp` sigue [la jerarquía](#jerarquia-de-configuracion).\n\nPara obtener más información, consulta la [página de documentación de CSP](@/blog/security/index.es.md).\n\n---\n\n## Indieweb\n\n### Webmentions\n\n| Página | Sección | `config.toml` | Sigue la jerarquía | Requiere JavaScript |\n|:------:|:-------:|:-------------:|:---------------:|:-------------------:|\n|   ❓   |   ❓    |      ✅       |        ❓       |         ✅          |\n\nComo se describe en el estándar W3C recomendado, [Webmention](https://www.w3.org/TR/webmention/#abstract-p-1) es una manera sencilla de notificar cualquier URL cuando la mencionas en tu sitio web. Desde la perspectiva del receptor, es una forma de solicitar notificaciones cuando otros sitios web la mencionan.\n\nPara sitios web estáticos, [webmention.io](https://webmention.io/) aloja un punto final de webmention que se puede utilizar para recibir webmentions. Esta función recupera las webmentions almacenadas en webmention.io y las muestra para una página. Necesitarás configurar una cuenta para tu sitio web en webmention.io. Cuando habilites la función, anunciará tu punto final de webmention.io y mostrará las webmentions para cualquier página.\n\nHabilita las webmentions para tu sitio web agregando lo siguiente a tu archivo `config.toml`.\n\n```toml\n[extra.webmentions]\nenable = true\n# Especifica el dominio registrado con webmention.io.\ndomain = \"www.example.com\"\n```\n\n❓: Para desactivar las webmentions para una sección o página específica, establece `webmentions = false` en la sección `[extra]` del front matter de esa sección o página.\n\nLa sección de webmentions se ve así:\n\n{{ dual_theme_image(light_src=\"blog/mastering-tabi-settings/img/webmention_light.webp\", dark_src=\"blog/mastering-tabi-settings/img/webmention_dark.webp\" alt=\"Captura de pantalla de webmentions mostrando reposts, me gusta, marcadores y comentarios\", full_width=true) }}\n\n### h-card representativa\n\n| Página | Sección | `config.toml` | Sigue Jerarquía | Requiere JavaScript |\n| :--: | :-----: | :-----------: | :---------------: | :-----------------: |\n|  ❌  |   ❌    |      ✅       |        ❌         |         ❌          |\n\nPor defecto, tabi añade una [h-card](https://microformats.org/wiki/h-card) representativa **oculta** a la página de inicio. Aunque es invisible para los visitantes, está disponible para los analizadores de microformatos. Puedes comprobar la validez de la tarjeta con la herramienta [Indiewebify.me](https://indiewebify.me/validate-h-card/).\n\nPara desactivar la h-card, establece `enable = false` en la sección `[extra.hcard]` de `config.toml`.\n\nLa h-card predeterminada incluye tu nombre, la URL del sitio web y los enlaces a redes sociales.\n\nPuedes establecer una imagen de perfil y una pequeña biografía con los ajustes `avatar` y `biography`.\n\nTodas las demás [propiedades de h-card](https://microformats.org/wiki/h-card#Properties) se pueden añadir listándolas en la sección `[extra.hcard]` del archivo de configuración. Simplemente reemplaza todos los caracteres `-` por `_`.\n\n---\n\n## Extendiendo elementos HTML en tabi\n\nAlgunos elementos HTML en tabi pueden extenderse para admitir casos de uso adicionales, como agregar JavaScript personalizado para comportamientos en todo el sitio al final de la etiqueta `<body>` o incluir contenido adicional al final del elemento `<head>` que no esté soportado por otras configuraciones de tabi.\n\nConsulta la tabla a continuación para ver los elementos que pueden extenderse:\n\n| Elemento | Plantilla                         |\n| :------: | :-------------------------------: |\n| `<head>` | `templates/tabi/extend_head.html` |\n| `<body>` | `templates/tabi/extend_body.html` |\n\nNo hay configuraciones explícitas que debas establecer para tu sitio o páginas. Simplemente crea el archivo de plantilla correspondiente para tu sitio, y tabi lo incluirá automáticamente.\n\n---\n\n[^1]: Si estás utilizando un repositorio Git remoto, es posible que quieras automatizar el proceso de actualización del campo `updated`. Aquí tienes una guía para eso: [Zola Git Hook: actualizando las fechas de las publicaciones](https://osc.garden/es/blog/zola-date-git-hook/).\n\n[^2]: Para codificar tu correo electrónico en base64 puedes utilizar [herramientas en línea](https://www.base64encode.org/) o, en tu terminal, ejecutar: `printf 'mail@example.com' | base64`\n"
  },
  {
    "path": "content/blog/mastering-tabi-settings/index.md",
    "content": "+++\ntitle = \"Mastering tabi Settings: A Comprehensive Guide\"\ndate = 2023-09-18\nupdated = 2026-01-31\ndescription = \"Discover the many ways you can customise your tabi site.\"\n\n[taxonomies]\ntags = [\"showcase\", \"tutorial\", \"FAQ\"]\n\n[extra]\npinned = true\nquick_navigation_buttons = true\nsocial_media_card = \"social_cards/blog_mastering_tabi_settings.jpg\"\n+++\n\nThis aims to be a comprehensive guide to every setting in tabi. If you have any questions, feel free to [open an issue on GitHub](https://github.com/welpo/tabi/issues/new) or [start a discussion](https://github.com/welpo/tabi/discussions).\n\n<details>\n    <summary><b>Table of Contents</b></summary>\n    <!-- toc -->\n</details>\n\n## Settings Hierarchy\n\ntabi has a hierarchy that allows you to customise your site at different levels. The hierarchy (from low to high priority) is as follows:\n\n1. **Global settings**: These are the settings that apply to your entire site. They are set in `config.toml`.\n2. **Section settings**: These are the settings that apply to a section of your site (e.g.`/blog` or `/projects`). They are set in the front matter of the `_index.md` file of the section.\n3. **Parent page settings**: For nested pages (pages within pages), these are the settings from the parent page.\n4. **Page settings**: These are the settings that apply to a single page. They are set in the front matter of the page.\n\nIn all cases, tabi's settings are set in the `[extra]` section.\n\nFor settings which follow this hierarchy, the value set on a page overrides the value for a section, which overrides the global value. In short: the more specific the setting, the higher priority it has, or `page > parent page/section > config.toml`.\n\n---\n\n## Search\n\n| Page | Section | `config.toml` | Follows Hierarchy | Requires JavaScript |\n|:----:|:-------:|:-------------:|:-----------------:|:-------------------:|\n|  ❌  |   ❌    |      ✅       |         ❌        |         ✅          |\n\ntabi supports accessible, local multi-lingual search with [Elasticlunr](http://elasticlunr.com/). To enable it, you need to:\n\n1. Set a `default_language` in `config.toml`.\n2. Set `build_search_index = true`.\n3. Optionally, configure the `[search]`.\n\nHere's an example configuration:\n\n```toml\nbase_url = \"https://example.com\"\ndefault_language = \"en\"\nbuild_search_index = true\n\n[search]\nindex_format = \"elasticlunr_json\" # Or the less efficient \"elasticlunr_javascript\".\ninclude_title = true\ninclude_description = true\ninclude_path = true\ninclude_content = true\n```\n\n**Note**: for Chinese/Japanese search support, you need to use a [custom Zola build](https://github.com/getzola/zola/blob/master/Cargo.toml#L54-L55).\n\n### Considerations for Zola 0.17.X Users\n\nZola 0.17.X doesn't provide access to the `search.index_format` variable ([bug report](https://github.com/getzola/zola/issues/2165)). When using tabi, this variable defaults to the more efficient JSON index. However, due to [another bug](https://github.com/getzola/zola/issues/2193) fixed in 0.18.0, the JSON index for multi-language sites is not generated correctly.\n\nUsers with Zola versions prior to 0.18.0 who want to use the JavaScript index need to set the `index_format` variable in two places:\n\n```toml\n[search]\nindex_format = \"elasticlunr_javascript\"\n\n[extra]\nindex_format = \"elasticlunr_javascript\"\n```\n\nThis ensures tabi loads the right files. We recommend upgrading to Zola 0.18.0 or later for optimal functionality.\n\n### Implementation Details\n\nFor technical details about the search implementation in tabi, including when the index loads, accessibility features, and other specifics, see the [Pull Request #250](https://github.com/welpo/tabi/pull/250).\n\n---\n\n## Multilingual Support\n\ntabi offers comprehensive multilingual support for your Zola site, from setting a default language to adding as many as you wish. Refer to the [multilingual FAQ](@/blog/faq-languages/index.md) for more information.\n\n---\n\n## Appearance\n\n### Home Page\n\nThe [main page](/) of this demo has a header with an image, a title and description:\n\n{{ dual_theme_image(light_src=\"blog/mastering-tabi-settings/img/header_light.webp\", dark_src=\"blog/mastering-tabi-settings/img/header_dark.webp\", alt=\"Main page header\") }}\n\n#### Heading\n\nTo set the image and title, you can use the `header` variable in the front matter of the section's `_index.md` file. For example:\n\n```toml\n[extra]\nheader = {title = \"Hello! I'm tabi~\", img = \"img/main.webp\", img_alt = \"Óscar Fernández, the theme's author\" }\n```\n\nThe description is regular Markdown content, set outside the front matter.\n\n#### Listing Recent Posts\n\nTo show posts on your main page, you first need to decide where these posts will be served from: the root path (`/`) or a subdirectory (e.g., `/blog`).\n\n**Option A: Serve posts from the root path (`/`)**\n\nSet `paginate_by` in the front matter of your `content/_index.md` file:\n\n```toml\ntitle = \"Latest posts\"\nsort_by = \"date\"\npaginate_by = 5  # Show 5 posts per page.\n\n[extra]\nheader = {title = \"Hello! I'm tabi~\", img = \"img/main.webp\", img_alt = \"Your Name\" }\n```\n\n{{ admonition(type=\"note\", text=\"The `paginate_by` setting goes in the main front matter, not in the `[extra]` section.\") }}\n\n**Option B: Serve posts from a subdirectory (e.g., `/blog`)**\n\nUse `section_path` in the `[extra]` section of your `content/_index.md` file:\n\n```toml\ntitle = \"Latest posts\"\nsort_by = \"date\"\n# Do not set `paginate_by` here.\n\n[extra]\nheader = {title = \"Hello! I'm tabi~\", img = \"img/main.webp\", img_alt = \"Your Name\" }\nsection_path = \"blog/_index.md\"  # Where to find your posts.\nmax_posts = 5  # Show up to 5 posts on the main page.\n```\n\n{{ admonition(type=\"warning\", text=\"Do not set both `paginate_by` and `section_path`. These settings are mutually exclusive and using both may result in no posts being displayed.\") }}\n\nAdditional notes:\n\n- The `title` in the front matter sets the header that appears above the posts.\n- Use the full path to the section's `_index.md` file for `section_path`. Using `section_path = \"blog/\"` will not work.\n\n##### Pinning Posts\n\nYou can pin posts to keep them at the top of the main page listing. In this demo, this post is pinned, so it appears first with a \"pinned\" icon and label:\n\n{{ dual_theme_image(light_src=\"blog/mastering-tabi-settings/img/pinned_post_light.webp\", dark_src=\"blog/mastering-tabi-settings/img/pinned_post_dark.webp\", alt=\"Pinned post\", full_width=true) }}\n\nPinned posts are shown first, maintaining their relative order of the section's `sort_by`, followed by regular posts.\n\nTo pin a post, add the following to its front matter:\n\n```toml\n[extra]\npinned = true\n```\n\n{{ admonition(type=\"info\", text=\"This setting only affects your site's main pages (like `/`, `/es/`, `/fr/`). Other sections like `blog/`, `tags/`, or `archive/` show posts in their normal order.\") }}\n\n{{ admonition(type=\"warning\", text='When using pagination (`paginate_by`), pinned posts may appear twice: once on top of page 1, and again in their normal chronological position on subsequent pages.') }}\n\n##### Display the Date of Posts in Listing\n\nBy default, when listing posts, the date of post creation is shown. You can configure which date(s) to display using the `post_listing_date` option. Available settings:\n\n- `date`: Show only the original date of the post (default).\n- `updated`: Show only the last updated date of the post. If there is no last updated date, it shows the original date.\n- `both`: Show both the original date and the last updated date.\n\n```toml\npost_listing_date = \"date\"\n```\n\n{% admonition(type=\"tip\") %}\nThis setting follows the hierarchy: you can set a global value in `config.toml` or override it for specific sections in their `_index.md` file. In both cases, add it to the `[extra]` section.\n{% end %}\n\n#### Listing Projects\n\nYou can showcase a selection of projects on your main page. To do this, you'll need to set up the `projects` directory first.\n\nOnce that's done, you configure the path to the projects in the `[extra]` section of your `_index.md` file:\n\n```toml\n[extra]\nprojects_path = \"projects/_index.md\"\n```\n\nBy default, this will show the 3 projects with the highest priority (smallest weight; same sorting as Projects page). To show more or fewer projects, you can set `max_projects` in the `[extra]` section.\n\nBy default, the featured projects will be shown after the posts. If you want to show the projects before the posts, set `show_projects_first = true`.\n\n### Light and Dark Mode Switcher\n\n| Page | Section | `config.toml` | Follows Hierarchy | Requires JavaScript |\n|:----:|:-------:|:-------------:|:-----------------:|:-------------------:|\n|  ❌  |   ❌    |      ✅       |         ❌        |         ✅          |\n\nThe light and dark mode switcher (the moon/sun icon on the top right) can be enabled by setting `theme_switcher = true` in  `config.toml`.\n\n### Default (Light/Dark) Mode\n\n| Page | Section | `config.toml` | Follows Hierarchy | Requires JavaScript |\n|:----:|:-------:|:-------------:|:-----------------:|:-------------------:|\n|  ❌  |   ❌    |      ✅       |         ❌        |         ❌          |\n\nThe default theme can be specified with the `default_theme` variable, which accepts either `\"dark\"` or `\"light\"`. If you don't set it, the default theme will be the one set in the user's browser.\n\n### Custom Skins\n\n| Page | Section | `config.toml` | Follows Hierarchy | Requires JavaScript |\n|:----:|:-------:|:-------------:|:-----------------:|:-------------------:|\n|  ❌  |   ❌    |      ✅       |         ❌        |         ❌          |\n\ntabi's skins change the main colour of the site. You can set the skin in `config.toml` with `skin = \"skin_name\"`. For example, `skin = \"lavender\"` looks like this (click to switch between light and dark mode):\n\n{{ image_toggler(default_src=\"blog/customise-tabi/skins/lavender_light.webp\", toggled_src=\"blog/customise-tabi/skins/lavender_dark.webp\", default_alt=\"lavender skin in light mode\", toggled_alt=\"lavender skin in dark mode\", full_width=true) }}\n\nExplore the available skins and learn how to create your own reading [the documentation](@/blog/customise-tabi/index.md#skins).\n\n### Sans-serif Font\n\n| Page | Section | `config.toml` | Follows Hierarchy | Requires JavaScript |\n|:----:|:-------:|:-------------:|:-----------------:|:-------------------:|\n|  ❌  |   ❌    |      ✅       |         ❌        |         ❌          |\n\ntabi uses a serif font for article paragraphs (the one you're seeing now). You can switch to using a sans-serif font (the one on the headers/menu) throughout your entire site by setting `override_serif_with_sans = true` in your `config.toml`.\n\nClick on the image below to compare the two looks:\n\n{{ image_toggler(default_src=\"blog/mastering-tabi-settings/img/serif.webp\", toggled_src=\"blog/mastering-tabi-settings/img/sans-serif.webp\", default_alt=\"Serif font\", toggled_alt=\"Sans-serif font\", full_width=true) }}\n\n### External Link Indicator\n\n| Page | Section | `config.toml` | Follows Hierarchy | Requires JavaScript |\n|:----:|:-------:|:-------------:|:-----------------:|:-------------------:|\n|  ❌  |   ❌    |      ✅       |         ❌        |         ❌          |\n\n{{ admonition(type=\"info\", text=\"Requires Zola 0.20.0 or later.\") }}\n\nIf you'd like to add an icon to external links, configure the `[markdown]` (not `[extra]`) section in your `config.toml`:\n\n```toml\n[markdown]\nexternal_links_class = \"external\"\n```\n\nThis will add a small icon next to external links:\n\n{{ dual_theme_image(light_src=\"blog/mastering-tabi-settings/img/external_link_light.webp\", dark_src=\"blog/mastering-tabi-settings/img/external_link_dark.webp\", alt=\"External link icon\", full_width=true) }}\n\n### Custom CSS\n\n| Page | Section | `config.toml` | Follows Hierarchy | Requires JavaScript |\n|:----:|:-------:|:-------------:|:-----------------:|:-------------------:|\n|  ✅  |   ❌    |      ✅       |         ❌        |         ❌          |\n\nYou can load custom CSS for the entire site or on specific pages with `stylesheets`, which takes a list of paths to CSS files. For example:\n\n```toml\nstylesheets = [\"css/custom.css\", \"css/another.css\"]\n```\n\n### Browser Theme Colour\n\n| Page | Section | `config.toml` | Follows Hierarchy | Requires JavaScript |\n|:----:|:-------:|:-------------:|:-----------------:|:-------------------:|\n|  ❌  |   ❌    |      ✅       |         ❌        |         ❌          |\n\nThe browser theme colour is the colour that appears in the browser's tab bar:\n\n{{ dual_theme_image(light_src=\"blog/mastering-tabi-settings/img/browser_theme_color_light.webp\", dark_src=\"blog/mastering-tabi-settings/img/browser_theme_color_dark.webp\" alt=\"tabi with a coloured browser theme\") }}\n\nYou can set it in `config.toml` like `browser_theme_color = \"#087e96\"`. If you'd like different colours for dark/light mode, you can set an array of colours with `browser_theme_color = [\"#ffffff\", \"#000000\"]`. The first colour will be used for light mode, the second for dark mode.\n\nThis variable accepts any valid CSS colour, so you can use keywords (e.g. `blue`), hex codes (e.g. `#087e96`) or RGB/HSL values (e.g. `rgb(8, 126, 150)`).\n\n### Compact Tags\n\n| Page | Section | `config.toml` | Follows Hierarchy | Requires JavaScript |\n|:----:|:-------:|:-------------:|:-----------------:|:-------------------:|\n|  ❌  |   ❌    |      ✅       |         ❌        |         ❌          |\n\nBy default, the [tags page](/tags) displays tags as:\n\n[TagName](#) — n post[s]\n\nSetting `compact_tags = true` will display them as:\n\n[TagName](#) <sup>n</sup>\n\n### Tags Sorting\n\n| Page | Section | `config.toml` | Follows Hierarchy | Requires JavaScript |\n|:----:|:-------:|:-------------:|:-----------------:|:-------------------:|\n|  ❌  |   ❌    |      ✅       |         ❌        |         ❌          |\n\nBy default, the [tags page](/tags) sorts tags alphabetically, given the default setting of `tag_sorting = \"name\"`.\n\nSetting `tag_sorting = \"frequency\"` will sort them by number-of-posts (descending).\n\n---\n\n## Series\n\nFor a detailed explanation of the series feature, see the [series documentation](@/blog/series/index.md).\n\n### Jump to posts link\n\n| Page | Section | `config.toml` | Follows Hierarchy | Requires JavaScript |\n|:----:|:-------:|:-------------:|:-----------------:|:-------------------:|\n|  ❌  |     ✅  |      ✅       |           ✅      |         ❌          |\n\nBy default, a \"Jump to posts\" link automatically appears next to the series title when a series has a content over 2000 characters:\n\n{{ dual_theme_image(light_src=\"blog/series/img/jump_to_series_posts_light.webp\", dark_src=\"blog/series/img/jump_to_series_posts_dark.webp\" alt=\"jump to series posts link\", full_width=true) }}\n\nSet `show_jump_to_posts = true` to force the feature on and `show_jump_to_posts = false` to force it off.\n\n### Series pages indexation\n\n| Page | Section | `config.toml` | Follows Hierarchy | Requires JavaScript |\n|:----:|:-------:|:-------------:|:-----------------:|:-------------------:|\n|  ❌  |     ✅  |      ✅       |           ✅      |         ❌          |\n\nBy default, series page are indexed (using a 1-based indexing) as per the series section `sort_by`.\n\nSet `post_listing_index_reversed = true` to reverse this index.\n\n---\n\n## Git Repository Integration\n\n| Page  | Section | `config.toml` | Follows Hierarchy | Requires JavaScript |\n|:-----:|:-------:|:-------------:|:-----------------:|:-------------------:|\n|  ❓   |   ❓    |      ✅       |         ❓        |         ❌          |\n\n❓: `show_remote_source` does follow [the hierarchy](#settings-hierarchy) and can be set on a page, section or globally. The rest of the settings can only be set in `config.toml`.\n\nThese settings allow you to link your tabi website with a public Git repository in GitHub, GitLab, Gitea or Codeberg. Example settings:\n\n```toml\nremote_repository_url = \"https://github.com/welpo/tabi\"\nremote_repository_git_platform = \"auto\"\nremote_repository_branch = \"main\"\nshow_remote_changes = true\nshow_remote_source = true\n```\n\nThis enables two features:\n\n1. `show_remote_source = true` adds a link to the source code of your site (your `remote_repository_url`) will be displayed on the footer:\n\n{{ dual_theme_image(light_src=\"blog/mastering-tabi-settings/img/site_source_light.webp\", dark_src=\"blog/mastering-tabi-settings/img/site_source_dark.webp\" alt=\"Page footer, showing a 'Site source' link\") }}\n\n1. `show_remote_changes = true` adds a \"See changes ↗\" link to the commit history of updated posts, next to the last updated date [^1]:\n\n{{ dual_theme_image(light_src=\"blog/mastering-tabi-settings/img/see_changes_light.webp\", dark_src=\"blog/mastering-tabi-settings/img/see_changes_dark.webp\" alt=\"Post title and metadata, showing a 'See changes' link\") }}\n\nClicking on this link will take you to the commit history of the post, where you can see the changes made to it:\n\n{{ dual_theme_image(light_src=\"blog/mastering-tabi-settings/img/commit_history_light.webp\", dark_src=\"blog/mastering-tabi-settings/img/commit_history_dark.webp\" alt=\"Commit history of a post\", full_width=true) }}\n\n---\n\n## Pages\n\n### Projects\n\ntabi has a built-in projects (cards) template. To enable it, you can create a directory in `content/projects/`. There, you can create a `_index.md` file with the following front matter:\n\n```toml\ntitle = \"Projects\"\nsort_by = \"weight\"\ntemplate = \"cards.html\"\ninsert_anchor_links = \"left\"\n\n[extra]\nshow_reading_time = false\nquick_navigation_buttons = true\n```\n\n- The `title` is the title of the page.\n- `sort_by` determines how the projects are sorted. You can sort by \"date\", \"update_date\", \"title\", \"title_bytes\", \"weight\", \"slug\" or \"none\".\n- `template = \"cards.html\"` sets the template to render the projects page.\n- `insert_anchor_links = \"left\"` adds anchor links to headers.\n- `show_reading_time = false` hides the [reading time](#reading-time).\n- `quick_navigation_buttons = true` shows the [quick navigation buttons](#quick-navigation-buttons) are shown.\n\nAlongside the `_index.md` file, you can create a file for each project. For example, this is the front matter for the [tabi project page](@/projects/tabi/index.md):\n\n```toml\ntitle = \"tabi\"\ndescription = \"A feature-rich modern Zola theme with first-class multi-language support.\"\nweight = 1\n\n[extra]\nlocal_image = \"img/tabi.webp\"\ninvertible_image = false\n```\n\n- `title` is the title of the project.\n- `description` is the description of the project.\n- `weight` determines the order in which the projects are shown. The lower the weight, the higher the project will appear.\n- `local_image` is the path to the image of the project. This image is shown on the projects page.\n- `local_image_dark` is an optional dark mode variant. Requires `local_image` to be set.\n- `remote_image` is a URL to an external image, as an alternative to `local_image`.\n- `remote_image_dark` is an optional dark mode variant. Requires `remote_image` to be set.\n- `invertible_image` inverts the image colours in dark mode. Useful for black and white logos/icons.\n\nWhen a user clicks on the image or title of a project, they will be taken to the project's page. If you'd rather have users go to an external link, you can set `link_to = \"https://example.com` in the `[extra]` section of the project's `.md` file.\n\nThe individual project's page is rendered with the default template, unless you set another one, e.g. `template = \"info-page.html\"`.\n\n#### Filtering Projects\n\nIf you add tags to your projects, you will see a tag filter:\n\n{{ dual_theme_image(light_src=\"blog/mastering-tabi-settings/img/projects_tag_filter_light.webp\", dark_src=\"blog/mastering-tabi-settings/img/projects_tag_filter_dark.webp\", alt=\"Projects page with tag filter\", full_width=true) }}\n\nThe tag filtering system uses progressive enhancement:\n\n- Without JavaScript: Tags link directly to dedicated tag pages (e.g. `/tags/tag-name`)\n- With JavaScript: Instant filtering, URL syncing (#tag-name), and keyboard navigation\n\nTo disable this feature, set `enable_cards_tag_filtering = false` in the `[extra]` section of the `projects/_index.md` file or in `config.toml`.\n\n{% admonition(type=\"tip\") %}\n\nTo filter projects by tags, you need to set tags in the front matter of each project. For example:\n\n```toml\ntitle = \"project name\"\nweight = 40\n\n[taxonomies]\ntags = [\"tag one\", \"tag 2\", \"third tag\"]\n```\n\n{% end %}\n\n### Archive\n\nAdding an archive page is similar to adding a projects page. You can create a directory in `content/archive/`. There, you can create a `_index.md` file with the following front matter:\n\n```toml\ntitle = \"Archive\"\ntemplate = \"archive.html\"\n```\n\nBy default, the archive will list posts located in `blog/`. To customise this, you can modify the `[extra]` section of the `_index.md` file:\n\n- **For a single source path**: Set `section_path = \"your-path/\"` to list posts from a specific directory. Make sure to include the trailing slash.\n\n- **For multiple source paths**: If you want to aggregate posts from various directories, `section_path` can be specified as a list of paths. For example:\n\n  ```toml\n  [extra]\n  section_path = [\"blog/\", \"notes/\", \"path-three/\"]\n  ```\n\nThe archive displays posts in reverse chronological order (newest first). You can reverse this order by setting `archive_reverse = true` in the `[extra]` section:\n\n```toml\n[extra]\narchive_reverse = true  # displays oldest posts first.\n```\n\n{{ admonition(type=\"note\", text=\"The Archive page will only list posts that have a date in their front matter.\") }}\n\n### Tags\n\ntabi has built-in support for tags. To enable them, simply add the taxonomy to your `config.toml`:\n\n```toml\ntaxonomies = [{name = \"tags\", feed = true}]\n```\n\nYou can then add tags to your posts by adding them to the `tags` array in the front matter of your post. For example:\n\n```toml,hl_lines=05-06\ntitle = \"Bears, Beets, Battlestar Galactica: The Dwight Schrute Guide to Life\"\ndate = 2007-04-26\ndescription = \"Lessons learned from beet farming and paper sales.\"\n\n[taxonomies]\ntags = [\"personal\", \"beets\"]\n```\n\n### About Page\n\nIf you'd like to have a non-article page for an \"About\" section, a \"Contact\" or \"Copyright\" page, etc., you can use the `info-page.html` template.\n\nFirst, create a directory inside `content/` with any name you like. For example, `content/pages/`. Then, create a `_index.md` file inside that directory. The file should look like this:\n\n```markdown\n+++\nrender = false\ninsert_anchor_links = \"left\"\n+++\n```\n\n- `render = false` tells Zola not to render the section.\n- `insert_anchor_links = \"left\"` adds anchor links to headers. This is optional.\n\nInside the directory, you can create any number of `.md` files.\n\nIn this demo, the [about](about/) page uses the `info-page.html` template. The front matter is as follows:\n\n```toml\ntitle = \"About\"\ntemplate = \"info-page.html\"\npath = \"about\"\n```\n\nNotice how the `path` is set to `about`. Zola will place the page at `$base_url/about/`. If you'd like to have the page available at `/contact/`, you'd set `path = \"contact\"`.\n\nThe `info-page.html` template can also be used to create landing pages at the path root (`\"/\"`). To do that, the `content/_index.md` file should look like this:\n\n```markdown\n+++\ntitle = \"Landing Page Title\"\ntemplate = \"info-page.html\"\n+++\n\nPlace your landing page Markdown content here.\n```\n\n---\n\n## SEO\n\ntabi takes care of most of the SEO for you (like Open Graph protocol tags, description, color-scheme…), but there are a few things you can customise.\n\n### Favicon\n\n| Page | Section | `config.toml` | Follows Hierarchy | Requires JavaScript |\n|:----:|:-------:|:-------------:|:-----------------:|:-------------------:|\n|  ❌  |   ❌    |      ✅       |         ❌        |         ❌          |\n\nThe favicon is the small icon that appears in the browser tab. You can set it in `config.toml` with `favicon = \"img/favicon.png\"`.\n\n### Emoji Favicon\n\n| Page | Section | `config.toml` | Follows Hierarchy | Requires JavaScript |\n|:----:|:-------:|:-------------:|:-----------------:|:-------------------:|\n|  ❌  |   ❌    |      ✅       |         ❌        |         ❌          |\n\nYou can also set an emoji as your favicon with `favicon_emoji`. For example, `favicon_emoji = \"👾\"`.\n\nNote: Some browsers don't support emoji favicons. See the compatibility table in [caniuse](https://caniuse.com/link-icon-svg).\n\n### Canonical URL\n\n| Page | Section | `config.toml` | Follows Hierarchy | Requires JavaScript |\n|:----:|:-------:|:-------------:|:-----------------:|:-------------------:|\n|  ✅  |   ✅    |      ✅       |         ❌        |         ❌          |\n\nThe canonical URL is a way to indicate to search engines what the preferred URL is for your website content. This is useful for SEO and avoiding duplicate content issues.\n\nBy default, the canonical URL is the URL of the page you're on. However, you can override this by setting `canonical_url` in the front matter of your page or section.\n\nIf you have a site with an identical structure and matching content, you can set `base_canonical_url` in your `config.toml`. The canonical URL will be crafted by replacing the `$base_url` of the current URL with the `$base_canonical_url` you set.\n\nFor example, if you set `base_canonical_url = \"https://example.com\"`, the canonical URL of the page `$base_url/blog/post1` will be `https://example.com/blog/post1`. This is useful if you have a site with multiple domains that share the same content.\n\n**Note**: to ensure that the canonical URL is correct, it's probably best to set `canonical_url` individually for each page.\n\n### Social media cards\n\n| Page | Section | `config.toml` | Follows Hierarchy | Requires JavaScript |\n|:----:|:-------:|:-------------:|:-----------------:|:-------------------:|\n|  ✅  |   ✅    |      ✅       |         ✅        |         ❌          |\n\nSocial media cards are the images that are displayed when you share a link on social media:\n\n{{ dimmable_image(src=\"img/with_social_media_card.webp\", alt=\"A screenshot of WhatsApp showing a link with a social media card\") }}\n\nYou can set the social media image with `social_media_card = \"img/social_media_card.png\"`.\n\nYou can specify both relative and absolute paths.\n\n- **Relative Path**: Place the image in the same folder as your blog post and specify its name. For example, `social_media_card = \"relative_image.png\"`.\n\n- **Absolute Path**: Put the image in a specific folder and specify the path from the root. For example, `social_media_card = \"/img/absolute_image.png\"`.\n\nIf both relative and absolute paths are valid, the relative path will take precedence.\n\nSince it follows the [hierarchy](#settings-hierarchy), if it's not set on a page, but is set on a section, the section's image will be used. If it's not set on a page or section, but is set in `config.toml`, the global image will be used.\n\n{{ admonition(type=\"tip\", title=\"PROTIP\", text=\"Automate their creation with a [script](https://github.com/welpo/osc.garden/blob/main/static/code/social-cards-zola): [Automating Link Previews for Zola Sites](https://osc.garden/blog/automating-social-media-cards-zola/).\") }}\n\n### Fediverse Creator\n\n| Page | Section | `config.toml` | Follows Hierarchy | Requires JavaScript |\n|:----:|:-------:|:-------------:|:-----------------:|:-------------------:|\n|  ❌  |   ❌    |      ✅       |         ❌        |         ❌          |\n\nYou can highlight your fediverse profile in Mastodon link previews by setting the `fediverse_creator` variable in your `config.toml`. For example, for @username@example.com, use:\n\n```toml\nfediverse_creator = { handle = \"username\", domain = \"example.com\" }\n```\n\nThis adds metadata to your HTML, allowing Mastodon to display the author's fediverse profile when your content is shared.\n\n---\n\n## Navigation\n\n### Navigation Bar\n\n| Page | Section | `config.toml` | Follows Hierarchy | Requires JavaScript |\n|:----:|:-------:|:-------------:|:-----------------:|:-------------------:|\n|  ❌  |   ❌    |      ✅       |         ❌        |         ❌          |\n\nThe navigation bar is the bar at the top of the page that contains the site title and the navigation menu. You can customise which items appear by setting `menu` in `config.toml`.\n\nThe menu supports both relative URLs for internal pages and absolute URLs for external links. For example:\n\n```toml\nmenu = [\n    { name = \"blog\", url = \"blog\", trailing_slash = true },\n    { name = \"archive\", url = \"archive\", trailing_slash = true },\n    { name = \"tags\", url = \"tags\", trailing_slash = true },\n    { name = \"projects\", url = \"projects\", trailing_slash = true },\n    { name = \"about\", url = \"about\", trailing_slash = true },\n    { name = \"github\", url = \"https://github.com/welpo/tabi\", trailing_slash = false },\n]\n```\n\n### Quick Navigation Buttons\n\n| Page | Section | `config.toml` | Follows Hierarchy | Requires JavaScript |\n|:----:|:-------:|:-------------:|:-----------------:|:-------------------:|\n|  ✅  |   ✅    |      ✅       |         ✅        |         ❌          |\n\nQuick navigation buttons are the buttons that appear on the bottom right of the screen. You should see them on this page, if you're not on mobile. They look like this:\n\n{{ dual_theme_image(light_src=\"blog/mastering-tabi-settings/img/quick_navigation_buttons_light.webp\", dark_src=\"blog/mastering-tabi-settings/img/quick_navigation_buttons_dark.webp\" alt=\"Quick navigation buttons\") }}\n\nThe buttons allow you to quickly navigate through an expandable mini-table of contents, to the comment section (if enabled), as well as to the top of the page.\n\nTo enable them, set `quick_navigation_buttons = true`.\n\n### Table of Contents\n\n| Page | Section | `config.toml` | Follows Hierarchy | Requires JavaScript |\n|:----:|:-------:|:-------------:|:-----------------:|:-------------------:|\n|  ✅  |   ✅    |      ✅       |         ✅        |         ❌          |\n\nEnable the table of contents right below the post's title and metadata with `toc = true`.\n\nRead more about the table of contents and how to customise it by reading [the docs](@/blog/toc/index.md).\n\n### Previous and Next Article Links\n\n| Page | Section | `config.toml` | Follows Hierarchy | Requires JavaScript |\n|:----:|:-------:|:-------------:|:-----------------:|:-------------------:|\n|  ✅  |   ✅    |      ✅       |         ✅        |         ❌          |\n\nDisplays links to the previous and next articles at the bottom of posts. It looks like this:\n\n{{ dual_theme_image(light_src=\"blog/mastering-tabi-settings/img/show_previous_next_article_links_light.webp\", dark_src=\"blog/mastering-tabi-settings/img/show_previous_next_article_links_dark.webp\" alt=\"Previous and next article links\", full_width=true) }}\n\nTo activate this feature, set `show_previous_next_article_links = true` and ensure your section has a `sort_by` value (e.g. `sort_by = \"date\"`).\n\nBy default, next articles will be on the left side of the page and previous articles will be on the right side.\nTo reverse the order (next articles on the right and previous articles on the left), set `invert_previous_next_article_links = true`.\n\nBy default, this navigation section will have the full width of the site (same as the navigation bar at the top).\nTo make it narrower, matching the article width, set `previous_next_article_links_full_width = false`.\n\nAll of these settings follow the hierarchy.\n\n### Footnote Backlinks\n\n{{ admonition(type=\"warning\", title=\"DEPRECATION WARNING\", text=\"Zola v0.19.0 and later can do this natively. Set `bottom_footnotes = true` in your config's `[markdown]` section.\") }}\n\n| Page | Section | `config.toml` | Follows Hierarchy | Requires JavaScript |\n|:----:|:-------:|:-------------:|:-----------------:|:-------------------:|\n|  ✅  |   ✅    |      ✅       |         ✅        |         ✅          |\n\nSetting `footnote_backlinks = true` will add backlinks to the footnotes of your posts, like this:\n\n{{ dual_theme_image(light_src=\"blog/mastering-tabi-settings/img/footnote_backlinks_light.webp\", dark_src=\"blog/mastering-tabi-settings/img/footnote_backlinks_dark.webp\" alt=\"Footnote backlinks\", full_width=true) }}\n\nWhen you click on a backlink (the arrow ↩), it will take you back to the text where the footnote was referenced.\n\n---\n\n## Usability\n\n### Copy Button on Code Blocks\n\n| Page | Section | `config.toml` | Follows Hierarchy | Requires JavaScript |\n|:----:|:-------:|:-------------:|:-----------------:|:-------------------:|\n|  ✅  |   ✅    |      ✅       |         ✅        |         ✅          |\n\nSetting `copy_button = true` will add a small copy button to the top right of code blocks, like this:\n\n{{ dual_theme_image(light_src=\"blog/mastering-tabi-settings/img/copy_button_on_code_blocks_light.webp\", dark_src=\"blog/mastering-tabi-settings/img/copy_button_on_code_blocks_dark.webp\" alt=\"Copy button on code blocks\", full_width=true) }}\n\n### Clickable Code Block Names\n\n| Page | Section | `config.toml` | Follows Hierarchy | Requires JavaScript |\n|:----:|:-------:|:-------------:|:-----------------:|:-------------------:|\n|  ✅  |   ✅    |      ✅       |         ✅        |         ✅          |\n\nSetting `code_block_name_links = true` enables URLs in code block names to become clickable links. Check out the [documentation](@/blog/shortcodes/index.md#show-source-or-path) for examples and usage.\n\n### Force Code Blocks LTR\n\n| Page | Section | `config.toml` | Follows Hierarchy | Requires JavaScript |\n|:----:|:-------:|:-------------:|:-----------------:|:-------------------:|\n|  ✅  |   ✅    |      ✅       |         ✅        |         ❌          |\n\nBy default, code blocks are rendered left-to-right, regardless of the overall text direction. Set `force_codeblock_ltr = false` to allow code blocks to follow the document's text direction. Useful for RTL languages needing RTL code blocks.\n\n### KaTeX Support\n\n| Page | Section | `config.toml` | Follows Hierarchy | Requires JavaScript |\n|:----:|:-------:|:-------------:|:-----------------:|:-------------------:|\n|  ✅  |   ✅    |      ✅       |         ✅        |         ✅          |\n\nKaTeX is a fast, easy-to-use JavaScript library for TeX math rendering on the web. You can enable it with `katex = true`. See what it looks like in tabi [here](/blog/markdown/#katex).\n\n### Mermaid Support\n\n| Page | Section | `config.toml` | Follows Hierarchy | Requires JavaScript |\n|:----:|:-------:|:-------------:|:-----------------:|:-------------------:|\n|  ✅  |    ✅    |      ✅       |         ✅         |         ✅          |\n\n[Mermaid](https://github.com/mermaid-js/mermaid) is a JavaScript-based diagramming and charting tool. You can enable it with `mermaid = true`.\n\nBy default, the Mermaid library is served locally. If you prefer to use a CDN, set `serve_local_mermaid = false` in `config.toml`. Using a CDN will serve the latest version of Mermaid; the local option will serve the version bundled with tabi.\n\nSee the [Mermaid documentation](@/blog/shortcodes/index.md#mermaid-diagrams) for usage instructions and examples.\n\n### Custom Font Subset\n\n| Page | Section | `config.toml` | Follows Hierarchy | Requires JavaScript |\n|:----:|:-------:|:-------------:|:-----------------:|:-------------------:|\n|  ❌  |   ❌    |      ✅       |         ❌        |         ❌          |\n\nCustom fonts cause flashing text in Firefox. To amend this, tabi loads a subset of glyphs for the header. Since this (slightly) increases the initial load time, it's a good idea to try and minimise the size of this subset, or disable it completely if you're not using a custom font in your skin.\n\nYou can create a custom subset tailored to your site, save it as `static/custom_subset.css`, and have it load with `custom_subset = true`. Disabling the subset can be done with `enable_subset = false`.\n\nFor more information, including instructions on how to create a custom subset, see the [docs](@/blog/custom-font-subset/index.md).\n\n### Full Content in Feed\n\n| Page | Section | `config.toml` | Follows Hierarchy | Requires JavaScript |\n|:----:|:-------:|:-------------:|:-----------------:|:-------------------:|\n|  ❌  |   ❌    |      ✅       |         ❌        |         ❌          |\n\nBy default, the Atom feed only contains the summary/description of your posts. You can include the entire posts' content by setting `full_content_in_feed = true` in `config.toml`.\n\n### Hiding Content from Feed\n\n| Page | Section | `config.toml` | Follows Hierarchy | Requires JavaScript |\n|:----:|:-------:|:-------------:|:-----------------:|:-------------------:|\n|  ✅  |   ✅    |      ✅       |         ✅        |         ❌          |\n\nYou can control how content appears in your feeds using two settings:\n\n1. `hide_from_feed = true`: Hides content from all feeds (main, section, and tag feeds)\n2. `hide_from_main_feed = true`: Hides content only from the main feed while keeping it visible in section and tag feeds\n\n### Comments {#adding-comments}\n\n| Page | Section | `config.toml` | Follows Hierarchy | Requires JavaScript |\n|:----:|:-------:|:-------------:|:-----------------:|:-------------------:|\n|  ✅  |   ❌    |      ✅       |         ❌        |         ✅          |\n\nTo enable comments on an individual page, set the name of the system you want to enable to `true` in the front matter. For example, `utterances = true`.\n\nTo enable a system globally (on all pages), set `enabled_for_all_posts = true` in the correct section of your `config.toml` (e.g. inside `[extra.giscus]`).\n\nIf you have enabled a system globally, but want to disable it on a specific page, set the name of the system to `false` in the front matter of that page. For example, `utterances = false`.\n\nRead [the docs](@/blog/comments/index.md) for more information on the available systems and their setup.\n\n### iine Like Buttons {#iine}\n\n| Page | Section | `config.toml` | Follows Hierarchy | Requires JavaScript |\n|:----:|:-------:|:-------------:|:-----------------:|:-------------------:|\n|  ✅  |   ✅    |      ✅       |         ✅        |         ❌          |\n\ntabi supports [iine](https://iine.to/) like buttons for anonymous appreciation of your content. These privacy-focused buttons work without JavaScript and don't track users.\n\nTo enable iine buttons globally:\n\n```toml\n[extra]\niine = true\n```\n\nYou can customise the icon used for the buttons (follows the hierarchy):\n\n```toml\n[extra]\niine_icon = \"thumbs_up\"  # Options: \"heart\", \"thumbs_up\", \"upvote\", or any emoji\n```\n\nFor multilingual sites, you can unify like counts across language versions of the same content (config-only setting; true by default):\n\n```toml\n[extra]\niine_unified_languages = true  # Likes on /es/blog/hello/ count towards /blog/hello/\n```\n\nYou can also enable iine buttons on individual pages or sections by setting `iine = true` in their front matter, or override the icon with `iine_icon = \"🚀\"`.\n\n### Analytics\n\n| Page | Section | `config.toml` | Follows Hierarchy | Requires JavaScript |\n|:----:|:-------:|:-------------:|:-----------------:|:-------------------:|\n|  ❌  |   ❌    |      ✅       |         ❌        |         ✅          |\n\ntabi supports 3 privacy-friendly analytics systems: [Plausible](https://plausible.io/), [GoatCounter](https://www.goatcounter.com/) and [Umami](https://umami.is/).\n\nYou can set them up in the `[extra.analytics]` section of your `config.toml`.\n\n- `service`: Specifies which analytics service to use. Supported options are `\"goatcounter\"`, `\"umami\"`, and `\"plausible\"`.\n\n- `id`: The unique identifier for your analytics service. This varies based on the service:\n  - For GoatCounter, it's the code chosen during signup. Self-hosted instances of GoatCounter don't require this field.\n  - For Umami, it's the website ID.\n  - For Plausible, it's either:\n    - **New format** (Plausible v3.1.0+): The random script name without the extension (e.g. `\"pa-XXXXXX\"`). Find this in your Plausible dashboard under Settings → Website Details → Script name.\n    - **Legacy format**: Your domain name (e.g. `\"example.com\"`). Use this if you need to send stats to multiple dashboards simultaneously, as the new format doesn't support this feature. See [Plausible's script update guide](https://plausible.io/docs/script-update-guide) for details.\n\n- `self_hosted_url`: Optional. Use this field to specify the URL for self-hosted instances of your chosen analytics service. The base URL differs based on your specific setup. Some examples:\n  - For GoatCounter: `\"https://stats.example.com\"`\n  - For Umami: `\"https://umami.example.com\"`\n  - For Plausible: `\"https://plausible.example.com\"`\n\n- `do_not_track`: (Umami only) Optional. When set to `true`, the generated tracking script will include the `data-do-not-track=\"true\"` attribute, which disables tracking for users whose browsers send a \"Do Not Track\" header.\n\nAn example configuration for non-self-hosted GoatCounter would look like this:\n\n```toml\n[extra.analytics]\nservice = \"goatcounter\"\nid = \"tabi\"\nself_hosted_url = \"\"\n```\n\n---\n\n## Footer\n\n### Social Media Icons\n\n| Page | Section | `config.toml` | Follows Hierarchy | Requires JavaScript |\n|:----:|:-------:|:-------------:|:-----------------:|:-------------------:|\n|  ❌  |   ❌    |      ✅       |         ❌        |         ❌          |\n\nYou can add social media icons to the footer with `socials`, which takes a list of social media objects. For example:\n\n```toml\nsocials = [\n    { name = \"github\", url = \"https://github.com/welpo/\", icon = \"github\" },\n    { name = \"soundcloud\", url = \"https://soundcloud.com/oskerwyld\", icon = \"soundcloud\" },\n    { name = \"instagram\", url = \"https://instagram.com/oskerwyld\", icon = \"instagram\" },\n    { name = \"youtube\", url = \"https://youtube.com/@oskerwyld\", icon = \"youtube\" },\n    { name = \"spotify\", url = \"https://open.spotify.com/artist/5Hv2bYBhMp1lUHFri06xkE\", icon = \"spotify\" },\n]\n```\n\nTo see a list of all the built-in icons, take a look at the [`static/social_icons` directory on GitHub](https://github.com/welpo/tabi/tree/main/static/social_icons).\n\nMissing an icon? If you think it would be a good addition to tabi, feel free to [open an issue](https://github.com/welpo/tabi/issues/new?assignees=&labels=enhancement&projects=&template=feature_request.md&title=) or submit a pull request ([example](https://github.com/welpo/tabi/pull/333)).\n\nTo use a custom icon, you can add it to your site's `static/social_icons` directory. For example, if you add `custom.svg`, you can reference it like this:\n\n```\n{ name = \"custom\", url = \"https://example.com\", icon = \"custom\" }\n```\n\n{{ admonition(type=\"note\", text=\"All social links include the `rel='me'` [attribute](https://developer.mozilla.org/docs/Web/HTML/Attributes/rel/me). This helps search engines and web services verify that the social media accounts are owned by you.\") }}\n\n### Feed Icon\n\n| Page | Section | `config.toml` | Follows Hierarchy | Requires JavaScript |\n|:----:|:-------:|:-------------:|:-----------------:|:-------------------:|\n|  ❌  |   ❌    |      ✅       |         ❌        |         ❌          |\n\nYou can add a link to your RSS/Atom feed to the footer with `feed_icon = true`.\n\nTo use a custom icon, set `feed_icon` to the icon name (e.g. `feed_icon = \"square-rss\"`). The icon must exist in `static/social_icons/` (without the `.svg` extension).\n\nNote for Zola 0.19.X users: when there are two filenames in `feed_filenames`, only the first one will be linked in the footer.\n\n### Footer Menu\n\n| Page | Section | `config.toml` | Follows Hierarchy | Requires JavaScript |\n|:----:|:-------:|:-------------:|:-----------------:|:-------------------:|\n|  ❌  |   ❌    |      ✅       |         ❌        |         ❌          |\n\nYou can add a menu to the footer with `footer_menu`, which takes a list of menu items. For example:\n\n```toml\nfooter_menu = [\n    {url = \"about\", name = \"about\", trailing_slash = true},\n    {url = \"privacy\", name = \"privacy\", trailing_slash = true},\n    {url = \"sitemap.xml\", name = \"sitemap\", trailing_slash = false},\n    {url = \"https://example.com\", name = \"external link\", trailing_slash = true},\n]\n```\n\n### Copyright\n\n| Page | Section | `config.toml` | Follows Hierarchy | Requires JavaScript |\n|:----:|:-------:|:-------------:|:-----------------:|:-------------------:|\n|  ❌  |   ❌    |      ✅       |         ❌        |         ❌          |\n\nTo add a copyright notice to your site, set `copyright`:\n\n```toml\ncopyright = \"© $CURRENT_YEAR Your Name $SEPARATOR Unless otherwise noted, the content in this website is available under the [CC BY-SA 4.0](https://creativecommons.org/licenses/by-sa/4.0/) license.\"\n```\n\nYou can use the following variables:\n\n- `$TITLE` will be replaced by `title` variable set in `config.toml`\n- `$CURRENT_YEAR` will be replaced by the current year\n- `$AUTHOR` will be replaced by the `author` variable\n- `$SEPARATOR` will be replaced by the [`separator` variable](#custom-separator)\n\nMarkdown is rendered. The example above:\n\n{{ dual_theme_image(light_src=\"blog/mastering-tabi-settings/img/copyright_light.webp\", dark_src=\"blog/mastering-tabi-settings/img/copyright_dark.webp\" alt=\"Copyright section\", full_width=true) }}\n\nIf you have a multilingual site and want to set different copyright notices for different languages, you can add the corresponding translation to `copyright_translations.{language_code}` for each language you want to support. The language code must match [tabi's language code](https://welpo.github.io/tabi/blog/faq-languages/#what-are-these-two-letter-codes). For example, for Spanish:\n\n```toml\ncopyright_translations.es = \"© $CURRENT_YEAR $AUTHOR $SEPARATOR A menos que se indique lo contrario, el contenido de esta web está disponible bajo la licencia [CC BY-SA 4.0](https://creativecommons.org/licenses/by-sa/4.0/).\"\n```\n\n---\n\n## Metadata\n\n### Show author\n\n| Page | Section | `config.toml` | Follows Hierarchy | Requires JavaScript |\n|:----:|:-------:|:-------------:|:-----------------:|:-------------------:|\n|  ✅  |   ✅    |      ✅       |         ✅        |         ❌          |\n\nTo show the author(s) below the post title, set `show_author = true`.\n\nThis will display the authors set on `authors = []` in the front matter of the post. If this is not available, it will fall back to `author = \"\"`in `config.toml`.\n\n### Reading Time\n\n| Page | Section | `config.toml` | Follows Hierarchy | Requires JavaScript |\n|:----:|:-------:|:-------------:|:-----------------:|:-------------------:|\n|  ✅  |   ✅    |      ✅       |         ✅        |         ❌          |\n\nYou can enable or hide the reading time of a post with `show_reading_time`. If you set it to `true`, it will be displayed in the post's metadata, like this:\n\n{{ dual_theme_image(light_src=\"blog/mastering-tabi-settings/img/see_changes_light.webp\", dark_src=\"blog/mastering-tabi-settings/img/see_changes_dark.webp\" alt=\"Post title and metadata, showing a 'See changes' link\") }}\n\nSince it follows [the hierarchy](#settings-hierarchy), you can enable it or hide it for specific pages or sections. For example, this demo sets `show_reading_time = false` in the [projects](https://welpo.github.io/tabi/projects/) section's [`_index.md`](https://github.com/welpo/tabi/blob/main/content/projects/_index.md?plain=1), so their individual posts don't show the reading time.\n\n### Show Date\n\n| Page | Section | `config.toml` | Follows Hierarchy | Requires JavaScript |\n|:----:|:-------:|:-------------:|:-----------------:|:-------------------:|\n|  ✅  |   ✅    |      ✅       |         ✅        |         ❌          |\n\nBy default, the date is shown below the post title. You can hide it with `show_date = false`. This setting follows [the hierarchy](#settings-hierarchy).\n\n### Date Format\n\n| Page | Section | `config.toml` | Follows Hierarchy | Requires JavaScript |\n|:----:|:-------:|:-------------:|:-----------------:|:-------------------:|\n|  ❌  |   ❌    |      ✅       |         ❌        |         ❌          |\n\ntabi has three date formats: `long_date_format`, `short_date_format` and `archive_date_format`. The short format is used in a post's metadata, while the long format is used when listing posts (i.e. on the [blog section](@/blog/_index.md) or the [main page](@/_index.md)). The archive format is used to display day and month on the archive page.\n\nThe default is \"6th July 2049\" for `long_date_format` and `short_date_format` in English. For other languages, the defaut is `\"%d %B %Y\"` for the long format and `\"%-d %b %Y\"` for the short format. The universal default for the archive format is `\"%d %b\"`.\n\nIn Zola, time formatting syntax is inspired fom strftime. A full reference is available in the [chrono docs](https://docs.rs/chrono/0.4.31/chrono/format/strftime/index.html).\n\n#### Per-language date formats\n\nYou can customise date formats for specific languages using the `date_formats` array in `config.toml`:\n\n```toml\ndate_formats = [\n    { lang = \"es\", long = \"%d de %B de %Y\", short = \"%-d %b %Y\", archive = \"%d de %b\" },\n    { lang = \"de\", long = \"%d. %B %Y\", short = \"%d.%m.%Y\", archive = \"%d. %b\" },\n]\n```\n\nThis allows different languages to use culturally appropriate date formatting (e.g. Spanish \"3 de febrero de 2024\" vs German \"3. Februar 2024\").\n\n### Custom Separator\n\n| Page | Section | `config.toml` | Follows Hierarchy | Requires JavaScript |\n|:----:|:-------:|:-------------:|:-----------------:|:-------------------:|\n|  ❌  |   ❌    |      ✅       |         ❌        |         ❌          |\n\nThe separator appears in various places: in the title tag, between the metadata of a post…\n\nThe default separator is a bullet point (`•`), but you can change by setting something like `separator = \"|\"`.\n\n### Title Tag Order\n\n| Page | Section | `config.toml` | Follows Hierarchy | Requires JavaScript |\n|:----:|:-------:|:-------------:|:-----------------:|:-------------------:|\n|  ❌  |   ❌    |      ✅       |         ❌        |         ❌          |\n\nThe title tag is the text that appears in the browser tab. By default, it's the site title followed by the page title. For example, the title tag of the blog section is \"~/tabi • Blog\".\n\nBy setting `invert_title_order = true`, you can invert the order of the site title and page title in the browser tab. For example, the title tag of the blog section would become \"Blog • ~/tabi\".\n\n---\n\n## Security\n\n### Encoded Email\n\n| Page | Section | `config.toml` | Follows Hierarchy | Requires JavaScript |\n|:----:|:-------:|:-------------:|:-----------------:|:-------------------:|\n|  ❌  |   ❌    |      ✅       |         ❌        |         ✅          |\n\nTo protect your email address from spambots, you can encode it in the footer. You can do this by setting `email` to a base64 encoded version of your email address[^2]. For example, `email = \"bWFpbEBleGFtcGxlLmNvbQ==\"` is the base64 encoded version of \"mail@example.com\".\n\nIf you don't want to encode your email yourself, tabi can encode it for you if you set `encode_plaintext_email = true`. This allows you to set a plaintext email on the config. Note that this only protects your email address on your site, not in public repositories.\n\nIf the email is encoded (either by you or by tabi), users with JavaScript disabled will not see the email icon.\n\n### CSP (Content Security Policy)\n\n| Page | Section | `config.toml` | Follows Hierarchy | Requires JavaScript |\n|:----:|:-------:|:-------------:|:-----------------:|:-------------------:|\n|  ❌  |   ❌    |      ✅       |         ❌        |         ❌          |\n\nContent Security Policy (CSP) is an added layer of security that helps to detect and mitigate certain types of attacks, including Cross Site Scripting (XSS) and data injection attacks. These attacks are used for everything from data theft to site defacement to distribution of malware.\n\ntabi has a default CSP that allows for remote images and videos, as well as YouTube and Vimeo embeds. You can customise it with `allowed_domains`, which takes a list of CSP directives. This is the default CSP:\n\n```toml\nallowed_domains = [\n    { directive = \"font-src\", domains = [\"'self'\", \"data:\"] },\n    { directive = \"img-src\", domains = [\"'self'\", \"https://*\", \"data:\"] },\n    { directive = \"script-src\", domains = [\"'self'\"] },\n    { directive = \"style-src\", domains = [\"'self'\"] },\n    { directive = \"frame-src\", domains = [\"player.vimeo.com\", \"https://www.youtube-nocookie.com\"] },\n]\n```\n\nThis feature is enabled by default. To disable it (and allow all connections), set `enable_csp = false` on a page, section or globally. The `enable_csp` setting follows the [hierarchy](#settings-hierarchy).\n\nSee the [CSP documentation page](@/blog/security/index.md) for more information.\n\n---\n\n## Indieweb\n\n### Webmentions\n\n| Page | Section | `config.toml` | Follows Hierarchy | Requires JavaScript |\n|:----:|:-------:|:-------------:|:-----------------:|:-------------------:|\n|  ❓  |   ❓    |      ✅       |         ❓        |         ✅          |\n\nAs described by the recommended W3C standard [Webmention](https://www.w3.org/TR/webmention/#abstract-p-1) is a simple way to notify any URL when you mention it on your site. From the receiver's perspective, it's a way to request notifications when other sites mention it.\n\nFor static sites [webmention.io](https://webmention.io/) hosts a webmention endpoint that can be used to receive webmentions. This feature fetches the webmentions stored at webmention.io and displays them for a page. You will need to have setup an account for your website at webmention.io. When you enable the webmention feature it will advertise your webmention.io endpoint and display the webmentions for all posts.\n\nEnable webmentions for your site by adding the following to your `config.toml` file.\n\n```toml\n[extra.webmentions]\nenable = true\n# Specify the domain registered with webmention.io.\ndomain = \"www.example.com\"\n```\n\n❓: To disable webmentions for a specific section or page, set `webmentions = false` in the `[extra]` section of that section or page's front matter.\n\nThe webmentions section looks like this:\n\n{{ dual_theme_image(light_src=\"blog/mastering-tabi-settings/img/webmention_light.webp\", dark_src=\"blog/mastering-tabi-settings/img/webmention_dark.webp\" alt=\"Webmentions screenshot showing reposts, likes, bookmarks, and comments\", full_width=true) }}\n\n### Representative h-card\n\n| Page | Section | `config.toml` | Follows Hierarchy | Requires JavaScript |\n| :--: | :-----: | :-----------: | :---------------: | :-----------------: |\n|  ❌  |   ❌    |      ✅       |        ❌         |         ❌          |\n\nBy default, tabi adds a **hidden** representative [h-card](https://microformats.org/wiki/h-card) to the homepage. While invisible to visitors, it's available to microformat parsers. You can check the validity of the card with the [Indiewebify.me](https://indiewebify.me/validate-h-card/) tool.\n\nTo disable the h-card, set `enable = false` in the `[extra.hcard]` section of `config.toml`.\n\nThe default h-card includes your name, website url and social media links.\n\nYou can set a profile picture and a small bio with the `avatar` and `biography` settings.\n\nAll other [h-card properties](https://microformats.org/wiki/h-card#Properties) can be added by listing them under the `[extra.hcard]`section of the config file. Simply replace all `-` characters by `_`.\n\n---\n\n## Extending HTML Elements in tabi\n\nSome HTML elements in tabi can be extended to support additional use cases such as adding custom JavaScript for site-wide behavior at the end of the `<body>` tag or including additional content at the end of the `<head>` element that is not otherwise supported by other tabi settings.\n\nSee the table below for elements that can be extended:\n\n| Element  | Template                          |\n| :------: | :-------------------------------: |\n| `<head>` | `templates/tabi/extend_head.html` |\n| `<body>` | `templates/tabi/extend_body.html` |\n\nThere are no explicit settings to configure for your site or pages. Simply create the relevant template file for your site, and tabi will automatically include it.\n\n---\n\n[^1]: If you're using a remote Git repository, you might want to automate the process of updating the `updated` field. Here's a guide for that: [Zola Git Pre-Commit Hook: Updating Post Dates](https://osc.garden/blog/zola-date-git-hook/).\n\n[^2]: To encode your email in base64 you can use [online tools](https://www.base64encode.org/) or, on your terminal, run: `printf 'mail@example.com' | base64`.\n"
  },
  {
    "path": "content/blog/security/index.ca.md",
    "content": "+++\ntitle = \"Seguretat per defecte\"\ndate = 2023-02-22\nupdated = 2026-01-14\ndescription = \"tabi té una Política de Seguretat de Contingut (CSP) fàcilment personalitzable amb valors segurs per defecte. Obtingues tranquil·litat i un A+ en l'Observatori de Mozilla.\"\n\n[taxonomies]\ntags = [\"seguretat\", \"funcionalitat\"]\n\n[extra]\nsocial_media_card = \"social_cards/ca_blog_security.jpg\"\n+++\n\nLa configuració per defecte del tema obté una puntuació A+ a l'[Observatori de Mozilla](https://observatory.mozilla.org).[^1]\n\nAixò s'aconsegueix configurant programàticament les capçaleres de la Política de Seguretat de Contingut (CSP) basant-se en una llista de dominis permesos definida per l'usuari en el fitxer `config.toml`. Aquí tens la configuració per defecte i recomanada (pots eliminar l'última directiva si no vols inserir vídeos de YouTube):\n\n```toml\n[extra]\nallowed_domains = [\n    { directive = \"font-src\", domains = [\"'self'\", \"data:\"] },\n    { directive = \"img-src\", domains = [\"'self'\", \"https://*\", \"data:\"] },\n    { directive = \"script-src\", domains = [\"'self'\"] },\n    { directive = \"style-src\", domains = [\"'self'\"] },\n    { directive = \"frame-src\", domains = [\"https://www.youtube-nocookie.com\"] },\n]\n```\n\nLa llista `allowed_domains` especifica les URLs a les quals el lloc web hauria de poder connectar-se, i cada domini de la llista està associat amb una directiva CSP com `frame-src`, `connect-src` o `script-src`. El fitxer `templates/partials/header.html` genera dinàmicament la capçalera CSP basant-se en aquesta llista.\n\nAquesta funcionalitat permet personalitzar fàcilment les capçaleres de seguretat del lloc web per permetre casos d'ús específics, com ara inserir vídeos de YouTube, carregar scripts o tipografies remotes ([no recomanat](https://www.albertovarela.net/blog/2022/11/stop-using-google-fonts/)).\n\nPots desactivar les capçaleres (permitint-ho tot) en una pàgina, secció, o globalment configurant `enable_csp = false` en el front matter o en el fitxer `config.toml`.\n\n**Notas**:\n\n- [Habilitar els comentaris](@/blog/comments/index.ca.md), [les analítiques](@/blog/mastering-tabi-settings/index.ca.md#analisi-web), o [els diagrames de mermaid](@/blog/shortcodes/index.ca.md#diagrames-de-mermaid) permet automàticament els scripts/frames/estils/conexions pertinents.\n- Per utilitzar un [tema de resaltat de sintaxis integrat a Zola](https://www.getzola.org/documentation/content/syntax-highlighting/), has de permetre `unsafe-inline` a la directiva `style-src`:\n\n    ```\n    { directive = \"style-src\", domains = [\"'self'\", \"'unsafe-inline'\"] },\n    ```\n\n---\n\n[^1]: Requereix una configuració adequada del servidor web (p. ex., redirigir el trànsit HTTP a HTTPS).\n"
  },
  {
    "path": "content/blog/security/index.es.md",
    "content": "+++\ntitle = \"Seguro por defecto\"\ndate = 2023-02-22\nupdated = 2026-01-14\ndescription = \"tabi tiene una Política de Seguridad de Contenido (CSP) fácilmente personalizable con configuraciones seguras. Obtén tranquilidad y una calificación de A+ en Mozilla Observatory.\"\n\n[taxonomies]\ntags = [\"seguridad\", \"funcionalidad\"]\n\n[extra]\nsocial_media_card = \"social_cards/es_blog_security.jpg\"\n+++\n\nLa configuración predeterminada del tema obtiene una calificación de A+ en [Mozilla Observatory](https://observatory.mozilla.org).[^1]\n\nEsto se logra configurando programáticamente las cabeceras de la Política de Seguridad de Contenido (CSP) en función de una lista de dominios permitidos definida por el usuario en el archivo `config.toml`. Aquí tienes la configuración predeterminada y recomendada (puedes eliminar la última directiva si no deseas insertar videos de YouTube):\n\n```toml\n[extra]\nallowed_domains = [\n    { directive = \"font-src\", domains = [\"'self'\", \"data:\"] },\n    { directive = \"img-src\", domains = [\"'self'\", \"https://*\", \"data:\"] },\n    { directive = \"script-src\", domains = [\"'self'\"] },\n    { directive = \"style-src\", domains = [\"'self'\"] },\n    { directive = \"frame-src\", domains = [\"https://www.youtube-nocookie.com\"] },\n]\n```\n\nLa lista `allowed_domains` especifica las URL a las que el sitio web debería poder conectarse, y cada dominio de la lista se asocia con una directiva CSP como `frame-src`, `connect-src` o `script-src`. El archivo `templates/partials/header.html` genera dinámicamente la cabecera CSP en función de esta lista.\n\nEsta función permite personalizar fácilmente las cabeceras de seguridad del sitio web para permitir casos de uso específicos, como la incrustación de videos de YouTube, la carga de scripts o  fuentes remotas ([no recomendado](https://www.albertovarela.net/blog/2022/11/stop-using-google-fonts/)).\n\nPuedes desactivar las cabeceras (permitiendo todo) en una página, sección, o globalmente configurando `enable_csp = false` en el front matter o en el archivo `config.toml`.\n\n**Notas**:\n\n- [Habilitar los comentarios](@/blog/comments/index.es.md), [las analíticas](@/blog/mastering-tabi-settings/index.es.md#analisis-web), o [los diagramas mermaid](@/blog/shortcodes/index.es.md#diagramas-de-mermaid) permite automáticamente los scripts/frames/estilos/conexiones pertinentes.\n- Para usar un [tema de resaltado de sintaxis integrado en Zola](https://www.getzola.org/documentation/content/syntax-highlighting/), has de permitir `unsafe-inline` en la directiva `style-src`:\n\n    ```\n    { directive = \"style-src\", domains = [\"'self'\", \"'unsafe-inline'\"] },\n    ```\n\n---\n\n[^1]: Requiere una configuración adecuada del servidor web (por ejemplo, redirigir el tráfico HTTP a HTTPS).\n"
  },
  {
    "path": "content/blog/security/index.md",
    "content": "+++\ntitle = \"Secure by default\"\ndate = 2023-02-22\nupdated = 2026-01-14\ndescription = \"tabi has an easily customizable Content Security Policy (CSP) with safe defaults. Get peace of mind and an A+ on Mozilla Observatory.\"\n\n[taxonomies]\ntags = [\"security\", \"showcase\"]\n\n[extra]\nsocial_media_card = \"social_cards/blog_security.jpg\"\n+++\n\nThe default configuration of the theme gets an A+ score on [Mozilla Observatory](https://observatory.mozilla.org).[^1]\n\nThis is accomplished by programatically configuring Content Security Policy (CSP) headers based on a user-defined list of allowed domains in the `config.toml` file. Here's the default and recommended setup (you could remove the last directive if you don't want to embed YouTube videos):\n\n```toml\n[extra]\nallowed_domains = [\n    { directive = \"font-src\", domains = [\"'self'\", \"data:\"] },\n    { directive = \"img-src\", domains = [\"'self'\", \"https://*\", \"data:\"] },\n    { directive = \"script-src\", domains = [\"'self'\"] },\n    { directive = \"style-src\", domains = [\"'self'\"] },\n    { directive = \"frame-src\", domains = [\"https://www.youtube-nocookie.com\"] },\n]\n```\n\nThe `allowed_domains` list specifies the URLs that the website should be able to connect to, and each domain in the list is associated with a CSP directive such as `frame-src`, `connect-src`, or `script-src`. The `templates/partials/header.html` file dynamically generates the CSP header based on this list.\n\nThis feature allows you to easily customize the website's security headers to allow for specific use cases, such as embedding YouTube videos, loading scripts or remote fonts ([not recommended](https://www.albertovarela.net/blog/2022/11/stop-using-google-fonts/)).\n\nYou can disable the CSP (allowing all connections) on a page, section, or globally by setting `enable_csp = false` in the front matter or `config.toml` file.\n\n**Notes**:\n\n- [Enabling comments](@/blog/comments/index.md), [analytics](@/blog/mastering-tabi-settings/index.md#analytics), or [mermaid diagrams](@/blog/shortcodes/index.md#mermaid-diagrams) automatically allows scripts/frames/styles/connections as needed.\n- To use a [Zola built-in syntax highlighting theme](https://www.getzola.org/documentation/content/syntax-highlighting/), you need to allow `unsafe-inline` in the `style-src` directive:\n\n    ```\n    { directive = \"style-src\", domains = [\"'self'\", \"'unsafe-inline'\"] },\n    ```\n\n---\n\n[^1]: Requires proper webserver configuration (e.g. redirecting HTTP traffic to HTTPS).\n"
  },
  {
    "path": "content/blog/series/index.ca.md",
    "content": "+++\ntitle = \"Guia completa sobre sèries\"\ndate = 2024-11-08\nupdated = 2025-02-21\ndescription = \"Aprèn a organitzar les teves publicacions en sèries seqüencials, perfectes per a tutorials, cursos i històries de diverses parts.\"\n\n[taxonomies]\ntags = [\"funcionalitat\", \"tutorial\", \"preguntes freqüents\", \"sèries\"]\n\n[extra]\nquick_navigation_buttons = true\ntoc = true\nmermaid = true\nsocial_media_card = \"social_cards/ca_blog_series.jpg\"\n+++\n\nUna sèrie organitza publicacions relacionades en ordre seqüencial, similar als capítols d'un llibre. A diferència de les etiquetes, que simplement agrupen contingut relacionat, les sèries suggereixen un ordre específic de lectura de principi a fi.\n\nLes publicacions dins d'una sèrie no necessiten publicar-se de forma consecutiva; la funció de sèries reuneix publicacions temàticament vinculades en una seqüència coherent.\n\nEl següent diagrama il·lustra com les publicacions de la sèrie (3, 5 i 8) existeixen dins del flux principal del blog mentre mantenen la seva pròpia seqüència ordenada dins de Sèrie 1.\n\n{% mermaid(full_width=true) %}\nflowchart\n    subgraph main[BLOG]\n        P1[Post 1]\n        P2[P2]\n        P3[P3]\n        P4[P4]\n        P5[P5]\n        P6[P6]\n        P7[P7]\n        P8[P8]\n        P9[P9]\n    end\n    subgraph series1[SÈRIE 1]\n        PS1[\"Post Sèrie 1 (=P3)\"]\n        PS2[\"Post Sèrie 2 (=P5)\"]\n        PS3[\"Post Sèrie 3 (=P8)\"]\n    end\n    P3 o-.-o PS1\n    P5 o-.-o PS2\n    P8 o-.-o PS3\n{% end %}\n\n## Inici ràpid\n\n1. Crea un directori per a la teva sèrie\n2. Crea `_index.md` al directori de la sèrie\n3. Configura el front matter de `_index.md`:\n\n    ```toml,name=series/_index.md\n    title = \"Aprenent Rust\"\n    template = \"series.html\"\n    sort_by = \"slug\"\n    transparent = true\n\n    [extra]\n    series = true\n    ```\n\n4. Crea els teus articles de la sèrie en aquest directori\n\nVols saber-ne més? Continua llegint!\n\n## Com funcionen les sèries?\n\nUna sèrie és simplement una secció que tabi gestiona de manera especial. Per a més detalls sobre seccions, consulta la [documentació de Zola](https://www.getzola.org/documentation/content/section/).\n\nPrenent l'exemple del diagrama anterior, l'estructura de directoris seria així:\n\n```txt\ncontent/\n    _index.md\n    blog/\n        _index.md\n        post1/\n            index.md\n        post2/\n            index.md\n        post4/\n            index.md\n        post6/\n            index.md\n        post7/\n            index.md\n        post9/\n            index.md\n        serie1/\n            _index.md\n            post3/\n                index.md\n            post5/\n                index.md\n            post8/\n                index.md\n```\n\nPer crear una sèrie, necessites:\n\n1. Utilitzar la plantilla `series.html`\n2. Establir `series = true` a la configuració `[extra]` de la secció\n3. Activar `transparent = true` per integrar les publicacions de la sèrie amb la secció del blog principal\n\nLa pàgina principal de la sèrie mostra un resum seguit d'una llista de totes les publicacions a la sèrie:\n\n{{ dual_theme_image(light_src=\"blog/series/img/series_light.webp\", dark_src=\"blog/series/img/series_dark.webp\" alt=\"una sèrie\", full_width=true) }}\n\n## Saltar a les publicacions\n\nSi el contingut d'una sèrie (el Markdown després del frontmatter a `_index.md`) supera els 2000 caràcters, apareix un enllaç \"Salta a les publicacions\" al costat del títol de la sèrie.\n\n{{ dual_theme_image(light_src=\"blog/series/img/jump_to_series_posts_light.webp\", dark_src=\"blog/series/img/jump_to_series_posts_dark.webp\" alt=\"enllaç per saltar a les publicacions de la sèrie\", full_width=true) }}\n\nPer forçar l'activació o desactivació d'aquesta funció, configura `show_jump_to_posts` a la secció `[extra]` de la teva secció de sèries o a `config.toml`. Aquesta configuració segueix [la jerarquia](@/blog/mastering-tabi-settings/index.ca.md#jerarquia-de-configuracio).\n\n## Pàgines de sèries i ordre\n\nTotes les pàgines a la secció de sèries seran pàgines de sèrie. Les pàgines s'ordenaran segons el `sort_by` de la secció.\n\nTot i que les sèries mantenen el seu propi ordre intern, romanen independents del flux cronològic de la secció principal (per exemple, `blog/`) gràcies a la configuració `transparent`.\n\n### Opcions d'ordre\n\nTria entre aquests mètodes d'ordre, cadascun amb els seus avantatges:\n\n{% wide_container() %}\n\n`sort_by` | avantatges | desavantatges\n---------|------------|---------------\n`slug`    | L'ordre de les pàgines és explícit a la ruta (per exemple, `example.com/blog/series1/01-series-post-un`). | Cada pàgina de la sèrie ha de tenir el prefix corresponent.\n`weight`  | L'ordre de les pàgines és fàcil de configurar de forma transparent.<br>La primera publicació té pes `1`, la segona pes `2` i així successivament. | Cada pàgina de la sèrie ha de tenir el seu pes configurat.\n`date`    | L'ordre de les pàgines es pot configurar una sola vegada a la configuració de la secció. No cal fer res a cada pàgina. | L'ordre de les pàgines s'ha d'invertir perquè la primera pàgina sol ser la més antiga. Això només es pot aconseguir paginant la secció (`paginate_by = 9999`) i invertint el seu ordre (`paginate_reversed = true`).\n\n{% end %}\n\n{{ admonition(type=\"danger\", title=\"Versió de Zola per ordenar per data\", text=\"Per invertir correctament les dates, es requereix Zola v0.19.3+ (no publicada) perquè la informació de paginació estigui disponible a través de la funció `get_section`. En cas contrari, qualsevol cosa que depengui de l'ordre de les pàgines de la sèrie no serà correcta (per exemple, pàgina anterior/següent, llistes ordenades i no ordenades...) Vegeu [Zola PR #2653](https://github.com/getzola/zola/pull/2653).\") }}\n\n### Indexació de pàgines\n\nLes pàgines en una sèrie s'indexen començant des d'1, seguint el seu ordre `sort_by`. Per invertir la indexació (fent que la primera pàgina tingui l'índex més alt), afegeix aquesta configuració a `_index.md` o `config.toml`:\n\n```toml\n[extra]\npost_listing_index_reversed = true  # Per defecte és false si no es configura\n```\n\n{{ dual_theme_image(light_src=\"blog/series/img/series_reversed_light.webp\", dark_src=\"blog/series/img/series_reversed_dark.webp\" alt=\"una sèrie amb índexs invertits\", full_width=true) }}\n\nAquesta configuració segueix [la jerarquia](@/blog/mastering-tabi-settings/index.ca.md#jerarquia-de-configuracio).\n\n## Plantilles d'introducció i conclusió\n\nEls articles d'una sèrie poden tenir seccions automàtiques d'introducció i conclusió. Aquestes es configuren al `_index.md` de la teva sèrie. Un exemple bàsic:\n\n```toml,name=series/_index.md\n[extra.series_intro_templates]\ndefault = \"Aquest article és part de la sèrie $SERIES_HTML_LINK.\"\n\n[extra.series_outro_templates]\ndefault = \"Gràcies per llegir la part $SERIES_PAGE_INDEX de $SERIES_HTML_LINK!\"\n```\n\nLes seccions d'introducció i conclusió tenen les seves pròpies classes CSS (`series-page-intro` i `series-page-outro`), que et permeten personalitzar la seva aparença mitjançant [CSS personalitzat](@/blog/mastering-tabi-settings/index.ca.md#estils-css-personalitzats).\n\n### Tipus de plantilles\n\nEl sistema de sèries utilitza diferents plantilles segons la posició de l'article a la sèrie:\n\n- `next_only` - Utilitzat per al primer article (té article següent però no anterior)\n- `middle` - Utilitzat per a articles amb articles anterior i següent\n- `prev_only` - Utilitzat per a l'últim article (té article anterior però no següent)\n- `default` - Plantilla per defecte utilitzada quan no existeix una plantilla específica per a la posició\n\nEl sistema determina automàticament quina plantilla utilitzar segons la posició de l'article. Les plantilles es defineixen a la configuració de la sèrie (`_index.md`), com `extra.series_intro_templates` i `extra.series_outro_templates`:\n\n```toml,name=series/_index.md\n[extra.series_intro_templates]\nnext_only = \"Benvingut a la part 1! Següent: $NEXT_HTML_LINK\"\nmiddle = \"Anterior: $PREV_HTML_LINK | Següent: $NEXT_HTML_LINK\"\nprev_only = \"El capítol final! Anteriorment: $PREV_HTML_LINK\"\ndefault = \"Part $SERIES_PAGE_INDEX de $SERIES_PAGES_NUMBER\"\n```\n\nTotes les plantilles són opcionals. La selecció de plantilles segueix un sistema de prioritat:\n\n1. Si existeix una plantilla específica per a la posició (`next_only`, `middle`, o `prev_only`), s'utilitzarà aquesta\n2. Si no, s'utilitza la plantilla `default`\n3. Si no es defineix cap plantilla, no es mostrarà informació de la sèrie\n\nMira l'[exemple de plantilla](#exemple-de-plantilla) per veure un exemple més elaborat.\n\n### Ubicació al contingut\n\nPer defecte:\n\n- Les introduccions de sèrie apareixen a l'inici del teu article\n- La conclusió apareix al final (abans de les notes al peu, si n'hi ha)\n\nPots controlar exactament on apareixen utilitzant `<!-- series_intro -->` i `<!-- series_outro -->` al teu Markdown:\n\n```markdown\nAquest paràgraf apareix abans de la introducció de la sèrie.\n\n<!-- series_intro -->\n\nContingut principal de l'article.\n\n<!-- series_outro -->\n\n## Recursos d'aprenentatge\n\nContingut addicional...\n\n[^1]: Les notes al peu sempre apareixeran al final.\n```\n\n## Variables\n\nLes plantilles de sèries utilitzen un sistema flexible de variables que et permet:\n\n1. Fer referència a informació de la sèrie (títol, enllaços)\n2. Afegir navegació entre articles\n3. Mostrar indicadors de progrés\n4. Incloure informació personalitzada utilitzant les teves pròpies variables\n\nLes variables són marcadors que comencen amb `$` i es reemplacen amb contingut real quan es construeix el teu lloc. Per exemple, `$SERIES_HTML_LINK` es converteix en un enllaç clicable a la pàgina índex de la teva sèrie.\n\nHi ha tres tipus de variables:\n\n- [Variables bàsiques de sèrie](#variables-basiques-de-serie): Informació general sobre la sèrie\n- [Variables de navegació](#variables-de-navegacio): Enllaços a articles anterior/següent\n- [Variables personalitzades](#variables-personalitzades): Els teus propis marcadors per a informació addicional\n\n### Variables bàsiques de sèrie\n\n{% wide_container() %}\n\n| Variable | Disponibilitat | Retorna | Descripció | Exemple d'ús | Exemple de sortida |\n|----------|---------------|----------|------------|--------------|-------------------|\n| `$SERIES_TITLE` | Sempre | Text | Títol de la sèrie en text pla | `Part de $SERIES_TITLE` | Part d'Aprenent Rust |\n| `$SERIES_PERMALINK` | Sempre | Text | URL a l'índex de la sèrie | `[Veure totes les publicacions]($SERIES_PERMALINK)` | [Veure totes les publicacions](/series/learn-rust) |\n| `$SERIES_HTML_LINK` | Sempre | HTML | Enllaç llest per usar a la sèrie | `Benvingut a $SERIES_HTML_LINK!` | Benvingut a <a href=\"/series/learn-rust\">Aprenent Rust</a>! |\n| `$SERIES_PAGES_NUMBER` | Sempre | Nombre | Total d'articles a la sèrie | `Una sèrie de $SERIES_PAGES_NUMBER parts` | Una sèrie de 5 parts |\n| `$SERIES_PAGE_INDEX` | Sempre | Nombre | Posició de l'article actual | `Part $SERIES_PAGE_INDEX de $SERIES_PAGES_NUMBER` | Part 3 de 5 |\n| `$SERIES_PAGES_OLIST` | Sempre | HTML | Llista ordenada de tots els articles | `Articles a la sèrie: $SERIES_PAGES_OLIST` | Articles a la sèrie: <ol><li>Article actual</li><li><a href=\"...\">Altres articles</a></li></ol> |\n| `$SERIES_PAGES_ULIST` | Sempre | HTML | Llista desordenada de tots els articles | `Articles a la sèrie: $SERIES_PAGES_ULIST` | Articles a la sèrie: <ul><li>Article actual</li><li><a href=\"...\">Altres articles</a></li></ul> |\n\n{% end %}\n\n{{ admonition(type=\"tip\", title=\"CONSELL: Text personalitzat amb permalinks\", text='Els enllaços markdown com `[text]($SERIES_PERMALINK)` seran marcats (i [estilitzats](@/blog/mastering-tabi-settings/index.ca.md#indicador-d-enllacos-externs)) com externs. Si necessites text personalitzat i vols evitar l\\'estil extern, utilitza HTML: `<a href=\\\"$SERIES_PERMALINK\\\">el teu text</a>`.') }}\n\n### Variables de navegació\n\n{% wide_container() %}\n\n| Variable | Disponibilitat | Retorna | Descripció | Exemple d'ús | Exemple de sortida |\n|----------|---------------|----------|------------|--------------|-------------------|\n| `$PREV_TITLE` | Existeix anterior | Text | Títol de l'article anterior | `Anteriorment: $PREV_TITLE` | Anteriorment: Configurant el teu entorn |\n| `$PREV_PERMALINK` | Existeix anterior | Text | URL a l'article anterior | `[← Enrere]($PREV_PERMALINK)` | [← Enrere](/series/learn-rust/setup) |\n| `$PREV_HTML_LINK` | Existeix anterior | HTML | Enllaç llest per usar a l'anterior | `Llegeix primer $PREV_HTML_LINK` | Llegeix primer <a href=\"/series/learn-rust/setup\">Configurant el teu entorn</a> |\n| `$PREV_DESCRIPTION` | Existeix anterior | Text | Descripció de l'article anterior | `Resum: $PREV_DESCRIPTION` | Resum: Configurant Rust |\n| `$NEXT_TITLE` | Existeix següent | Text | Títol del següent article | `Següent: $NEXT_TITLE` | Següent: Patrons avançats |\n| `$NEXT_PERMALINK` | Existeix següent | Text | URL al següent article | `[Continuar →]($NEXT_PERMALINK)` | [Continuar →](/series/learn-rust/patterns) |\n| `$NEXT_HTML_LINK` | Existeix següent | HTML | Enllaç llest per usar al següent | `Continua amb $NEXT_HTML_LINK` | Continua amb <a href=\"/series/learn-rust/patterns\">Patrons avançats</a> |\n| `$NEXT_DESCRIPTION` | Existeix següent | Text | Descripció del següent article | `Properament: $NEXT_DESCRIPTION` | Properament: Aprèn sobre les característiques avançades de pattern matching en Rust |\n\n{% end %}\n\n### Referència al primer article\n\n{% wide_container() %}\n\n| Variable | Disponibilitat | Retorna | Descripció | Exemple d'ús | Exemple de sortida |\n|----------|---------------|----------|------------|--------------|-------------------|\n| `$FIRST_TITLE` | Sempre | Text | Títol del primer article | `Comença amb $FIRST_TITLE` | Comença amb Introducció a Rust |\n| `$FIRST_HTML_LINK` | Sempre | HTML | Enllaç llest per usar al primer article | `Comença a $FIRST_HTML_LINK` | Comença a <a href=\"/series/learn-rust/intro\">Introducció a Rust</a> |\n\n{% end %}\n\n### Exemple de plantilla\n\n{{ admonition(type=\"tip\", title=\"Variables HTML vs text\", text=\"Utilitza variables HTML (que acaben en `_HTML_LINK`) quan vulguis enllaços preparats per usar. Utilitza variables de text (que acaben en `_TITLE` o `_PERMALINK`) quan vulguis més control sobre el format.\") }}\n\n```toml,name=series/_index.md\n# Introducció\n[extra.series_intro_templates]\nnext_only = \"\"\"\nBenvingut a $SERIES_HTML_LINK! Aquesta sèrie de $SERIES_PAGES_NUMBER parts t'ensenyarà Rust des de zero.\n\nSegüent: $NEXT_HTML_LINK - $NEXT_DESCRIPTION\n\"\"\"\n\nmiddle = \"\"\"\n📚 Part $SERIES_PAGE_INDEX de $SERIES_PAGES_NUMBER a $SERIES_HTML_LINK\n\nAnterior: $PREV_HTML_LINK\nSegüent: $NEXT_HTML_LINK\n\"\"\"\n\nprev_only = \"\"\"\nBenvingut a l'última part de $SERIES_HTML_LINK!\nEts nou? Comença amb $FIRST_HTML_LINK per construir una base sòlida.\n\nAnterior: $PREV_HTML_LINK\n\"\"\"\n\n# Plantilla de respatller\ndefault = \"Aquest article és part de la sèrie $SERIES_HTML_LINK.\"\n\n# Conclusió\n[extra.series_outro_templates]\nnext_only = \"\"\"\nGràcies per llegir! 🙌\n\nContinua el teu viatge amb $NEXT_HTML_LINK, on $NEXT_DESCRIPTION\nO revisa l'esquema complet de la sèrie [$SERIES_TITLE]($SERIES_PERMALINK).\n\"\"\"\n\nmiddle = \"\"\"\n---\n📝 Navegació de la sèrie\n\n- Anterior: $PREV_HTML_LINK\n- Següent: $NEXT_HTML_LINK\n- [Resum de la sèrie]($SERIES_PERMALINK)\n\"\"\"\n\nprev_only = \"\"\"\n🎉 Felicitats! Has completat $SERIES_HTML_LINK.\n\nVols repassar? Aquí vam començar: $FIRST_HTML_LINK\nO revisa el que acabem de veure a $PREV_HTML_LINK.\n\"\"\"\n\n# Respatller.\ndefault = \"\"\"\n---\nAquest article és la part $SERIES_PAGE_INDEX de $SERIES_PAGES_NUMBER a $SERIES_HTML_LINK.\n\"\"\"\n```\n\n### Variables personalitzades\n\nLes plantilles de sèries admeten variables personalitzades per incloure informació addicional a tota la teva sèrie. El procés té dos passos:\n\n1. Primer, defineix els teus **marcadors** a la configuració de la teva sèrie (`_index.md`):\n\n```toml,name=series/_index.md\n[extra]\nseries = true\nseries_template_placeholders = [\"$POSITION\", \"$TOPIC\", \"$DIFFICULTY\"]\n```\n\n2. Després, a cada article de la sèrie, proporciona els valors per a aquests marcadors a `series_template_variables`:\n\n```toml,name=series/article.md\n[extra.series_template_variables]\nposition = \"primer\"\ntopic = \"Variables i tipus\"\ndifficulty = \"Principiant\"\n```\n\n### Ús de variables personalitzades\n\nPots usar les teves variables personalitzades a qualsevol plantilla, juntament amb les variables integrades:\n\n```toml,name=series/_index.md\n[extra.series_intro_templates]\ndefault = \"\"\"\nAquest és l'article $POSITION a $SERIES_HTML_LINK.\nTema d'avui: $TOPIC\nNivell de dificultat: $DIFFICULTY\n\"\"\"\n```\n\n{{ admonition(type=\"warning\", text=\"Encara que els marcadors es defineixen en majúscules (`$POSITION`), els noms de variables a `series_template_variables` han d'estar en minúscules (`position`).\") }}\n\n### Exemple amb variables personalitzades\n\n```toml,name=series/_index.md\n# A la configuració de la sèrie.\n[extra]\nseries = true\nseries_template_placeholders = [\"$LEARNING_TIME\", \"$KEY_CONCEPTS\"]\n\nseries_intro_templates.default = \"\"\"\n📚 Part $SERIES_PAGE_INDEX de $SERIES_PAGES_NUMBER\n⏱️ Temps estimat: $LEARNING_TIME\n🔑 Conceptes clau: $KEY_CONCEPTS\n\"\"\"\n```\n\n```toml,name=series/02-learning-rust/index.md\n# En un article de la sèrie.\n[extra.series_template_variables]\nlearning_time = \"30 minuts\"\nkey_concepts = \"Funcions, gestió d'errors, coincidència de patrons\"\n```\n\nAixò generarà:\n\n```txt\n📚 Part 2 de 5\n⏱️ Temps estimat: 30 minuts\n🔑 Conceptes clau: Funcions, gestió d'errors, coincidència de patrons\n```\n\n{{ admonition(type=\"warning\", title=\"Variables que falten\", text=\"Si uses un marcador a les teves plantilles però no proporciones el seu valor a `series_template_variables`, la compilació fallarà amb un error que llista les variables que falten.\") }}\n"
  },
  {
    "path": "content/blog/series/index.es.md",
    "content": "+++\ntitle = \"Guía completa sobre series\"\ndate = 2024-11-08\nupdated = 2025-02-21\ndescription = \"Aprende a organizar tus publicaciones en series secuenciales, perfectas para tutoriales, cursos e historias de varias partes.\"\n\n[taxonomies]\ntags = [\"funcionalidad\", \"tutorial\", \"preguntas frecuentes\", \"series\"]\n\n[extra]\nquick_navigation_buttons = true\ntoc = true\nmermaid = true\nsocial_media_card = \"social_cards/es_blog_series.jpg\"\n+++\n\nUna serie organiza publicaciones relacionadas en orden secuencial, similar a los capítulos de un libro. A diferencia de las etiquetas, que simplemente agrupan contenido relacionado, las series sugieren un orden específico de lectura de principio a fin.\n\nLas publicaciones dentro de una serie no necesitan publicarse de forma consecutiva; la función de series reúne publicaciones temáticamente vinculadas en una secuencia coherente.\n\nEl siguiente diagrama ilustra cómo las publicaciones de la serie (3, 5 y 8) existen dentro del flujo principal del blog mientras mantienen su propia secuencia ordenada dentro de Serie 1.\n\n{% mermaid(full_width=true) %}\nflowchart\n    subgraph main[BLOG]\n        P1[Post 1]\n        P2[P2]\n        P3[P3]\n        P4[P4]\n        P5[P5]\n        P6[P6]\n        P7[P7]\n        P8[P8]\n        P9[P9]\n    end\n    subgraph series1[SERIE 1]\n        PS1[\"Post Serie 1 (=P3)\"]\n        PS2[\"Post Serie 2 (=P5)\"]\n        PS3[\"Post Serie 3 (=P8)\"]\n    end\n    P3 o-.-o PS1\n    P5 o-.-o PS2\n    P8 o-.-o PS3\n{% end %}\n\n## Inicio rápido\n\n1. Crea un directorio para tu serie\n2. Crea `_index.md` en el directorio de la serie\n3. Configura el front matter de `_index.md`:\n\n    ```toml,name=series/_index.md\n    title = \"Aprendiendo Rust\"\n    template = \"series.html\"\n    sort_by = \"slug\"\n    transparent = true\n\n    [extra]\n    series = true\n    ```\n\n4. Crea tus artículos de la serie en este directorio\n\n¿Quieres saber más? ¡Sigue leyendo!\n\n## ¿Cómo funcionan las series?\n\nUna serie es simplemente una sección que tabi maneja de manera especial. Para más detalles sobre secciones, consulta la [documentación de Zola](https://www.getzola.org/documentation/content/section/).\n\nTomando el ejemplo del diagrama anterior, la estructura de directorios sería así:\n\n```txt\ncontent/\n    _index.md\n    blog/\n        _index.md\n        post1/\n            index.md\n        post2/\n            index.md\n        post4/\n            index.md\n        post6/\n            index.md\n        post7/\n            index.md\n        post9/\n            index.md\n        serie1/\n            _index.md\n            post3/\n                index.md\n            post5/\n                index.md\n            post8/\n                index.md\n```\n\nPara crear una serie, necesitas:\n\n1. Usar la plantilla `series.html`\n2. Establecer `series = true` en la configuración `[extra]` de la sección\n3. Activar `transparent = true` para integrar las publicaciones de la serie con la sección del blog principal\n\nLa página principal de la serie muestra un resumen seguido de una lista de todas las publicaciones en la serie:\n\n{{ dual_theme_image(light_src=\"blog/series/img/series_light.webp\", dark_src=\"blog/series/img/series_dark.webp\" alt=\"una serie\", full_width=true) }}\n\n## Saltar a las publicaciones\n\nSi el contenido de una serie (el Markdown después del frontmatter en `_index.md`) supera los 2000 caracteres, aparece un enlace \"Saltar a publicaciones\" junto al título de la serie.\n\n{{ dual_theme_image(light_src=\"blog/series/img/jump_to_series_posts_light.webp\", dark_src=\"blog/series/img/jump_to_series_posts_dark.webp\" alt=\"enlace para saltar a las publicaciones de la serie\", full_width=true) }}\n\nPara forzar la activación o desactivación de esta función, configura `show_jump_to_posts` en la sección `[extra]` de tu sección de series o en `config.toml`. Esta configuración sigue [la jerarquía](@/blog/mastering-tabi-settings/index.es.md#jerarquia-de-configuracion).\n\n## Páginas de series y orden\n\nTodas las páginas en la sección de series serán páginas de serie. Las páginas se ordenarán según el `sort_by` de la sección.\n\nAunque las series mantienen su propio orden interno, permanecen independientes del flujo cronológico de la sección principal (por ejemplo, `blog/`) gracias a la configuración `transparent`.\n\n### Opciones de orden\n\nElige entre estos métodos de orden, cada uno con sus ventajas:\n\n{% wide_container() %}\n\n`sort_by` | ventajas                                                                                                                                      | desventajas\n---------|-------------------------------------------------------------------------------------------------------------------------------------------|---------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------\n`slug`    | El orden de las páginas es explícito en la ruta (por ejemplo, `example.com/blog/series1/01-series-post-uno`).                        | Cada página de la serie debe tener el prefijo correspondiente.\n`weight`  | El orden de las páginas es fácil de configurar de forma transparente.<br>La primera publicación tiene peso `1`, la segunda peso `2` y así sucesivamente. | Cada página de la serie debe tener su peso configurado.\n`date`    | El orden de las páginas se puede configurar una sola vez en la configuración de la sección. No hay que hacer nada en cada página.            | El orden de las páginas debe invertirse porque la primera página suele ser la más antigua. Esto solo se puede lograr paginando la sección (`paginate_by = 9999`) e invirtiendo su orden (`paginate_reversed = true`).\n\n{% end %}\n\n{{ admonition(type=\"danger\", title=\"Versión de Zola para ordenar por fecha\", text=\"Para invertir correctamente las fechas, se requiere Zola v0.19.3+ (no publicada) para que la información de paginación esté disponible a través de la función `get_section`. De lo contrario, cualquier cosa que dependa del orden de las páginas de la serie no será correcta (por ejemplo, página anterior/siguiente, listas ordenadas y no ordenadas...) Ver [Zola PR #2653](https://github.com/getzola/zola/pull/2653).\") }}\n\n### Indexación de páginas\n\nLas páginas en una serie se indexan empezando desde 1, siguiendo su orden `sort_by`. Para invertir la indexación (haciendo que la primera página tenga el índice más alto), añade esta configuración a `_index.md` o `config.toml`:\n\n```toml\n[extra]\npost_listing_index_reversed = true  # Por defecto es false si no se configura\n```\n\n{{ dual_theme_image(light_src=\"blog/series/img/series_reversed_light.webp\", dark_src=\"blog/series/img/series_reversed_dark.webp\" alt=\"una serie con índices invertidos\", full_width=true) }}\n\nEsta configuración sigue [la jerarquía](@/blog/mastering-tabi-settings/index.es.md#jerarquia-de-configuracion).\n\n## Plantillas de introducción y conclusión\n\nLos artículos de una serie pueden tener secciones automáticas de introducción y conclusión. Estas se configuran en el `_index.md` de tu serie. Un ejemplo básico:\n\n```toml,name=series/_index.md\n[extra.series_intro_templates]\ndefault = \"Este artículo es parte de la serie $SERIES_HTML_LINK.\"\n\n[extra.series_outro_templates]\ndefault = \"¡Gracias por leer la parte $SERIES_PAGE_INDEX de $SERIES_HTML_LINK!\"\n```\n\nLas secciones de introducción y conclusión tienen sus propias clases CSS (`series-page-intro` y `series-page-outro`), lo que te permite personalizar su apariencia mediante [CSS personalizado](@/blog/mastering-tabi-settings/index.es.md#estilos-css-personalizados).\n\n### Tipos de plantillas\n\nEl sistema de series usa diferentes plantillas según la posición del artículo en la serie:\n\n- `next_only` - Usado para el primer artículo (tiene artículo siguiente pero no anterior)\n- `middle` - Usado para artículos con artículos anterior y siguiente\n- `prev_only` - Usado para el último artículo (tiene artículo anterior pero no siguiente)\n- `default` - Plantilla por defecto usada cuando no existe una plantilla específica para la posición\n\nEl sistema determina automáticamente qué plantilla usar según la posición del artículo. Las plantillas se definen en la configuración de la serie (`_index.md`), como `extra.series_intro_templates` y `extra.series_outro_templates`:\n\n```toml,name=series/_index.md\n[extra.series_intro_templates]\nnext_only = \"¡Bienvenido a la parte 1! Siguiente: $NEXT_HTML_LINK\"\nmiddle = \"Anterior: $PREV_HTML_LINK | Siguiente: $NEXT_HTML_LINK\"\nprev_only = \"¡El capítulo final! Anteriormente: $PREV_HTML_LINK\"\ndefault = \"Parte $SERIES_PAGE_INDEX de $SERIES_PAGES_NUMBER\"\n```\n\nTodas las plantillas son opcionales. La selección de plantillas sigue un sistema de prioridad:\n\n1. Si existe una plantilla específica para la posición (`next_only`, `middle`, o `prev_only`), se usará esa\n2. Si no, se usa la plantilla `default`\n3. Si no se define ninguna plantilla, no se mostrará información de la serie\n\nMira el [ejemplo de plantilla](#ejemplo-de-plantilla) para ver un ejemplo más elaborado.\n\n### Ubicación en el contenido\n\nPor defecto:\n\n- Las introducciones de serie aparecen al inicio de tu artículo\n- La conclusión aparece al final (antes de las notas al pie, si las hay)\n\nPuedes controlar exactamente dónde aparecen usando `<!-- series_intro -->` y `<!-- series_outro -->` en tu Markdown:\n\n```markdown\nEste párrafo aparece antes de la introducción de la serie.\n\n<!-- series_intro -->\n\nContenido principal del artículo.\n\n<!-- series_outro -->\n\n## Recursos de aprendizaje\n\nContenido adicional...\n\n[^1]: Las notas al pie siempre aparecerán al final.\n```\n\n## Variables\n\nLas plantillas de series usan un sistema flexible de variables que te permite:\n\n1. Hacer referencia a información de la serie (título, enlaces)\n2. Añadir navegación entre artículos\n3. Mostrar indicadores de progreso\n4. Incluir información personalizada usando tus propias variables\n\nLas variables son marcadores que comienzan con `$` y se reemplazan con contenido real cuando se construye tu sitio. Por ejemplo, `$SERIES_HTML_LINK` se convierte en un enlace clicable a la página índice de tu serie.\n\nHay tres tipos de variables:\n\n- [Variables básicas de serie](#variables-basicas-de-serie): Información general sobre la serie\n- [Variables de navegación](#variables-de-navegacion): Enlaces a artículos anterior/siguiente\n- [Variables personalizadas](#variables-personalizadas): Tus propios marcadores para información adicional\n\n### Variables básicas de serie\n\n{% wide_container() %}\n\n| Variable | Disponibilidad | Devuelve | Descripción | Ejemplo de uso | Ejemplo de salida |\n|----------|---------------|-----------|-------------|----------------|-------------------|\n| `$SERIES_TITLE` | Siempre | Texto | Título de la serie en texto plano | `Parte de $SERIES_TITLE` | Parte de Aprendiendo Rust |\n| `$SERIES_PERMALINK` | Siempre | Texto | URL al índice de la serie | `[Ver todas las publicaciones]($SERIES_PERMALINK)` | [Ver todas las publicaciones](/series/learn-rust) |\n| `$SERIES_HTML_LINK` | Siempre | HTML | Enlace listo para usar a la serie | `¡Bienvenido a $SERIES_HTML_LINK!` | ¡Bienvenido a <a href=\"/series/learn-rust\">Aprendiendo Rust</a>! |\n| `$SERIES_PAGES_NUMBER` | Siempre | Número | Total de artículos en la serie | `Una serie de $SERIES_PAGES_NUMBER partes` | Una serie de 5 partes |\n| `$SERIES_PAGE_INDEX` | Siempre | Número | Posición del artículo actual | `Parte $SERIES_PAGE_INDEX de $SERIES_PAGES_NUMBER` | Parte 3 de 5 |\n| `$SERIES_PAGES_OLIST` | Siempre | HTML | Lista ordenada de todos los artículos | `Artículos en la serie: $SERIES_PAGES_OLIST` | Artículos en la serie: <ol><li>Artículo actual</li><li><a href=\"...\">Otros artículos</a></li></ol> |\n| `$SERIES_PAGES_ULIST` | Siempre | HTML | Lista desordenada de todos los artículos | `Artículos en la serie: $SERIES_PAGES_ULIST` | Artículos en la serie: <ul><li>Artículo actual</li><li><a href=\"...\">Otros artículos</a></li></ul> |\n\n{% end %}\n\n{{ admonition(type=\"tip\", title=\"CONSEJO: Texto personalizado con permalinks\", text='Los enlaces markdown como `[texto]($SERIES_PERMALINK)` serán marcados (y [estilizados](@/blog/mastering-tabi-settings/index.es.md#indicador-enlaces-externos)) como externos. Si necesitas texto personalizado y quieres evitar el estilo externo, usa HTML: `<a href=\\\"$SERIES_PERMALINK\\\">tu texto</a>`.') }}\n\n### Variables de navegación\n\n{% wide_container() %}\n\n| Variable | Disponibilidad | Devuelve | Descripción | Ejemplo de uso | Ejemplo de salida |\n|----------|---------------|-----------|-------------|----------------|-------------------|\n| `$PREV_TITLE` | Existe anterior | Texto | Título del artículo anterior | `Anteriormente: $PREV_TITLE` | Anteriormente: Configurando tu entorno |\n| `$PREV_PERMALINK` | Existe anterior | Texto | URL al artículo anterior | `[← Atrás]($PREV_PERMALINK)` | [← Atrás](/series/learn-rust/setup) |\n| `$PREV_HTML_LINK` | Existe anterior | HTML | Enlace listo para usar al anterior | `Lee primero $PREV_HTML_LINK` | Lee primero <a href=\"/series/learn-rust/setup\">Configurando tu entorno</a> |\n| `$PREV_DESCRIPTION` | Existe anterior | Texto | Descripción del artículo anterior | `Resumen: $PREV_DESCRIPTION` | Resumen: Configurando Rust |\n| `$NEXT_TITLE` | Existe siguiente | Texto | Título del siguiente artículo | `Siguiente: $NEXT_TITLE` | Siguiente: Patrones avanzados |\n| `$NEXT_PERMALINK` | Existe siguiente | Texto | URL al siguiente artículo | `[Continuar →]($NEXT_PERMALINK)` | [Continuar →](/series/learn-rust/patterns) |\n| `$NEXT_HTML_LINK` | Existe siguiente | HTML | Enlace listo para usar al siguiente | `Continúa con $NEXT_HTML_LINK` | Continúa con <a href=\"/series/learn-rust/patterns\">Patrones avanzados</a> |\n| `$NEXT_DESCRIPTION` | Existe siguiente | Texto | Descripción del siguiente artículo | `Próximamente: $NEXT_DESCRIPTION` | Próximamente: Aprende sobre las características avanzadas de pattern matching en Rust |\n\n{% end %}\n\n### Referencia al primer artículo\n\n{% wide_container() %}\n\n| Variable | Disponibilidad | Devuelve | Descripción | Ejemplo de uso | Ejemplo de salida |\n|----------|---------------|-----------|-------------|----------------|-------------------|\n| `$FIRST_TITLE` | Siempre | Texto | Título del primer artículo | `Comienza con $FIRST_TITLE` | Comienza con Introducción a Rust |\n| `$FIRST_HTML_LINK` | Siempre | HTML | Enlace listo para usar al primer artículo | `Empieza en $FIRST_HTML_LINK` | Empieza en <a href=\"/series/learn-rust/intro\">Introducción a Rust</a> |\n\n{% end %}\n\n### Ejemplo de plantilla\n\n{{ admonition(type=\"tip\", title=\"Variables HTML vs texto\", text=\"Usa variables HTML (que terminan en `_HTML_LINK`) cuando quieras enlaces listos para usar. Usa variables de texto (que terminan en `_TITLE` o `_PERMALINK`) cuando quieras más control sobre el formato.\") }}\n\n```toml,name=series/_index.md\n# Introducción.\n[extra.series_intro_templates]\nnext_only = \"\"\"\n¡Bienvenido a $SERIES_HTML_LINK! Esta serie de $SERIES_PAGES_NUMBER partes te enseñará Rust desde cero.\n\nSiguiente: $NEXT_HTML_LINK - $NEXT_DESCRIPTION\n\"\"\"\n\nmiddle = \"\"\"\n📚 Parte $SERIES_PAGE_INDEX de $SERIES_PAGES_NUMBER en $SERIES_HTML_LINK\n\nAnterior: $PREV_HTML_LINK\nSiguiente: $NEXT_HTML_LINK\n\"\"\"\n\nprev_only = \"\"\"\n¡Bienvenido a la última parte de $SERIES_HTML_LINK!\n¿Eres nuevo? Comienza con $FIRST_HTML_LINK para construir una base sólida.\n\nAnterior: $PREV_HTML_LINK\n\"\"\"\n\n# Plantilla de respaldo.\ndefault = \"Este artículo es parte de la serie $SERIES_HTML_LINK.\"\n\n# Conclusión.\n[extra.series_outro_templates]\nnext_only = \"\"\"\n¡Gracias por leer! 🙌\n\nContinúa tu viaje con $NEXT_HTML_LINK, donde $NEXT_DESCRIPTION\nO revisa el esquema completo de la serie [$SERIES_TITLE]($SERIES_PERMALINK).\n\"\"\"\n\nmiddle = \"\"\"\n---\n📝 Navegación de la serie\n\n- Anterior: $PREV_HTML_LINK\n- Siguiente: $NEXT_HTML_LINK\n- [Resumen de la serie]($SERIES_PERMALINK)\n\"\"\"\n\nprev_only = \"\"\"\n🎉 ¡Felicidades! Has completado $SERIES_HTML_LINK.\n\n¿Quieres repasar? Aquí comenzamos: $FIRST_HTML_LINK\nO revisa lo que acabamos de ver en $PREV_HTML_LINK.\n\"\"\"\n\n# Respaldo.\ndefault = \"\"\"\n---\nEste artículo es la parte $SERIES_PAGE_INDEX de $SERIES_PAGES_NUMBER en $SERIES_HTML_LINK.\n\"\"\"\n```\n\n### Variables personalizadas\n\nLas plantillas de series admiten variables personalizadas para incluir información adicional en toda tu serie. El proceso tiene dos pasos:\n\n1. Primero, define tus **marcadores** en la configuración de tu serie (`_index.md`):\n\n```toml,name=series/_index.md\n[extra]\nseries = true\nseries_template_placeholders = [\"$POSITION\", \"$TOPIC\", \"$DIFFICULTY\"]\n```\n\n2. Luego, en cada artículo de la serie, proporciona los valores para estos marcadores en `series_template_variables`:\n\n```toml,name=series/article.md\n[extra.series_template_variables]\nposition = \"primero\"\ntopic = \"Variables y tipos\"\ndifficulty = \"Principiante\"\n```\n\n### Uso de variables personalizadas\n\nPuedes usar tus variables personalizadas en cualquier plantilla, junto con las variables integradas:\n\n```toml,name=series/_index.md\n[extra.series_intro_templates]\ndefault = \"\"\"\nEste es el artículo $POSITION en $SERIES_HTML_LINK.\nTema de hoy: $TOPIC\nNivel de dificultad: $DIFFICULTY\n\"\"\"\n```\n\n{{ admonition(type=\"warning\", text=\"Aunque los marcadores se definen en mayúsculas (`$POSITION`), los nombres de variables en `series_template_variables` deben estar en minúsculas (`position`).\") }}\n\n### Ejemplo con variables personalizadas\n\n```toml,name=series/_index.md\n# En la configuración de la serie.\n[extra]\nseries = true\nseries_template_placeholders = [\"$LEARNING_TIME\", \"$KEY_CONCEPTS\"]\n\nseries_intro_templates.default = \"\"\"\n📚 Parte $SERIES_PAGE_INDEX de $SERIES_PAGES_NUMBER\n⏱️ Tiempo estimado: $LEARNING_TIME\n🔑 Conceptos clave: $KEY_CONCEPTS\n\"\"\"\n```\n\n```toml,name=series/02-learning-rust/index.md\n# En un artículo de la serie.\n[extra.series_template_variables]\nlearning_time = \"30 minutos\"\nkey_concepts = \"Funciones, manejo de errores, coincidencia de patrones\"\n```\n\nEsto generará:\n\n```txt\n📚 Parte 2 de 5\n⏱️ Tiempo estimado: 30 minutos\n🔑 Conceptos clave: Funciones, manejo de errores, coincidencia de patrones\n```\n\n{{ admonition(type=\"warning\", title=\"Variables faltantes\", text=\"Si usas un marcador en tus plantillas pero no proporcionas su valor en `series_template_variables`, la compilación fallará con un error que lista las variables faltantes.\") }}\n"
  },
  {
    "path": "content/blog/series/index.md",
    "content": "+++\ntitle = \"A Complete Guide to Series\"\ndate = 2024-11-08\nupdated = 2025-02-21\ndescription = \"Learn how to organize your posts into sequential series, perfect for tutorials, courses, and multi-part stories.\"\n\n[taxonomies]\ntags = [\"showcase\", \"tutorial\", \"FAQ\", \"series\"]\n\n[extra]\nquick_navigation_buttons = true\ntoc = true\nmermaid = true\nsocial_media_card = \"social_cards/es_blog_series.jpg\"\n+++\n\nA series organizes related posts in a sequential order, similar to chapters in a book. Unlike tags, which simply group related content, series suggest a specific reading order from start to finish.\n\nPosts within a series do not need to be published consecutively; the series feature brings together thematically linked posts in a coherent sequence.\n\nThe diagram below illustrates how series posts (3, 5, and 8) exist within the main blog flow while maintaining their own ordered sequence within Series 1.\n\n{% mermaid(full_width=true) %}\nflowchart\n    subgraph main[BLOG]\n        P1[Post 1]\n        P2[P2]\n        P3[P3]\n        P4[P4]\n        P5[P5]\n        P6[P6]\n        P7[P7]\n        P8[P8]\n        P9[P9]\n    end\n    subgraph series1[SERIES 1]\n        PS1[\"Series Post 1 (=P3)\"]\n        PS2[\"Series Post 2 (=P5)\"]\n        PS3[\"Series Post 3 (=P8)\"]\n    end\n    P3 o-.-o PS1\n    P5 o-.-o PS2\n    P8 o-.-o PS3\n{% end %}\n\n## Quick Start\n\n1. Create a directory for your series.\n2. Create `_index.md` in the series directory.\n3. Set up the `_index.md` front matter:\n\n    ```toml,name=series/_index.md\n    title = \"Learning Rust\"\n    template = \"series.html\"\n    sort_by = \"slug\"\n    transparent = true\n\n    [extra]\n    series = true\n    ```\n\n4. Create your series articles in this directory.\n\nWant more? Keep reading!\n\n## How Do Series Work?\n\nA series is just a section which is handled in a special way by tabi. For more details on sections, see the [Zola documentation](https://www.getzola.org/documentation/content/section/).\n\nTaking the example from the diagram above, the directory structure would be as follow:\n\n```txt\ncontent/\n    _index.md\n    blog/\n        _index.md\n        post1/\n            index.md\n        post2/\n            index.md\n        post4/\n            index.md\n        post6/\n            index.md\n        post7/\n            index.md\n        post9/\n            index.md\n        series1/\n            _index.md\n            post3/\n                index.md\n            post5/\n                index.md\n            post8/\n                index.md\n```\n\nTo create a series, you need to:\n\n1. Use the `series.html` template\n2. Set `series = true` in the section's `[extra]` configuration\n3. Enable `transparent = true` to integrate series posts with the parent blog section\n\nThe series main page displays an overview followed by a list of all posts in the series:\n\n{{ dual_theme_image(light_src=\"blog/series/img/series_light.webp\", dark_src=\"blog/series/img/series_dark.webp\" alt=\"a series\", full_width=true) }}\n\n## Jump to Posts\n\nIf the content of a series (the Markdown after the front matter in `_index.md`) is over 2000 characters, a \"Jump to posts\" link appears next to the series title.\n\n{{ dual_theme_image(light_src=\"blog/series/img/jump_to_series_posts_light.webp\", dark_src=\"blog/series/img/jump_to_series_posts_dark.webp\" alt=\"jump to series posts link\", full_width=true) }}\n\nTo force the feature on or off, set `show_jump_to_posts` in the `[extra]` section of your series section or in `config.toml`. This setting follows [the hierarchy](@/blog/mastering-tabi-settings/index.md#settings-hierarchy).\n\n## Series Pages and Order\n\nAll pages in the series section will be a series page. The series pages will be ordered as per the series section `sort_by`.\n\nWhile series maintain their own internal order, they remain independent from the main section's (e.g. `blog/`) chronological flow thanks to the `transparent` setting.\n\n### Sorting Options\n\nChoose from these sorting methods, each with its own advantages:\n\n{% wide_container() %}\n\n`sort_by` | pros                                                                                                                                      | cons\n---------|-------------------------------------------------------------------------------------------------------------------------------------------|---------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------\n `slug`    | The series pages order is made explicit in the path (e.g. `example.com/blog/series1/01-series-post-one`).                        | Each series page must be prefixed accordingly.\n `weight`  | The series pages order is easy to set up transparently.<br>First series post has weight `1`, second series post has weight `2` and so on. | Each series page must have its weight set accordingly.\n `date`    | The series pages order can be configured once in the series section configuration. No need to do anything on each series page.            | The series pages order has to be reversed because the first page is usually the oldest. This can only be achieved by paginating the series section (`paginate_by = 9999`) and reversing its order (`paginate_reversed = true`).\n\n{% end %}\n\n{{ admonition(type=\"danger\", title=\"Zola version to sort by date\", text=\"In order to properly reverse dates, Zola v0.19.3+ (unreleased) is required so that pagination information is available through the `get_section` function. Anything relying on the series pages order won't be correct in a series page otherwise (e.g. previous/next series page, ordered and unordered list…) See [Zola PR #2653](https://github.com/getzola/zola/pull/2653).\") }}\n\n### Page Indexing\n\nPages in a series are indexed starting from 1, following their `sort_by` order. To reverse the indexing (making the first page have the highest index instead), add this setting to `_index.md` or `config.toml`:\n\n```toml\n[extra]\npost_listing_index_reversed = true  # Defaults to false if unset.\n```\n\n{{ dual_theme_image(light_src=\"blog/series/img/series_reversed_light.webp\", dark_src=\"blog/series/img/series_reversed_dark.webp\" alt=\"a series with indexes reversed\", full_width=true) }}\n\nThis setting follows [the hierarchy](@/blog/mastering-tabi-settings/index.md#settings-hierarchy).\n\n## Intro and Outro Templates\n\nSeries articles can have automatic introduction and conclusion sections. These are configured in your series' `_index.md`. A basic example:\n\n```toml,name=series/_index.md\n[extra.series_intro_templates]\ndefault = \"This article is part of the $SERIES_HTML_LINK series.\"\n\n[extra.series_outro_templates]\ndefault = \"Thanks for reading part $SERIES_PAGE_INDEX of $SERIES_HTML_LINK!\"\n```\n\nThe intro and outro sections each have their own CSS classes (`series-page-intro` and `series-page-outro`), allowing you to customize their appearance through [custom CSS](@/blog/mastering-tabi-settings/index.md#custom-css).\n\n### Template Types\n\nThe series system uses different templates based on an article's position in the series:\n\n- `next_only` - Used for the first article (has next article but no previous)\n- `middle` - Used for articles with both previous and next articles\n- `prev_only` - Used for the last article (has previous article but no next)\n- `default` - Fallback template used when a specific position template isn't defined\n\nThe system automatically determines which template to use based on the article's position. The templates are defined in the series configuration (`_index.md`), as `extra.series_intro_templates` and `extra.series_outro_templates`.:\n\n```toml,name=series/_index.md\n[extra.series_intro_templates]\nnext_only = \"Welcome to part 1! Next up: $NEXT_HTML_LINK\"\nmiddle = \"Previous: $PREV_HTML_LINK | Next: $NEXT_HTML_LINK\"\nprev_only = \"The final chapter! Previously: $PREV_HTML_LINK\"\ndefault = \"Part $SERIES_PAGE_INDEX of $SERIES_PAGES_NUMBER\"\n```\n\nAll templates are optional. Template selection follows a priority system:\n\n1. If a position-specific template exists (`next_only`, `middle`, or `prev_only`), it will be used\n2. Otherwise, the `default` template is used\n3. If no templates are defined at all, no series information will be displayed\n\nSee the [template example](#template-example) for a more elaborate example.\n\n### Placement in Content\n\nBy default:\n\n- Series introductions appear at the start of your article\n- Series outro appears at the end (before footnotes, if any)\n\nYou can control exactly where these appear using `<!-- series_intro -->` and `<!-- series_outro -->` in your Markdown:\n\n```markdown\nThis paragraph appears before the series introduction.\n\n<!-- series_intro -->\n\nMain content of the article.\n\n<!-- series_outro -->\n\n## Learning Resources\n\nExtra content…\n\n[^1]: Footnotes will always appear at the end.\n```\n\n## Variables\n\nSeries templates use a flexible variable system that lets you:\n\n1. Reference series information (title, links)\n2. Add navigation between articles\n3. Show progress indicators\n4. Include custom information using your own variables\n\nVariables are placeholders starting with `$` that get replaced with actual content when your site builds. For example, `$SERIES_HTML_LINK` becomes a clickable link to your series index page.\n\nThere are three types of variables:\n\n- [**Basic Series Variables**](#basic-series-variables): General information about the series\n- [**Navigation Variables**](#navigation-variables): Links to previous/next articles\n- [**Custom Variables**](#custom-variables): Your own placeholders for additional information\n\n### Basic Series Variables\n\n{% wide_container() %}\n\n| Variable | Availability | Returns | Description | Example Usage | Example Output |\n|----------|-------------|---------|-------------|---------------|----------------|\n| `$SERIES_TITLE` | Always | Text | Plain text title of the series | `Part of $SERIES_TITLE` | Part of Learn Rust |\n| `$SERIES_PERMALINK` | Always | Text | URL to series index | `[See all posts]($SERIES_PERMALINK)` | [See all posts](/series/learn-rust) |\n| `$SERIES_HTML_LINK` | Always | HTML | Ready-to-use link to series | `Welcome to $SERIES_HTML_LINK!` | Welcome to <a href=\"/series/learn-rust\">Learn Rust</a>! |\n| `$SERIES_PAGES_NUMBER` | Always | Number | Total articles in series | `A $SERIES_PAGES_NUMBER part series` | A 5 part series |\n| `$SERIES_PAGE_INDEX` | Always | Number | Current article's position | `Part $SERIES_PAGE_INDEX of $SERIES_PAGES_NUMBER` | Part 3 of 5 |\n| `$SERIES_PAGES_OLIST` | Always | HTML | Ordered list of all articles | `Articles in series: $SERIES_PAGES_OLIST` | Articles in series: <ol><li>Current article</li><li><a href=\"...\">Other articles</a></li></ol> |\n| `$SERIES_PAGES_ULIST` | Always | HTML | Unordered list of all articles | `Articles in series: $SERIES_PAGES_ULIST` | Articles in series: <ul><li>Current article</li><li><a href=\"...\">Other articles</a></li></ul> |\n\n{% end %}\n\n{{ admonition(type=\"tip\", title=\"TIP: Custom text with permalinks\", text='Markdown links like `[text]($SERIES_PERMALINK)` will be marked (and [styled](@/blog/mastering-tabi-settings/index.md#external-link-indicator)) as external. If you need custom text and want to avoid external styling, use HTML: `<a href=\\\"$SERIES_PERMALINK\\\">your text</a>`.') }}\n\n### Navigation Variables\n\n{% wide_container() %}\n\n| Variable | Availability | Returns | Description | Example Usage | Example Output |\n|----------|-------------|---------|-------------|---------------|----------------|\n| `$PREV_TITLE` | Previous exists | Text | Previous article's title | `Previously: $PREV_TITLE` | Previously: Setting Up Your Environment |\n| `$PREV_PERMALINK` | Previous exists | Text | URL to previous article | `[← Back]($PREV_PERMALINK)` | [← Back](/series/learn-rust/setup) |\n| `$PREV_HTML_LINK` | Previous exists | HTML | Ready-to-use link to previous | `Read $PREV_HTML_LINK first` | Read <a href=\"/series/learn-rust/setup\">Setting Up Your Environment</a> first |\n| `$PREV_DESCRIPTION` | Previous exists | Text | Description of previous article | `Recap: $PREV_DESCRIPTION` | Recap: Setting up Rust |\n| `$NEXT_TITLE` | Next exists | Text | Next article's title | `Next up: $NEXT_TITLE` | Next up: Advanced Patterns |\n| `$NEXT_PERMALINK` | Next exists | Text | URL to next article | `[Continue →]($NEXT_PERMALINK)` | [Continue →](/series/learn-rust/patterns) |\n| `$NEXT_HTML_LINK` | Next exists | HTML | Ready-to-use link to next | `Continue with $NEXT_HTML_LINK` | Continue with <a href=\"/series/learn-rust/patterns\">Advanced Patterns</a> |\n| `$NEXT_DESCRIPTION` | Next exists | Text | Description of next article | `Coming up: $NEXT_DESCRIPTION` | Coming up: Learn about Rust's advanced pattern matching features |\n\n{% end %}\n\n### First Article Reference\n\n{% wide_container() %}\n\n| Variable | Availability | Returns | Description | Example Usage | Example Output |\n|----------|-------------|---------|-------------|---------------|----------------|\n| `$FIRST_TITLE` | Always | Text | First article's title | `Start with $FIRST_TITLE` | Start with Introduction to Rust |\n| `$FIRST_HTML_LINK` | Always | HTML | Ready-to-use link to first article | `Begin at $FIRST_HTML_LINK` | Begin at <a href=\"/series/learn-rust/intro\">Introduction to Rust</a> |\n\n{% end %}\n\n### Template Example\n\n{{ admonition(type=\"tip\", title=\"HTML vs text variables\", text=\"Use HTML variables (ending in `_HTML_LINK`) when you want ready-made links. Use text variables (ending in `_TITLE` or `_PERMALINK`) when you want more control over the formatting.\") }}\n\n```toml,name=series/_index.md\n# Introduction.\n[extra.series_intro_templates]\nnext_only = \"\"\"\nWelcome to $SERIES_HTML_LINK! This $SERIES_PAGES_NUMBER-part series will teach you Rust from scratch.\n\nUp next: $NEXT_HTML_LINK - $NEXT_DESCRIPTION\n\"\"\"\n\nmiddle = \"\"\"\n📚 Part $SERIES_PAGE_INDEX of $SERIES_PAGES_NUMBER in $SERIES_HTML_LINK\n\nPreviously: $PREV_HTML_LINK\nNext up: $NEXT_HTML_LINK\n\"\"\"\n\nprev_only = \"\"\"\nWelcome to the final part of $SERIES_HTML_LINK!\nNew here? Start with $FIRST_HTML_LINK to build a strong foundation.\n\nPreviously: $PREV_HTML_LINK\n\"\"\"\n\n# Fallback template.\ndefault = \"This article is part of the $SERIES_HTML_LINK series.\"\n\n# Outro.\n[extra.series_outro_templates]\nnext_only = \"\"\"\nThanks for reading! 🙌\n\nContinue your journey with $NEXT_HTML_LINK, where $NEXT_DESCRIPTION\nOr check out the complete [$SERIES_TITLE]($SERIES_PERMALINK) series outline.\n\"\"\"\n\nmiddle = \"\"\"\n---\n📝 Series Navigation\n\n- Previous: $PREV_HTML_LINK\n- Next: $NEXT_HTML_LINK\n- [Series Overview]($SERIES_PERMALINK)\n\"\"\"\n\nprev_only = \"\"\"\n🎉 Congratulations! You've completed $SERIES_HTML_LINK.\n\nWant to review? Here's where we started: $FIRST_HTML_LINK\nOr check what we just covered in $PREV_HTML_LINK.\n\"\"\"\n\n# Fallback.\ndefault = \"\"\"\n---\nThis article is part $SERIES_PAGE_INDEX of $SERIES_PAGES_NUMBER in $SERIES_HTML_LINK.\n\"\"\"\n```\n\n### Custom Variables\n\nSeries templates support custom variables for additional information you want to include across your series. The process takes two steps:\n\n1. First, define your **placeholders** in your series configuration (`_index.md`):\n\n```toml,name=series/_index.md\n[extra]\nseries = true\nseries_template_placeholders = [\"$POSITION\", \"$TOPIC\", \"$DIFFICULTY\"]\n```\n\n2. Then, in each series article, provide the values for these placeholders in `series_template_variables`:\n\n```toml,name=series/article.md\n[extra.series_template_variables]\nposition = \"first\"\ntopic = \"Variables and Types\"\ndifficulty = \"Beginner\"\n```\n\n### Using Custom Variables\n\nYou can use your custom variables in any template, alongside the built-in variables:\n\n```toml,name=series/_index.md\n[extra.series_intro_templates]\ndefault = \"\"\"\nThis is the $POSITION article in $SERIES_HTML_LINK.\nToday's topic: $TOPIC\nDifficulty level: $DIFFICULTY\n\"\"\"\n```\n\n{{ admonition(type=\"warning\", text=\"While placeholders are defined with uppercase (`$POSITION`), the variable names in `series_template_variables` must be lowercase (`position`).\") }}\n\n### Example with Custom Variables\n\n```toml,name=series/_index.md\n# In the series configuration.\n[extra]\nseries = true\nseries_template_placeholders = [\"$LEARNING_TIME\", \"$KEY_CONCEPTS\"]\n\nseries_intro_templates.default = \"\"\"\n📚 Part $SERIES_PAGE_INDEX of $SERIES_PAGES_NUMBER\n⏱️ Estimated time: $LEARNING_TIME\n🔑 Key concepts: $KEY_CONCEPTS\n\"\"\"\n```\n\n```toml,name=series/02-learning-rust/index.md\n# In an article of the series.\n[extra.series_template_variables]\nlearning_time = \"30 minutes\"\nkey_concepts = \"Functions, Error Handling, Pattern Matching\"\n```\n\nThis will output:\n\n```txt\n📚 Part 2 of 5\n⏱️ Estimated time: 30 minutes\n🔑 Key concepts: Functions, Error Handling, Pattern Matching\n```\n\n{{ admonition(type=\"warning\", title=\"Missing Variables\", text=\"If you use a placeholder in your templates but don't provide its value in `series_template_variables`, the build will fail with an error listing the missing variables.\") }}\n"
  },
  {
    "path": "content/blog/shortcodes/index.ca.md",
    "content": "+++\ntitle = \"Shortcodes personalitzats\"\ndate = 2023-02-19\nupdated = 2025-10-21\ndescription = \"Aquest tema inclou alguns shortcodes personalitzats útils que pots utilitzar per millorar les teves publicacions. Ja sigui per mostrar imatges que s'adapten als temes clar i fosc, o per donar format a una secció de referències amb un aspecte professional, aquests shortcodes personalitzats t'ajudaran.\"\n\n[taxonomies]\ntags = [\"funcionalitat\", \"shortcodes\"]\n\n[extra]\ntoc = true\ntoc_levels = 2\nquick_navigation_buttons = true\ncode_block_name_links = true\nmermaid = true\nsocial_media_card = \"social_cards/ca_blog_shortcodes.jpg\"\n+++\n\n## Shortcodes de diagrames\n\n### Diagrames de Mermaid\n\n[Mermaid](https://github.com/mermaid-js/mermaid) és una eina de diagramació i gràfics que utilitza text i codi per generar diagrames. Admet diagrames de flux, diagrames de seqüència, gràfics de Gantt i més.\n\nPer incloure un diagrama Mermaid a la teva publicació, cal fer dues coses:\n\n1. Estableix `mermaid = true` a la secció `[extra]` del front matter de la teva pàgina, secció o `config.toml`. Això carregarà el JavaScript necessari per renderitzar els diagrames.\n\n2. Utilitza el shortcode `mermaid()` per definir el teu diagrama. Per exemple:\n\n```plain\n{%/* mermaid() */%}\nclassDiagram\n    class DistorsionsCognitives {\n        +PensamentTotORes()\n        +Sobregeneralitzacio()\n        +FiltreMental()\n        +TreureConclusionsPrecipitades()\n    }\n    class PensamentTotORes {\n        +VeureEnExtrems()\n    }\n    class Sobregeneralitzacio {\n        +GeneralitzarDUnic()\n    }\n    class FiltreMental {\n        +EnfocarseEnNegatiu()\n    }\n    class TreureConclusionsPrecipitades {\n        +FerSuposicions()\n    }\n    DistorsionsCognitives *-- PensamentTotORes\n    DistorsionsCognitives *-- Sobregeneralitzacio\n    DistorsionsCognitives *-- FiltreMental\n    DistorsionsCognitives *-- TreureConclusionsPrecipitades\n{%/* end */%}\n```\n\nEl diagrama es renderitzarà així:\n\n{% mermaid() %}\nclassDiagram\n    class DistorsionsCognitives {\n        +PensamentTotORes()\n        +Sobregeneralitzacio()\n        +FiltreMental()\n        +TreureConclusionsPrecipitades()\n    }\n    class PensamentTotORes {\n        +VeureEnExtrems()\n    }\n    class Sobregeneralitzacio {\n        +GeneralitzarDUnic()\n    }\n    class FiltreMental {\n        +EnfocarseEnNegatiu()\n    }\n    class TreureConclusionsPrecipitades {\n        +FerSuposicions()\n    }\n    DistorsionsCognitives *-- PensamentTotORes\n    DistorsionsCognitives *-- Sobregeneralitzacio\n    DistorsionsCognitives *-- FiltreMental\n    DistorsionsCognitives *-- TreureConclusionsPrecipitades\n{% end %}\n\nEl shortcode de Mermaid admet dos paràmetres:\n\n- `invertible`: Si s'estableix a `true` (per defecte), el diagrama invertirà els seus colors en mode fosc, igual que les [imatges invertibles](#imatge-invertible).\n- `full_width`: Permet que el diagrama ocupi l'amplada de la capçalera. Mira [imatge d'amplada completa](#imatge-d-amplada-completa).\n\n{{ admonition(type=\"tip\", title=\"CONSELL\", text=\"Empra l'[editor de Mermaid](https://mermaid.live/) per crear i previsualitzar els teus diagrames.\") }}\n\n#### Ús\n\n```\n{%/* mermaid(invertible=true, full_width=false) */%}\n\nEl teu codi Mermaid va aquí.\n\n`invertible` or `full_width` poden ometre's per emprar els valors per defecte.\n\n{%/* end */%}\n```\n\n## Shortcodes d'imatge\n\nTots els shortcodes d'imatge admeten rutes absolutes, rutes relatives, i fonts remotes en el paràmetre `src`.\n\nTots els shortcodes d'imatge tenen els següents paràmetres opcionals:\n\n- `raw_path`. Per defecte és `false`. Si es configura a `true`, el paràmetre `src` s'utilitzarà tal qual. Útil per a actius ubicats a la mateixa carpeta que tenen un slug personalitzat (vegeu [Zola issue #2598](https://github.com/getzola/zola/issues/2598)).\n- `inline`. Valor predeterminat: `false`. Si s'estableix a `true`, la imatge es mostrarà en línia amb el text.\n- `full_width`. Valor predeterminat: `false` (vegeu [a sota](#imatge-d-amplada-completa)).\n- `lazy_loading`. Valor predeterminat: `true`.\n\n### Imatges per a temes duals\n\nÚtil si vols utilitzar una imatge diferent pels temes clar i fosc:\n\n{{ dual_theme_image(light_src=\"img/paris_day.webp\", dark_src=\"img/paris_night.webp\" alt=\"La Torre Eiffel\") }}\n\n#### Ús\n```\n{{/* dual_theme_image(light_src=\"img/paris_day.webp\", dark_src=\"img/paris_night.webp\" alt=\"La Torre Eiffel\") */}}\n```\n\n### Imatge invertible\n\nÚtil per a gràfics, dibuixos de línies, diagrames… Inverteix els colors de la imatge. La imatge original s'utilitzarà per al tema clar.\n\n{{ invertible_image(src=\"img/graph.webp\", alt=\"Gràfic invertible\") }}\n\n#### Ús\n\n```\n{{/* invertible_image(src=\"img/graph.webp\", alt=\"Gràfic invertible\") */}}\n```\n\n### Imatge regulable\n\nLes imatges amb massa brillantor o contrast poden ser molestes en un fons fosc. Aquí tens un exemple d'una fotografia que s'atenua quan s'activa el tema fosc.\n\n{{ dimmable_image(src=\"img/desert_by_oskerwyld.webp\", alt=\"Fotografia d'un desert, cel celestial\") }}\n\n#### Ús\n\n```\n{{/* dimmable_image(src=\"img/desert_by_oskerwyld.webp\", alt=\"Fotografia d'un desert, cel celestial\") */}}\n```\n\n### Canvi d'imatge en passar el cursor\n\nLa imatge mostrada canvia quan l'usuari passa el cursor per sobre. Útil per a comparacions d'abans i després, per exemple.\n\n{{ image_hover(default_src=\"img/edited.webp\", hovered_src=\"img/raw.webp\", default_alt=\"Foto editada\", hovered_alt=\"Foto original\") }}\n\n#### Ús\n\n```\n{{/* image_hover(default_src=\"img/before.webp\", hovered_src=\"img/after.webp\", default_alt=\"Foto editada\", hovered_alt=\"Foto original\") */}}\n```\n\n### Canvi d'imatge via clic\n\nMostra una imatge i canvia a una diferent en fer clic. Ideal per destacar diferències o cridar l'atenció sobre detalls.\n\n{{ image_toggler(default_src=\"img/mojave_day.webp\", toggled_src=\"img/mojave_night.webp\", default_alt=\"Mojave de dia\", toggled_alt=\"Mojave de nit\") }}\n\n#### Ús\n\n```\n{{/* image_toggler(default_src=\"img/mojave_day.webp\", toggled_src=\"img/mojave_night.webp\", default_alt=\"Mojave de dia\", toggled_alt=\"Mojave de nit\") */}}\n```\n\n### Imatge d'amplada completa\n\nLa imatge s'expandirà per coincidir amb l'amplada de la capçalera, que normalment és més ampla que el text de l'article (excepte en mòbil/finestres petites).\n\nTots els altres shortcodes d'imatges poden utilizar l'amplada completa assignant `true` al paràmetre opcional `full_width`.\n\n{{ full_width_image(src=\"img/amsterdam_by_oskerwyld.webp\", alt=\"Fotografia d'un canal a Àmsterdam\") }}\n\n#### Ús\n\n```\n{{/* full_width_image(src=\"img/amsterdam_by_oskerwyld.webp\", alt=\"Fotografia d'un canal a Àmsterdam\") */}}\n```\n\n## Shortcodes socials\n\n### iine\n\n{{ aside(text=\"Per afegir-lo a totes les publicacions, estableix `iine = true` a la secció `[extra]` del teu `config.toml`.\") }}\n\nAquest shortcode et permet afegir botons addicionals d'[iine.to](https://iine.to) a les teves publicacions, com aquest:\n\n{{ iine(slug=\"/blog/shortcodes/demo-button\") }}\n\n#### Ús\n\n```\n{{/* iine(icon=\"heart\", slug=\"/post/el-meu-slug-de-post/like\", label=\"M'agrada aquesta publicació\") */}}\n```\n\nEl shortcode accepta els següents paràmetres opcionals:\n\n- `icon`: La icona a mostrar. Pot ser `heart`, `thumbs_up`, `upvote`, o qualsevol emoji.\n- `slug`: Un identificador únic. Per defecte és la ruta de la pàgina actual. Útil si vols més d'un botó a la mateixa pàgina.\n- `label`: L'etiqueta d'accessibilitat per al botó. Per defecte és \"M'agrada aquesta publicació\".\n\n## Shortcodes de codi\n\n### Mostrar ruta o URL\n\nPots mostrar una ruta o URL per a un bloc de codi utilitzant la sintaxi nativa de Zola:\n\n{{ aside(text=\"Requereix Zola 0.20.0 o superior.\") }}\n\n````\n```rust,name=src/main.rs\nfn main() {\n    println!(\"Hola, món!\");\n}\n```\n````\n\nAixò renderitza:\n\n```rust,name=src/main.rs\nfn main() {\n    println!(\"Hola, món!\");\n}\n```\n\nSi estableixes el `name` com una URL (és a dir, comença amb `http` o `https`), pots convertir-lo en un enllaç clicable. Això és particularment útil quan s'utilitza juntament amb el [shortcode de text remot](#text-remot).\n\n{{ admonition(type=\"warning\", title=\"JavaScript necessari\", text=\"La funció d'URLs clicables requereix JavaScript. Per habilitar-la, configura `code_block_name_links = true` a la secció `[extra]` de la teva pàgina, secció, o `config.toml`.\") }}\n\n```plain,name=https://github.com/welpo/doteki/blob/main/.gitignore\n__pycache__/\n*coverage*\n.vscode/\ndist/\n```\n\n### Suport de shortcode heretat\n\nEl shortcode `add_src_to_code_block` segueix funcionant per retrocompatibilitat però serà descontinuat en una versió futura. Si us plau, utilitza la sintaxi nativa de Zola:\n\n```\n# Forma antiga (descontinuada):\n{{/* add_src_to_code_block(src=\"ruta/al/fitxer.rs\") */}}\n\n# Forma nova (recomanada):\n```rust,name=ruta/al/fitxer.rs\n```\n\n#### Ús\n\n````\n{{/* add_src_to_code_block(src=\"https://github.com/welpo/doteki/blob/main/.gitignore\") */}}\n```plain\n**pycache**/\n*coverage*\n.vscode/\ndist/\n```\n````\n\n## Shortcodes de text\n\n### Aside (nota al marge)\n\nAfegeix contingut complementari als marges en pantalles amples, o com a blocs distintius en mòbil.\n\n{{ aside(text=\"*Nota al marge* ve de *nota* (del llatí, 'marca' o 'senyal') i *marge* (del llatí *margo*, 'vora' o 'límit').\") }}\n\nEl shortcode accepta dos paràmetres:\n\n- `position`: Establir com a `\"right\"` per col·locar al marge dret (per defecte, esquerre)\n- El contingut es pot proporcionar mitjançant el paràmetre `text` o entre les etiquetes del shortcode\n\n#### Ús\n\n{{ admonition(type=\"warning\", text=\"Separa la definició de la nota del shortcode amb dues línies en blanc per evitar errors de renderització.\") }}\n\nFent servir el paràmetre `text`:\n\n```\n{{/* aside(text=\"*Nota al marge* ve de *nota* (del llatí, 'marca' o 'senyal') i *marge* (del llatí *margo*, 'vora' o 'límit').\") */}}\n```\n\nFent servir el cos del contingut i indicant la posició a la dreta:\n\n```\n{%/* aside(position=\"right\") */%}\nUna nota més llarga que\npot ocupar diverses línies.\n\nS'admet *Markdown*.\n{%/* end */%}\n```\n\n### Text remot\n\nAfegeix text des d'una URL remota o un arxiu local.\n\nEl shortcode accepta tres paràmetres:\n\n- `src`: L'URL d'origen o ruta del fitxer (obligatori)\n- `start`: Primera línia a mostrar (opcional, comença a 1)\n- `end`: Número de l'última línia (opcional, per defecte és 0, l'última línia)\n\n{{ admonition(type=\"info\", text=\"`start` i `end` són inclusius. `start=3, end=3` mostrarà només la tercera línia.\") }}\n\n**Important**:\n\n- **Arxius remots VS arxius locals**: Si `src` comença amb \"http\", es tractarà com un arxiu remot. D'altra banda, s'assumeix que és una ruta d'arxiu local.\n- **Accés a arxius**: Atès que utilitza la funció [`load_data`](https://www.getzola.org/documentation/templates/overview/#load-data) de Zola, els arxius locals han d'estar dins del directori de Zola —vegeu la [lògica de cerca d'arxius](https://www.getzola.org/documentation/templates/overview/#file-searching-logic). Desde [tabi 2.16.0](https://github.com/welpo/tabi/releases/tag/v2.16.0), el shortcode admet també rutes relatives.\n- **Formateig de blocs de codi**: Per mostrar el text com un bloc de codi, has d'afegir manualment les tanques de codi Markdown (cometes inverses) i, opcionalment, especificar el llenguatge de programació per al ressaltat sintàctic.\n\n#### Ús\n\nAfegeix un script de Python remot dins d'un bloc de codi amb ressaltat sintàctic:\n\n````\n```python\n{{/* remote_text(src=\"https://example.com/script.py\") */}}\n```\n````\n\nMostra el text d'un arxiu local:\n\n```\n{{/* remote_text(src=\"ruta/a/arxiu.txt\") */}}\n```\n\nMostreu només les línies 3 a 5 d'un arxiu local:\n\n```\n{{/* remote_text(src=\"ruta/a/arxiu.txt\", start=3, end=5) */}}\n```\n\n### Advertències\n\nDestaca informació amb aquests shortcodes d'advertència/alerta. Hi ha cinc tipus (`type`): `note`, `tip`, `info`, `warning`, i `danger`.\n\n{{ admonition(type=\"note\", text=\"Contingut amb **sintaxi** *Markdown*. Consulta [aquesta `api`](#).\") }}\n\n{{ admonition(type=\"tip\", text=\"Contingut amb **sintaxi** *Markdown*. Consulta [aquesta `api`](#).\") }}\n\n{{ admonition(type=\"info\", text=\"Contingut amb **sintaxi** *Markdown*. Consulta [aquesta `api`](#).\") }}\n\n{{ admonition(type=\"warning\", text=\"Contingut amb **sintaxi** *Markdown*. Consulta [aquesta `api`](#).\") }}\n\n{{ admonition(type=\"danger\", text=\"Contingut amb **sintaxi** *Markdown*. Consulta [aquesta `api`](#).\") }}\n\nPots canviar el `title` i la `icon` de l'advertència. Ambdós paràmetres accepten text i per defecte coincideixen amb el tipus d'advertència. `icon` pot ser qualsevol dels tipus d'advertència disponibles.\n\n{{ admonition(type=\"note\", icon=\"tip\", title=\"Títol i icona personalitzats\", text=\"Contingut amb **sintaxi** *Markdown*. Consulta [aquesta `api`](#).\") }}\n\n#### Ús\n\nPots utilitzar les advertències de dues maneres:\n\n1. En línia amb paràmetres:\n\n```md\n{{/* admonition(type=\"danger\", icon=\"tip\", title=\"Un consell important\", text=\"Mantingues-te hidratat\") */}}\n```\n\n2. Amb contingut al cos:\n\n```md\n{%/* admonition(type=\"danger\", icon=\"tip\", title=\"Un consell important\") */%}\nMantingues-te hidratat\n\nAquest mètode és especialment útil per a contingut llarg o múltiples paràgrafs.\n{%/* end */%}\n```\n\nAmbdós mètodes admeten els mateixos paràmetres (`type`, `icon`, i `title`).\n\n### Cites multillenguatge\n\nAquest shortcode permet mostrar una cita traduïda i en el llenguatge original:\n\n{{ multilingual_quote(original=\"Die Logik ist zwar unerschütterlich, aber einem Menschen, der leben will, widersteht sie nicht.\", translated=\"La lògica, encara que inquebrantable, no resisteix a un home que vol viure.\", author=\"Franz Kafka\") }}\n\n#### Ús\n\n```\n{{/* multilingual_quote(original=\"Die Logik ist zwar unerschütterlich, aber einem Menschen, der leben will, widersteht sie nicht.\", translated=\"La lògica, encara que inquebrantable, no resisteix a un home que vol viure.\", author=\"Franz Kafka\") */}}\n```\n\n### Referències amb sagnat invertit\n\nAquest shortcode formata una secció de referència amb un sagnat invertit de la següent manera:\n\n{% references() %}\n\nAlderson, E. (2015). Ciberseguretat i justícia social: Una crítica a la hegemonia corporativa en un món digital. *New York Journal of Technology, 11*(2), 24-39. [https://doi.org/10.1007/s10198-022-01497-6](https://doi.org/10.1007/s10198-022-01497-6).\n\nFunkhouser, M. (2012). Les normes socials d'indecència: Un anàlisi del comportament desviat a la societat contemporània. *Los Angeles Journal of Sociology, 16*(3), 41-58. [https://doi.org/10.1093/jmp/jhx037](https://doi.org/10.1093/jmp/jhx037).\n\nSchrute, D. (2005). La revolució de l'agricultura de remolatxa: Un anàlisi de la innovació agrícola. *Scranton Agricultural Quarterly, 38*(3), 67-81.\n\nSteinbrenner, G. (1997). L'anàlisi cost-benefici de George Costanza: Un anàlisi del comportament de presa de riscos en el lloc de treball. *New York Journal of Business, 12*(4), 112-125.\n\nWinger, J. A. (2010). L'art del debat: Un examen de la retòrica en el model de les Nacions Unides del Greendale Community College. *Colorado Journal of Communication Studies, 19*(2), 73-86. [https://doi.org/10.1093/6seaons/1movie](https://doi.org/10.1093/6seaons/1movie).\n\n{% end %}\n\n#### Ús\n\n```\n{%/* references() */%}\n\nLes teves referències van aquí.\n\nCada una en una nova línia. Es renderitzarà el Markdown (enllaços, cursiva…).\n\n{%/* end */%}\n```\n\n### Spoiler\n\nAquest shortcode amaga el text fins que l'usuari fa clic per revelar-lo. Per exemple: A l'antiga Roma, el *vomitorium* era {{ spoiler(text=\"l'entrada a través de la qual les multituds entraven i sortien d'un estadi, no un espai especial utilitzat per a vomitar durant els àpats. Sí, [de debó](https://ca.wikipedia.org/wiki/Vomitori).\") }}\n\nCom veus, el Markdown es renderitza.\n\nAquest shortcode té l'opció `fixed_blur` per difuminar el text \"SPOILER\", en lloc de difuminar el contingut real. Per exemple: és {{ spoiler(text=\"innecessari\", fixed_blur=true)}} esperar 24 hores abans de denunciar la desaparició d'una persona.\n\n#### Ús\n\n```\n{{/* spoiler(text=\"text a amagar\", fixed_blur=false) */}}\n```\n\n## Contenidors\n\n### Contenidor ample\n\nUtilitza aquest codi curt si vols tenir una taula, paràgraf, bloc de codi… més ample. A l'escriptori, ocuparà l'amplada de la capçalera. A mòbils no tindrà efecte, excepte per les taules, que guanyaran scroll horitzontal.\n\n{% wide_container() %}\n\n| Títol             |  Any  | Director              | Director de fotografia | Gènere         | IMDb  | Durada       |\n|-------------------|-------|----------------------|-------------------------|----------------|-------|--------------|\n| Beoning           | 2018  | Lee Chang-dong       | Hong Kyung-pyo          | Drama/Misteri  | 7.5   | 148 min      |\n| The Master        | 2012  | Paul Thomas Anderson | Mihai Mălaimare Jr.     | Drama/Història | 7.1   | 137 min      |\n| The Tree of Life  | 2011  | Terrence Malick      | Emmanuel Lubezki        | Drama          | 6.8   | 139 min      |\n\n{% end %}\n\n#### Ús\n\n```\n{%/* wide_container() */%}\n\nPosa el teu bloc de codi, paràgraf, taula… aquí.\n\nEl Markdown, per suposat, serà interpretat.\n\n{%/* end */%}\n```\n\n### Forçar direcció del text\n\nForça la direcció del text d'un bloc de contingut. Substitueix tant la configuració global `force_codeblock_ltr` com la direcció general del document.\n\nAccepta el paràmetre `direction`: la direcció de text desitjada. Pot ser \"ltr\" (d'esquerra a dreta) o \"rtl\" (de dreta a esquerra). Per defecte és \"ltr\".\n\n{% force_text_direction(direction=\"rtl\") %}\n```python\ndef مرحبا_بالعالم():\n    print(\"مرحبا بالعالم!\")\n```\n{% end %}\n\n#### Ús\n\nEn una pàgina LTR podem forçar que un bloc de codi sigui RTL (com es mostra a dalt) de la següent manera:\n\n````\n{%/* force_text_direction(direction=\"rtl\") */%}\n\n```python\ndef مرحبا_بالعالم():\n    print(\"مرحبا بالعالم!\")\n```\n\n{%/* end */%}\n````\n"
  },
  {
    "path": "content/blog/shortcodes/index.es.md",
    "content": "+++\ntitle = \"Shortcodes personalizados\"\ndate = 2023-02-19\nupdated = 2025-10-21\ndescription = \"Este tema incluye algunos shortcodes personalizados útiles que puedes utilizar para mejorar tus publicaciones. Puedes mostrar imágenes que se adapten a los temas claro y oscuro, dar formato a una sección de referencias con un aspecto profesional, y más.\"\n\n[taxonomies]\ntags = [\"funcionalidad\", \"shortcodes\"]\n\n[extra]\ntoc = true\ntoc_levels = 2\nquick_navigation_buttons = true\ncode_block_name_links = true\nmermaid = true\nsocial_media_card = \"social_cards/es_blog_shortcodes.jpg\"\n+++\n\n## Shortcodes de diagramas\n\n### Diagramas de Mermaid\n\n[Mermaid](https://github.com/mermaid-js/mermaid) es una herramienta de diagramación y gráficos que usa texto y código para generar diagramas. Admite diagramas de flujo, diagramas de secuencia, gráficos de Gantt y más.\n\nPara incluir un diagrama Mermaid en tu publicación, sigue estos dos pasos:\n\n1. Establece `mermaid = true` en la sección `[extra]` del front matter de tu página, sección o `config.toml`. Esto cargará el JavaScript necesario para renderizar los diagramas.\n\n2. Usa el shortcode `mermaid()` para definir tu diagrama. Por ejemplo:\n\n```txt\n{%/* mermaid() */%}\nclassDiagram\n    class DistorsionesCognitivas {\n        +PensamientoTodoONada()\n        +Sobregeneralizacion()\n        +FiltroMental()\n        +SacarConclusionesPrecipitadas()\n    }\n    class PensamientoTodoONada {\n        +VerEnExtremos()\n    }\n    class Sobregeneralizacion {\n        +GeneralizarDeUnicoEjemplo()\n    }\n    class FiltroMental {\n        +EnfocarseEnNegativo()\n    }\n    class SacarConclusionesPrecipitadas {\n        +HacerSuposiciones()\n    }\n    DistorsionesCognitivas *-- PensamientoTodoONada\n    DistorsionesCognitivas *-- Sobregeneralizacion\n    DistorsionesCognitivas *-- FiltroMental\n    DistorsionesCognitivas *-- SacarConclusionesPrecipitadas\n{%/* end */%}\n```\n\nEl diagrama se renderizará así:\n\n{% mermaid() %}\nclassDiagram\n    class DistorsionesCognitivas {\n        +PensamientoTodoONada()\n        +Sobregeneralizacion()\n        +FiltroMental()\n        +SacarConclusionesPrecipitadas()\n    }\n    class PensamientoTodoONada {\n        +VerEnExtremos()\n    }\n    class Sobregeneralizacion {\n        +GeneralizarDeUnicoEjemplo()\n    }\n    class FiltroMental {\n        +EnfocarseEnNegativo()\n    }\n    class SacarConclusionesPrecipitadas {\n        +HacerSuposiciones()\n    }\n    DistorsionesCognitivas *-- PensamientoTodoONada\n    DistorsionesCognitivas *-- Sobregeneralizacion\n    DistorsionesCognitivas *-- FiltroMental\n    DistorsionesCognitivas *-- SacarConclusionesPrecipitadas\n{% end %}\n\nEl shortcode de Mermaid admite dos parámetros:\n\n- `invertible`: Si se establece en `true` (por defecto), el diagrama se invertirá en modo oscuro, igual que las [imágenes invertibles](#imagen-invertible).\n- `full_width`: Permite que el diagrama ocupe el ancho del encabezado. Mira [imagen a ancho completo](#imagen-a-ancho-completo).\n\n{{ admonition(type=\"tip\", title=\"CONSEJO\", text=\"Puedes usar el [editor de Mermaid](https://mermaid.live/) para crear y previsualizar tus diagramas.\") }}\n\n#### Uso\n\n```\n{%/* mermaid(invertible=true, full_width=false) */%}\n\nTu diagrama Mermaid va aquí. Puedes omitir los parámetros para usar los valores predeterminados.\n\n{%/* end */%}\n```\n\n## Shortcodes de imagen\n\nTodos los shortcodes de imagen admiten rutas absolutas, rutas relativas, y fuentes remotas en el parámetro `src`.\n\nTodos los shortcodes de imagen tienen los siguientes parámetros opcionales:\n\n- `raw_path`. Por defecto es `false`. Si se establece en `true`, el parámetro `src` se usará tal cual. Útil para activos ubicados en la misma carpeta que tienen un slug personalizado (ver [Zola issue #2598](https://github.com/getzola/zola/issues/2598)).\n- `inline`. Valor predeterminado: `false`. Si se establece `true`, la imagen se mostrará en línea con el texto.\n- `full_width`. Valor predeterminado: `false` (ver [más abajo](#imagen-a-ancho-completo)).\n- `lazy_loading`. Valor predeterminado: `true`.\n\n### Imágenes de doble tema\n\nÚtil si deseas usar una imagen diferente para los temas claro y oscuro:\n\n{{ dual_theme_image(light_src=\"img/paris_day.webp\", dark_src=\"img/paris_night.webp\" alt=\"La Torre Eiffel\") }}\n\n#### Uso\n```\n{{/* dual_theme_image(light_src=\"img/paris_day.webp\", dark_src=\"img/paris_night.webp\" alt=\"La Torre Eiffel\") */}}\n```\n\n### Imagen invertible\n\nIdeal para gráficos, dibujos lineales, diagramas... Invierte los colores de la imagen. La imagen de origen se utilizará para el tema claro.\n\n{{ invertible_image(src=\"img/graph.webp\", alt=\"Gráfico invertible\") }}\n\n#### Uso\n\n```\n{{/* invertible_image(src=\"img/graph.webp\", alt=\"Gráfico invertible\") */}}\n```\n\n\n### Imagen atenuable\n\nLas imágenes con demasiado brillo o contraste pueden ser demasiado discordantes en un fondo oscuro. Aquí tienes un ejemplo de una fotografía que se atenúa cuando el tema oscuro está activo.\n\n{{ dimmable_image(src=\"img/desert_by_oskerwyld.webp\", alt=\"Fotografía de un desierto, cielo celestial\") }}\n\n#### Uso\n\n```\n{{/* dimmable_image(src=\"img/desert_by_oskerwyld.webp\", alt=\"Fotografía de un desierto, cielo celestial\") */}}\n```\n\n### Cambio de imagen al pasar el cursor\n\nLa imagen mostrada cambia cuando el usuario pasa el cursor por encima. Útil para comparaciones de antes y después, por ejemplo.\n\n{{ image_hover(default_src=\"img/edited.webp\", hovered_src=\"img/raw.webp\", default_alt=\"Foto editada\", hovered_alt=\"Foto original\") }}\n\n#### Uso\n\n```\n{{/* image_hover(default_src=\"img/before.webp\", hovered_src=\"img/after.webp\", default_alt=\"Imagen editada\", hovered_alt=\"Toma original\") */}}\n```\n\n### Cambio de imagen vía click\n\nMuestra una imagen y cambia a una diferente al hacer clic. Ideal para destacar diferencias o llamar la atención sobre detalles.\n\n{{ image_toggler(default_src=\"img/mojave_day.webp\", toggled_src=\"img/mojave_night.webp\", default_alt=\"Mojave durante el día\", toggled_alt=\"Mojave durante la noche\") }}\n\n#### Uso\n\n```\n{{/* image_toggler(default_src=\"img/mojave_day.webp\", toggled_src=\"img/mojave_night.webp\", default_alt=\"Mojave durante el día\", toggled_alt=\"Mojave durante la noche\") */}}\n```\n\n### Imagen a ancho completo\n\nLa imagen se expandirá para coincidir con el ancho del encabezado, que generalmente es más ancho que el texto del artículo (excepto en móvil/ventanas pequeñas).\n\nTodos los otros shortcodes de imágenes pueden usar el ancho completo asignando el valor `true` al parámetro opcional `full_width`.\n\n{{ full_width_image(src=\"img/amsterdam_by_oskerwyld.webp\", alt=\"Fotografía de un canal en Ámsterdam\") }}\n\n#### Uso\n\n```\n{{/* full_width_image(src=\"img/amsterdam_by_oskerwyld.webp\", alt=\"Fotografía de un canal en Ámsterdam\") */}}\n```\n\n## Shortcodes sociales\n\n### iine\n\n{{ aside(text=\"Para añadirlo a todas las publicaciones, establece `iine = true` en la sección `[extra]` de tu `config.toml`.\") }}\n\nEste shortcode te permite añadir botones adicionales de [iine.to](https://iine.to) a tus publicaciones, como este:\n\n{{ iine(slug=\"/blog/shortcodes/demo-button\") }}\n\n#### Uso\n\n```\n{{/* iine(icon=\"heart\", slug=\"/post/mi-slug-de-post/like\", label=\"Me gusta esta publicación\") */}}\n```\n\nEl shortcode acepta los siguientes parámetros opcionales:\n\n- `icon`: El icono a mostrar. Puede ser `heart`, `thumbs_up`, `upvote`, o cualquier emoji.\n- `slug`: Un identificador único. Por defecto es la ruta de la página actual. Útil si quieres más de un botón en la misma página.\n- `label`: La etiqueta de accesibilidad para el botón. Por defecto es \"Me gusta esta publicación\".\n\n## Shortcodes de código\n\n### Mostrar ruta o URL\n\nPuedes mostrar una ruta o URL para un bloque de código usando la sintaxis nativa de Zola:\n\n{{ aside(text=\"Requiere Zola 0.20.0 o superior.\") }}\n\n````\n```rust,name=src/main.rs\nfn main() {\n    println!(\"¡Hola, mundo!\");\n}\n```\n````\n\nEsto renderiza:\n\n```rust,name=src/main.rs\nfn main() {\n    println!(\"¡Hola, mundo!\");\n}\n```\n\nSi estableces el `name` como una URL (es decir, comienza con `http` o `https`), puedes convertirlo en un enlace clickable. Esto es particularmente útil cuando se usa junto con el [shortcode de texto remoto](#texto-remoto).\n\n{{ admonition(type=\"warning\", title=\"JavaScript requerido\", text=\"La función de URLs clickables requiere JavaScript. Para habilitarla, configura `code_block_name_links = true` en la sección `[extra]` de tu página, sección, o `config.toml`.\") }}\n\n```plain,name=https://github.com/welpo/doteki/blob/main/.gitignore\n__pycache__/\n*coverage*\n.vscode/\ndist/\n```\n\n### Soporte de shortcode heredado\n\nEl shortcode `add_src_to_code_block` sigue funcionando por retrocompatibilidad, pero será descontinuado en una versión futura. Por favor, usa la sintaxis nativa de Zola:\n\n```\n# Forma antigua (descontinuada):\n{{/* add_src_to_code_block(src=\"ruta/al/archivo.rs\") */}}\n\n# Forma nueva (recomendada):\n```rust,name=ruta/al/archivo.rs\n```\n\n## Shortcodes de texto\n\n### Aside (nota al margen)\n\nAñade contenido complementario en los márgenes en pantallas anchas, o como bloques distintivos en móvil.\n\n{{ aside(text=\"*Nota al margen* viene de *nota* (del latín, 'marca' o 'señal') y *margen* (del latín *margo*, 'borde' o 'límite').\") }}\n\nEl shortcode acepta dos parámetros:\n\n- `position`: Establecer como `\"right\"` para colocar en el margen derecho (por defecto, izquierdo)\n- El contenido puede proporcionarse mediante el parámetro `text` o entre las etiquetas del shortcode\n\n#### Uso\n\n{{ admonition(type=\"warning\", text=\"Separa la llamada al shortcode con saltos de línea para evitar errores de renderizado.\") }}\n\nUsando el parámetro `text`:\n\n```\n{{/* aside(text=\"*Nota al margen* viene de *nota* (del latín, 'marca' o 'señal') y *margen* (del latín *margo*, 'borde' o 'límite').\") */}}\n```\n\nUsando el cuerpo del contenido e indicando la posición:\n\n```\n{%/* aside(position=\"right\") */%}\nUna nota más larga que\npuede ocupar varias líneas.\n\nSe admite *Markdown*.\n{%/* end */%}\n```\n\n### Texto remoto\n\nAñade texto desde una URL remota o un archivo local.\n\nEl shortcode acepta tres parámetros:\n\n- `src`: La URL de origen o ruta del archivo (obligatorio)\n- `start`: Primera línea a mostrar (opcional, empieza en 1)\n- `end`: Número de la última línea (opcional, por defecto es 0, la última línea)\n\n{{ admonition(type=\"info\", text=\"`start` y `end` son inclusivos. `start=3, end=3` mostrará solo la tercera línea.\") }}\n\n**Importante**:\n\n- **Archivos remotos VS archivos locales**: Si `src` empieza con \"http\", se tratará como un archivo remoto. De lo contrario, se asume que es una ruta de archivo local.\n- **Acceso a archivos**: Dado que utiliza la función [`load_data`](https://www.getzola.org/documentation/templates/overview/#load-data) de Zola, los archivos locales deben estar dentro del directorio de Zola —ver la [lógica de búsqueda de archivos](https://www.getzola.org/documentation/templates/overview/#file-searching-logic). Desde [tabi 2.16.0](https://github.com/welpo/tabi/releases/tag/v2.16.0), el shortcode admite también rutas relativas.\n- **Formateo de bloques de código**: Para mostrar el texto como un bloque de código, debes añadir manualmente las cercas de código Markdown (comillas invertidas) y, opcionalmente, especificar el lenguaje de programación para el resaltado sintáctico.\n\n#### Uso\n\nAñade un script de Python remoto dentro de un bloque de código con resaltado sintáctico:\n\n````\n```python\n{{/* remote_text(src=\"https://example.com/script.py\") */}}\n```\n````\n\nVisualización de texto de un archivo local:\n\n```\n{{/* remote_text(src=\"ruta/a/archivo.txt\") */}}\n```\n\nMostar sólo las líneas 3 a 5 de un archivo remoto:\n\n```\n{{/* remote_text(src=\"https://example.com/script.py\", start=3, end=5) */}}\n```\n\n### Advertencias\n\nDestaca información con estos shortcodes de advertencia/alerta. Hay cinco tipos (`type`): `note`, `tip`, `info`, `warning`, y `danger`.\n\n{{ admonition(type=\"note\", text=\"Contenido con **sintaxis** *Markdown*. Consulta [esta `api`](#).\") }}\n\n{{ admonition(type=\"tip\", text=\"Contenido con **sintaxis** *Markdown*. Consulta [esta `api`](#).\") }}\n\n{{ admonition(type=\"info\", text=\"Contenido con **sintaxis** *Markdown*. Consulta [esta `api`](#).\") }}\n\n{{ admonition(type=\"warning\", text=\"Contenido con **sintaxis** *Markdown*. Consulta [esta `api`](#).\") }}\n\n{{ admonition(type=\"danger\", text=\"Contenido con **sintaxis** *Markdown*. Consulta [esta `api`](#).\") }}\n\nPuedes cambiar el `title` y el `icon` de la advertencia. Ambos parámetros aceptan texto y por defecto coinciden con el tipo de advertencia. `icon` puede ser cualquiera de los tipos de advertencia disponibles.\n\n{{ admonition(type=\"note\", icon=\"tip\", title=\"Título e icono personalizados\", text=\"Contenido con **sintaxis** *Markdown*. Consulta [esta `api`](#).\") }}\n\n#### Uso\n\nPuedes usar las advertencias de dos formas:\n\n1. En línea con parámetros:\n\n```md\n{{/* admonition(type=\"danger\", icon=\"tip\", title=\"Un consejo importante\", text=\"Mantente hidratado\") */}}\n```\n\n2. Con contenido en el cuerpo:\n\n```md\n{%/* admonition(type=\"danger\", icon=\"tip\", title=\"Un consejo importante\") */%}\nMantente hidratado\n\nEste método es especialmente útil para contenido largo o múltiples párrafos.\n{%/* end */%}\n```\n\nAmbos métodos admiten los mismos parámetros (`type`, `icon`, y `title`).\n\n### Citas multilenguaje\n\nEste shortcode permite mostrar una cita traducida y en su lenguaje original:\n\n{{ multilingual_quote(original=\"Ce qui est terrible, ce n’est pas de souffrir ni de mourir, mais de mourir en vain.\", translated=\"Lo terrible no es sufrir o morir, sino morir en vano.\", author=\"Jean-Paul Sartre\") }}\n\n#### Uso\n\n```\n{{/* multilingual_quote(original=\"Ce qui est terrible, ce n’est pas de souffrir ni de mourir, mais de mourir en vain.\", translated=\"Lo terrible no es sufrir o morir, sino morir en vano.\", author=\"Jean-Paul Sartre\") */}}\n```\n\n### Referencias con sangría francesa\n\nEste shortcode formatea una sección de referencias con sangría francesa de la siguiente manera:\n\n{% references() %}\n\nAlderson, E. (2015). Ciberseguridad y justicia social: Una crítica a la hegemonía corporativa en un mundo digital. *New York Journal of Technology, 11*(2), 24-39. [https://doi.org/10.1007/s10198-022-01497-6](https://doi.org/10.1007/s10198-022-01497-6).\n\nFunkhouser, M. (2012). Las normas sociales de indecencia: Un análisis del comportamiento desviado en la sociedad contemporánea. *Los Angeles Journal of Sociology, 16*(3), 41-58. [https://doi.org/10.1093/jmp/jhx037](https://doi.org/10.1093/jmp/jhx037).\n\nSchrute, D. (2005). La revolución de la agricultura de remolacha: Un análisis de la innovación agrícola. *Scranton Agricultural Quarterly, 38*(3), 67-81.\n\nSteinbrenner, G. (1997). El análisis costo-beneficio de George Costanza: Un examen del comportamiento de toma de riesgos en el lugar de trabajo. *New York Journal of Business, 12*(4), 112-125.\n\nWinger, J. A. (2010). El arte del debate: Un examen de la retórica en el modelo de las Naciones Unidas del Greendale Community College. *Colorado Journal of Communication Studies, 19*(2), 73-86. [https://doi.org/10.1093/6seaons/1movie](https://doi.org/10.1093/6seaons/1movie).\n\n{% end %}\n\n#### Uso\n\n```\n{%/* references() */%}\n\nTus referencias van aquí.\n\nCada una en una línea nueva. Se renderizará Markdown (enlaces, cursivas…).\n\n{%/* end */%}\n```\n\n### Spoilers\n\nEste shortcode permite ocultar texto que se revelará al hacer clic. Por ejemplo: las galletas de la fortuna tiene su origen en {{ spoiler(text=\"Japón. Sí, [en serio](https://es.wikipedia.org/wiki/Galleta_de_la_suerte#Historia_y_origen).\") }}\n\nComo ves, el Markdown se renderiza.\n\nEste shortcode tiene la opción `fixed_blur` para difuminar el texto \"SPOILER\", en lugar de difuminar el contenido real. Por ejemplo: es {{ spoiler(text=\"innecesario\", fixed_blur=true)}} esperar 24 horas antes de denunciar la desaparición de una persona.\n\n\n#### Uso\n\n```\n{{/* spoiler(text=\"texto que ocultar\", fixed_blur=false) */}}\n```\n\n## Contenedores\n\n### Contenedor ancho\n\nUtiliza este código corto si deseas tener una tabla, párrafo, bloque de código… más ancho. En escritorio, ocupará el ancho del encabezado. En móviles no tendrá efecto, excepto para las tablas, que ganarán scroll horizontal.\n\n{% wide_container() %}\n\n| Título            |  Año  | Director             | Director de Fotografía| Género        | IMDb  | Duración     |\n|-------------------|-------|----------------------|-----------------------|---------------|-------|--------------|\n| Beoning           | 2018  | Lee Chang-dong       | Hong Kyung-pyo        | Drama/Misterio| 7.5   | 148 min      |\n| The Master        | 2012  | Paul Thomas Anderson | Mihai Mălaimare Jr.   | Drama/Historia| 7.1   | 137 min      |\n| The Tree of Life  | 2011  | Terrence Malick      | Emmanuel Lubezki      | Drama         | 6.8   | 139 min      |\n\n{% end %}\n\n#### Uso\n\n```\n{%/* wide_container() */%}\n\nColoca tu bloque de código, párrafo, tabla… aquí.\n\nEl Markdown, por supuesto, será interpretado.\n\n{%/* end */%}\n```\n\n### Forzar dirección del texto\n\nFuerza la dirección del texto de un bloque de contenido. Anula tanto la configuración global `force_codeblock_ltr` como la dirección general del documento.\n\nAcepta el parámetro `direction`: la dirección de texto deseada. Puede ser \"ltr\" (de izquierda a derecha) o \"rtl\" (de derecha a izquierda). Por defecto es \"ltr\".\n\n{% force_text_direction(direction=\"rtl\") %}\n```python\ndef مرحبا_بالعالم():\n    print(\"مرحبا بالعالم!\")\n```\n{% end %}\n\n#### Uso\n\nEn una página LTR podemos forzar que un bloque de código sea RTL (como se muestra arriba) de la siguiente manera:\n\n````\n{%/* force_text_direction(direction=\"rtl\") */%}\n\n```python\ndef مرحبا_بالعالم():\n    print(\"مرحبا بالعالم!\")\n```\n\n{%/* end */%}\n````\n"
  },
  {
    "path": "content/blog/shortcodes/index.md",
    "content": "+++\ntitle = \"Custom shortcodes\"\ndate = 2023-02-19\nupdated = 2025-10-21\ndescription = \"This theme includes some useful custom shortcodes that you can use to enhance your posts. Whether you want to display images that adapt to light and dark themes, or format a professional-looking reference section, these custom shortcodes have got you covered.\"\n\n[taxonomies]\ntags = [\"showcase\", \"shortcodes\"]\n\n[extra]\ntoc = true\ntoc_levels = 2\nquick_navigation_buttons = true\ncode_block_name_links = true\nmermaid = true\nsocial_media_card = \"social_cards/blog_shortcodes.jpg\"\n+++\n\n## Diagram shortcode\n\n### Mermaid diagrams\n\n[Mermaid](https://github.com/mermaid-js/mermaid) is a a diagramming and charting tool that uses text and code to generate diagrams. It supports flowcharts, sequence diagrams, Gantt charts, and more.\n\nTo include a Mermaid diagram in your post, there are two steps:\n\n1. Set `mermaid = true` in the `[extra]` section of the front matter of your page, section or `config.toml`. This will load the JavaScript needed to render the diagrams.\n\n2. Use the `mermaid()` shortcode to define your diagram in your posts. For example:\n\n```txt\n{%/* mermaid() */%}\nclassDiagram\n    class CognitiveDistortions {\n        +AllOrNothingThinking()\n        +Overgeneralization()\n        +MentalFilter()\n        +JumpingToConclusions()\n    }\n    class AllOrNothingThinking {\n        +SeeInExtremes()\n    }\n    class Overgeneralization {\n        +GeneralizeFromSingle()\n    }\n    class MentalFilter {\n        +FocusOnNegative()\n    }\n    class JumpingToConclusions {\n        +MakeAssumptions()\n    }\n    CognitiveDistortions *-- AllOrNothingThinking\n    CognitiveDistortions *-- Overgeneralization\n    CognitiveDistortions *-- MentalFilter\n    CognitiveDistortions *-- JumpingToConclusions\n{%/* end */%}\n```\n\nThe diagram will be rendered as follows:\n\n{% mermaid() %}\nclassDiagram\n    class CognitiveDistortions {\n        +AllOrNothingThinking()\n        +Overgeneralization()\n        +MentalFilter()\n        +JumpingToConclusions()\n    }\n    class AllOrNothingThinking {\n        +SeeInExtremes()\n    }\n    class Overgeneralization {\n        +GeneralizeFromSingle()\n    }\n    class MentalFilter {\n        +FocusOnNegative()\n    }\n    class JumpingToConclusions {\n        +MakeAssumptions()\n    }\n    CognitiveDistortions *-- AllOrNothingThinking\n    CognitiveDistortions *-- Overgeneralization\n    CognitiveDistortions *-- MentalFilter\n    CognitiveDistortions *-- JumpingToConclusions\n{% end %}\n\nThe Mermaid shortcode supports two parameters:\n\n- `invertible`: If set to `true` (default), the diagram will be inverted in dark mode, just like [invertible images](#invertible-image).\n- `full_width`: Allows the diagram to take up the width of the header. See [full-width image](#full-width-image).\n\n{{ admonition(type=\"tip\", text=\"You can use the [Mermaid Live Editor](https://mermaid.live/) to create and preview your diagrams.\") }}\n\n#### Usage\n\n```\n{%/* mermaid(invertible=true, full_width=false) */%}\n\nYour diagram goes here.\n\n`invertible` or `full_width` can be omitted if default values are used.\n\n{%/* end */%}\n```\n\n## Image shortcodes\n\nAll image shortcodes support absolute paths, relative paths, and remote sources in the `src` parameter.\n\nAll image shortcodes have these optional parameters:\n\n- `raw_path`. Defaults to `false`. If set to `true`, the `src` parameter will be used as is. Useful for colocated assets with a custom slug (see [Zola issue #2598](https://github.com/getzola/zola/issues/2598)).\n- `inline`. Defaults to `false`. If set to `true`, the image will be displayed inline with the text.\n- `full_width`. Defaults to `false` (see [below](#full-width-image))\n- `lazy_loading`. Defaults to `true`.\n\n### Dual theme images\n\nUseful if you want to use a different image for the light and dark themes:\n\n{{ dual_theme_image(light_src=\"img/paris_day.webp\", dark_src=\"img/paris_night.webp\" alt=\"The Eiffel tower\") }}\n\n#### Usage\n\n```\n{{/* dual_theme_image(light_src=\"img/paris_day.webp\", dark_src=\"img/paris_night.webp\" alt=\"The Eiffel tower\") */}}\n```\n\n### Invertible image\n\nGood for graphs, line drawings, diagrams… Inverts the colours of the image. The source image will be used for the light theme.\n\n{{ invertible_image(src=\"img/graph.webp\", alt=\"Invertible graph\") }}\n\n#### Usage\n\n```\n{{/* invertible_image(src=\"img/graph.webp\", alt=\"Invertible graph\") */}}\n```\n\n### Dimmable image\n\nImages with too much brightness or contrast can be jarring against a dark background. Here's an example of a photograph that dims when the dark theme is active.\n\n{{ dimmable_image(src=\"img/desert_by_oskerwyld.webp\", alt=\"Photograph of a desert, heavenly sky\") }}\n\n#### Usage\n\n```\n{{/* dimmable_image(src=\"img/desert_by_oskerwyld.webp\", alt=\"Photograph of a desert, heavenly sky\") */}}\n```\n\n### Swap image on hover\n\nProvides an interaction where the image displayed changes as the user hovers over it. Useful for before-after comparisons, for example.\n\n{{ image_hover(default_src=\"img/edited.webp\", hovered_src=\"img/raw.webp\", default_alt=\"Edited picture\", hovered_alt=\"Original shot\") }}\n\n#### Usage\n\n```\n{{/* image_hover(default_src=\"img/before.webp\", hovered_src=\"img/after.webp\", default_alt=\"Edited picture\", hovered_alt=\"Original shot\") */}}\n```\n\n### Interactive image toggle\n\nDisplay an image and switch to a different one on click. Ideal for highlighting differences or drawing attention to details.\n\n{{ image_toggler(default_src=\"img/mojave_day.webp\", toggled_src=\"img/mojave_night.webp\", default_alt=\"Mojave during the day\", toggled_alt=\"Mojave at night\") }}\n\n#### Usage\n\n```\n{{/* image_toggler(default_src=\"img/mojave_day.webp\", toggled_src=\"img/mojave_night.webp\", default_alt=\"Mojave during the day\", toggled_alt=\"Mojave at night\") */}}\n```\n\n### Full-width image\n\nThe image will expand to match the width of the header, which is usually wider than the article text (except on mobile/small windows).\n\nAll other image shortcodes can be made into full-width by setting the optional parameter `full_width` to `true`.\n\n{{ full_width_image(src=\"img/amsterdam_by_oskerwyld.webp\", alt=\"Photograph of a canal in Amsterdam\") }}\n\n#### Usage\n\n```\n{{/* full_width_image(src=\"img/amsterdam_by_oskerwyld.webp\", alt=\"Photograph of a canal in Amsterdam\") */}}\n```\n\n## Engagement shortcodes\n\n### iine\n\n{{ aside(text=\"To add it to all posts, set `iine = true` in the `[extra]` section of your `config.toml`.\") }}\n\nThis shortcode allows you to add extra [iine.to](https://iine.to) buttons to your posts, like this:\n\n{{ iine(slug=\"/blog/shortcodes/demo-button\") }}\n\n#### Usage\n\n```\n{{/* iine(icon=\"heart\", slug=\"/post/my-post-slug/like\", label=\"Like this post\") */}}\n```\n\nThe shortcode takes the following optional parameters:\n\n- `icon`: The icon to display. Can be `heart`, `thumbs_up`, `upvote`, or any emoji.\n- `slug`: A unique identifier. Defaults to the current page's path. Useful if you want more than one button on the same page.\n- `label`: The accessibility label for the button. Defaults to \"Like this post\".\n\n## Code shortcodes\n\n### Show source or path\n\nYou can display a path or URL for a code block using Zola's native syntax:\n\n{{ aside(text=\"Requires Zola 0.20.0 or later.\") }}\n\n````\n```rust,name=src/main.rs\nfn main() {\n    println!(\"Hello, world!\");\n}\n```\n````\n\nThis renders:\n\n```rust,name=src/main.rs\nfn main() {\n    println!(\"Hello, world!\");\n}\n```\n\nIf you set the `name` to a URL (i.e. it starts with `http` or `https`), you can turn it into a clickable link. This is particularly useful when used in conjunction with the [remote text shortcode](#remote-text).\n\n{{ admonition(type=\"warning\", title=\"JavaScript required\", text=\"The clickable URL feature requires JavaScript. To enable it, set `code_block_name_links = true` on the `[extra]` section of your page, section, or `config.toml`.\") }}\n\n```plain,name=https://github.com/welpo/doteki/blob/main/.gitignore\n__pycache__/\n*coverage*\n.vscode/\ndist/\n```\n\n### Legacy shortcode support\n\nThe `add_src_to_code_block` shortcode is still supported for backward compatibility but will be deprecated in a future release. Please use Zola's native syntax shown above instead:\n\n```\n# Old way (deprecated):\n{{/* add_src_to_code_block(src=\"path/to/file.rs\") */}}\n\n# New way (preferred):\n```rust,name=path/to/file.rs\n```\n\n## Text shortcodes\n\n### Aside (side/margin note)\n\nAdd supplementary content in the margins on wide screens, or as distinct blocks on mobile.\n\n{{ aside(text=\"*Sidenote* comes from Latin *nota* ('mark') + Old English *síde* ('side').\") }}\n\nThe shortcode accepts two parameters:\n\n- `position`: Set to `\"right\"` to place in right margin (defaults to left)\n- Content can be provided via `text` parameter or between shortcode tags\n\n#### Usage\n\n{{ admonition(type=\"warning\", text=\"Place the aside shortcode on its own line to prevent formatting issues.\") }}\n\nUsing the `text` parameter:\n\n```\n{{/* aside(text=\"*Sidenote* comes from Latin *nota* ('mark') + Old English *síde* ('side').\") */}}\n```\n\nUsing the content body and setting the position to right:\n\n```\n{%/* aside(position=\"right\") */%}\nA longer note that\ncan span multiple lines.\n\n_Markdown_ is supported.\n{%/* end */%}\n```\n\n### Remote text\n\nEmbed text from a remote URL or a local file. To display the path or URL on the code block, see the [show source or path shortcode](#show-source-or-path).\n\nThe shortcode accepts three parameters:\n\n- `src`: The source URL or file path (required)\n- `start`: First line to display (optional, starts at 1)\n- `end`: The ending line number (optional, defaults to 0, meaning the last line)\n\n{{ admonition(type=\"info\", text=\"`start` and `end` are inclusive. `start=3, end=3` will display only the third line.\") }}\n\n**Important**:\n\n- **Remote VS local files**: If `src` starts with \"http\", it will be treated as a remote file. Otherwise, it assumes a local file path.\n- **Files access**: As it uses Zola's [`load_data`](https://www.getzola.org/documentation/templates/overview/#load-data), local files must be inside the Zola directory—see [File searching logic](https://www.getzola.org/documentation/templates/overview/#file-searching-logic). As of [tabi 2.16.0](https://github.com/welpo/tabi/releases/tag/v2.16.0), the shortcode supports both relative and absolute paths.\n- **Code block formatting**: To display the text as a code block, you must manually add the Markdown code fences (backticks) and, optionally, specify the programming language for syntax highlighting.\n\n#### Usage\n\nEmbedding a remote Python script within a code block with syntax highlighting:\n\n````\n```python\n{{/* remote_text(src=\"https://example.com/script.py\") */}}\n```\n````\n\nDisplaying text from a local file:\n\n```\n{{/* remote_text(src=\"path/to/file.txt\") */}}\n```\n\nDisplay lines 3 to 7 (both inclusive) of a local file:\n\n```\n{{/* remote_text(src=\"path/to/file.txt\", start=3, end=7) */}}\n```\n\n### Admonitions\n\nBring attention to information with these admonition/alert shortcodes. They come in five `type`s: `note`, `tip`, `info`, `warning`, and `danger`.\n\n{{ admonition(type=\"note\", text=\"Some **content** with _Markdown_ `syntax`. Check [this `api`](#).\") }}\n\n{{ admonition(type=\"tip\", text=\"Some **content** with _Markdown_ `syntax`. Check [this `api`](#).\") }}\n\n{{ admonition(type=\"info\", text=\"Some **content** with _Markdown_ `syntax`. Check [this `api`](#).\") }}\n\n{{ admonition(type=\"warning\", text=\"Some **content** with _Markdown_ `syntax`. Check [this `api`](#).\") }}\n\n{{ admonition(type=\"danger\", text=\"Some **content** with _Markdown_ `syntax`. Check [this `api`](#).\") }}\n\nYou can change the `title` and `icon` of the admonition. Both parameters take a string and default to the type of admonition. `icon` can be any of the available admonition types.\n\n{{ admonition(type=\"note\", icon=\"tip\", title=\"Custom title and icon\", text=\"Some **content** with _Markdown_ `syntax`. Check [this `api`](#).\") }}\n\n#### Usage\n\nYou can use admonitions in two ways:\n\n1. Inline with parameters:\n\n```md\n{{/* admonition(type=\"danger\", icon=\"tip\", title=\"An important tip\", text=\"Stay hydrated~\") */}}\n```\n\n2. With a content body:\n\n```md\n{%/* admonition(type=\"danger\", icon=\"tip\", title=\"An important tip\") */%}\nStay hydrated~\n\nThis method is particularly useful for longer content or multiple paragraphs.\n{%/* end */%}\n```\n\nBoth methods support the same parameters (`type`, `icon`, and `title`), with the content either passed as the `text` parameter or as the body between tags.\n\n### Multilingual quotes\n\nThis shortcode allows you to display both the translated and original text for a quote. The quotation marks will be added automatically:\n\n{{ multilingual_quote(original=\"Qué sosiego, ir por la vida en silencio, saludando sólo a los amigos.\", translated=\"What tranquillity, to go through life in silence, greeting only friends.\", author=\"Francisco Umbral\") }}\n\n#### Usage\n\n```\n{{/* multilingual_quote(original=\"Qué sosiego, ir por la vida en silencio, saludando sólo a los amigos.\", translated=\"What tranquillity, to go through life in silence, greeting only friends.\", author=\"Francisco Umbral\") */}}\n```\n\n### References with hanging indent\n\nThis shortcode formats a reference section with a hanging indent like so:\n\n{% references() %}\n\nAlderson, E. (2015). Cybersecurity and Social Justice: A Critique of Corporate Hegemony in a Digital World. *New York Journal of Technology, 11*(2), 24-39. [https://doi.org/10.1007/s10198-022-01497-6](https://doi.org/10.1007/s10198-022-01497-6).\n\nFunkhouser, M. (2012). The Social Norms of Indecency: An Analysis of Deviant Behavior in Contemporary Society. *Los Angeles Journal of Sociology, 16*(3), 41-58. [https://doi.org/10.1093/jmp/jhx037](https://doi.org/10.1093/jmp/jhx037).\n\nSchrute, D. (2005). The Beet Farming Revolution: An Analysis of Agricultural Innovation. *Scranton Agricultural Quarterly, 38*(3), 67-81.\n\nSteinbrenner, G. (1997). The Cost-Benefit Analysis of George Costanza: An Examination of Risk-Taking Behavior in the Workplace. *New York Journal of Business, 12*(4), 112-125.\n\nWinger, J. A. (2010). The Art of Debate: An Examination of Rhetoric in Greendale Community College's Model United Nations. *Colorado Journal of Communication Studies, 19*(2), 73-86. [https://doi.org/10.1093/6seaons/1movie](https://doi.org/10.1093/6seaons/1movie).\n\n{% end %}\n\n#### Usage\n\n```\n{%/* references() */%}\n\nYour references go here.\n\nEach in a new line. Markdown (links, italics…) will be rendered.\n\n{%/* end */%}\n```\n\n### Spoiler\n\nThis shortcode allows you to blur text until the user clicks on it. Like this: Goldfish have a memory span of a few {{ spoiler(text=\"months. Yes, [really](https://en.wikipedia.org/wiki/Goldfish#Cognitive_abilities).\") }}\n\nAs you can see, Markdown is rendered. You can even add newlines with `<br>`.\n\nThis shortcode has the optional flag `fixed_blur` to blur a fixed placeholder (\"SPOILER\"), instead of blurring the actual contents. Like this: it is {{ spoiler(text=\"not necessary\", fixed_blur=true)}} to wait 24 hours before filing a missing person report.\n\n#### Usage\n\n```\n{{/* spoiler(text=\"text to hide\", fixed_blur=false) */}}\n```\n\n## Containers\n\n### Wide container\n\nUse this shortcode if you want to have a wider table, paragraph, code block… On desktop, it will take up the width of the header. It will have no effect on mobile, except for tables, which will get a horizontal scroll.\n\n{% wide_container() %}\n\n| Title             |  Year | Director             | Cinematographer       | Genre         | IMDb  | Duration     |\n|-------------------|-------|----------------------|-----------------------|---------------|-------|--------------|\n| Beoning           | 2018  | Lee Chang-dong       | Hong Kyung-pyo        | Drama/Mystery | 7.5   | 148 min      |\n| The Master        | 2012  | Paul Thomas Anderson | Mihai Mălaimare Jr.   | Drama/History | 7.1   | 137 min      |\n| The Tree of Life  | 2011  | Terrence Malick      | Emmanuel Lubezki      | Drama         | 6.8   | 139 min      |\n\n{% end %}\n\n#### Usage\n\n```\n{%/* wide_container() */%}\n\nPlace your code block, paragraph, table… here.\n\nMarkdown will of course be rendered.\n\n{%/* end */%}\n```\n\n### Force text direction\n\nForce the text direction of a content block. Overrides both the global `force_codeblock_ltr` setting and the document's overall direction.\n\nAccepts the parameter `direction`: the desired text direction. This can be either \"ltr\" (left-to-right) or \"rtl\" (right-to-left). Defaults to \"ltr\".\n\n{% force_text_direction(direction=\"rtl\") %}\n\n```python\ndef مرحبا_بالعالم():\n    print(\"مرحبا بالعالم!\")\n```\n\n{% end %}\n\n#### Usage\n\nIn a LTR page we can force a code block to be RTL (as shown above) like so:\n\n````\n{%/* force_text_direction(direction=\"rtl\") */%}\n\n```python\ndef مرحبا_بالعالم():\n    print(\"مرحبا بالعالم!\")\n```\n\n{%/* end */%}\n````\n"
  },
  {
    "path": "content/blog/toc/index.ca.md",
    "content": "+++\ntitle = \"Taula de contingut\"\ndate = 2022-11-01\nupdated = 2024-02-16\ndescription = \"Una publicació que mostra la taula de contingut opcional i la seva configuració.\"\n\n[taxonomies]\ntags = [\"funcionalitat\", \"markdown\", \"tutorial\"]\n\n[extra]\ntoc = true\nquick_navigation_buttons = true\nsocial_media_card = \"social_cards/ca_blog_toc.jpg\"\n+++\n\n## Documentació\n### Habilitant (i posicionant) la Taula de Contingut\n\nHi ha dues formes d'habilitar la Taula de Contingut (TdC). Si vols que estigui just a sota del capçalera (com en aquesta pàgina), configura aquesta variable en el front matter del teu post:\n\n```toml\n[extra]\ntoc = true\n```\n\nSi prefereixes col·locar la TdC a un altre lloc (per exemple, després d'una introducció), pots fer-ho afegint una línia amb aquest contingut allà on vulguis que aparegui la TdC:\n\n```markdown\n<!-- toc -->\n```\n\nTambé pots utilitzar el shortcode `{{/* toc() */}}`, que simplement inserirà aquest text per tu ([idea de Michael Clayton](https://github.com/getzola/zola/issues/584#issuecomment-1546086781)).\n\nAquest mètode renderitzarà la TdC sense el capçalera \"Taula de Contingut\". Això et permet utilitzar un capçalera diferent (o cap) per la TdC, o fins i tot ocultar-la de forma predeterminada:\n\n<details>\n    <summary>TdC oculta</summary>\n    <!-- toc -->\n</details>\n\nEl codi per aconseguir-ho:\n\n```markdown\n<details>\n    <summary>TdC oculta</summary>\n    <!-- toc -->\n</details>\n```\n\n*Nota*: Si actives la TdC amb `toc = true` i també afegeixes `<!-- toc -->` en algun lloc del teu text, obtindràs múltiples TdCs.\n\nSi col·loques la TdC en un lloc diferent del predeterminat i li afegeixes un capçalera, segurament voldràs ocultar aquest capçalera de la TdC (consulta la [secció per ocultar capçaleres](#ocultant-capcaleres-de-la-tdc)). Pots fer-ho així:\n\n```markdown,hl_lines=06 11-13\n+++\ntitle = \"El títol del teu post\"\ndate = 2034-01-11\n\n[extra]\ntoc_ignore_pattern = \"^(Taula de contingut)\"\n+++\n\nAquí va algun text introductori.\n\n### Taula de contingut\n\n<!-- toc -->\n\n## Primer encapçalament de contingut\n```\n\n### Establint la profunditat màxima\n\nPots establir la profunditat màxima per la TdC especificant la variable `toc_levels`, que accepta un número enter entre 1 i 4:\n\n```toml\n[extra]\ntoc_levels = 2\n```\n\nEn aquest exemple, només els dos primers nivells d'encapçalaments s'inclourien a la TdC, independentment de les seves etiquetes HTML reals (`h1`, `h2`, `h3`, etc.). Si vols incloure només el nivell principal d'encapçalaments, estableix `toc_levels = 1`. El valor per defecte de `toc_levels` és `3`.\n\nTingues en compte als teus lectors quan establertis `toc_levels`. Encara que pot ser temptador incloure molts nivells imbricats per a una navegació detallada, una TdC més curta i senzilla sovint és més amigable i menys aclaparadora.\n\n### Ocultant capçaleres de la TdC\n\nÉs possible que vulguis amagar certes capçaleres. Per exemple, si el teu article té moltes Figures o Taules, aquestes podrien saturar la TdC. Pots ocultar capçaleres específiques a la TdC configurant la variable `toc_ignore_pattern` en la secció `[extra]` del front matter del teu post.\n\nAquesta variable espera una expressió regular (regex), ja que utilitza el test [matching](https://keats.github.io/tera/docs/#matching) de Tera. El `toc_ignore_pattern` es prova contra el text del capçalera. Per exemple, per a la capçalera `### Lectura addicional`, només el text `Lectura addicional` s'utilitzaria per comprovar si coincideix amb el patró.\n\nAquí tens alguns valors d'exemple per a `toc_ignore_pattern` juntament amb les capçaleres que amagarien:\n\n| `toc_ignore_pattern`             | Exclou capçaleres que…                                                 |\n|----------------------------------|------------------------------------------------------------------------|\n| `Taula`                          | continguin \"Taula\"                                                     |\n| `^Figura`                        | comencin amb \"Figura\"                                                  |\n| <code>(?i)(taula\\|figura)</code> | comencin amb \"Taula\" o \"Figura\" (insensible a majúscules/minúscules)   |\n| `\\[Esborrany\\]$`                 | acabin amb \"[Esborrany]\".                                              |\n\nPots provar la teva expressió regular en plataformes com [regex101](https://regex101.com/r/2dI7U2/1) per assegurar-te que funciona com esperes.\n\n*Nota*: Les capacitats de \"look-around\", incloent look-ahead i look-behind, no estan suportades.\n\n# Capçalera 1\nLorem ipsum dolor sit amet, consectetur adipiscing elit. Mauris sed mollis augue, vel efficitur lacus. Pellentesque eu egestas mi. Etiam ultrices lectus sit amet aliquet ullamcorper. Praesent in erat quis est sagittis finibus. Etiam nec sapien in nulla interdum faucibus. Integer iaculis lorem quis arcu lobortis, non malesuada neque vehicula. Aenean nec tellus eu metus bibendum tempus. Sed rutrum urna ut commodo tempor. Vestibulum aliquet elit posuere turpis maximus condimentum. Sed urna libero, ornare eu tellus vitae, laoreet condimentum risus. Aenean elit lectus, mattis quis nibh nec, faucibus rutrum sapien. Sed iaculis consectetur mi, eget posuere turpis finibus et.\n\n## Capçalera 2\nLorem ipsum dolor sit amet, consectetur adipiscing elit. Mauris sed mollis augue, vel efficitur lacus. Pellentesque eu egestas mi. Etiam ultrices lectus sit amet aliquet ullamcorper. Praesent in erat quis est sagittis finibus. Etiam nec sapien in nulla interdum faucibus. Integer iaculis lorem quis arcu lobortis, non malesuada neque vehicula. Aenean nec tellus eu metus bibendum tempus. Sed rutrum urna ut commodo tempor. Vestibulum aliquet elit posuere turpis maximus condimentum. Sed urna libero, ornare eu tellus vitae, laoreet condimentum risus. Aenean elit lectus, mattis quis nibh nec, faucibus rutrum sapien. Sed iaculis consectetur mi, eget posuere turpis finibus et.\n### Capçalera 3.1\nLorem ipsum dolor sit amet, consectetur adipiscing elit. Mauris sed mollis augue, vel efficitur lacus. Pellentesque eu egestas mi. Etiam ultrices lectus sit amet aliquet ullamcorper. Praesent in erat quis est sagittis finibus. Etiam nec sapien in nulla interdum faucibus. Integer iaculis lorem quis arcu lobortis, non malesuada neque vehicula. Aenean nec tellus eu metus bibendum tempus. Sed rutrum urna ut commodo tempor. Vestibulum aliquet elit posuere turpis maximus condimentum. Sed urna libero, ornare eu tellus vitae, laoreet condimentum risus. Aenean elit lectus, mattis quis nibh nec, faucibus rutrum sapien. Sed iaculis consectetur mi, eget posuere turpis finibus et.\n#### Capçalera 4.1\nLorem ipsum dolor sit amet, consectetur adipiscing elit. Mauris sed mollis augue, vel efficitur lacus. Pellentesque eu egestas mi. Etiam ultrices lectus sit amet aliquet ullamcorper. Praesent in erat quis est sagittis finibus. Etiam nec sapien in nulla interdum faucibus. Integer iaculis lorem quis arcu lobortis, non malesuada neque vehicula. Aenean nec tellus eu metus bibendum tempus. Sed rutrum urna ut commodo tempor. Vestibulum aliquet elit posuere turpis maximus condimentum. Sed urna libero, ornare eu tellus vitae, laoreet condimentum risus. Aenean elit lectus, mattis quis nibh nec, faucibus rutrum sapien. Sed iaculis consectetur mi, eget posuere turpis finibus et.\n#### Capçalera 4.2\nLorem ipsum dolor sit amet, consectetur adipiscing elit. Mauris sed mollis augue, vel efficitur lacus. Pellentesque eu egestas mi. Etiam ultrices lectus sit amet aliquet ullamcorper. Praesent in erat quis est sagittis finibus. Etiam nec sapien in nulla interdum faucibus. Integer iaculis lorem quis arcu lobortis, non malesuada neque vehicula. Aenean nec tellus eu metus bibendum tempus. Sed rutrum urna ut commodo tempor. Vestibulum aliquet elit posuere turpis maximus condimentum. Sed urna libero, ornare eu tellus vitae, laoreet condimentum risus. Aenean elit lectus, mattis quis nibh nec, faucibus rutrum sapien. Sed iaculis consectetur mi, eget posuere turpis finibus et.\n\n### Capçalera 3.2\nLorem ipsum dolor sit amet, consectetur adipiscing elit. Mauris sed mollis augue, vel efficitur lacus. Pellentesque eu egestas mi. Etiam ultrices lectus sit amet aliquet ullamcorper. Praesent in erat quis est sagittis finibus. Etiam nec sapien in nulla interdum faucibus. Integer iaculis lorem quis arcu lobortis, non malesuada neque vehicula. Aenean nec tellus eu metus bibendum tempus. Sed rutrum urna ut commodo tempor. Vestibulum aliquet elit posuere turpis maximus condimentum. Sed urna libero, ornare eu tellus vitae, laoreet condimentum risus. Aenean elit lectus, mattis quis nibh nec, faucibus rutrum sapien. Sed iaculis consectetur mi, eget posuere turpis finibus et.\n#### Capçalera 4\nLorem ipsum dolor sit amet, consectetur adipiscing elit. Mauris sed mollis augue, vel efficitur lacus. Pellentesque eu egestas mi. Etiam ultrices lectus sit amet aliquet ullamcorper. Praesent in erat quis est sagittis finibus. Etiam nec sapien in nulla interdum faucibus. Integer iaculis lorem quis arcu lobortis, non malesuada neque vehicula. Aenean nec tellus eu metus bibendum tempus. Sed rutrum urna ut commodo tempor. Vestibulum aliquet elit posuere turpis maximus condimentum. Sed urna libero, ornare eu tellus vitae, laoreet condimentum risus. Aenean elit lectus, mattis quis nibh nec, faucibus rutrum sapien. Sed iaculis consectetur mi, eget posuere turpis finibus et.\n#### Capçalera 4\nLorem ipsum dolor sit amet, consectetur adipiscing elit. Mauris sed mollis augue, vel efficitur lacus. Pellentesque eu egestas mi. Etiam ultrices lectus sit amet aliquet ullamcorper. Praesent in erat quis est sagittis finibus. Etiam nec sapien in nulla interdum faucibus. Integer iaculis lorem quis arcu lobortis, non malesuada neque vehicula. Aenean nec tellus eu metus bibendum tempus. Sed rutrum urna ut commodo tempor. Vestibulum aliquet elit posuere turpis maximus condimentum. Sed urna libero, ornare eu tellus vitae, laoreet condimentum risus. Aenean elit lectus, mattis quis nibh nec, faucibus rutrum sapien. Sed iaculis consectetur mi, eget posuere turpis finibus et.\n\n## Capçalera 2\nLorem ipsum dolor sit amet, consectetur adipiscing elit. Mauris sed mollis augue, vel efficitur lacus. Pellentesque eu egestas mi. Etiam ultrices lectus sit amet aliquet ullamcorper. Praesent in erat quis est sagittis finibus. Etiam nec sapien in nulla interdum faucibus. Integer iaculis lorem quis arcu lobortis, non malesuada neque vehicula. Aenean nec tellus eu metus bibendum tempus. Sed rutrum urna ut commodo tempor. Vestibulum aliquet elit posuere turpis maximus condimentum. Sed urna libero, ornare eu tellus vitae, laoreet condimentum risus. Aenean elit lectus, mattis quis nibh nec, faucibus rutrum sapien. Sed iaculis consectetur mi, eget posuere turpis finibus et.\n### Capçalera 3.1\nLorem ipsum dolor sit amet, consectetur adipiscing elit. Mauris sed mollis augue, vel efficitur lacus. Pellentesque eu egestas mi. Etiam ultrices lectus sit amet aliquet ullamcorper. Praesent in erat quis est sagittis finibus. Etiam nec sapien in nulla interdum faucibus. Integer iaculis lorem quis arcu lobortis, non malesuada neque vehicula. Aenean nec tellus eu metus bibendum tempus. Sed rutrum urna ut commodo tempor. Vestibulum aliquet elit posuere turpis maximus condimentum. Sed urna libero, ornare eu tellus vitae, laoreet condimentum risus. Aenean elit lectus, mattis quis nibh nec, faucibus rutrum sapien. Sed iaculis consectetur mi, eget posuere turpis finibus et.\n#### Capçalera 4.1\nLorem ipsum dolor sit amet, consectetur adipiscing elit. Mauris sed mollis augue, vel efficitur lacus. Pellentesque eu egestas mi. Etiam ultrices lectus sit amet aliquet ullamcorper. Praesent in erat quis est sagittis finibus. Etiam nec sapien in nulla interdum faucibus. Integer iaculis lorem quis arcu lobortis, non malesuada neque vehicula. Aenean nec tellus eu metus bibendum tempus. Sed rutrum urna ut commodo tempor. Vestibulum aliquet elit posuere turpis maximus condimentum. Sed urna libero, ornare eu tellus vitae, laoreet condimentum risus. Aenean elit lectus, mattis quis nibh nec, faucibus rutrum sapien. Sed iaculis consectetur mi, eget posuere turpis finibus et.\n#### Capçalera 4.2\nLorem ipsum dolor sit amet, consectetur adipiscing elit. Mauris sed mollis augue, vel efficitur lacus. Pellentesque eu egestas mi. Etiam ultrices lectus sit amet aliquet ullamcorper. Praesent in erat quis est sagittis finibus. Etiam nec sapien in nulla interdum faucibus. Integer iaculis lorem quis arcu lobortis, non malesuada neque vehicula. Aenean nec tellus eu metus bibendum tempus. Sed rutrum urna ut commodo tempor. Vestibulum aliquet elit posuere turpis maximus condimentum. Sed urna libero, ornare eu tellus vitae, laoreet condimentum risus. Aenean elit lectus, mattis quis nibh nec, faucibus rutrum sapien. Sed iaculis consectetur mi, eget posuere turpis finibus et.\n\n### Capçalera 3.2\nLorem ipsum dolor sit amet, consectetur adipiscing elit. Mauris sed mollis augue, vel efficitur lacus. Pellentesque eu egestas mi. Etiam ultrices lectus sit amet aliquet ullamcorper. Praesent in erat quis est sagittis finibus. Etiam nec sapien in nulla interdum faucibus. Integer iaculis lorem quis arcu lobortis, non malesuada neque vehicula. Aenean nec tellus eu metus bibendum tempus. Sed rutrum urna ut commodo tempor. Vestibulum aliquet elit posuere turpis maximus condimentum. Sed urna libero, ornare eu tellus vitae, laoreet condimentum risus. Aenean elit lectus, mattis quis nibh nec, faucibus rutrum sapien. Sed iaculis consectetur mi, eget posuere turpis finibus et.\n#### Capçalera 4.1\nLorem ipsum dolor sit amet, consectetur adipiscing elit. Mauris sed mollis augue, vel efficitur lacus. Pellentesque eu egestas mi. Etiam ultrices lectus sit amet aliquet ullamcorper. Praesent in erat quis est sagittis finibus. Etiam nec sapien in nulla interdum faucibus. Integer iaculis lorem quis arcu lobortis, non malesuada neque vehicula. Aenean nec tellus eu metus bibendum tempus. Sed rutrum urna ut commodo tempor. Vestibulum aliquet elit posuere turpis maximus condimentum. Sed urna libero, ornare eu tellus vitae, laoreet condimentum risus. Aenean elit lectus, mattis quis nibh nec, faucibus rutrum sapien. Sed iaculis consectetur mi, eget posuere turpis finibus et.\n#### Capçalera 4.1\nLorem ipsum dolor sit amet, consectetur adipiscing elit. Mauris sed mollis augue, vel efficitur lacus. Pellentesque eu egestas mi. Etiam ultrices lectus sit amet aliquet ullamcorper. Praesent in erat quis est sagittis finibus. Etiam nec sapien in nulla interdum faucibus. Integer iaculis lorem quis arcu lobortis, non malesuada neque vehicula. Aenean nec tellus eu metus bibendum tempus. Sed rutrum urna ut commodo tempor. Vestibulum aliquet elit posuere turpis maximus condimentum. Sed urna libero, ornare eu tellus vitae, laoreet condimentum risus. Aenean elit lectus, mattis quis nibh nec, faucibus rutrum sapien. Sed iaculis consectetur mi, eget posuere turpis finibus et.\n"
  },
  {
    "path": "content/blog/toc/index.es.md",
    "content": "+++\ntitle = \"Tabla de contenido\"\ndate = 2022-11-01\nupdated = 2024-02-16\ndescription = \"Una publicación que muestra la tabla de contenido opcional así como su configuración.\"\n\n[taxonomies]\ntags = [\"funcionalidad\", \"markdown\", \"tutorial\"]\n\n[extra]\ntoc = true\nquick_navigation_buttons = true\nsocial_media_card = \"social_cards/es_blog_toc.jpg\"\n+++\n\n## Documentación\n### Habilitando (y posicionando) la Tabla de Contenido\n\nHay dos formas de habilitar la Tabla de Contenido (TdC). Si quieres que esté justo debajo del encabezado (como en esta página), configura esta variable en el front matter de tu post:\n\n```toml\n[extra]\ntoc = true\n```\n\nSi prefieres situar la TdC en otro lugar de tu post (por ejemplo, después de una introducción), puedes hacerlo añadiendo una línea con este contenido ahí donde quieras que aparezca la TdC:\n\n```markdown\n<!-- toc -->\n```\n\nTambién puedes usar el shortcode `{{/* toc() */}}`, que simplemente insertará ese texto por ti ([idea de Michael Clayton](https://github.com/getzola/zola/issues/584#issuecomment-1546086781)).\n\nEste método renderizará la TdC sin el encabezado \"Tabla de contenido\". Esto te permite usar un encabezado diferente (o ninguno) para la TdC, o incluso ocultarla de forma predeterminada:\n\n<details>\n    <summary>TdC oculta</summary>\n    <!-- toc -->\n</details>\n\nEl código para lograr esto:\n\n```markdown\n<details>\n    <summary>TdC oculta</summary>\n    <!-- toc -->\n</details>\n```\n\n*Nota*: Si activas la TdC a través de `toc = true` y además añades `<!-- toc -->` en algún lugar de tu texto, obtendrás múltiples TdCs.\n\nSi colocas la TdC en algún lugar distinto al predeterminado y le añades un encabezado, seguramente quieras ocultar dicho encabezado de la TdC (consulta la [sección para ocultar encabezados](#ocultando-encabezados-de-la-tdc)). Puedes lograrlo así:\n\n```markdown,hl_lines=06 11-13\n+++\ntitle = \"El título de tu post\"\ndate = 2034-01-11\n\n[extra]\ntoc_ignore_pattern = \"^(Tabla de contenido)\"\n+++\n\nAquí va algún texto introductorio.\n\n### Tabla de contenido\n\n<!-- toc -->\n\n## Primer encabezado de contenido\n```\n\n### Estableciendo la profundidad máxima\n\nPuedes establecer la profundidad máxima para la TdC especificando la variable `toc_levels`, que acepta un número entero entre 1 y 4:\n\n```toml\n[extra]\ntoc_levels = 2\n```\n\nEn este ejemplo, sólo los dos primeros niveles de encabezados se incluirían en la TdC, independientemente de sus etiquetas HTML reales (`h1`, `h2`, `h3`, etc.). Si quieres incluir sólo el nivel principal de encabezados, establece `toc_levels = 1`. El valor predeterminado de `toc_levels` es `3`.\n\nTen en cuenta a tus lectores al establecer `toc_levels`. Aunque puede ser tentador incluir muchos niveles anidados para una navegación detallada, una TdC más corta y sencilla suele ser más amigable y menos abrumadora.\n\n### Ocultando encabezados de la TdC\n\nEs posible que quieras ocultar ciertos encabezados. Por ejemplo, si tu artículo tiene muchas Figuras o Tablas, éstas podrían saturar la TdC. Puedes ocultar encabezados específicos en la TdC configurando la variable `toc_ignore_pattern` en la sección `[extra]` del front matter de tu post.\n\nEsta variable espera una expresión regular (regex), ya que utiliza el test [matching](https://keats.github.io/tera/docs/#matching) de Tera. El `toc_ignore_pattern` se prueba contra el texto del encabezado. Por ejemplo, para el encabezado `### Lectura adicional`, sólo el texto `Lectura adicional` se usaría para comprobar si concuerda con el patrón.\n\nAquí tienes algunos valores de ejemplo para `toc_ignore_pattern` junto con los encabezados que ocultarían:\n\n| `toc_ignore_pattern`             | Excluye encabezados que…                                             |\n|----------------------------------|----------------------------------------------------------------------|\n| `Tabla`                          | contengan \"Tabla\"                                                    |\n| `^Figura`                        | empiecen por \"Figura\"                                                |\n| <code>(?i)(tabla\\|figura)</code> | empiecen por \"Tabla\" o \"Figura\" (insensible a mayúsculas/minúsculas) |\n| `\\[Borrador\\]$`                  | terminen con \"[Borrador]\".                                           |\n\nPuedes probar tu expresión regular en plataformas como [regex101](https://regex101.com/r/2dI7U2/1) para asegurarte de que funciona como esperas.\n\n*Nota*: Las capacidades de \"look-around\", incluyendo look-ahead y look-behind, no están soportadas.\n\n# Encabezado 1\nLorem ipsum dolor sit amet, consectetur adipiscing elit. Mauris sed mollis augue, vel efficitur lacus. Pellentesque eu egestas mi. Etiam ultrices lectus sit amet aliquet ullamcorper. Praesent in erat quis est sagittis finibus. Etiam nec sapien in nulla interdum faucibus. Integer iaculis lorem quis arcu lobortis, non malesuada neque vehicula. Aenean nec tellus eu metus bibendum tempus. Sed rutrum urna ut commodo tempor. Vestibulum aliquet elit posuere turpis maximus condimentum. Sed urna libero, ornare eu tellus vitae, laoreet condimentum risus. Aenean elit lectus, mattis quis nibh nec, faucibus rutrum sapien. Sed iaculis consectetur mi, eget posuere turpis finibus et.\n\n## Encabezado 2\nLorem ipsum dolor sit amet, consectetur adipiscing elit. Mauris sed mollis augue, vel efficitur lacus. Pellentesque eu egestas mi. Etiam ultrices lectus sit amet aliquet ullamcorper. Praesent in erat quis est sagittis finibus. Etiam nec sapien in nulla interdum faucibus. Integer iaculis lorem quis arcu lobortis, non malesuada neque vehicula. Aenean nec tellus eu metus bibendum tempus. Sed rutrum urna ut commodo tempor. Vestibulum aliquet elit posuere turpis maximus condimentum. Sed urna libero, ornare eu tellus vitae, laoreet condimentum risus. Aenean elit lectus, mattis quis nibh nec, faucibus rutrum sapien. Sed iaculis consectetur mi, eget posuere turpis finibus et.\n### Encabezado 3.1\nLorem ipsum dolor sit amet, consectetur adipiscing elit. Mauris sed mollis augue, vel efficitur lacus. Pellentesque eu egestas mi. Etiam ultrices lectus sit amet aliquet ullamcorper. Praesent in erat quis est sagittis finibus. Etiam nec sapien in nulla interdum faucibus. Integer iaculis lorem quis arcu lobortis, non malesuada neque vehicula. Aenean nec tellus eu metus bibendum tempus. Sed rutrum urna ut commodo tempor. Vestibulum aliquet elit posuere turpis maximus condimentum. Sed urna libero, ornare eu tellus vitae, laoreet condimentum risus. Aenean elit lectus, mattis quis nibh nec, faucibus rutrum sapien. Sed iaculis consectetur mi, eget posuere turpis finibus et.\n#### Encabezado 4.1\nLorem ipsum dolor sit amet, consectetur adipiscing elit. Mauris sed mollis augue, vel efficitur lacus. Pellentesque eu egestas mi. Etiam ultrices lectus sit amet aliquet ullamcorper. Praesent in erat quis est sagittis finibus. Etiam nec sapien in nulla interdum faucibus. Integer iaculis lorem quis arcu lobortis, non malesuada neque vehicula. Aenean nec tellus eu metus bibendum tempus. Sed rutrum urna ut commodo tempor. Vestibulum aliquet elit posuere turpis maximus condimentum. Sed urna libero, ornare eu tellus vitae, laoreet condimentum risus. Aenean elit lectus, mattis quis nibh nec, faucibus rutrum sapien. Sed iaculis consectetur mi, eget posuere turpis finibus et.\n#### Encabezado 4.2\nLorem ipsum dolor sit amet, consectetur adipiscing elit. Mauris sed mollis augue, vel efficitur lacus. Pellentesque eu egestas mi. Etiam ultrices lectus sit amet aliquet ullamcorper. Praesent in erat quis est sagittis finibus. Etiam nec sapien in nulla interdum faucibus. Integer iaculis lorem quis arcu lobortis, non malesuada neque vehicula. Aenean nec tellus eu metus bibendum tempus. Sed rutrum urna ut commodo tempor. Vestibulum aliquet elit posuere turpis maximus condimentum. Sed urna libero, ornare eu tellus vitae, laoreet condimentum risus. Aenean elit lectus, mattis quis nibh nec, faucibus rutrum sapien. Sed iaculis consectetur mi, eget posuere turpis finibus et.\n\n### Encabezado 3.2\nLorem ipsum dolor sit amet, consectetur adipiscing elit. Mauris sed mollis augue, vel efficitur lacus. Pellentesque eu egestas mi. Etiam ultrices lectus sit amet aliquet ullamcorper. Praesent in erat quis est sagittis finibus. Etiam nec sapien in nulla interdum faucibus. Integer iaculis lorem quis arcu lobortis, non malesuada neque vehicula. Aenean nec tellus eu metus bibendum tempus. Sed rutrum urna ut commodo tempor. Vestibulum aliquet elit posuere turpis maximus condimentum. Sed urna libero, ornare eu tellus vitae, laoreet condimentum risus. Aenean elit lectus, mattis quis nibh nec, faucibus rutrum sapien. Sed iaculis consectetur mi, eget posuere turpis finibus et.\n#### Encabezado 4\nLorem ipsum dolor sit amet, consectetur adipiscing elit. Mauris sed mollis augue, vel efficitur lacus. Pellentesque eu egestas mi. Etiam ultrices lectus sit amet aliquet ullamcorper. Praesent in erat quis est sagittis finibus. Etiam nec sapien in nulla interdum faucibus. Integer iaculis lorem quis arcu lobortis, non malesuada neque vehicula. Aenean nec tellus eu metus bibendum tempus. Sed rutrum urna ut commodo tempor. Vestibulum aliquet elit posuere turpis maximus condimentum. Sed urna libero, ornare eu tellus vitae, laoreet condimentum risus. Aenean elit lectus, mattis quis nibh nec, faucibus rutrum sapien. Sed iaculis consectetur mi, eget posuere turpis finibus et.\n#### Encabezado 4\nLorem ipsum dolor sit amet, consectetur adipiscing elit. Mauris sed mollis augue, vel efficitur lacus. Pellentesque eu egestas mi. Etiam ultrices lectus sit amet aliquet ullamcorper. Praesent in erat quis est sagittis finibus. Etiam nec sapien in nulla interdum faucibus. Integer iaculis lorem quis arcu lobortis, non malesuada neque vehicula. Aenean nec tellus eu metus bibendum tempus. Sed rutrum urna ut commodo tempor. Vestibulum aliquet elit posuere turpis maximus condimentum. Sed urna libero, ornare eu tellus vitae, laoreet condimentum risus. Aenean elit lectus, mattis quis nibh nec, faucibus rutrum sapien. Sed iaculis consectetur mi, eget posuere turpis finibus et.\n\n## Encabezado 2\nLorem ipsum dolor sit amet, consectetur adipiscing elit. Mauris sed mollis augue, vel efficitur lacus. Pellentesque eu egestas mi. Etiam ultrices lectus sit amet aliquet ullamcorper. Praesent in erat quis est sagittis finibus. Etiam nec sapien in nulla interdum faucibus. Integer iaculis lorem quis arcu lobortis, non malesuada neque vehicula. Aenean nec tellus eu metus bibendum tempus. Sed rutrum urna ut commodo tempor. Vestibulum aliquet elit posuere turpis maximus condimentum. Sed urna libero, ornare eu tellus vitae, laoreet condimentum risus. Aenean elit lectus, mattis quis nibh nec, faucibus rutrum sapien. Sed iaculis consectetur mi, eget posuere turpis finibus et.\n### Encabezado 3.1\nLorem ipsum dolor sit amet, consectetur adipiscing elit. Mauris sed mollis augue, vel efficitur lacus. Pellentesque eu egestas mi. Etiam ultrices lectus sit amet aliquet ullamcorper. Praesent in erat quis est sagittis finibus. Etiam nec sapien in nulla interdum faucibus. Integer iaculis lorem quis arcu lobortis, non malesuada neque vehicula. Aenean nec tellus eu metus bibendum tempus. Sed rutrum urna ut commodo tempor. Vestibulum aliquet elit posuere turpis maximus condimentum. Sed urna libero, ornare eu tellus vitae, laoreet condimentum risus. Aenean elit lectus, mattis quis nibh nec, faucibus rutrum sapien. Sed iaculis consectetur mi, eget posuere turpis finibus et.\n#### Encabezado 4.1\nLorem ipsum dolor sit amet, consectetur adipiscing elit. Mauris sed mollis augue, vel efficitur lacus. Pellentesque eu egestas mi. Etiam ultrices lectus sit amet aliquet ullamcorper. Praesent in erat quis est sagittis finibus. Etiam nec sapien in nulla interdum faucibus. Integer iaculis lorem quis arcu lobortis, non malesuada neque vehicula. Aenean nec tellus eu metus bibendum tempus. Sed rutrum urna ut commodo tempor. Vestibulum aliquet elit posuere turpis maximus condimentum. Sed urna libero, ornare eu tellus vitae, laoreet condimentum risus. Aenean elit lectus, mattis quis nibh nec, faucibus rutrum sapien. Sed iaculis consectetur mi, eget posuere turpis finibus et.\n#### Encabezado 4.2\nLorem ipsum dolor sit amet, consectetur adipiscing elit. Mauris sed mollis augue, vel efficitur lacus. Pellentesque eu egestas mi. Etiam ultrices lectus sit amet aliquet ullamcorper. Praesent in erat quis est sagittis finibus. Etiam nec sapien in nulla interdum faucibus. Integer iaculis lorem quis arcu lobortis, non malesuada neque vehicula. Aenean nec tellus eu metus bibendum tempus. Sed rutrum urna ut commodo tempor. Vestibulum aliquet elit posuere turpis maximus condimentum. Sed urna libero, ornare eu tellus vitae, laoreet condimentum risus. Aenean elit lectus, mattis quis nibh nec, faucibus rutrum sapien. Sed iaculis consectetur mi, eget posuere turpis finibus et.\n\n### Encabezado 3.2\nLorem ipsum dolor sit amet, consectetur adipiscing elit. Mauris sed mollis augue, vel efficitur lacus. Pellentesque eu egestas mi. Etiam ultrices lectus sit amet aliquet ullamcorper. Praesent in erat quis est sagittis finibus. Etiam nec sapien in nulla interdum faucibus. Integer iaculis lorem quis arcu lobortis, non malesuada neque vehicula. Aenean nec tellus eu metus bibendum tempus. Sed rutrum urna ut commodo tempor. Vestibulum aliquet elit posuere turpis maximus condimentum. Sed urna libero, ornare eu tellus vitae, laoreet condimentum risus. Aenean elit lectus, mattis quis nibh nec, faucibus rutrum sapien. Sed iaculis consectetur mi, eget posuere turpis finibus et.\n#### Encabezado 4.1\nLorem ipsum dolor sit amet, consectetur adipiscing elit. Mauris sed mollis augue, vel efficitur lacus. Pellentesque eu egestas mi. Etiam ultrices lectus sit amet aliquet ullamcorper. Praesent in erat quis est sagittis finibus. Etiam nec sapien in nulla interdum faucibus. Integer iaculis lorem quis arcu lobortis, non malesuada neque vehicula. Aenean nec tellus eu metus bibendum tempus. Sed rutrum urna ut commodo tempor. Vestibulum aliquet elit posuere turpis maximus condimentum. Sed urna libero, ornare eu tellus vitae, laoreet condimentum risus. Aenean elit lectus, mattis quis nibh nec, faucibus rutrum sapien. Sed iaculis consectetur mi, eget posuere turpis finibus et.\n#### Encabezado 4.1\nLorem ipsum dolor sit amet, consectetur adipiscing elit. Mauris sed mollis augue, vel efficitur lacus. Pellentesque eu egestas mi. Etiam ultrices lectus sit amet aliquet ullamcorper. Praesent in erat quis est sagittis finibus. Etiam nec sapien in nulla interdum faucibus. Integer iaculis lorem quis arcu lobortis, non malesuada neque vehicula. Aenean nec tellus eu metus bibendum tempus. Sed rutrum urna ut commodo tempor. Vestibulum aliquet elit posuere turpis maximus condimentum. Sed urna libero, ornare eu tellus vitae, laoreet condimentum risus. Aenean elit lectus, mattis quis nibh nec, faucibus rutrum sapien. Sed iaculis consectetur mi, eget posuere turpis finibus et.\n"
  },
  {
    "path": "content/blog/toc/index.md",
    "content": "+++\ntitle = \"Table of Contents\"\ndate = 2022-11-01\nupdated = 2024-02-16\ndescription = \"A post showcasing the optional Table of Contents and its options.\"\n\n[taxonomies]\ntags = [\"showcase\", \"markdown\", \"tutorial\"]\n\n[extra]\ntoc = true\nquick_navigation_buttons = true\nsocial_media_card = \"social_cards/blog_toc.jpg\"\n+++\n\n## Documentation\n### Enabling (and positioning) the Table of Contents\n\nThere are two ways to enable the Table of Contents (ToC). If you want it to be right below the header (like in this page) set this variable on the post's front matter:\n\n```toml\n[extra]\ntoc = true\n```\nIf you'd rather show the ToC elsewhere on your post (e.g. after an introduction), you can do so by adding a line with this content wherever you'd like the ToC to appear:\n\n```markdown\n<!-- toc -->\n```\n\nYou can also use the simple `{{/* toc() */}}` shortcode, which will simply write that string for you, effectively inserting the ToC ([Michael Clayton's idea](https://github.com/getzola/zola/issues/584#issuecomment-1546086781)).\n\nThis method will render the ToC without the \"Table of Contents\" header. This allows you to use a different (or no) header for the ToC, or hide it like this:\n\n<details>\n    <summary>Hidden ToC</summary>\n    <!-- toc -->\n</details>\n\nThe code to achieve this:\n\n```markdown\n<details>\n    <summary>Hidden ToC</summary>\n    <!-- toc -->\n</details>\n```\n\n*Note*: If you both set `toc = true` and have `<!-- toc -->` somewhere in your text, you'll get multiple ToCs.\n\nIf you set a custom position and a custom header for the ToC, you'll probably want to hide it (see the [section below](#hiding-headers-from-the-toc)) like this:\n\n```markdown,hl_lines=06 11-13\n+++\ntitle = \"Your Post's Title\"\ndate = 2034-01-11\n\n[extra]\ntoc_ignore_pattern = \"^(Table of Contents)\"\n+++\n\nHere goes some introductory text.\n\n### Table of Contents\n\n<!-- toc -->\n\n## First content header\n```\n\n### Setting a maximum depth\n\nYou can set the maximum depth for the ToC by specifying the `toc_levels` variable, which takes an integer between 1 and 4:\n\n```toml\n[extra]\ntoc_levels = 2\n```\n\nIn this example, only the first two levels of headers would be included in the ToC, regardless of their actual HTML tags (`h1`, `h2`, `h3`, etc.). If you want to include only the main level of headers, set `toc_levels = 1`. The default `toc_levels` value is `3`.\n\nKeep your readers in mind when setting the `toc_levels`. While it can be tempting to include many nested levels for detailed navigation, a shorter and simpler ToC can often be more reader-friendly and less overwhelming. Adjust the depth according to the complexity and length of your content for the best reader experience.\n\n### Hiding headers from the ToC\n\nYou might want to hide certain headers. For example, if your article has many Figures or Tables, they might clutter the ToC. You can hide specific headers in the ToC with the `toc_ignore_pattern` variable.\n\nThis variable expects a regular expression (regex), as it's using Tera's [matching](https://keats.github.io/tera/docs/#matching) test. The `toc_ignore_pattern` is tested against the text of the header, excluding the `#` character(s). For example, for the header `### Further reading`, the text `Further reading` would be checked against.\n\nHere are some example values for `toc_ignore_pattern` along with the headers they can hide:\n\n| `toc_ignore_pattern`             | Excludes headers which…                           |\n|----------------------------------|---------------------------------------------------|\n| `Table`                          | contain \"Table\"                                   |\n| `^Figure`                        | start with \"Figure\"                               |\n| <code>(?i)(table\\|figure)</code> | start with \"Table\" or \"Figure\" (case insensitive) |\n| `\\[Draft\\]$`                     | end with \"[Draft]\".                               |\n\nYou can test your regular expression on a site like [regex101](https://regex101.com/r/2dI7U2/1) to ensure it works as expected.\n\n*Note*: \"Look-around\" capabilities, including look-ahead and look-behind, are not supported.\n\n# Heading 1\nLorem ipsum dolor sit amet, consectetur adipiscing elit. Mauris sed mollis augue, vel efficitur lacus. Pellentesque eu egestas mi. Etiam ultrices lectus sit amet aliquet ullamcorper. Praesent in erat quis est sagittis finibus. Etiam nec sapien in nulla interdum faucibus. Integer iaculis lorem quis arcu lobortis, non malesuada neque vehicula. Aenean nec tellus eu metus bibendum tempus. Sed rutrum urna ut commodo tempor. Vestibulum aliquet elit posuere turpis maximus condimentum. Sed urna libero, ornare eu tellus vitae, laoreet condimentum risus. Aenean elit lectus, mattis quis nibh nec, faucibus rutrum sapien. Sed iaculis consectetur mi, eget posuere turpis finibus et.\n\n## Heading 2\nLorem ipsum dolor sit amet, consectetur adipiscing elit. Mauris sed mollis augue, vel efficitur lacus. Pellentesque eu egestas mi. Etiam ultrices lectus sit amet aliquet ullamcorper. Praesent in erat quis est sagittis finibus. Etiam nec sapien in nulla interdum faucibus. Integer iaculis lorem quis arcu lobortis, non malesuada neque vehicula. Aenean nec tellus eu metus bibendum tempus. Sed rutrum urna ut commodo tempor. Vestibulum aliquet elit posuere turpis maximus condimentum. Sed urna libero, ornare eu tellus vitae, laoreet condimentum risus. Aenean elit lectus, mattis quis nibh nec, faucibus rutrum sapien. Sed iaculis consectetur mi, eget posuere turpis finibus et.\n### Heading 3.1\nLorem ipsum dolor sit amet, consectetur adipiscing elit. Mauris sed mollis augue, vel efficitur lacus. Pellentesque eu egestas mi. Etiam ultrices lectus sit amet aliquet ullamcorper. Praesent in erat quis est sagittis finibus. Etiam nec sapien in nulla interdum faucibus. Integer iaculis lorem quis arcu lobortis, non malesuada neque vehicula. Aenean nec tellus eu metus bibendum tempus. Sed rutrum urna ut commodo tempor. Vestibulum aliquet elit posuere turpis maximus condimentum. Sed urna libero, ornare eu tellus vitae, laoreet condimentum risus. Aenean elit lectus, mattis quis nibh nec, faucibus rutrum sapien. Sed iaculis consectetur mi, eget posuere turpis finibus et.\n#### Heading 4.1\nLorem ipsum dolor sit amet, consectetur adipiscing elit. Mauris sed mollis augue, vel efficitur lacus. Pellentesque eu egestas mi. Etiam ultrices lectus sit amet aliquet ullamcorper. Praesent in erat quis est sagittis finibus. Etiam nec sapien in nulla interdum faucibus. Integer iaculis lorem quis arcu lobortis, non malesuada neque vehicula. Aenean nec tellus eu metus bibendum tempus. Sed rutrum urna ut commodo tempor. Vestibulum aliquet elit posuere turpis maximus condimentum. Sed urna libero, ornare eu tellus vitae, laoreet condimentum risus. Aenean elit lectus, mattis quis nibh nec, faucibus rutrum sapien. Sed iaculis consectetur mi, eget posuere turpis finibus et.\n#### Heading 4.2\nLorem ipsum dolor sit amet, consectetur adipiscing elit. Mauris sed mollis augue, vel efficitur lacus. Pellentesque eu egestas mi. Etiam ultrices lectus sit amet aliquet ullamcorper. Praesent in erat quis est sagittis finibus. Etiam nec sapien in nulla interdum faucibus. Integer iaculis lorem quis arcu lobortis, non malesuada neque vehicula. Aenean nec tellus eu metus bibendum tempus. Sed rutrum urna ut commodo tempor. Vestibulum aliquet elit posuere turpis maximus condimentum. Sed urna libero, ornare eu tellus vitae, laoreet condimentum risus. Aenean elit lectus, mattis quis nibh nec, faucibus rutrum sapien. Sed iaculis consectetur mi, eget posuere turpis finibus et.\n\n### Heading 3.2\nLorem ipsum dolor sit amet, consectetur adipiscing elit. Mauris sed mollis augue, vel efficitur lacus. Pellentesque eu egestas mi. Etiam ultrices lectus sit amet aliquet ullamcorper. Praesent in erat quis est sagittis finibus. Etiam nec sapien in nulla interdum faucibus. Integer iaculis lorem quis arcu lobortis, non malesuada neque vehicula. Aenean nec tellus eu metus bibendum tempus. Sed rutrum urna ut commodo tempor. Vestibulum aliquet elit posuere turpis maximus condimentum. Sed urna libero, ornare eu tellus vitae, laoreet condimentum risus. Aenean elit lectus, mattis quis nibh nec, faucibus rutrum sapien. Sed iaculis consectetur mi, eget posuere turpis finibus et.\n#### Heading 4\nLorem ipsum dolor sit amet, consectetur adipiscing elit. Mauris sed mollis augue, vel efficitur lacus. Pellentesque eu egestas mi. Etiam ultrices lectus sit amet aliquet ullamcorper. Praesent in erat quis est sagittis finibus. Etiam nec sapien in nulla interdum faucibus. Integer iaculis lorem quis arcu lobortis, non malesuada neque vehicula. Aenean nec tellus eu metus bibendum tempus. Sed rutrum urna ut commodo tempor. Vestibulum aliquet elit posuere turpis maximus condimentum. Sed urna libero, ornare eu tellus vitae, laoreet condimentum risus. Aenean elit lectus, mattis quis nibh nec, faucibus rutrum sapien. Sed iaculis consectetur mi, eget posuere turpis finibus et.\n#### Heading 4\nLorem ipsum dolor sit amet, consectetur adipiscing elit. Mauris sed mollis augue, vel efficitur lacus. Pellentesque eu egestas mi. Etiam ultrices lectus sit amet aliquet ullamcorper. Praesent in erat quis est sagittis finibus. Etiam nec sapien in nulla interdum faucibus. Integer iaculis lorem quis arcu lobortis, non malesuada neque vehicula. Aenean nec tellus eu metus bibendum tempus. Sed rutrum urna ut commodo tempor. Vestibulum aliquet elit posuere turpis maximus condimentum. Sed urna libero, ornare eu tellus vitae, laoreet condimentum risus. Aenean elit lectus, mattis quis nibh nec, faucibus rutrum sapien. Sed iaculis consectetur mi, eget posuere turpis finibus et.\n\n## Heading 2\nLorem ipsum dolor sit amet, consectetur adipiscing elit. Mauris sed mollis augue, vel efficitur lacus. Pellentesque eu egestas mi. Etiam ultrices lectus sit amet aliquet ullamcorper. Praesent in erat quis est sagittis finibus. Etiam nec sapien in nulla interdum faucibus. Integer iaculis lorem quis arcu lobortis, non malesuada neque vehicula. Aenean nec tellus eu metus bibendum tempus. Sed rutrum urna ut commodo tempor. Vestibulum aliquet elit posuere turpis maximus condimentum. Sed urna libero, ornare eu tellus vitae, laoreet condimentum risus. Aenean elit lectus, mattis quis nibh nec, faucibus rutrum sapien. Sed iaculis consectetur mi, eget posuere turpis finibus et.\n### Heading 3.1\nLorem ipsum dolor sit amet, consectetur adipiscing elit. Mauris sed mollis augue, vel efficitur lacus. Pellentesque eu egestas mi. Etiam ultrices lectus sit amet aliquet ullamcorper. Praesent in erat quis est sagittis finibus. Etiam nec sapien in nulla interdum faucibus. Integer iaculis lorem quis arcu lobortis, non malesuada neque vehicula. Aenean nec tellus eu metus bibendum tempus. Sed rutrum urna ut commodo tempor. Vestibulum aliquet elit posuere turpis maximus condimentum. Sed urna libero, ornare eu tellus vitae, laoreet condimentum risus. Aenean elit lectus, mattis quis nibh nec, faucibus rutrum sapien. Sed iaculis consectetur mi, eget posuere turpis finibus et.\n#### Heading 4.1\nLorem ipsum dolor sit amet, consectetur adipiscing elit. Mauris sed mollis augue, vel efficitur lacus. Pellentesque eu egestas mi. Etiam ultrices lectus sit amet aliquet ullamcorper. Praesent in erat quis est sagittis finibus. Etiam nec sapien in nulla interdum faucibus. Integer iaculis lorem quis arcu lobortis, non malesuada neque vehicula. Aenean nec tellus eu metus bibendum tempus. Sed rutrum urna ut commodo tempor. Vestibulum aliquet elit posuere turpis maximus condimentum. Sed urna libero, ornare eu tellus vitae, laoreet condimentum risus. Aenean elit lectus, mattis quis nibh nec, faucibus rutrum sapien. Sed iaculis consectetur mi, eget posuere turpis finibus et.\n#### Heading 4.2\nLorem ipsum dolor sit amet, consectetur adipiscing elit. Mauris sed mollis augue, vel efficitur lacus. Pellentesque eu egestas mi. Etiam ultrices lectus sit amet aliquet ullamcorper. Praesent in erat quis est sagittis finibus. Etiam nec sapien in nulla interdum faucibus. Integer iaculis lorem quis arcu lobortis, non malesuada neque vehicula. Aenean nec tellus eu metus bibendum tempus. Sed rutrum urna ut commodo tempor. Vestibulum aliquet elit posuere turpis maximus condimentum. Sed urna libero, ornare eu tellus vitae, laoreet condimentum risus. Aenean elit lectus, mattis quis nibh nec, faucibus rutrum sapien. Sed iaculis consectetur mi, eget posuere turpis finibus et.\n\n### Heading 3.2\nLorem ipsum dolor sit amet, consectetur adipiscing elit. Mauris sed mollis augue, vel efficitur lacus. Pellentesque eu egestas mi. Etiam ultrices lectus sit amet aliquet ullamcorper. Praesent in erat quis est sagittis finibus. Etiam nec sapien in nulla interdum faucibus. Integer iaculis lorem quis arcu lobortis, non malesuada neque vehicula. Aenean nec tellus eu metus bibendum tempus. Sed rutrum urna ut commodo tempor. Vestibulum aliquet elit posuere turpis maximus condimentum. Sed urna libero, ornare eu tellus vitae, laoreet condimentum risus. Aenean elit lectus, mattis quis nibh nec, faucibus rutrum sapien. Sed iaculis consectetur mi, eget posuere turpis finibus et.\n#### Heading 4.1\nLorem ipsum dolor sit amet, consectetur adipiscing elit. Mauris sed mollis augue, vel efficitur lacus. Pellentesque eu egestas mi. Etiam ultrices lectus sit amet aliquet ullamcorper. Praesent in erat quis est sagittis finibus. Etiam nec sapien in nulla interdum faucibus. Integer iaculis lorem quis arcu lobortis, non malesuada neque vehicula. Aenean nec tellus eu metus bibendum tempus. Sed rutrum urna ut commodo tempor. Vestibulum aliquet elit posuere turpis maximus condimentum. Sed urna libero, ornare eu tellus vitae, laoreet condimentum risus. Aenean elit lectus, mattis quis nibh nec, faucibus rutrum sapien. Sed iaculis consectetur mi, eget posuere turpis finibus et.\n#### Heading 4.1\nLorem ipsum dolor sit amet, consectetur adipiscing elit. Mauris sed mollis augue, vel efficitur lacus. Pellentesque eu egestas mi. Etiam ultrices lectus sit amet aliquet ullamcorper. Praesent in erat quis est sagittis finibus. Etiam nec sapien in nulla interdum faucibus. Integer iaculis lorem quis arcu lobortis, non malesuada neque vehicula. Aenean nec tellus eu metus bibendum tempus. Sed rutrum urna ut commodo tempor. Vestibulum aliquet elit posuere turpis maximus condimentum. Sed urna libero, ornare eu tellus vitae, laoreet condimentum risus. Aenean elit lectus, mattis quis nibh nec, faucibus rutrum sapien. Sed iaculis consectetur mi, eget posuere turpis finibus et.\n"
  },
  {
    "path": "content/pages/_index.ar.md",
    "content": "+++\nrender = false\ninsert_anchor_links = \"right\"\n\n[extra]\nhide_from_feed = true\n+++\n"
  },
  {
    "path": "content/pages/_index.ca.md",
    "content": "+++\nrender = false\ninsert_anchor_links = \"left\"\n\n[extra]\nhide_from_feed = true\n+++\n"
  },
  {
    "path": "content/pages/_index.es.md",
    "content": "+++\nrender = false\ninsert_anchor_links = \"left\"\n\n[extra]\nhide_from_feed = true\n+++\n"
  },
  {
    "path": "content/pages/_index.md",
    "content": "+++\nrender = false\ninsert_anchor_links = \"left\"\n\n[extra]\nhide_from_feed = true\n+++\n"
  },
  {
    "path": "content/pages/about/index.ar.md",
    "content": "+++\ntitle = \"عني\"\ntemplate = \"info-page.html\"\npath = \"/ar/about\"\n\n[extra]\nquick_navigation_buttons = true\nsocial_media_card = \"social_cards/about.ar.jpg\"\n+++\n\nمرحبًا بكم في العرض التقديمي ل[**تابي**]، وهو قالب لـ[Zola]، مولد المواقع الثابتة السريع.\n\n**تابي** هو من ابتكار أوسكار فرنانديز ([welpo] على GitHub)، وقد صُمم في البداية لموقعه الشخصي [osc.garden]. هو قالب بشكل حديث غني بالميزات يوفر إعدادات افتراضية منطقية وآمنة.\n\nيمكنك العثور على وثائق تشرح طريقة إستخدام **تابي** في [المدونة] و[الأرشيف]، سوف تجد بهم مقالات تشرح القالب وتقدم لك نظرة عامة حوله.\n\n## الميزات\n\nيركز **تابي** على الاداء ويدعم [تعدد اللغات] أي انه يمكن لمدونتك دعم أكثر من لغة، ويمكنك [تخصيص مظهرها]، ويمكنك [تفعيل التعليقات بها] أي السماح للأخرين بالتعليق على منشوراتك. و **تابي** لا يجبر مدونتك على إستخدام جافاسكربت، حيث انها [إختيارية] ولكنك سوف تفقد المميزات التي تحتاج إليها.\n\n## المساهمة\n\nنحن نقدر المساهمات! نرحب بالتبليغ على الأخطاء، وتحسينات الترجمة أو الوثائق (بغض النظر عن حجمها)، واقتراحات الميزات... تحقق من [إرشادات المساهمة] لمعرفة كيفية المساعدة. شكرًا لك!\n\n## الترخيص\n\nالكود متاح بموجب [رخصة MIT].\n\n[**تابي**]: https://github.com/welpo/tabi\n[Zola]: https://www.getzola.org\n[welpo]: https://github.com/welpo\n[osc.garden]: https://osc.garden\n[المدونة]: https://welpo.github.io/tabi/blog\n[الأرشيف]: https://welpo.github.io/tabi/archive\n[تعدد اللغات]: https://welpo.github.io/tabi/blog/faq-languages\n[تخصيص مظهرها]: https://welpo.github.io/tabi/blog/customise-tabi\n[تفعيل التعليقات بها]: https://welpo.github.io/tabi/blog/comments\n[إختيارية]: https://welpo.github.io/tabi/blog/javascript\n[إرشادات المساهمة]: https://github.com/welpo/tabi/blob/main/CONTRIBUTING.md\n[رخصة MIT]: https://choosealicense.com/licenses/mit\n"
  },
  {
    "path": "content/pages/about/index.ca.md",
    "content": "+++\ntitle = \"Sobre mi\"\ntemplate = \"info-page.html\"\npath = \"/ca/about\"\n\n[extra]\nquick_navigation_buttons = true\n+++\n\nBenvingut a la demo de [**tabi**](https://github.com/welpo/tabi), un tema per a [Zola](https://www.getzola.org/), un generador de llocs web estàtics rapidíssim.\n\n**tabi** és una obra d'Óscar Fernández ([welpo](https://github.com/welpo/) a GitHub), inicialment creat per al [seu lloc personal](https://osc.garden/ca/). És un tema modern i ple de funcionalitats que ofereix configuracions predefinides raonables i segures.\n\nBusques instruccions detallades o consells sobre com utilitzar **tabi**? Les seccions del [blog](https://welpo.github.io/tabi/ca/blog/) i de l'[arxiu](https://welpo.github.io/tabi/ca/archive/) contenen la documentació del tema, que inclou des de mostres de funcions fins a guies detallades.\n\n## Característiques\n\n**tabi** ofereix suport per a [diversos idiomes](https://welpo.github.io/tabi/ca/blog/faq-languages/), [aparença personalitzable](https://welpo.github.io/tabi/ca/blog/customise-tabi/), [integració de comentaris](https://welpo.github.io/tabi/blog/comments/), i molt més, tot amb un enfocament en el rendiment ([tot el JavaScript és opcional](https://welpo.github.io/tabi/ca/blog/javascript/)) i la [seguretat](https://welpo.github.io/tabi/ca/blog/security/).\n\n## Contribucions\n\nTota aportació és benvinguda! Agraïm els informes d'errors, millores en traduccions o documentació (per mínimes que siguin), sol·licituds de noves funcions… Consulta les [Pautes de Contribució](https://github.com/welpo/tabi/blob/main/CONTRIBUTING.md) per saber com pots contribuir. Gràcies!\n\n## Llicència\n\nEl codi està disponible sota la [llicència MIT](https://choosealicense.com/licenses/mit/).\n"
  },
  {
    "path": "content/pages/about/index.es.md",
    "content": "+++\ntitle = \"Sobre mí\"\ntemplate = \"info-page.html\"\npath = \"/es/about\"\n\n[extra]\nquick_navigation_buttons = true\n+++\n\nBienvenido a la demo de [**tabi**](https://github.com/welpo/tabi), un tema para [Zola](https://www.getzola.org/), un rapidísimo generador de sitios estáticos.\n\n**tabi** es obra de Óscar Fernández ([welpo](https://github.com/welpo/) en GitHub), inicialmente creado para [su página personal](https://osc.garden/es/). Es un tema moderno y lleno de funcionalidades que ofrece configuraciones predeterminadas sensatas y seguras.\n\n¿Buscas instrucciones detalladas o consejos sobre cómo utilizar **tabi**? Las secciones del [blog](https://welpo.github.io/tabi/es/blog/) y del [archivo](https://welpo.github.io/tabi/es/archive/) contienen la documentación del tema, que incluye desde muestras de funciones hasta guías detalladas.\n\n## Características\n\n**tabi** ofrece soporte para [varios idiomas](https://welpo.github.io/tabi/es/blog/faq-languages/), [apariencia personalizable](https://welpo.github.io/tabi/es/blog/customise-tabi/), [integración de comentarios](https://welpo.github.io/tabi/es/blog/comments/), y mucho más, todo con un enfoque en el rendimiento ([todo el JavaScript es opcional](https://welpo.github.io/tabi/es/blog/javascript/)) y la [seguridad](https://welpo.github.io/tabi/es/blog/security/).\n\n## Contribuciones\n\n¡Toda aportación es bienvenida! Agradecemos los reportes de errores, mejoras en traducciones o documentación (por mínimas que sean), solicitudes de nuevas funciones… Consulta las [Pautas de Contribución](https://github.com/welpo/tabi/blob/main/CONTRIBUTING.md) para saber cómo puedes contribuir. ¡Gracias!\n\n## Licencia\n\nEl código está disponible bajo la [licencia MIT](https://choosealicense.com/licenses/mit/).\n"
  },
  {
    "path": "content/pages/about/index.md",
    "content": "+++\ntitle = \"About\"\ntemplate = \"info-page.html\"\npath = \"about\"\n\n[extra]\nquick_navigation_buttons = true\n+++\n\nWelcome to the demo of [**tabi**](https://github.com/welpo/tabi), a theme for [Zola](https://www.getzola.org/), a fast static site generator.\n\n**tabi** is the creation of Óscar Fernández ([welpo](https://github.com/welpo/) on GitHub), initially designed for [his personal site](https://osc.garden/). It is a feature-rich, modern theme that provides sane (and safe) defaults.\n\nLooking for detailed instructions or tips on using **tabi**? The [blog](https://welpo.github.io/tabi/blog/) and [archive](https://welpo.github.io/tabi/archive/) sections feature the theme's documentation, offering a variety of articles from feature overviews to step-by-step guides.\n\n## Features\n\n**tabi** supports [multiple languages](https://welpo.github.io/tabi/blog/faq-languages/), [customisable appearance](https://welpo.github.io/tabi/blog/customise-tabi/), [comment integrations](https://welpo.github.io/tabi/blog/comments/), and much more, all with an emphasis on performance ([all JavaScript is optional](https://welpo.github.io/tabi/blog/javascript/)) and [security](https://welpo.github.io/tabi/blog/security/).\n\n## Contributing\n\nContributions are much appreciated! We appreciate bug reports, improvements to translations or documentation (however minor), feature requests… Check out the [Contributing Guidelines](https://github.com/welpo/tabi/blob/main/CONTRIBUTING.md) to learn how you can help. Thank you!\n\n## License\n\nThe code is available under the [MIT license](https://choosealicense.com/licenses/mit/).\n"
  },
  {
    "path": "content/pages/privacy/index.ar.md",
    "content": "+++\ntitle = \"سياسة الخصوصية\"\npath = \"/ar/privacy\"\ndate = 2023-10-31\nupdated = 2024-05-12\n+++\n\nهذه السياسة توضح كيف نقوم بجمع ومعالجة بياناتك على هذا الموقع.\n\n{{ toc() }}\n\n## ما البيانات التي نجمعها؟\n\n### التصفح العام {#what-general}\n\nأثناء تصفح الموقع، لا يتم جمع أي معلومات شخصية.\n\n### التعليقات {#what-comments}\n\nنحن لا نجمع أي بيانات عند إرسال تعليق على تدوينة، لكن GitHub يجمع البيانات لتوفير الخدمة، حيث انه يتم توفيرها من خلاله.\n\n### التحليلات {#what-analytics}\n\nلتحسين الموقع، يتم جمع البيانات غير الشخصية التالية:\n\n- **المرجع**: المصدر الذي قادك إلى هذا الموقع.\n- **الرابط المطلوب**: الصفحة المحددة التي قمت بزيارتها.\n- **وكيل المستخدم**: يحدد المتصفح ونظام التشغيل الذي تستخدمه (مثل \"Safari 17.0، Mac OS X\").\n- **اسم البلد**: البلد الذي زرت الموقع منه، يتم تحديده بواسطة عنوان IP الخاص بك.\n- **حجم الشاشة**: أبعاد شاشة جهازك.\n- **الوقت**: الوقت الذي وصلت فيه إلى الموقع.\n- **جلسة التصفح**: رمز تعريف مؤقت يتم إنشاؤه من عنوان IP الخاص بك، ومعلومات المتصفح، ورقم عشوائي. يُستخدم هذا للتعرف على جلسة التصفح لمدة 8 ساعات. بعد ذلك، يتم حذف الرمز من الذاكرة ولا يُخزن في أي مكان.\n\nنحن لا نتتبع الزوار عبر الجلسات، ولا نتتبع مدة بقائك في الموقع أو أين تذهب بعد مغادرتك.\n\n## كيف نجمع هذه البيانات؟\n\n### التعليقات {#how-comments}\n\nيتم جمع البيانات المرتبطة بالتعليقات باستخدام [Giscus](https://giscus.app/)، وهي منصة تمكن من إضافة التعليقات، المبنية على GitHub.\n\n### التحليلات {#how-analytics}\n\nيتم جمع البيانات غير الشخصية باستخدام نسخة مستضافة ذاتيًا من [GoatCounter](https://www.goatcounter.com/)، وهي منصة مفتوحة المصدر لتحليلات الويب تركز على الخصوصية.\n\n## كيف سنستخدم البيانات؟\n\nتُستخدم البيانات المرسلة إلى GitHub لعرض تعليقك على الموقع.\n\nتُستخدم البيانات غير الشخصية لإنشاء إحصائيات حول الموقع، مثل عدد الزوار يوميًا، أو الصفحات والمراجع الأكثر شعبية. تُستخدم هذه البيانات لتحسين هذا الموقع التجريبي وقالب تابي. يمكنك رؤية الإحصائيات الناتجة عن هذه البيانات على [صفحة الإحصائيات العامة](https://tabi-stats.osc.garden/).\n\nجميع البيانات المجمعة متاحة للجمهور، سواء في شكل تعليقات أو إحصائيات.\n\nنحن لا نستخدم البيانات لأي غرض آخر.\n\n## كيف نقوم بتخزين البيانات؟\n\nبيانات التعليقات لا تُخزن بواسطة Giscus، كما هو موضح في [سياسة الخصوصية الخاصة بهم](https://github.com/giscus/giscus/blob/main/PRIVACY-POLICY.md#what-data-do-we-collect). يتم تخزين البيانات على خوادم GitHub. راجع [سياسة الخصوصية الخاصة بGitHub](https://docs.github.com/en/site-policy/privacy-policies/github-privacy-statement).\n\nيتم تخزين بيانات التحليلات على خادم مستضاف بواسطة [Vultr](https://www.vultr.com/). يقع الخادم في باريس، فرنسا.\n\n## كم من الوقت سنقوم بتخزين البيانات؟\n\nتُخزن التعليقات إلى أجل غير مسمى، أو حتى تطلب حذفها.\n\nتُخزن باقي البيانات إلى أجل غير مسمى.\n\n## ما هي حقوقك في حماية البيانات؟\n\nاعتمادًا على المعالجة والأساس القانوني، هناك عدد من الخيارات المتاحة لك للحفاظ على السيطرة على بياناتك الشخصية:\n\n- حق الوصول إلى بياناتك\n- حق تعديل بياناتك\n- حق الاعتراض على معالجة بياناتك الشخصية\n- حق تقييد معالجة بياناتك\n- حق حذف بياناتك\n- حق سحب موافقتك\n\nإذا قدمت طلبًا، لدينا شهر واحد للرد عليك. إذا كنت ترغب في ممارسة أي من هذه الحقوق، يرجى الاتصال بنا باستخدام أيقونة البريد الإلكتروني في تذييل الموقع.\n\n## الكوكيز\n\nالموقع لا يستخدم الكوكيز.\n\n## سياسات الخصوصية لمواقع أخرى\n\nيحتوي هذا الموقع على روابط لمواقع أخرى. تنطبق سياسة الخصوصية هذه فقط على هذا الموقع، لذا إذا قمت بالنقر على رابط لموقع آخر، يجب عليك قراءة سياسة الخصوصية الخاصة بهم.\n\n## التغييرات على سياسة الخصوصية\n\nنحن نراجع هذه السياسة بانتظام ونضع أي تحديثات على هذه الصفحة. يمكنك التحقق من تاريخ تحديث هذه السياسة، بالإضافة إلى سجل التغييرات تحت عنوان الصفحة.\n\n## كيف تتواصل معنا\n\nإذا كانت لديك أي أسئلة حول سياسة الخصوصية هذه، أو البيانات التي نحتفظ بها عنك، أو إذا كنت ترغب في ممارسة أحد حقوق حماية البيانات الخاصة بك، فلا تتردد في الاتصال بنا باستخدام أيقونة البريد الإلكتروني في تذييل الموقع."
  },
  {
    "path": "content/pages/privacy/index.ca.md",
    "content": "+++\ntitle = \"Política de privacitat\"\npath = \"/ca/privacy\"\ndate = 2023-10-31\nupdated = 2024-05-12\n+++\n\nAquesta política de privacitat detalla com recollim i processem les teves dades en aquest lloc web.\n\n{{ toc() }}\n\n## Quines dades recollim?\n\n### Navegació general {#what-general}\n\nMentre navegues pel lloc, no es recull cap informació personal.\n\n### Comentaris {#what-comments}\n\nNo recollim cap dada quan envieu un comentari o reacció, però GitHub sí que ho fa per proporcionar el servei.\n\n### Anàlisis {#what-analytics}\n\nPer a la millora del lloc web, es recullen les dades no personals següents:\n\n- **Referent**: la font que t'ha portat a aquest lloc.\n- **URL sol·licitat**: la pàgina específica que visites.\n- **Agent d'usuari**: identifica el navegador i el sistema operatiu que utilitzes (per exemple, \"Safari 17.0, Mac OS X\").\n- **Nom del país**: el país des d'on estàs visitant, determinat per la teva adreça IP.\n- **Mida de pantalla**: les dimensions de la pantalla del teu dispositiu.\n- **Data i hora**: quan accedeixes al lloc.\n- **Sessió de navegació**: un codi d'identificació temporal generat a partir de la adreça IP, informació del navegador i un número aleatori. Aquest s'utilitza per reconèixer la sessió de navegació durant 8 hores. Passat aquest temps, el codi s'esborra de la memòria i no s'emmagatzema enlloc.\n\nNo seguim els visitants únics a través de sessions, ni seguim quant de temps et quedes al lloc o on vas en marxar.\n\n## Com recollim aquestes dades?\n\n### Comentaris {#how-comments}\n\nLes dades associades als comentaris es recullen utilitzant [giscus](https://giscus.app/), una plataforma que permet comentaris basada en GitHub.\n\n### Anàlisis {#how-analytics}\n\nLes dades no personals es recullen utilitzant una instància autoallotjada de [GoatCounter](https://www.goatcounter.com/), una plataforma d'anàlisis web de codi obert i respectuosa amb la privacitat.\n\n## Com utilitzarem les dades?\n\nLes dades enviades a GitHub s'utilitzen per mostrar el teu comentari al lloc.\n\nLes dades no personals s'utilitzen per generar estadístiques sobre el lloc, com el nombre de visitants per dia, o les pàgines i referents més populars. Aquestes dades s'utilitzen per millorar aquest lloc de demostració i el tema tabi. Pots veure les estadístiques generades a partir d'aquestes dades a la [pàgina d'estadístiques públiques](https://tabi-stats.osc.garden/).\n\nTotes les dades recollides estan públicament disponibles, ja sigui en forma de comentaris o estadístiques.\n\nNo utilitzem les dades per cap altre propòsit.\n\n## Com emmagatzemem les dades?\n\nLes dades dels comentaris no s'emmagatzemen per Giscus, tal com s'especifica a la seva [política de privacitat](https://github.com/giscus/giscus/blob/main/PRIVACY-POLICY.md#what-data-do-we-collect). Les dades s'emmagatzemen als servidors de GitHub. Vegeu la [política de privacitat de GitHub](https://docs.github.com/en/site-policy/privacy-policies/github-privacy-statement).\n\nLes dades d'anàlisis s'emmagatzemen en un servidor allotjat per [Vultr](https://www.vultr.com/). El servidor es troba a París, França.\n\nEl servidor segueix les millors pràctiques de la indústria per a la seguretat, incloent actualitzacions de seguretat automàtiques, una política de seguretat de contingut estricta, un tallafocs, accés SSH basat en claus, etc.\n\n## Quant de temps emmagatzemarem les dades?\n\nEls comentaris s'emmagatzemen indefinidament, o fins que sol·licitis la seva eliminació.\n\nLa resta de dades s'emmagatzemen indefinidament.\n\n## Quins són els teves drets de protecció de dades?\n\nDepenent del processament i de la base legal, hi ha diverses possibilitats disponibles per a que mantinguis el control sobre les teves dades personals:\n\n- Dret d'accés a les teves dades\n- Dret de modificar les teves dades\n- Dret d'oposar-se al processament de les teves dades personals\n- Dret de limitar el processament de les teves dades\n- Dret de supressió de les teves dades\n- Dret de retirar el teu consentiment\n\nSi fas una sol·licitud, tenim un mes per respondre't. Si vols exercir algun d'aquests drets, si us plau, posa't en contacte amb nosaltres utilitzant la icona de correu electrònic al peu de pàgina del lloc.\n\n## Galetes (cookies)\n\nEl lloc no utilitza galetes.\n\n## Polítiques de privacitat d'altres llocs web\n\nAquest lloc web conté enllaços a altres llocs web. Aquesta política de privacitat només s'aplica a aquest lloc web, així que si cliques en un enllaç cap a un altre lloc web, hauries de llegir la seva política de privacitat.\n\n## Canvis a la política de privacitat\n\nMantindrem aquesta política de privacitat sota revisió regular i col·locarem qualsevol actualització en aquesta pàgina web. Pots veure l'última data d'actualització d'aquesta política de privacitat, així com l'historial de canvis sota el títol d'aquesta pàgina.\n\n## Com posar-te en contacte amb nosaltres\n\nSi tens alguna pregunta sobre aquesta política de privacitat, les dades que tenim sobre tu, o vols exercir algun dels teus drets de protecció de dades, si us plau, no dubtis a posar-te en contacte amb nosaltres utilitzant la icona de correu electrònic al peu de pàgina del lloc.\n"
  },
  {
    "path": "content/pages/privacy/index.es.md",
    "content": "+++\ntitle = \"Política de privacidad\"\npath = \"/es/privacy\"\ndate = 2023-10-31\nupdated = 2024-05-12\n+++\n\nEsta política de privacidad describe cómo recopilamos y procesamos tus datos en este sitio web.\n\n{{ toc() }}\n\n## ¿Qué datos recopilamos?\n\n### Navegación general {#what-general}\n\nMientras navegas por el sitio, no se recopila ninguna información personal.\n\n### Comentarios {#what-comments}\n\nNo recopilamos ningún dato cuando envías un comentario o reacción, pero GitHub sí lo hace para proporcionar el servicio.\n\n### Análisis {#what-analytics}\n\nPara mejorar el sitio web, se recopila la siguiente información no personal:\n\n- **Referente**: la fuente que te llevó a este sitio.\n- **URL solicitado**: la página específica que visitas.\n- **Agente de usuario**: identifica el navegador y el sistema operativo que utilizas (por ejemplo, \"Safari 17.0, Mac OS X\").\n- **Nombre del país**: el país desde el que estás visitando, determinado por tu dirección IP.\n- **Tamaño de pantalla**: las dimensiones de la pantalla de tu dispositivo.\n- **Fecha y hora**: cuándo accedes al sitio.\n- **Sesión de navegación**: un código de identificación temporal generado a partir de tu dirección IP, información del navegador y un número aleatorio. Este código se utiliza para reconocer la sesión de navegación durante 8 horas. Después de ese tiempo, el código se elimina de la memoria y no se almacena en ningún lugar.\n\nNo rastreamos visitantes únicos a través de sesiones, ni el tiempo que permaneces en el sitio o a dónde vas después de salir.\n\n## ¿Cómo recopilamos estos datos?\n\n### Comentarios {#how-comments}\n\nLos datos asociados con los comentarios se recopilan utilizando [giscus](https://giscus.app/), una plataforma que habilita comentarios basados en GitHub.\n\n### Análisis {#how-analytics}\n\nLos datos no personales se recopilan mediante una instancia autoalojada de [GoatCounter](https://www.goatcounter.com/), una plataforma de análisis web de código abierto y respetuosa con la privacidad.\n\n## ¿Cómo utilizaremos los datos?\n\nLos datos enviados a GitHub se utilizan para mostrar tu comentario en el sitio.\n\nLos datos no personales se utilizan para generar estadísticas sobre el sitio, como el número de visitantes por día o las páginas y referentes más populares. Estos datos se utilizan para mejorar este sitio de demostración y el tema tabi. Puedes ver las estadísticas generadas a partir de estos datos en la [página de estadísticas públicas](https://tabi-stats.osc.garden/).\n\nTodos los datos recopilados están públicamente disponibles, ya sea en forma de comentarios o estadísticas.\n\nNo utilizamos los datos para ningún otro propósito.\n\n## ¿Cómo almacenamos los datos?\n\nLos datos de los comentarios no son almacenados por Giscus, como se especifica en su [política de privacidad](https://github.com/giscus/giscus/blob/main/PRIVACY-POLICY.md#what-data-do-we-collect). Los datos se almacenan en servidores de GitHub. Consulta la [política de privacidad de GitHub](https://docs.github.com/en/site-policy/privacy-policies/github-privacy-statement).\n\nLos datos de análisis se almacenan en un servidor alojado por [Vultr](https://www.vultr.com/). El servidor está ubicado en París, Francia.\n\nEl servidor sigue las mejores prácticas de la industria en cuanto a seguridad, incluidas actualizaciones de seguridad automáticas, una estricta Política de Seguridad de Contenido, un cortafuegos, acceso SSH basado en clave, etc.\n\n## ¿Cuánto tiempo almacenaremos los datos?\n\nLos comentarios se almacenan indefinidamente o hasta que solicites su eliminación.\n\nEl resto de los datos se almacena indefinidamente.\n\n## ¿Cuáles son tus derechos de protección de datos?\n\nDependiendo del procesamiento y la base legal, dispones de varias opciones para mantener el control sobre tus datos personales:\n\n- Derecho a acceder a tus datos\n- Derecho a enmendar tus datos\n- Derecho a oponerte al procesamiento de tus datos personales\n- Derecho a limitar el procesamiento de tus datos\n- Derecho a que se eliminen tus datos\n- Derecho a retirar tu consentimiento\n\nSi realizas una solicitud, tenemos un mes para responderte. Si deseas ejercer alguno de estos derechos, contáctanos utilizando el icono de correo electrónico en el pie de página del sitio.\n\n## Cookies\n\nEl sitio no utiliza cookies.\n\n## Políticas de privacidad de otros sitios web\n\nEste sitio web contiene enlaces a otros sitios web. Esta política de privacidad sólo se aplica a este sitio web, por lo que si haces clic en un enlace a otro sitio web, debes leer su política de privacidad.\n\n## Cambios en la política de privacidad\n\nMantenemos esta política de privacidad bajo revisión regular y colocamos cualquier actualización en esta página web. Puedes consultar la fecha de actualización de esta política de privacidad, así como el historial de cambios bajo el título de la página.\n\n## Cómo contactarnos\n\nSi tienes alguna pregunta sobre esta política de privacidad, los datos que tenemos sobre ti o si te gustaría ejercer alguno de tus derechos de protección de datos, no dudes en contactarnos utilizando el icono de correo electrónico en el pie de página del sitio.\n"
  },
  {
    "path": "content/pages/privacy/index.md",
    "content": "+++\ntitle = \"Privacy Policy\"\npath = \"privacy\"\ndate = 2023-10-31\nupdated = 2024-05-12\n+++\n\nThis privacy policy outlines how we collect and process your data on this website.\n\n{{ toc() }}\n\n## What data do we collect?\n\n### General browsing {#what-general}\n\nWhile browsing the site, no personal information is collected.\n\n### Comments {#what-comments}\n\nWe do not collect any data when you send a comment or reaction, but GitHub does in order to provide the service.\n\n### Analytics {#what-analytics}\n\nFor website improvement, the following non-personal data is collected:\n\n- **Referrer**: the source that led you to this site.\n- **Requested URL**: the specific page you visited.\n- **User-Agent**: identifies the browser and operating system you use (e.g. \"Safari 17.0, Mac OS X\").\n- **Country name**: the country you are visiting from, determined by your IP address.\n- **Screen size**: the dimensions of your device's screen.\n- **Time**: when you accessed the site.\n- **Browsing session**: a temporary identification code generated from your IP address, browser information, and a random number. This is used to recognize a browsing session for 8 hours. After that time, the code is deleted from memory and is not stored anywhere.\n\nWe do not track unique visitors across sessions, and we do not track how long you stay on the site or where you go after you leave.\n\n## How do we collect this data?\n\n### Comments {#how-comments}\n\nThe data associated with comments is collected using [giscus](https://giscus.app/), a platform that enables GitHub-based comments.\n\n### Analytics {#how-analytics}\n\nThe non-personal data is collected using a self-hosted instance of [GoatCounter](https://www.goatcounter.com/), an open-source privacy-friendly web analytics platform.\n\n## How will we use the data?\n\nThe data sent to GitHub is used to display your comment on the site.\n\nThe non personal data is used to generate statistics about the site, such as the number of visitors per day, or the most popular pages and referrers. This data is used to improve this demo site and the tabi theme. You can see the statistics generated from this data on the [public stats page](https://tabi-stats.osc.garden/).\n\nAll data collected is publicly available, either in the form of comments or statistics.\n\nWe do not use the data for any other purpose.\n\n## How do we store the data?\n\nThe comments data is not stored by Giscus, as specified in their [privacy policy](https://github.com/giscus/giscus/blob/main/PRIVACY-POLICY.md#what-data-do-we-collect). The data is stored on GitHub servers. See the [GitHub's privacy policy](https://docs.github.com/en/site-policy/privacy-policies/github-privacy-statement).\n\nThe analytics data is stored on a server hosted by [Vultr](https://www.vultr.com/). The server is located in Paris, France.\n\nThe server follows industry best practices for security, including automatic security updates, a strict Content Security Policy, a firewall, key-based SSH access, etc.\n\n## How long will we store the data?\n\nThe comments are stored indefinitely, or until you request their deletion.\n\nThe rest of the data is stored indefinitely.\n\n## What are your data protection rights?\n\nDepending on the processing and the legal basis, there are a number of possibilities available to you to keep control over your personal data:\n\n- Right to access your data\n- Right to amend your data\n- Right to object to the processing of your personal data\n- Right to limit the processing of your data\n- Right to have your data deleted\n- Right to withdraw your consent\n\nIf you make a request, we have one month to respond to you. If you would like to exercise any of these rights, please contact us using the e-mail icon in the footer of the site.\n\n## Cookies\n\nThe site does not use cookies.\n\n## Privacy policies of other websites\n\nThis website contains links to other websites. This privacy policy applies only to this website, so if you click on a link to another website, you should read their privacy policy.\n\n## Changes to the privacy policy\n\nWe keep this privacy policy under regular review and place any updates on this web page. You can check the date this privacy policy was updated, as well as the history of changes under the page title.\n\n## How to contact us\n\nIf you have any questions about this privacy policy, the data we hold on you, or you would like to exercise one of your data protection rights, please do not hesitate to contact us using the e-mail icon in the footer of the site.\n"
  },
  {
    "path": "content/projects/_index.ar.md",
    "content": "+++\ntitle = \"مشاريعي\"\nsort_by = \"weight\"\ntemplate = \"cards.html\"\ninsert_anchor_links = \"right\"\n\n[extra]\nshow_reading_time = false\nquick_navigation_buttons = true\n+++\n"
  },
  {
    "path": "content/projects/_index.ca.md",
    "content": "+++\ntitle = \"Projectes\"\nsort_by = \"weight\"\ntemplate = \"cards.html\"\ninsert_anchor_links = \"left\"\n\n[extra]\nsocial_media_card = \"projects/ca_projects.jpg\"\nshow_reading_time = false\nquick_navigation_buttons = true\n+++\n"
  },
  {
    "path": "content/projects/_index.es.md",
    "content": "+++\ntitle = \"Proyectos\"\nsort_by = \"weight\"\ntemplate = \"cards.html\"\ninsert_anchor_links = \"left\"\n\n[extra]\nsocial_media_card = \"projects/es_projects.jpg\"\nshow_reading_time = false\nquick_navigation_buttons = true\n+++\n"
  },
  {
    "path": "content/projects/_index.md",
    "content": "+++\ntitle = \"Projects\"\nsort_by = \"weight\"\ntemplate = \"cards.html\"\ninsert_anchor_links = \"left\"\n\n[extra]\nsocial_media_card = \"projects/projects.jpg\"\nshow_reading_time = false\nquick_navigation_buttons = true\niine_icon = 'heart'\n+++\n"
  },
  {
    "path": "content/projects/bunbu/index.ca.md",
    "content": "+++\ntitle = \"bunbu\"\ndescription = \"Analitza la freqüència de paraules en textos japonesos.\"\nweight = 34\n\n[taxonomies]\ntags = [\"Japonès\", \"web app\", \"web\", \"JavaScript\", \"PWA\"]\n\n[extra]\nlocal_image = \"projects/bunbu/bunbu_logo.webp\"\ncanonical_url = \"https://osc.garden/es/projects/bunbu/\"\nsocial_media_card = \"social_cards/projects_bunbu.jpg\"\n+++\n\n**bunbu** és una aplicació web progressiva per analitzar la freqüència de paraules en textos en japonès. Enganxa o carrega qualsevol text en japonès i bunbu el desglossarà en paraules individuals i et mostrarà amb quina freqüència apareix cadascuna.\n\n#### [Prova-la ara](https://bunbu.osc.garden) • [GitHub](https://github.com/welpo/bunbu) {.centered-text}\n\n## Característiques\n\n- Anàlisi morfològica de text japonès impulsada per [Goya](https://github.com/Leko/goya)\n- Taula de freqüència de paraules amb recomptes, percentatges i categories gramaticals\n- Furigana per als kanjis\n- Enllaços a diccionaris en línia (Jisho, Weblio, Kotobank, Goo, Wiktionary)\n- Referències de pronunciació (Forvo, YouGlish, ImmersionKit) i text a veu\n- Filtra els resultats per categoria gramatical (substantius, verbs, adjectius…)\n- Múltiples mètodes d'entrada: enganxar, escriure o importar fitxers\n- Processament 100% local: el teu text mai surt del teu dispositiu\n- Funciona fora de línia com a aplicació web progressiva (PWA)\n- Instal·lable en mòbils i escriptori\n\n[![targeta per a xarxes socials de bunbu](social_cards/projects_bunbu.jpg)](https://bunbu.osc.garden)\n"
  },
  {
    "path": "content/projects/bunbu/index.es.md",
    "content": "+++\ntitle = \"bunbu\"\ndescription = \"Analiza la frecuencia de palabras en textos japoneses.\"\nweight = 34\n\n[taxonomies]\ntags = [\"Japonés\", \"web app\", \"web\", \"JavaScript\", \"PWA\"]\n\n[extra]\nlocal_image = \"projects/bunbu/bunbu_logo.webp\"\ncanonical_url = \"https://osc.garden/es/projects/bunbu/\"\nsocial_media_card = \"social_cards/projects_bunbu.jpg\"\n+++\n\n**bunbu** es una aplicación web progresiva (PWA) para analizar la frecuencia de palabras en textos en japonés. Pega o carga cualquier texto en japonés y bunbu lo desglosará en palabras individuales, mostrándote con qué frecuencia aparece cada una.\n\n#### [Pruébala ahora](https://bunbu.osc.garden) • [GitHub](https://github.com/welpo/bunbu) {.centered-text}\n\n## Características\n\n- Análisis morfológico de texto japonés impulsado por [Goya](https://github.com/Leko/goya)\n- Tabla de frecuencia de palabras con conteos, porcentajes y categorías gramaticales\n- Furigana para los kanjis\n- Enlaces a diccionarios online (Jisho, Weblio, Kotobank, Goo, Wiktionary)\n- Referencias de pronunciación (Forvo, YouGlish, ImmersionKit) y texto a voz\n- Filtrado de resultados por categoría gramatical (sustantivos, verbos, adjetivos…)\n- Múltiples métodos de entrada: pegar, escribir o importar archivos\n- Procesamiento 100% local: tu texto nunca sale de tu dispositivo\n- Funciona sin conexión como aplicación web progresiva (PWA)\n- Instalable en móviles y escritorio\n\n[![tarjeta para redes sociales de bunbu](social_cards/projects_bunbu.jpg)](https://bunbu.osc.garden)\n"
  },
  {
    "path": "content/projects/bunbu/index.md",
    "content": "+++\ntitle = \"bunbu\"\ndescription = \"Word frequency analysis of Japanese text.\"\nweight = 34\n\n[taxonomies]\ntags = [\"Japanese\", \"web app\", \"web\", \"JavaScript\", \"PWA\"]\n\n[extra]\nlocal_image = \"projects/bunbu/bunbu_logo.webp\"\ncanonical_url = \"https://osc.garden/projects/bunbu/\"\nsocial_media_card = \"social_cards/projects_bunbu.jpg\"\n+++\n\n**bunbu** is a Progressive Web App to analyse word frequency in Japanese text. Paste or load any Japanese text and bunbu will break it down into individual words, showing you how often each one appears.\n\n#### [Try it now](https://bunbu.osc.garden) • [GitHub](https://github.com/welpo/bunbu) {.centered-text}\n\n## Features\n\n- Morphological analysis of Japanese text powered by [Goya](https://github.com/Leko/goya)\n- Word frequency table with counts, percentages, and parts of speech\n- Furigana for kanji\n- Links to online dictionaries (Jisho, Weblio, Kotobank, Goo, Wiktionary)\n- Pronunciation references (Forvo, YouGlish, ImmersionKit) and text-to-speech\n- Filter results by part of speech (nouns, verbs, adjectives…)\n- Multiple input methods: paste, type, or import files\n- 100% local processing—your text never leaves your device\n- Works offline as a Progressive Web App\n- Installable on mobile and desktop\n\n[![bunbu social media card](social_cards/projects_bunbu.jpg)](https://bunbu.osc.garden)\n"
  },
  {
    "path": "content/projects/doteki/index.ca.md",
    "content": "+++\ntitle = \"dōteki\"\ndescription = \"Afegeix contingut dinàmic al teu perfil de GitHub amb un sistema intuïtiu de plugins.\"\nweight = 30\n\n[taxonomies]\ntags = [\"GitHub Actions\", \"automatització\", \"Python\"]\n\n[extra]\nlocal_image = \"projects/doteki/doteki_logo.webp\"\nsocial_media_card = \"social_cards/projects_doteki.jpg\"\ncanonical_url = \"https://osc.garden/ca/projects/doteki/\"\n+++\n\n**dōteki** actualitza el teu perfil de GitHub automàticament. Afegeix les teves últimes publicacions del blog, la música que escoltes o qualsevol altre contingut dinàmic mitjançant plugins.\n\n![logo de dōteki: un riu passant per un bosc de bambú](https://cdn.jsdelivr.net/gh/welpo/doteki@main/website/static/img/logo.png)\n\n#### [GitHub](https://github.com/welpo/doteki) • [Lloc web](https://doteki.org/) • [Documentació](https://doteki.org/docs/) {.centered-text}\n\n## Com funciona\n\n1. Afegeix marcadors al teu README:\n\n```md,name=README.md\n<!-- blog start -->\n<!-- blog end -->\n```\n\n2. Configura què hi va:\n\n```toml,name=doteki.toml\n[sections.blog]\nplugin = \"feed\"\nurl = \"https://osc.garden/atom.xml\"  # Substitueix amb el teu feed.\n\n[sections.last_updated]\nplugin = \"current_date\"\ninline = true\n```\n\n3. Configura l'[Acció de GitHub](https://github.com/welpo/doteki-action).\n\nAixò és tot! El teu README s'actualitzarà automàticament.\n\n## Característiques\n\n- **Sistema de plugins**: Mostra [entrades del blog](https://doteki.org/docs/plugins/feed), [música](https://doteki.org/docs/plugins/lastfm), o [crea el teu propi plugin](https://doteki.org/docs/developer-guide/plugin-standard)\n- **Configuració simple**: Un arxiu TOML, una Acció de GitHub\n- **Flexible**: Cada plugin té les seves pròpies opcions (ordre, entrades màximes, format…)\n- **[Documentació detallada](https://doteki.org/docs/)**: Informació detallada sobre com configurar i utilitzar **dōteki** i els seus plugins. Inclou [instruccions clares per als desenvolupadors](https://doteki.org/docs/developer-guide/) que vulguin contribuir.\n\n## Documentació\n\nConsulta la [documentació](https://doteki.org/docs/) per a:\n\n- [Guia d'inici](https://doteki.org/docs/)\n- [Plugins disponibles](https://doteki.org/docs/category/plugins)\n- [Desenvolupament de plugins](https://doteki.org/docs/developer-guide/)\n- [Opcions de configuració](https://doteki.org/docs/configuration/)\n"
  },
  {
    "path": "content/projects/doteki/index.es.md",
    "content": "+++\ntitle = \"dōteki\"\ndescription = \"Añade contenido dinámico a tu perfil de GitHub con un sistema intuitivo de plugins.\"\nweight = 30\n\n[taxonomies]\ntags = [\"GitHub Actions\", \"automatización\", \"Python\"]\n\n[extra]\nlocal_image = \"projects/doteki/doteki_logo.webp\"\nsocial_media_card = \"social_cards/projects_doteki.jpg\"\ncanonical_url = \"https://osc.garden/es/projects/doteki/\"\n+++\n\n**dōteki** actualiza tu perfil de GitHub automáticamente. Añade tus últimas publicaciones del blog, la música que escuchas o cualquier otro contenido dinámico mediante plugins.\n\n![logo de dōteki: un río pasando por un bosque de bambú](https://cdn.jsdelivr.net/gh/welpo/doteki@main/website/static/img/logo.png)\n\n#### [GitHub](https://github.com/welpo/doteki) • [Sitio web](https://doteki.org/) • [Documentación](https://doteki.org/docs/) {.centered-text}\n\n## Cómo funciona\n\n1. Añade marcadores a tu README:\n\n```md,name=README.md\n<!-- blog start -->\n<!-- blog end -->\n```\n\n2. Configura qué va ahí:\n\n```toml,name=doteki.toml\n[sections.blog]\nplugin = \"feed\"\nurl = \"https://osc.garden/atom.xml\"  # Reemplaza con tu feed.\n\n[sections.last_updated]\nplugin = \"current_date\"\ninline = true\n```\n\n3. Configura la [Acción de GitHub](https://github.com/welpo/doteki-action).\n\n¡Eso es todo! Tu README se actualizará automáticamente.\n\n## Características\n\n- **Sistema de plugins**: Muestra [entradas del blog](https://doteki.org/docs/plugins/feed), [música](https://doteki.org/docs/plugins/lastfm), o [crea tu propio plugin](https://doteki.org/docs/developer-guide/plugin-standard)\n- **Configuración simple**: Un archivo TOML, una Acción de GitHub\n- **Flexible**: Cada plugin tiene sus propias opciones (orden, entradas máximas, formato…)\n- **[Documentación detallada](https://doteki.org/docs/)**: Información detallada sobre cómo configurar y usar **dōteki** y sus plugins. Incluye [instrucciones claras para los desarrolladores](https://doteki.org/docs/developer-guide/) que quieran contribuir.\n\n## Documentación\n\nConsulta la [documentación](https://doteki.org/docs/) para:\n\n- [Guía de inicio rápido](https://doteki.org/docs/)\n- [Plugins disponibles](https://doteki.org/docs/category/plugins)\n- [Desarrollo de plugins](https://doteki.org/docs/developer-guide/)\n- [Opciones de configuración](https://doteki.org/docs/configuration/)\n"
  },
  {
    "path": "content/projects/doteki/index.md",
    "content": "+++\ntitle = \"dōteki\"\ndescription = \"Add dynamic content to your GitHub profile through an intuitive plugin system.\"\nweight = 30\n\n[taxonomies]\ntags = [\"GitHub Actions\", \"automation\", \"Python\"]\n\n[extra]\nlocal_image = \"projects/doteki/doteki_logo.webp\"\nsocial_media_card = \"social_cards/projects_doteki.jpg\"\ncanonical_url = \"https://osc.garden/projects/doteki/\"\n+++\n\n**dōteki** updates your GitHub profile README automatically. Add your latest blog posts, music you're listening to, or any other dynamic content using plugins.\n\n![doteki logo: a river passing through a bamboo forest](https://cdn.jsdelivr.net/gh/welpo/doteki@main/website/static/img/logo.png)\n\n#### [GitHub](https://github.com/welpo/doteki) • [Website](https://doteki.org/) • [Documentation](https://doteki.org/docs/) {.centered-text}\n\n## How it works\n\n1. Add markers to your README:\n\n```md,name=README.md\n<!-- blog start -->\n<!-- blog end -->\n```\n\n2. Configure what goes there:\n\n```toml,name=doteki.toml\n[sections.blog]\nplugin = \"feed\"\nurl = \"https://osc.garden/atom.xml\"  # Replace with your feed.\n\n[sections.last_updated]\nplugin = \"current_date\"\ninline = true\n```\n\n3. Set up the [GitHub Action](https://github.com/welpo/doteki-action).\n\nThat's it! Your README will stay updated automatically.\n\n## Features\n\n- **Plugin system**: Show [blog posts](https://doteki.org/docs/plugins/feed), [music](https://doteki.org/docs/plugins/lastfm), or [build your own plugin](https://doteki.org/docs/developer-guide/plugin-standard)\n- **Simple setup**: One TOML file, one GitHub Action\n- **Flexible**: Each plugin has its own options (sort order, max entries, format…)\n- **[Extensive documentation](https://doteki.org/docs/)**: Detailed information on how to set up and use **dōteki** and its plugins. It includes [clear instructions for developers](https://doteki.org/docs/developer-guide/) looking to contribute.\n\n## Documentation\n\nCheck the [docs](https://doteki.org/docs/) for:\n\n- [Getting started guide](https://doteki.org/docs/)\n- [Available plugins](https://doteki.org/docs/category/plugins)\n- [Plugin development](https://doteki.org/docs/developer-guide/)\n- [Configuration options](https://doteki.org/docs/configuration/)\n"
  },
  {
    "path": "content/projects/git-sumi/index.ca.md",
    "content": "+++\ntitle = \"git-sumi\"\ndescription = \"El linter de missatges de commit no opinat basat en Rust.\"\nweight = 10\n\n[taxonomies]\ntags = [\"Git\", \"Rust\", \"Continuous Integration\", \"GitHub Actions\", \"CLI\", \"automatització\"]\n\n[extra]\nlocal_image = \"projects/git-sumi/git-sumi_logo.webp\"\nsocial_media_card = \"social_cards/projects_git-sumi.jpg\"\ncanonical_url = \"https://osc.garden/ca/projects/git-sumi/\"\n+++\n\n**git-sumi** és el linter de missatges de commit no opinat escrit en Rust.\n\n{% wide_container() %}\n<video controls src=\"https://cdn.jsdelivr.net/gh/welpo/git-sumi@main/assets/git-sumi_demo.mp4\" title=\"demo de git-sumi\"></video>\n{% end %}\n\n#### [GitHub](https://github.com/welpo/git-sumi) • [Lloc web](https://sumi.rs/) • [Documentació](https://sumi.rs/docs/) {.centered-text}\n\n## Característiques principals\n\n- **Regles personalitzables**: Configura regles per a Conventional Commits, límits de longitud, ús de [Gitmoji](https://gitmoji.dev/) i [més](https://sumi.rs/docs/rules).\n- **Informe d'errors clar**: Proporciona errors detallats, fent que la correcció sigui senzilla i educativa.\n- **Integració senzilla**: Com a binari únic, git-sumi s'integra fàcilment al teu flux de treball. També pots fer servir l'[Acció de GitHub](https://github.com/welpo/git-sumi-action) per validar commits (o títols de PR) sense instal·lar res.\n\n## Bones pràctiques de desenvolupament\n\n- **Cobertura de codi**: 98% de cobertura en tests; un linter ha de ser fiable.\n- **[Integració](https://github.com/welpo/git-sumi/blob/main/.github/workflows/ci.yml) i [publicació](https://github.com/welpo/git-sumi/blob/main/.github/workflows/release.yml) contínua**: Fluxos automatitzats per a testing i publicació de binaris multiplataforma a crates.io, PyPI i GitHub releases.\n- **Documentació**: [Documentació completa](https://sumi.rs/docs/) amb [guia ràpida](https://sumi.rs/docs/), [exemples](https://sumi.rs/docs/examples), [regles](https://sumi.rs/docs/rules), [integració](https://sumi.rs/docs/integration), [FAQ](https://sumi.rs/docs/faq)...\n"
  },
  {
    "path": "content/projects/git-sumi/index.es.md",
    "content": "+++\ntitle = \"git-sumi\"\ndescription = \"El linter de mensajes de commit no opinado basado en Rust.\"\nweight = 10\n\n[taxonomies]\ntags = [\"Git\", \"Rust\", \"Continuous Integration\", \"GitHub Actions\", \"CLI\", \"automatización\"]\n\n[extra]\nlocal_image = \"projects/git-sumi/git-sumi_logo.webp\"\nsocial_media_card = \"social_cards/projects_git-sumi.jpg\"\ncanonical_url = \"https://osc.garden/es/projects/git-sumi/\"\n+++\n\n**git-sumi** es el linter de mensajes de commit no opinado escrito en Rust.\n\n{% wide_container() %}\n<video controls src=\"https://cdn.jsdelivr.net/gh/welpo/git-sumi@main/assets/git-sumi_demo.mp4\" title=\"git-sumi demo\"></video>\n{% end %}\n\n#### [GitHub](https://github.com/welpo/git-sumi) • [Sitio web](https://sumi.rs/) • [Documentación](https://sumi.rs/docs/) {.centered-text}\n\n## Características principales\n\n- **Reglas personalizables**: Configura reglas para [Conventional Commits](https://www.conventionalcommits.org/), límites de longitud, uso de [Gitmoji](https://gitmoji.dev/) y [más](https://sumi.rs/docs/rules).\n- **Reporte de errores claro**: Proporciona errores detallados, haciendo que la corrección sea sencilla y educativa.\n- **Integración sencilla**: Al ser único binario, git-sumi se integra fácilmente en tu flujo de trabajo. Puedes usar la [Acción de GitHub](https://github.com/welpo/git-sumi-action) para validar commits (o títulos de PR) sin instalar nada.\n\n## Buenas prácticas de desarrollo\n\n- **Cobertura de código**: 98% de cobertura de código; un linter debe ser robusto.\n- **[Integración](https://github.com/welpo/git-sumi/blob/main/.github/workflows/ci.yml) y [publicación](https://github.com/welpo/git-sumi/blob/main/.github/workflows/release.yml) continua**: Flujos automatizados para testing y publicación de binarios multiplataforma en crates.io, PyPI y GitHub releases.\n- **Documentación**: [Documentación completa](https://sumi.rs/docs/) con [guía rápida](https://sumi.rs/docs/), [ejemplos](https://sumi.rs/docs/examples), [reglas](https://sumi.rs/docs/rules), [integración](https://sumi.rs/docs/integration), [FAQ](https://sumi.rs/docs/faq)...\n"
  },
  {
    "path": "content/projects/git-sumi/index.md",
    "content": "+++\ntitle = \"git-sumi\"\ndescription = \"The non-opinionated Rust-based commit message linter.\"\nweight = 10\n\n[taxonomies]\ntags = [\"Git\", \"Rust\", \"Continuous Integration\", \"GitHub Actions\", \"CLI\", \"automation\"]\n\n[extra]\nlocal_image = \"projects/git-sumi/git-sumi_logo.webp\"\nsocial_media_card = \"social_cards/projects_git-sumi.jpg\"\ncanonical_url = \"https://osc.garden/projects/git-sumi/\"\n+++\n\n**git-sumi** is the non-opinionated commit message linter written in Rust.\n\n{% wide_container() %}\n<video controls src=\"https://cdn.jsdelivr.net/gh/welpo/git-sumi@main/assets/git-sumi_demo.mp4\" title=\"git-sumi demo\"></video>\n{% end %}\n\n#### [GitHub](https://github.com/welpo/git-sumi) • [Website](https://sumi.rs/) • [Documentation](https://sumi.rs/docs/) {.centered-text}\n\n## Main features\n\n- **Customizable rules**: Configure rules to enforce [Conventional Commits](https://www.conventionalcommits.org/), length limits, [Gitmoji](https://gitmoji.dev/) usage, and [more](https://sumi.rs/docs/rules).\n- **Clear error reporting**: Provides detailed error reporting, making fixing commit messages straightforward and educational.\n- **Seamless integration**: As a single binary, git-sumi easily integrates into your existing workflow with minimal setup. You can even use the [GitHub Action](https://github.com/welpo/git-sumi-action) to lint your commits (or PR titles) without installing anything.\n\n## Development best practices\n\n- **Comprehensive code coverage**: 98% test coverage; linting needs to be reliable.\n- **Continuous [integration](https://github.com/welpo/git-sumi/blob/main/.github/workflows/ci.yml) and [deployment](https://github.com/welpo/git-sumi/blob/main/.github/workflows/release.yml)**: Automated workflows for testing and releasing cross-compiled binaries to crates.io, PyPI and GitHub releases.\n- **Documentation**: [Comprehensive documentation](https://sumi.rs/docs/) with a [quick start guide](https://sumi.rs/docs/), [examples](https://sumi.rs/docs/examples), [rules](https://sumi.rs/docs/rules), [integration](https://sumi.rs/docs/integration), [FAQ](https://sumi.rs/docs/faq)…\n"
  },
  {
    "path": "content/projects/iine/index.ca.md",
    "content": "+++\ntitle = \"iine\"\ndescription = \"Botons d'apreciació centrats en la privadesa i accessibles per a la web petita/independent.\"\nweight = 15\n\n[taxonomies]\ntags = [\"web\", \"JavaScript\", \"privadesa\", \"backend\"]\n\n[extra]\nlocal_image = \"projects/iine/iine_logo.webp\"\nsocial_media_card = \"social_cards/projects_iine.jpg\"\ncanonical_url = \"https://osc.garden/projects/iine/\"\n+++\n\nBotons d'apreciació per al teu blog, jardí digital, portafoli… No cal compte. Sense seguiment.\n\n#### [GitHub](https://github.com/welpo/iine) • [Web](https://iine.to) {.centered-text}\n\n## Característiques\n\n- Sense marques de temps, sense IPs, sense seguiment\n- ~3KB de JavaScript vanilla\n- Funciona sense JavaScript\n- Diverses icones: cor, polze amunt, vot positiu, o qualsevol emoji\n- Construït per a la web petita/independent\n- Gratuït per sempre\n- Accessible per disseny\n- Auto-hostejable\n\n## Inici ràpid\n\nAfegeix botons d'apreciació al teu lloc amb dues línies. Carrega l'script:\n\n```html\n<script defer src=\"https://cdn.jsdelivr.net/gh/welpo/iine@main/iine.mini.js\"></script>\n```\n\nAfegeix el botó:\n\n```html\n<button class=\"iine-button\" aria-hidden=\"true\"></button>\n```\n\nFet. Els teus visitants ja poden mostrar apreciació anònima pel teu contingut.\n\n## Per què iine?\n\nVolia afegir botons de m'agrada/felicitacions/alegria/apreciació al [meu blog](https://osc.garden) sense registrar-me a res, i assegurant-me que els meus visitants no fossin rastrejats.\n\nEsperant que altres ho trobin útil, vaig fer el servei públic (codi obert) i gratuït per sempre (almenys fins que doni suport a més de 100k llocs web).\n\nConstruït per a blogs, portafolis, jardins digitals, i qualsevol que cregui que la web hauria de ser més humana.\n"
  },
  {
    "path": "content/projects/iine/index.es.md",
    "content": "+++\ntitle = \"iine\"\ndescription = \"Botones de apreciación centrados en la privacidad y accesibles para la web pequeña.\"\nweight = 15\n\n[taxonomies]\ntags = [\"web\", \"JavaScript\", \"privacidad\", \"backend\"]\n\n[extra]\nlocal_image = \"projects/iine/iine_logo.webp\"\nsocial_media_card = \"social_cards/projects_iine.jpg\"\ncanonical_url = \"https://osc.garden/projects/iine/\"\n+++\n\nBotones de apreciación para tu blog, jardín digital, portafolio… Sin cuenta requerida. Sin seguimiento.\n\n#### [GitHub](https://github.com/welpo/iine) • [Sitio web](https://iine.to) {.centered-text}\n\n## Características\n\n- Sin marcas de tiempo, sin IPs, sin seguimiento\n- ~3KB de JavaScript vanilla\n- Funciona sin JavaScript\n- Varios iconos: corazón, pulgar arriba, voto positivo, o cualquier emoji\n- Construido para la web pequeña/independiente\n- Gratis para siempre\n- Accesible por diseño\n- Auto-hosteable\n\n## Inicio rápido\n\nAñade botones de apreciación a tu sitio con dos líneas. Carga el script:\n\n```html\n<script defer src=\"https://cdn.jsdelivr.net/gh/welpo/iine@main/iine.mini.js\"></script>\n```\n\nAñade el botón:\n\n```html\n<button class=\"iine-button\" aria-hidden=\"true\"></button>\n```\n\nListo. Tus visitantes ya pueden mostrar apreciación anónima por tu contenido.\n\n## ¿Por qué iine?\n\nQuería añadir botones de me gusta/felicitaciones/alegría/apreciación a [mi blog](https://osc.garden) sin registrarme en nada, y asegurándome de que mis visitantes no fueran rastreados.\n\nEsperando que otros lo encuentren útil, hice el servicio público (código abierto) y gratis para siempre (al menos hasta que soporte más de 100k sitios web).\n\nConstruido para blogs, portafolios, jardines digitales, y cualquiera que crea que la web debería ser más humana.\n"
  },
  {
    "path": "content/projects/iine/index.md",
    "content": "+++\ntitle = \"iine\"\ndescription = \"Privacy-focused & accessible like buttons for the small web.\"\nweight = 15\n\n[taxonomies]\ntags = [\"web\", \"JavaScript\", \"privacy\", \"backend\"]\n\n[extra]\nlocal_image = \"projects/iine/iine_logo.webp\"\nsocial_media_card = \"social_cards/projects_iine.jpg\"\ncanonical_url = \"https://osc.garden/projects/iine/\"\n+++\n\nAppreciation buttons for your blog, digital garden, portfolio… No account required. No tracking.\n\n#### [GitHub](https://github.com/welpo/iine) • [Website](https://iine.to) {.centered-text}\n\n## Features\n\n- No timestamps, no IPs, no tracking\n- ~3KB of vanilla JavaScript\n- Works without JavaScript\n- Various icons: heart, thumbs up, upvote, or any emoji\n- Built for the small/indie web\n- Free forever\n- Accessible by design\n- Self-hostable\n\n## Quick start\n\nAdd appreciation buttons to your site with two lines. Load the script:\n\n```html\n<script defer src=\"https://cdn.jsdelivr.net/gh/welpo/iine@main/iine.mini.js\"></script>\n```\n\nAdd the button:\n\n```html\n<button class=\"iine-button\" aria-hidden=\"true\"></button>\n```\n\nDone. Your visitors can now show anonymous appreciation for your content.\n\n## Why iine?\n\nI wanted to add like/kudos/cheers/appreciation buttons to [my blog](https://osc.garden) without signing up for anything, and making sure my visitors weren't tracked.\n\nHoping others find it useful, I made the service public (open source) and free forever (at least until it supports over 100k websites).\n\nBuilt for blogs, portfolios, digital gardens, and anyone who believes the web should be more human.\n"
  },
  {
    "path": "content/projects/nani/index.ca.md",
    "content": "+++\ntitle = \"nani\"\ndescription = \"Script Bash per crear URLs públiques a partir d'arxius o text en servidors remots.\"\nweight = 50\n\n[taxonomies]\ntags = [\"bash\", \"CLI\"]\n\n[extra]\nlocal_image = \"projects/nani/nani_logo.webp\"\ncanonical_url = \"https://osc.garden/ca/projects/tabi/\"\nsocial_media_card = \"social_cards/ca_projects_nani.jpg\"\n+++\n\nSi treballes en un servidor remot, saps que compartir arxius amb altres persones pot ser un procés feixuc. `nani` és un script en Bash dissenyat per simplificar aquesta tasca. Amb una sola comanda, pots convertir arxius locals o URLs en enllaços accessibles, facilitant el procés de compartir directament des del teu servidor.\n\n[![nani logo](nani_logo.webp)](https://github.com/welpo/nani/)\n\n#### [Veure a GitHub](https://github.com/welpo/nani) {.centered-text}\n\n## Característiques clau\n\n- **Tot tipus d'arxius**: gestiona directoris, arxius FLAC, arxius de text i fins i tot URLs a vídeos.\n- **Personalitzable**: adapta els ajustos editant l'script o un arxiu de configuració.\n- **Notificacions**: notificacions a l'escriptori i integració amb el portaretrats per a una millor experiència.\n\n## Inici ràpid\n\n1. Col·loca `nani` en un directori dins del teu PATH.\n2. Fes que l'script sigui executable.\n\nPer a passos d'instal·lació més detallats, [consulta la documentació completa](https://github.com/welpo/nani#-install).\n\n## Ús\n\n```bash\n$ nani Ruta/A/foto.png\nhttps://example.com/nani/hjRGLZB.png\n```\n\nCompartir un directori mantenint el seu nom original:\n\n```bash\n$ nani -o Ruta/A/Directori\nhttps://example.com/nani/Directori.zip\n```\n\nPots configurar diverses opcions a través dels paràmetres. Aquí tens la sortida de `nani --help`:\n\n{% wide_container() %}\n\n```\nUsage: nani [options] <infile>\nProvides public URL from input.\n\nInput handling:\n    Directory                       Will be stored using zip (or symbolic link)\n    FLAC                            Can be transcoded to MP3\n    Text (html, php...)             Extension can be set to .txt\n    Other files                     New copy/hard link/symbolic link at output directory\n    URL to video (e.g: youtube)     Downloaded using yt-dlp\n    Other URLs                      Downloaded using wget\n\nModify the first lines of the script to change how nani behaves: quiet mode,\nenabling/disabling transcoding, length of the string, extension truncation...\n\nSettings and options:\n    -a, --alias      Revert the hard link setting\n    -c, --cleanup    Remove all files on /nani/ except index.html\n    -h, --help       Display this help and exit\n    -i, --insert     Open nano to enter text. Saved in output directory as .txt\n    -k, --keep       Output dir becomes /nani/k/, to set different cleanup rules\n    -l, --list       List files in output directory /nani/\n    -n, --name       Use custom name (e.g. nani -n DesiredName <file>)\n    -N, --notify     Revert the notify option\n    -o, --original   Preserve original file name\n    -p, --push       Send push notification\n    -q, --quiet      Revert the quiet setting\n    -s, --string     Force a certain string length (e.g. nani -s 32 <file>)\n    -t, --transcode  Revert the transcode setting\n    -x, --xclip      Revert the xclip setting\n    -y, --symbolic   Create a symbolic link for files and directories\n```\n\n{% end %}\n"
  },
  {
    "path": "content/projects/nani/index.es.md",
    "content": "+++\ntitle = \"nani\"\ndescription = \"Script Bash para crear URLs públicas a partir de archivos o texto en servidores remotos.\"\nweight = 50\n\n[taxonomies]\ntags = [\"bash\", \"CLI\"]\n\n[extra]\nlocal_image = \"projects/nani/nani_logo.webp\"\ncanonical_url = \"https://osc.garden/es/projects/tabi/\"\nsocial_media_card = \"social_cards/es_projects_nani.jpg\"\n+++\n\nSi trabajas en un servidor remoto, sabrás que compartir archivos con otras personas puede ser un proceso tedioso. `nani` es un script en Bash diseñado para simplificar esta tarea. Con un solo comando, puedes convertir archivos locales en enlaces públicos, facilitando el proceso de compartir directamente desde tu servidor.\n\n[![nani logo](nani_logo.webp)](https://github.com/welpo/nani/)\n\n#### [Ver en GitHub](https://github.com/welpo/nani) {.centered-text}\n\n## Características clave\n\n- **Todo tipo de archivos**: maneja directorios, archivos FLAC, archivos de texto e incluso URLs a vídeos.\n- **Personalizable**: adapta los ajustes editando el script o un archivo de configuración.\n- **Notificaciones**: notificaciones en el escritorio e integración con el portapapeles para una mejor experiencia.\n\n## Inicio rápido\n\n1. Coloca `nani` en un directorio dentro de tu PATH.\n2. Haz el script ejecutable.\n\nPara pasos de instalación más detallados, [consulta la documentación completa](https://github.com/welpo/nani#-install).\n\n## Uso\n\n```bash\n$ nani Ruta/A/foto.png\nhttps://example.com/nani/hjRGLZB.png\n```\n\nCompartir un directorio manteniendo su nombre original:\n\n```bash\n$ nani -o Ruta/A/Directorio\nhttps://example.com/nani/Directorio.zip\n```\n\nPuedes configurar varias opciones a través de los parámetros. Aquí tienes la salida de `nani` --help:\n{% wide_container() %}\n\n```\nUsage: nani [options] <infile>\nProvides public URL from input.\n\nInput handling:\n    Directory                       Will be stored using zip (or symbolic link)\n    FLAC                            Can be transcoded to MP3\n    Text (html, php...)             Extension can be set to .txt\n    Other files                     New copy/hard link/symbolic link at output directory\n    URL to video (e.g: youtube)     Downloaded using yt-dlp\n    Other URLs                      Downloaded using wget\n\nModify the first lines of the script to change how nani behaves: quiet mode,\nenabling/disabling transcoding, length of the string, extension truncation...\n\nSettings and options:\n    -a, --alias      Revert the hard link setting\n    -c, --cleanup    Remove all files on /nani/ except index.html\n    -h, --help       Display this help and exit\n    -i, --insert     Open nano to enter text. Saved in output directory as .txt\n    -k, --keep       Output dir becomes /nani/k/, to set different cleanup rules\n    -l, --list       List files in output directory /nani/\n    -n, --name       Use custom name (e.g. nani -n DesiredName <file>)\n    -N, --notify     Revert the notify option\n    -o, --original   Preserve original file name\n    -p, --push       Send push notification\n    -q, --quiet      Revert the quiet setting\n    -s, --string     Force a certain string length (e.g. nani -s 32 <file>)\n    -t, --transcode  Revert the transcode setting\n    -x, --xclip      Revert the xclip setting\n    -y, --symbolic   Create a symbolic link for files and directories\n```\n\n{% end %}\n"
  },
  {
    "path": "content/projects/nani/index.md",
    "content": "+++\ntitle = \"nani\"\ndescription = \"Bash script to create public URLs from files or text on remote servers.\"\nweight = 50\n\n[taxonomies]\ntags = [\"bash\", \"CLI\"]\n\n[extra]\nlocal_image = \"projects/nani/nani_logo.webp\"\ncanonical_url = \"https://osc.garden/projects/tabi/\"\nsocial_media_card = \"social_cards/projects_nani.jpg\"\n+++\n\nIf you're working on a remote server, you know that sharing files with others can often involve multiple steps. `nani` is a Bash script designed to streamline this process. By executing a single command, you can convert local files or URLs into accessible links, allowing for easier sharing right from your server.\n\n[![nani logo](nani_logo.webp)](https://github.com/welpo/nani/)\n\n#### [View on GitHub](https://github.com/welpo/nani) {.centered-text}\n\n## Key Features\n\n- **Multiple File Types**: Handles directories, FLAC files, text files, and even URLs to videos.\n- **Customisable**: Tailor settings via a config file or runtime flags.\n- **Notifications**: Desktop notifications and clipboard integration for a better experience.\n\n## Quick Start\n\n1. Place `nani` in a directory within your PATH.\n2. Make the script executable.\n\nFor detailed installation steps, [read the full documentation](https://github.com/welpo/nani#-install).\n\n## Usage\n\n```bash\n$ nani Path/To/picture.png\nhttps://example.com/nani/hjRGLZB.png\n```\n\n**Share a directory keeping its original name**:\n\n```bash\n$ nani -o Path/To/Directory\nhttps://example.com/nani/Directory.zip\n```\n\nAdditional control is available through flags. Here's the output of `nani --help`:\n\n{% wide_container() %}\n\n```\nUsage: nani [options] <infile>\nProvides public URL from input.\n\nInput handling:\n    Directory                       Will be stored using zip (or symbolic link)\n    FLAC                            Can be transcoded to MP3\n    Text (html, php...)             Extension can be set to .txt\n    Other files                     New copy/hard link/symbolic link at output directory\n    URL to video (e.g: youtube)     Downloaded using yt-dlp\n    Other URLs                      Downloaded using wget\n\nModify the first lines of the script to change how nani behaves: quiet mode,\nenabling/disabling transcoding, length of the string, extension truncation...\n\nSettings and options:\n    -a, --alias      Revert the hard link setting\n    -c, --cleanup    Remove all files on /nani/ except index.html\n    -h, --help       Display this help and exit\n    -i, --insert     Open nano to enter text. Saved in output directory as .txt\n    -k, --keep       Output dir becomes /nani/k/, to set different cleanup rules\n    -l, --list       List files in output directory /nani/\n    -n, --name       Use custom name (e.g. nani -n DesiredName <file>)\n    -N, --notify     Revert the notify option\n    -o, --original   Preserve original file name\n    -p, --push       Send push notification\n    -q, --quiet      Revert the quiet setting\n    -s, --string     Force a certain string length (e.g. nani -s 32 <file>)\n    -t, --transcode  Revert the transcode setting\n    -x, --xclip      Revert the xclip setting\n    -y, --symbolic   Create a symbolic link for files and directories\n```\n\n{% end %}\n"
  },
  {
    "path": "content/projects/nemui/index.ca.md",
    "content": "+++\ntitle = \"nemui\"\ndescription = \"Ajusta gradualment el teu horari de son amb suport per horari d'estiu.\"\nweight = 22\n\n[taxonomies]\ntags = [\"son\", \"interactiu\", \"web app\", \"web\", \"JavaScript\"]\n\n[extra]\nlocal_image = \"projects/nemui/nemui_logo.webp\"\ncanonical_url = \"https://osc.garden/ca/projects/tabi/\"\nsocial_media_card = \"social_cards/projects_nemui.jpg\"\n+++\n\nnemui és una aplicació web que t'ajuda a fer una transició suau a un nou horari de son. El seu nom ve de les paraules japoneses per dormir (<ruby>眠<rt>nemu</rt></ruby>) i transició (<ruby>移<rt>i</rt></ruby>), que es llegeix com <ruby>眠い<rt>nemui</rt></ruby> (somnolent).\n\n#### [Prova-la ara](https://nemui.osc.garden) • [GitHub](https://github.com/welpo/nemui) • [Blog](https://osc.garden/ca/blog/nemui-sleep-schedule-planner/) {.centered-text}\n\n## Característiques\n\n- Interfície interactiva de rellotge inspirada en Apple\n- Ajust gradual de l'horari de son basat en la ciència del son\n- Suport complet per a l'horari d'estiu (DST)\n- Exportació a calendari (.ics) amb recordatoris per anar a dormir\n- Emmagatzematge local per seguir el teu progrés\n- Accessible: compatible amb navegació per teclat i lectors de pantalla\n\n## Per què nemui?\n\nA diferència dels canvis bruscos que poden alterar el teu ritme circadià, nemui t'ajuda a ajustar el teu horari de son de manera gradual. És especialment útil per a:\n\n- Adaptar-te a nous horaris de feina/estudi\n- Preparar-te per a canvis de zona horària\n- Fer una transició suau durant els canvis d'hora\n- Corregir un horari de son desajustat\n"
  },
  {
    "path": "content/projects/nemui/index.es.md",
    "content": "+++\ntitle = \"nemui\"\ndescription = \"Ajusta gradualmente tu horario de sueño con soporte para horario de verano.\"\nweight = 32\n\n[taxonomies]\ntags = [\"sueño\", \"interactivo\", \"web app\", \"web\", \"JavaScript\"]\n\n[extra]\nlocal_image = \"projects/nemui/nemui_logo.webp\"\ncanonical_url = \"https://osc.garden/es/projects/tabi/\"\nsocial_media_card = \"social_cards/projects_nemui.jpg\"\n+++\n\nnemui es una aplicación web que te ayuda a hacer una transición suave a un nuevo horario de sueño. Su nombre viene de las palabras japonesas para dormir (<ruby>眠<rt>nemu</rt></ruby>) y transición (<ruby>移<rt>i</rt></ruby>), que se lee como <ruby>眠い<rt>nemui</rt></ruby> (somnoliento).\n\n#### [Pruébala ahora](https://nemui.osc.garden) • [GitHub](https://github.com/welpo/nemui) • [Blog](https://osc.garden/es/blog/nemui-sleep-schedule-planner/) {.centered-text}\n\n## Características\n\n- Interfaz interactiva de reloj inspirada en Apple\n- Ajuste gradual del horario de sueño basado en la ciencia del sueño\n- Soporte completo para el horario de verano (DST)\n- Exportación a calendario (.ics) con recordatorios para dormir\n- Almacenamiento local para seguir tu progreso\n- Accesible: compatible con navegación por teclado y lectores de pantalla\n\n## ¿Por qué nemui?\n\nA diferencia de los cambios bruscos que pueden alterar tu ritmo circadiano, nemui te ayuda a ajustar tu horario de sueño de forma gradual. Es especialmente útil para:\n\n- Adaptarte a nuevos horarios de trabajo/estudio\n- Prepararte para cambios de zona horaria\n- Hacer una transición suave durante los cambios de hora\n- Corregir un horario de sueño desajustado\n"
  },
  {
    "path": "content/projects/nemui/index.md",
    "content": "+++\ntitle = \"nemui\"\ndescription = \"Gradually adjust your sleep schedule with support for DST transitions.\"\nweight = 32\n\n[taxonomies]\ntags = [\"sleep\", \"interactive\", \"web app\", \"web\", \"JavaScript\"]\n\n[extra]\nlocal_image = \"projects/nemui/nemui_logo.webp\"\ncanonical_url = \"https://osc.garden/projects/tabi/\"\nsocial_media_card = \"social_cards/projects_nemui.jpg\"\n+++\n\nnemui is a web app that helps you smoothly transition to a new sleep schedule. Named after the Japanese words for sleep (<ruby>眠<rt>nemu</rt></ruby>) and transition (<ruby>移<rt>i</rt></ruby>), reading as <ruby>眠い<rt>nemui</rt></ruby> (sleepy).\n\n#### [Try it now](https://nemui.osc.garden) • [GitHub](https://github.com/welpo/nemui) • [Blog post](https://osc.garden/blog/nemui-sleep-schedule-planner/) {.centered-text}\n\n## Features\n\n- Interactive clock interface inspired by Apple\n- Gradual sleep schedule adjustment based on sleep science\n- Full Daylight Saving Time (DST) support\n- Calendar (.ics) export with bedtime reminders\n- Local storage for progress tracking\n- Accessible: supports keyboard navigation and screen readers\n\n## Why nemui?\n\nUnlike abrupt changes that can disrupt your circadian rhythm, nemui helps you adjust your sleep schedule gradually. It's particularly useful for:\n\n- Adapting to new work/study schedules\n- Preparing for timezone changes\n- Smoothly transitioning through DST changes\n- Fixing a misaligned sleep schedule\n"
  },
  {
    "path": "content/projects/ramu/index.ca.md",
    "content": "+++\ntitle = \"ramu\"\ndescription = \"Una aplicació web per practicar la lectura i comprensió auditiva de nombres en japonès.\"\nweight = 35\n\n[taxonomies]\ntags = [\"Japonès\", \"interactiu\", \"web app\", \"web\", \"PWA\", \"JavaScript\"]\n\n[extra]\nlocal_image = \"projects/ramu/ramu_logo.webp\"\ncanonical_url = \"https://osc.garden/ca/projects/ramu/\"\nsocial_media_card = \"social_cards/projects_ramu.jpg\"\niine_icon = '🐏'\n+++\n\n<ruby>ラ<rt>ra</rt>ム<rt>mu</rt></ruby> és una aplicació web progressiva per practicar la lectura i comprensió auditiva de nombres en japonès. El nom reflecteix el seu propòsit: aconseguir accès aleatori (RAM; memòria d'accés aleatori) als nombres, en contraposició a una memòria seqüencial (1, 2, 3…).\n\n{% wide_container() %}\n<video controls src=\"media/ラム_demo.mp4\" title=\"demo de ramu\"></video>\n{% end %}\n\n#### [Prova-la ara](https://ramu.osc.garden) • [GitHub](https://github.com/welpo/ramu) • [Article](https://osc.garden/ca/blog/ramu-japanese-numbers-practice-web-app/) {.centered-text}\n\n## Característiques\n\n- Pràctica amb nombres aràbics (123…) i japonesos (一二三…)\n- Dos modes de pràctica: lectura i comprensió auditiva\n- Rangs numèrics configurables (des de 0 fins a més de 100.000.000)\n- Pràctica de comptadors (個、本、匹…)\n- Funciona sense connexió com a aplicació web progressiva\n- Control complet per teclat (<kbd>espai</kbd>/<kbd>→</kbd> per següent, <kbd>esc</kbd> per aturar)\n- Compatible amb lectors de pantalla per a la pràctica amb nombres aràbics\n\n[![targeta social de ramu](social_cards/projects_ramu.jpg)](https://ramu.osc.garden)\n"
  },
  {
    "path": "content/projects/ramu/index.es.md",
    "content": "+++\ntitle = \"ramu\"\ndescription = \"Una aplicación web para practicar la lectura y comprensión auditiva de números en japonés.\"\nweight = 35\n\n[taxonomies]\ntags = [\"Japonés\", \"interactivo\", \"web app\", \"web\", \"PWA\", \"JavaScript\"]\n\n[extra]\nlocal_image = \"projects/ramu/ramu_logo.webp\"\ncanonical_url = \"https://osc.garden/es/projects/ramu/\"\nsocial_media_card = \"social_cards/projects_ramu.jpg\"\niine_icon = '🐏'\n+++\n\n<ruby>ラ<rt>ra</rt>ム<rt>mu</rt></ruby> es una aplicación web progresiva para practicar la lectura y comprensión auditiva de números en japonés. El nombre refleja su propósito: lograr acceso aleatorio (RAM; memoria de acceso aleatorio) a los números, en contraposición a una memoria secuencial (1, 2, 3…).\n\n{% wide_container() %}\n<video controls src=\"media/ラム_demo.mp4\" title=\"demo de ramu\"></video>\n{% end %}\n\n#### [Pruébala ahora](https://ramu.osc.garden) • [GitHub](https://github.com/welpo/ramu) • [Artículo](https://osc.garden/es/blog/ramu-japanese-numbers-practice-web-app/) {.centered-text}\n\n## Características\n\n- Práctica con números arábigos (123…) y japoneses (一二三…)\n- Dos modos de práctica: lectura y comprensión auditiva\n- Rangos numéricos configurables (desde 0 hasta más de 100.000.000)\n- Práctica de contadores (個、本、匹…)\n- Funciona sin conexión como aplicación web progresiva\n- Control por teclado (<kbd>espacio</kbd>/<kbd>→</kbd> para siguiente, <kbd>esc</kbd> para detener)\n- Compatible con lectores de pantalla para la práctica con números arábigos\n\n[![tarjeta social de ramu](social_cards/projects_ramu.jpg)](https://ramu.osc.garden)\n"
  },
  {
    "path": "content/projects/ramu/index.md",
    "content": "+++\ntitle = \"ramu\"\ndescription = \"A web app to practice reading and listening to Japanese numbers.\"\nweight = 35\n\n[taxonomies]\ntags = [\"Japanese\", \"interactive\", \"web app\", \"web\", \"PWA\", \"JavaScript\"]\n\n[extra]\nlocal_image = \"projects/ramu/ramu_logo.webp\"\ncanonical_url = \"https://osc.garden/projects/ramu/\"\nsocial_media_card = \"social_cards/projects_ramu.jpg\"\niine_icon = '🐏'\n+++\n\n<ruby>ラ<rt>ra</rt>ム<rt>mu</rt></ruby> is a Progressive Web App to practice reading and listening to Japanese numbers. The name reflects its purpose: achieving RAM (Random Access Memory) to numbers, as opposed to sequential memory (1, 2, 3…).\n\n{% wide_container() %}\n<video controls src=\"media/ラム_demo.mp4\" title=\"ramu demo\"></video>\n{% end %}\n\n#### [Try it now](https://ramu.osc.garden) • [GitHub](https://github.com/welpo/ramu) • [Blog post](https://osc.garden/blog/ramu-japanese-numbers-practice-web-app/) {.centered-text}\n\n## Features\n\n- Practice with both Arabic (123…) and Japanese (一二三…) numerals\n- Two practice modes: reading and listening comprehension\n- Configurable number ranges (from 0 to over 100,000,000)\n- Counter word practice (個、本、匹…)\n- Works offline as a Progressive Web App\n- Full keyboard control (<kbd>space</kbd>/<kbd>→</kbd> for next, <kbd>esc</kbd> to stop)\n- Screen reader friendly for Arabic numeral practice\n\n[![ramu social media card](social_cards/projects_ramu.jpg)](https://ramu.osc.garden)\n"
  },
  {
    "path": "content/projects/shuku/index.ca.md",
    "content": "+++\ntitle = \"shuku\"\ndescription = \"Condensa pel·lícules i sèries per quedar-te només amb el diàleg. Dissenyat per aprendre idiomes.\"\nweight = 20\n\n[taxonomies]\ntags = [\"Python\", \"media\", \"linguistics\", \"CLI\"]\n\n[extra]\nlocal_image = \"projects/shuku/shuku_logo.webp\"\nsocial_media_card = \"social_cards/projects_shuku.jpg\"\ncanonical_url = \"https://osc.garden/ca/projects/shuku/\"\n+++\n\n**shuku** (<ruby><rb>縮</rb><rt>しゅく</rt></ruby><ruby><rb>小</rb><rt>しょう</rt></ruby>: «minificació») crea versions condensades de pel·lícules i sèries conservant només els diàlegs.\n\n<video class=\"invertible-image\" controls muted width=\"800\" loop=\"true\" autoplay=\"autoplay\" title=\"demo de shuku\" src=\"https://cdn.jsdelivr.net/gh/welpo/shuku/assets/animation_demo/shuku_demo.mov\"></video>\n\n#### [GitHub](https://github.com/welpo/shuku) • [Blog](https://osc.garden/ca/blog/shuku-condensed-media-language-learning/) • [Documentació](https://github.com/welpo/shuku#readme) • [PyPI](https://pypi.org/project/shuku/) {.centered-text}\n\n## Característiques\n\n### Gestió intel·ligent de continguts\n\n- Detecció i correspondència automàtica de subtítols amb cerca difusa (fuzzy matching)\n- Selecció intel·ligent de pistes d'àudio/subtítols\n- Extracció de metadades (títol, temporada, número d'episodi)\n\n### Output flexible\n\n- Àudio condensat (MP3, FLAC, AAC, Opus…)\n- Vídeo condensat\n- Subtítols sincronitzats (SRT, ASS, o LRC per a apps tipus karaoke)\n\n### Alta personalització\n\n- Qualitat i còdecs d'àudio/vídeo configurables\n- Ajust de temps de subtítols i farciment\n- Filtra subtítols (efectes de so, lletres, capítols específics)\n- Suport per a arguments personalitzats de FFmpeg\n\n### Experiència de l'usuari\n\n- Multiplataforma: GNU+Linux, macOS i Windows\n- Logging detallat amb indicadors de progrés\n- Suport per a processament per lots\n\n[![targeta social de shuku](/img/social_cards/projects_shuku.jpg)](https://github.com/welpo/shuku)\n"
  },
  {
    "path": "content/projects/shuku/index.es.md",
    "content": "+++\ntitle = \"shuku\"\ndescription = \"Condensa películas y series para quedarte solo con el diálogo. Diseñado para aprender idiomas.\"\nweight = 20\n\n[taxonomies]\ntags = [\"Python\", \"media\", \"linguistics\", \"CLI\"]\n\n[extra]\nlocal_image = \"projects/shuku/shuku_logo.webp\"\nsocial_media_card = \"social_cards/projects_shuku.jpg\"\ncanonical_url = \"https://osc.garden/es/projects/shuku/\"\n+++\n\n**shuku** (<ruby><rb>縮</rb><rt>しゅく</rt></ruby><ruby><rb>小</rb><rt>しょう</rt></ruby>: «minificación») crea versiones condensadas de películas y series conservando solo los diálogos.\n\n<video class=\"invertible-image\" controls muted width=\"800\" loop=\"true\" autoplay=\"autoplay\" title=\"demo de shuku\" src=\"https://cdn.jsdelivr.net/gh/welpo/shuku/assets/animation_demo/shuku_demo.mov\"></video>\n\n#### [GitHub](https://github.com/welpo/shuku) • [Blog](https://osc.garden/es/blog/shuku-condensed-media-language-learning/) • [Documentación](https://github.com/welpo/shuku#readme) • [PyPI](https://pypi.org/project/shuku/) {.centered-text}\n\n## Características\n\n### Manejo inteligente de medios\n\n- Detección y correspondencia automática de subtítulos con búsqueda difusa (fuzzy matching)\n- Selección inteligente de pistas de audio/subtítulos\n- Extracción de metadatos (título, temporada, número de episodio)\n\n### Output flexible\n\n- Audio condensado (MP3, FLAC, AAC, Opus…)\n- Video condensado\n- Subtítulos sincronizados (SRT, ASS, o LRC para apps tipo karaoke)\n\n### Alta personalización\n\n- Calidad y códecs de audio/video configurables\n- Ajuste de tiempo de subtítulos y relleno\n- Filtra subtítulos (efectos de sonido, letras, capítulos específicos)\n- Soporte para argumentos personalizados de FFmpeg\n\n### Experiencia del usuario\n\n- Multiplataforma: GNU+Linux, macOS y Windows\n- Logging detallado con indicadores de progreso\n- Soporte para procesamiento por lotes\n\n[![tarjeta social de shuku](/img/social_cards/projects_shuku.jpg)](https://github.com/welpo/shuku)\n"
  },
  {
    "path": "content/projects/shuku/index.md",
    "content": "+++\ntitle = \"shuku\"\ndescription = \"Shrink media to keep only the dialogue. For immersion language learning.\"\nweight = 20\n\n[taxonomies]\ntags = [\"Python\", \"media\", \"linguistics\", \"CLI\"]\n\n[extra]\nlocal_image = \"projects/shuku/shuku_logo.webp\"\nsocial_media_card = \"social_cards/projects_shuku.jpg\"\ncanonical_url = \"https://osc.garden/projects/shuku/\"\n+++\n\n**shuku** (<ruby><rb>縮</rb><rt>しゅく</rt></ruby><ruby><rb>小</rb><rt>しょう</rt></ruby>: \"minification\") creates condensed versions of films and TV shows by keeping only the dialogue.\n\n<video class=\"invertible-image\" controls muted width=\"800\" loop=\"true\" autoplay=\"autoplay\" title=\"shuku demo\" src=\"https://cdn.jsdelivr.net/gh/welpo/shuku/assets/animation_demo/shuku_demo.mov\"></video>\n\n#### [GitHub](https://github.com/welpo/shuku) • [Blog post](https://osc.garden/blog/shuku-condensed-media-language-learning/) • [Documentation](https://github.com/welpo/shuku#readme) • [PyPI](https://pypi.org/project/shuku/) {.centered-text}\n\n## Features\n\n### Smart media handling\n\n- Automatic subtitle detection and matching with fuzzy search\n- Intelligent audio/subtitle track selection\n- Metadata extraction (title, season, episode number)\n\n### Flexible output\n\n- Condensed audio (MP3, FLAC, AAC, Opus...)\n- Condensed video\n- Synchronized subtitles (including LRC for karaoke-style review)\n- Clean filenames in output\n\n### High customization\n\n- Configurable audio/video quality and codecs\n- Subtitle timing adjustment and padding\n- Skip unwanted content (sound effects, lyrics, specific chapters)\n- Custom FFmpeg arguments support\n\n### User experience\n\n- Cross-platform: GNU+Linux, macOS, and Windows\n- Detailed logging with progress indicators\n- Batch processing support\n\n## Development best practices\n\n- Comprehensive testing: 100% code coverage\n- Clean code: Type-hinted Python with clear responsibilities\n- Continuous Integration/Deployment\n- Comprehensive documentation\n\n[![shuku social media card](/img/social_cards/projects_shuku.jpg)](https://github.com/welpo/shuku)\n"
  },
  {
    "path": "content/projects/streaming-royalties-calculator/index.ca.md",
    "content": "+++\ntitle = \"Calculadora de royalties de streaming\"\ndescription = \"Una eina per calcular els royalties de streaming per a músics.\"\nweight = 45\n\n[taxonomies]\ntags = [\"música\", \"interactiu\", \"web app\", \"web\", \"JavaScript\", \"anàlisi de dades\"]\n\n[extra]\nlocal_image = \"projects/streaming-royalties-calculator/streaming-royalties-calculator_logo.webp\"\ncanonical_url = \"https://osc.garden/ca/projects/streaming-royalties-calculator/\"\nsocial_media_card = \"social_cards/ca_projects_streaming_royalties_calculator.jpg\"\n+++\n\nLa Calculadora de royalties de streaming permet als músics estimar els seus guanys de plataformes com Spotify, Apple Music, Instagram, TikTok i més.\n\nPots introduir una quantitat objectiu de guanys per veure quantes reproduccions es necessiten a cada plataforma, o introduir el nombre de reproduccions per servei per calcular les royalties esperades. Aquí tens una captura de pantalla:\n\n<a href=\"https://osc.garden/ca/royalties-calculator/\" target=\"_blank\">\n   {{ dual_theme_image(light_src=\"https://cdn.jsdelivr.net/gh/welpo/osc.garden@main/content/blog/data-analysis-music-streaming/img/calculator_light.ca.webp\", dark_src=\"https://cdn.jsdelivr.net/gh/welpo/osc.garden@main/content/blog/data-analysis-music-streaming/img/calculator_dark.ca.webp\" alt=\"Calculadora de royalties de streaming\") }}\n</a>\n\n#### [Prova-la!](https://osc.garden/ca/royalties-calculator/) • [Codi font (JavaScript)](https://github.com/welpo/osc.garden/blob/main/content/pages/royalties-calculator/js/streamsMonthCalculator.js) {.centered-text}\n\n## Característiques principals\n\n- **Dades precises**: Basades en l'últim any de [les meves pròpies dades de royalties](https://osc.garden/ca/blog/data-analysis-music-streaming/).\n- **Múltiples plataformes**: Inclou Tidal, Spotify, Apple Music, Facebook, Deezer, TikTok i més.\n- **Modes de càlcul**: Utilitza la taxa de pagament mitjana, mediana, mínima o màxima per estimar els guanys.\n"
  },
  {
    "path": "content/projects/streaming-royalties-calculator/index.es.md",
    "content": "+++\ntitle = \"Calculadora de royalties de streaming\"\ndescription = \"Una herramienta para calcular los royalties de streaming para músicos.\"\nweight = 45\n\n[taxonomies]\ntags = [\"música\", \"interactivo\", \"web app\", \"web\", \"JavaScript\", \"análisis de datos\"]\n\n[extra]\nlocal_image = \"projects/streaming-royalties-calculator/streaming-royalties-calculator_logo.webp\"\ncanonical_url = \"https://osc.garden/es/projects/streaming-royalties-calculator/\"\nsocial_media_card = \"social_cards/es_projects_streaming_royalties_calculator.jpg\"\n+++\n\nLa Calculadora de royalties de streaming permite a los músicos estimar sus ganancias de plataformas como Spotify, Apple Music, Instagram, TikTok y más.\n\nPuedes introducir una cantidad objetivo de ganancias para ver cuántas reproducciones se necesitan en cada plataforma, o introducir el número de reproducciones por servicio para calcular las royalties esperadas. Aquí tienes una captura de pantalla:\n\n<a href=\"https://osc.garden/es/royalties-calculator/\" target=\"_blank\">\n   {{ dual_theme_image(light_src=\"https://cdn.jsdelivr.net/gh/welpo/osc.garden@main/content/blog/data-analysis-music-streaming/img/calculator_light.es.webp\", dark_src=\"https://cdn.jsdelivr.net/gh/welpo/osc.garden@main/content/blog/data-analysis-music-streaming/img/calculator_dark.es.webp\" alt=\"Calculadora de royalties de streaming\") }}\n</a>\n\n#### [¡Pruébala!](https://osc.garden/es/royalties-calculator/) • [Código fuente (JavaScript)](https://github.com/welpo/osc.garden/blob/main/content/pages/royalties-calculator/js/streamsMonthCalculator.js) {.centered-text}\n\n## Características principales\n\n- **Datos precisos**: Basada en el último año de [mis propios datos de royalties](https://osc.garden/es/blog/data-analysis-music-streaming/).\n- **Múltiples plataformas**: Incluye Tidal, Spotify, Apple Music, Facebook, Deezer, TikTok y más.\n- **Modos de cálculo**: Utiliza la tasa de pago media, mediana, mínima o máxima para estimar las ganancias.\n"
  },
  {
    "path": "content/projects/streaming-royalties-calculator/index.md",
    "content": "+++\ntitle = \"Streaming Royalties Calculator\"\ndescription = \"A tool to calculate streaming royalties for musicians.\"\nweight = 45\n\n[taxonomies]\ntags = [\"music\", \"interactive\", \"web app\", \"web\", \"JavaScript\", \"data analysis\"]\n\n[extra]\nlocal_image = \"projects/streaming-royalties-calculator/streaming-royalties-calculator_logo.webp\"\ncanonical_url = \"https://osc.garden/projects/streaming-royalties-calculator/\"\nsocial_media_card = \"social_cards/projects_streaming_royalties_calculator.jpg\"\n+++\n\nThe Streaming Royalties Calculator allows musicians to estimate their earnings from platforms like Spotify, Apple Music, Instagram, TikTok, and more.\n\nYou can either input a target earnings amount to see how many streams are needed on each platform, or enter the number of streams per service to calculate the expected royalties. Here's a screenshot:\n\n<a href=\"https://osc.garden/royalties-calculator/\" target=\"_blank\">\n    {{ dual_theme_image(light_src=\"https://cdn.jsdelivr.net/gh/welpo/osc.garden@main/content/blog/data-analysis-music-streaming/img/calculator_light.webp\", dark_src=\"https://cdn.jsdelivr.net/gh/welpo/osc.garden@main/content/blog/data-analysis-music-streaming/img/calculator_dark.webp\" alt=\"Streaming Royalties Calculator\") }}\n</a>\n\n#### [Try it!](https://osc.garden/royalties-calculator/) • [JavaScript Source](https://github.com/welpo/osc.garden/blob/main/content/pages/royalties-calculator/js/streamsMonthCalculator.js) {.centered-text}\n\n## Main Features\n\n- **Accurate data**: Based on the last year of [my own royalties data](https://osc.garden/blog/data-analysis-music-streaming/).\n- **Multiple platforms**: Includes Tidal, Spotify, Apple Music, Facebook, Deezer, TikTok, and more.\n- **Calculation modes**: Use the mean, median, minimum or maximum values to estimate the earnings.\n"
  },
  {
    "path": "content/projects/tabi/index.ar.md",
    "content": "+++\ntitle = \"تابي\"\ndescription = \"سمة زولا عصرية غنية بالمميزات مع دعم متميز لتعدد اللغات\"\nweight = 40\n\n[taxonomies]\ntags = [\"ويب\", \"جافا سكريبت\"]\n\n[extra]\nlocal_image = \"projects/tabi/tabi.webp\"\n+++\n\n[**تابي**](https://github.com/welpo/tabi) هي سمة عصرية غنية بالمميزات لـ[Zola](https://www.getzola.org/)، منشئ مواقع ثابتة سريع.\n\n{{ full_width_image(src=\"https://cdn.jsdelivr.net/gh/welpo/tabi@main/light_dark_screenshot.png\", alt=\"تابي في الوضعين الفاتح والداكن\") }}\n\n#### [عرض على GitHub](https://github.com/welpo/tabi) • [عرض توضيحي وتوثيق](https://welpo.github.io/tabi/) {.centered-text}\n\n## المميزات\n\n- دعم كامل [للغة العربية والكتابة من اليمين إلى اليسار](https://welpo.github.io/tabi/blog/faq-languages/#how-do-i-set-a-default-language-for-my-site)\n- [دعم للسلاسل](https://welpo.github.io/tabi/blog/series/) لإنشاء محتوى متسلسل مثل الدروس والدورات\n- سمتان داكنة وفاتحة، مع التبديل التلقائي حسب إعدادات النظام\n- [دعم التعليقات](https://welpo.github.io/tabi/blog/comments/) باستخدام giscus أو utterances أو Hyvor Talk أو Isso\n- [دعم KaTeX](https://katex.org/) للمعادلات الرياضية\n- [دعم Indieweb](https://indieweb.org/) مع microformats وh-card وwebmentions\n- [دعم Mermaid](https://welpo.github.io/tabi/blog/shortcodes/#mermaid-diagrams) لإنشاء المخططات\n- [بحث محلي](https://welpo.github.io/tabi/blog/mastering-tabi-settings/#search) متعدد اللغات\n- تصميم متجاوب يعمل على جميع الأجهزة\n- [صفحة مشاريع](https://welpo.github.io/tabi/projects/) و[أرشيف](https://welpo.github.io/tabi/archive/)\n\n## ممارسات التطوير\n\n- **[الالتزامات التقليدية](https://www.conventionalcommits.org) و[Gitmoji](https://gitmoji.dev/)**: تتبع رسائل الالتزام تنسيقات موحدة لتحسين القراءة\n- **تتبع المشكلات**: لكل خطأ أو ميزة جديدة تذكرة مخصصة، مرتبطة بأي التزامات برمجية وطلبات سحب ذات صلة\n- **تعليقات شاملة**: يتم توثيق التذاكر بالصور ومقاطع الفيديو والأوصاف التفصيلية لتسهيل التواصل الفعال وحل المشكلات\n- **الإحالات المرجعية**: نربط جميع التذاكر بالتزامات الشفرة وطلبات السحب أو المشكلات ذات الصلة للتتبع الكامل\n\n## تطور المشروع\n\nتم تصميم **تابي** في الأصل لموقعي الشخصي، [osc.garden](https://osc.garden). على الرغم من أصولها للاستخدام الشخصي، تم تنفيذ أفضل الممارسات منذ البداية لضمان الجودة وقابلية الصيانة. ومنذ ذلك الحين نمت السمة لتجذب مجتمعًا نشطًا من المساهمين على GitHub.\n\n## ابدأ رحلة الكتابة مع تابي\n\nلديك شيء تريد قوله. ربما عن تجربتك في تعلم لغة جديدة، أو عن رحلتك في استكشاف [فن الفلامنكو](https://en.wikipedia.org/wiki/Palo_(flamenco))، أو عن كيف نجحت في حل مشكلة برمجية في مشروع مفتوح المصدر.\n\n**تابي** توفر الأساس المثالي لمساحة كتابتك، مما يتيح لك التركيز على كلماتك بينما يهتم Zola وتابي بالجانب التقني. ادخل عالم التدوين مع نظام يجعل كل تدوينة متعة في الكتابة والقراءة. صوتك له قيمة—شاركه مع العالم.\n"
  },
  {
    "path": "content/projects/tabi/index.ca.md",
    "content": "+++\ntitle = \"tabi\"\ndescription = \"Un tema de Zola ràpid, lleuger i modern amb suport multilingüe.\"\nweight = 40\n\n[taxonomies]\ntags = [\"web\", \"JavaScript\"]\n\n[extra]\nlocal_image = \"projects/tabi/tabi.webp\"\ncanonical_url = \"https://osc.garden/ca/projects/tabi/\"\nsocial_media_card = \"social_cards/ca_projects_tabi.jpg\"\niine_icon = '🌱'\n+++\n\n[**tabi**](https://github.com/welpo/tabi) és un tema modern i ric en funcionalitat per a [Zola](https://www.getzola.org/), un generador de llocs web estàtics molt ràpid.\n\n{{ full_width_image(src=\"https://cdn.jsdelivr.net/gh/welpo/tabi@main/light_dark_screenshot.png\", alt=\"Modes clar i fosc de tabi\") }}\n\n#### [Veure a GitHub](https://github.com/welpo/tabi) • [Demo i documentación](https://welpo.github.io/tabi/ca/) {.centered-text}\n\n## Característiques\n\n- [Estableix qualsevol idioma com a predeterminat](https://welpo.github.io/tabi/ca/blog/faq-languages/#com-estableixo-la-llengua-predeterminada-del-meu-lloc). Configura el teu lloc en xinès, espanyol, francès, hindi… o qualsevol [altre idioma compatible](https://welpo.github.io/tabi/ca/blog/faq-languages/#quines-llengues-admet-tabi). La interfície del tema es traduirà en conseqüència.\n- [Integració amb repositoris remots](https://welpo.github.io/tabi/ca/blog/mastering-tabi-settings#integracio-amb-repositoris-git) a GitHub, GitLab, Gitea i Codeberg per a l'historial de commits i mostrar el codi font del lloc.\n- Tema clar i fosc. S'adapta a la configuració del sistema operatiu, amb un interruptor a la barra de navegació.\n- [Suport multilingüe complet](https://welpo.github.io/tabi/ca/blog/faq-languages/#com-gestiona-tabi-el-suport-multilingue). Afegeix tants idiomes com vulguis i deixa que els teus usuaris triin amb el selector d'idioma.\n- [Suport per a sèries](https://welpo.github.io/tabi/ca/blog/series/) per crear contingut seqüencial com tutorials, cursos i històries multipart.\n- Puntuació perfecta en Lighthouse (Rendiment, Accessibilitat, Millors Pràctiques i SEO).\n- Suport per a [Indieweb](https://indieweb.org/) amb microformats, suport per a [hcard](https://welpo.github.io/tabi/ca/blog/mastering-tabi-settings/#h-card-representativa) i [webmentions](https://welpo.github.io/tabi/ca/blog/mastering-tabi-settings/#webmentions).\n- Suport per a [diagrames de Mermaid](https://welpo.github.io/tabi/ca/blog/shortcodes/#diagrames-de-mermaid) per a crear diagrames i gràfics amb text.\n- Botons de «m'agrada» d'[iine](https://iine.to/) per mostrar apreciació anònima pel teu contingut.\n- Ressaltat de sintaxi de codi amb colors basats en [Catppuccin](https://github.com/catppuccin/catppuccin) Frappé.\n- Suport per a [comentaris usant giscus, utterances, Hyvor Talk o Isso](https://welpo.github.io/tabi/ca/blog/comments/).\n- [Cerca local](https://welpo.github.io/tabi/ca/blog/mastering-tabi-settings/#cerca) amb una interfície accessible i multilingüe.\n- Tot el JavaScript es pot [deshabilitar completament](https://welpo.github.io/tabi/ca/blog/javascript/).\n- [Codificació de correu](https://welpo.github.io/tabi/ca/blog/mastering-tabi-settings/#correu-electronic-codificat) per a protecció contra spam.\n- [Mapa del lloc estilitzat i llegible per humans](https://welpo.github.io/tabi/sitemap.xml).\n- [Capçaleres de seguretat personalitzables](https://welpo.github.io/tabi/ca/blog/security/).\n- [Feed Atom estilitzat i llegible per humans](https://welpo.github.io/tabi/ca/atom.xml).\n- [Enllaços de retrocés per a notes al peu](https://welpo.github.io/tabi/ca/blog/mastering-tabi-settings/#enllacos-de-retorn-a-les-notes-a-peu-de-pagina).\n- [Taula de continguts personalitzable](https://welpo.github.io/tabi/ca/blog/toc/).\n- [Avís de drets d'autor personalitzat](https://welpo.github.io/tabi/ca/blog/mastering-tabi-settings/#copyright).\n- [Botó de copiar per a blocs de codi](https://welpo.github.io/tabi/ca/blog/mastering-tabi-settings/#boto-de-copiar-en-blocs-de-codi).\n- [URL canòniques personalitzables](https://welpo.github.io/tabi/ca/blog/mastering-tabi-settings/#url-canonica).\n- [Targetes per a xarxes socials](https://welpo.github.io/tabi/ca/blog/mastering-tabi-settings/#targetes-per-a-xarxes-socials).\n- [Botons de navegació ràpida](https://welpo.github.io/tabi/ca/blog/mastering-tabi-settings/#botons-de-navegacio-rapida).\n- [Shortcodes personalitzats](https://welpo.github.io/tabi/ca/blog/shortcodes/).\n- [Skins personalitzables](https://welpo.github.io/tabi/ca/blog/customise-tabi/).\n- [Publicacions fixades](https://welpo.github.io/tabi/ca/blog/mastering-tabi-settings/#fixar-entrades).\n- [Pàgina de projectes](https://welpo.github.io/tabi/ca/projects/).\n- Disseny responsive.\n- Suport de [KaTeX](https://katex.org/).\n- [Enllaços socials](https://welpo.github.io/tabi/ca/blog/mastering-tabi-settings/#icones-de-xarxes-socials).\n- [Pàgina d'arxiu](https://welpo.github.io/tabi/ca/archive/).\n- [Etiquetes](https://welpo.github.io/tabi/ca/blog/mastering-tabi-settings/#etiquetes).\n\n## Pràctiques de desenvolupament\n\n- **[Conventional Commits](https://www.conventionalcommits.org) i [Gitmoji](https://gitmoji.dev/)**: els missatges de commit segueixen formats estandarditzats per facilitar la llegibilitat.\n- **Seguiment d'incidències**: cada error o nova funcionalitat té el seu propi tiquet, que s'enllaça amb els commits de codi i PRs o problemes relacionats.\n- **Comentaris detallats**: els tiquets es documenten amb imatges, vídeos i descripcions detallades per facilitar una comunicació i resolució de problemes efectives.\n- **Referències creuades**: enllacem tots els tiquets amb els commits de codi, pull requests o problemes relacionats per a una traçabilitat completa.\n\n## Evolució del projecte\n\n**tabi** va néixer com a disseny per al meu lloc web personal, [osc.garden](https://osc.garden). Malgrat les seves arrels personals, des del principi es van implementar bones pràctiques per assegurar la qualitat i la mantenibilitat. Des d'aleshores, el tema ha aconseguit atraure una comunitat activa de col·laboradors a GitHub.\n\n## Inicia el teu recorregut com a escriptor amb tabi\n\nTens alguna cosa a dir. Potser vols parlar sobre com els lingüistes encara no han acordat una [definició de \"paraula\"](https://ca.wikipedia.org/wiki/Mot), o sobre la teva experiència explorant els diferents [palos del flamenc](https://ca.wikipedia.org/wiki/Estils_flamencs), o de com vas aconseguir resoldre un error d'un projecte de codi obert popular.\n\n**tabi** t'ofereix la base ideal per al teu espai d'escriptura, permetent-te centrar-te en les teves paraules mentre Zola i tabi s'encarreguen de l'aspecte tècnic. Submergeix-te en el món dels blogs amb un sistema que fa que cada publicació sigui un plaer escriure i llegir. La teva veu té valor; comparteix-la amb el món.\n"
  },
  {
    "path": "content/projects/tabi/index.es.md",
    "content": "+++\ntitle = \"tabi\"\ndescription = \"Un tema de Zola rápido, ligero y moderno con soporte multilingüe.\"\nweight = 40\n\n[taxonomies]\ntags = [\"web\", \"JavaScript\"]\n\n[extra]\nlocal_image = \"projects/tabi/tabi.webp\"\ncanonical_url = \"https://osc.garden/es/projects/tabi/\"\nsocial_media_card = \"social_cards/es_projects_tabi.jpg\"\niine_icon = '🌱'\n+++\n\n[**tabi**](https://github.com/welpo/tabi) es un tema moderno y rico en funcionalidad para [Zola](https://www.getzola.org/), un generador de sitios web estáticos muy rápido.\n\n{{ full_width_image(src=\"https://cdn.jsdelivr.net/gh/welpo/tabi@main/light_dark_screenshot.png\", alt=\"Modos claro y oscuro de tabi\") }}\n\n#### [Ver en GitHub](https://github.com/welpo/tabi) • [Demo y documentación](https://welpo.github.io/tabi/es/) {.centered-text}\n\n## Características\n\n- [Establece cualquier idioma como predeterminado](https://welpo.github.io/tabi/es/blog/faq-languages/#como-establezco-el-idioma-predeterminado-de-mi-sitio). Configura tu sitio en chino, español, francés, hindi… o cualquier [otro idioma compatible](https://welpo.github.io/tabi/es/blog/faq-languages/#que-idiomas-admite-tabi). La interfaz del tema se traducirá en consecuencia.\n- [Integración con repositorios remotos](https://welpo.github.io/tabi/es/blog/mastering-tabi-settings/#integracion-con-repositorios-git) en GitHub, GitLab, Gitea y Codeberg para el historial de commits y mostrar el código fuente del sitio.\n- [Soporte multilingüe completo](https://welpo.github.io/tabi/es/blog/faq-languages/#como-gestiona-tabi-el-soporte-multilingue). Añade tantos idiomas como desees y deja que tus usuarios elijan con un selector de idioma.\n- Tema claro y oscuro. Se adapta a la configuración del sistema operativo, con un interruptor en la barra de navegación.\n- [Soporte para series](https://welpo.github.io/tabi/es/blog/series/) para crear contenido secuencial como tutoriales, cursos e historias en varias partes.\n- Puntuación perfecta en Lighthouse (Rendimiento, Accesibilidad, Mejores Prácticas y SEO).\n- Soporte para [Indieweb](https://indieweb.org/) con microformatos, soporte para [hcard](https://welpo.github.io/tabi/es/blog/mastering-tabi-settings/#h-card-representativa) y [webmentions](https://welpo.github.io/tabi/blog/mastering-tabi-settings/#webmentions).\n- Botones de «me gusta» de [iine](https://iine.to/) para mostrar aprecio anónimo por tu contenido.\n- Soporte para [diagramas de Mermaid](https://welpo.github.io/tabi/es/blog/shortcodes/#diagramas-de-mermaid) para crear diagramas y gráficos con texto.\n- Resaltado de sintaxis de código con colores basados en [Catppuccin](https://github.com/catppuccin/catppuccin) Frappé.\n- Soporte para [comentarios usando giscus, utterances, Hyvor Talk o Isso](https://welpo.github.io/tabi/es/blog/comments/).\n- Todo el JavaScript se puede [deshabilitar completamente](https://welpo.github.io/tabi/es/blog/javascript/).\n- [Búsqueda local](https://welpo.github.io/tabi/es/blog/mastering-tabi-settings/#busqueda) con una interfaz accesible y multilingüe.\n- [Codificación de correo](https://welpo.github.io/tabi/es/blog/mastering-tabi-settings/#correo-electronico-codificado) para protección contra spam.\n- [Mapa de sitio web estilizado y legible por humanos](https://welpo.github.io/tabi/sitemap.xml).\n- [Feed de Atom estilizado y legible por humanos](https://welpo.github.io/tabi/es/atom.xml).\n- [Aviso de derechos de autor personalizado](https://welpo.github.io/tabi/es/blog/mastering-tabi-settings/#copyright).\n- [Cabeceras de seguridad personalizables](https://welpo.github.io/tabi/es/blog/security/).\n- [Botón de copiar para bloques de código](https://welpo.github.io/tabi/es/blog/mastering-tabi-settings/#boton-de-copiar-en-bloques-de-codigo).\n- [Enlaces de retroceso para notas al pie](https://welpo.github.io/tabi/es/blog/mastering-tabi-settings/#enlaces-de-retorno-en-notas-al-pie).\n- [Tabla de contenidos personalizable](https://welpo.github.io/tabi/es/blog/toc/).\n- [URL canónicas personalizables](https://welpo.github.io/tabi/es/blog/mastering-tabi-settings/#url-canonica).\n- [Botones de navegación rápida](https://welpo.github.io/tabi/es/blog/mastering-tabi-settings/#botones-de-navegacion-rapida).\n- [Tarjetas para redes sociales](https://welpo.github.io/tabi/es/blog/mastering-tabi-settings/#tarjetas-para-redes-sociales).\n- [Shortcodes personalizados](https://welpo.github.io/tabi/es/blog/shortcodes/).\n- [Skins personalizables](https://welpo.github.io/tabi/es/blog/customise-tabi/).\n- [Publicaciones fijadas](https://welpo.github.io/tabi/es/blog/mastering-tabi-settings/#fijar-publicaciones).\n- [Página de proyectos](https://welpo.github.io/tabi/es/projects/).\n- Diseño responsive.\n- Soporte de [KaTeX](https://katex.org/).\n- [Página de archivo](https://welpo.github.io/tabi/es/archive/).\n- [Enlaces sociales](https://welpo.github.io/tabi/es/blog/mastering-tabi-settings/#iconos-de-redes-sociales).\n- [Etiquetas](https://welpo.github.io/tabi/es/blog/mastering-tabi-settings/#etiquetas).\n\n## Prácticas de desarrollo\n\n- **[Conventional Commits](https://www.conventionalcommits.org) y [Gitmoji](https://gitmoji.dev/)**: los mensajes de commit siguen formatos estandarizados para mejorar la legibilidad.\n- **Seguimiento de problemas**: cada error o nueva funcionalidad tiene su propio ticket, que se vincula a los commits de código y PRs o problemas relacionados.\n- **Comentarios detallados**: los tickets se documentan con imágenes, vídeos y descripciones detalladas para facilitar una comunicación y resolución de problemas efectivas.\n- **Referencias cruzadas**: enlazamos todos los tickets con los commits de código, pull requests o problemas relacionados para una rastreabilidad completa.\n\n## Evolución del proyecto\n\n**tabi** nació como diseño para mi sitio personal, [osc.garden](https://osc.garden/es/). A pesar de sus raíces personales, desde el principio se implementaron buenas prácticas para asegurar la calidad y mantenibilidad. Desde entonces, el tema ha logrado atraer a una comunidad activa de contribuyentes en GitHub.\n\n## Empieza tu aventura escribiendo con tabi\n\nTienes algo que decir. Tal vez se trate de cómo los lingüistas aún no han acordado una [definición de \"palabra\"](https://es.wikipedia.org/wiki/Palabra), o sobre tu experiencia explorando los diferentes [palos del flamenco](https://es.wikipedia.org/wiki/Flamenco#Palos), o de cómo lograste resolver un fallo de un proyecto de código abierto popular.\n\n**tabi** te ofrece la base ideal para tu espacio de escritura, permitiéndote centrarte en tus palabras mientras Zola y tabi se encargan del aspecto técnicos. Sumérgete en el mundo de los blogs con un sistema que hace que cada publicación sea un placer escribir y leer. Tu voz tiene valor; compártela con el mundo.\n"
  },
  {
    "path": "content/projects/tabi/index.md",
    "content": "+++\ntitle = \"tabi\"\ndescription = \"A feature-rich modern Zola theme with first-class multi-language support.\"\nweight = 40\n\n[taxonomies]\ntags = [\"web\", \"JavaScript\"]\n\n[extra]\nlocal_image = \"projects/tabi/tabi.webp\"\nsocial_media_card = \"social_cards/projects_tabi.jpg\"\niine_icon = '🌱'\n+++\n\n[**tabi**](https://github.com/welpo/tabi) is a modern, feature-rich theme for [Zola](https://www.getzola.org/), a fast static site generator.\n\n{{ full_width_image(src=\"https://cdn.jsdelivr.net/gh/welpo/tabi@main/light_dark_screenshot.png\", alt=\"tabi light and dark mode\") }}\n\n#### [View on GitHub](https://github.com/welpo/tabi) • [Demo & Documentation](https://welpo.github.io/tabi/) {.centered-text}\n\n## Features\n\n- [Set any language as default](https://welpo.github.io/tabi/blog/faq-languages/#how-do-i-set-a-default-language-for-my-site). Set your base site to Chinese, Spanish, French, Hindi… or any [other supported language](https://welpo.github.io/tabi/blog/faq-languages/#what-languages-does-tabi-support). The theme's interface will be translated accordingly.\n- [Integration with remote repositories](https://welpo.github.io/tabi/blog/mastering-tabi-settings/#git-repository-integration) on GitHub, GitLab, Gitea & Codeberg for commit history and showing the site source.\n- [Series support](https://welpo.github.io/tabi/blog/series/) for creating sequential content like tutorials, courses, and multi-part stories.\n- Dark and light themes. Defaults to the OS setting, with a switcher in the navigation bar.\n- Thorough documentation. See [Mastering tabi Settings: A Comprehensive Guide](https://welpo.github.io/tabi/blog/mastering-tabi-settings/).\n- Perfect Lighthouse score (Performance, Accessibility, Best Practices and SEO).\n- [Comprehensive multi-language support](https://welpo.github.io/tabi/blog/faq-languages/#how-does-tabi-handle-multilingual-support). Add as many languages as you wish.\n- Support for [comments using giscus, utterances, Hyvor Talk, or Isso](https://welpo.github.io/tabi/blog/comments/).\n- [Indieweb](https://indieweb.org/) ready with microformats, [hcard](https://welpo.github.io/tabi/blog/mastering-tabi-settings/#representative-h-card) and [webmentions](https://welpo.github.io/tabi/blog/mastering-tabi-settings/#webmentions) support.\n- Code syntax highlighting with colours based on [Catppuccin](https://github.com/catppuccin/catppuccin) Frappé.\n- [iine like buttons](https://iine.to/) for anonymous appreciation of your content.\n- [Mermaid support](https://welpo.github.io/tabi/blog/shortcodes/#mermaid-diagrams) to create diagrams and charts with text.\n- [Local search](https://welpo.github.io/tabi/blog/mastering-tabi-settings/#search) with an accessible, multi-lingual interface.\n- [Custom Twitter card](https://welpo.github.io/tabi/blog/mastering-tabi-settings/#social-media-cards) and automatic Open Graph tags.\n- [KaTeX](https://katex.org/) support for mathematical notation.\n- [Stylized and human readable Atom feed](https://welpo.github.io/tabi/atom.xml).\n- [Stylized and human readable sitemap](https://welpo.github.io/tabi/sitemap.xml).\n- [Mail encoding](https://welpo.github.io/tabi/blog/mastering-tabi-settings/#encoded-email) for spam protection.\n- All JavaScript can be [fully disabled](https://welpo.github.io/tabi/blog/javascript/).\n- [Customizable Table of Contents](https://welpo.github.io/tabi/blog/toc/).\n- [Customizable secure headers](https://welpo.github.io/tabi/blog/security/).\n- [Copy button for code blocks](https://welpo.github.io/tabi/blog/mastering-tabi-settings/#copy-button-on-code-blocks).\n- [Quick navigation buttons](https://welpo.github.io/tabi/blog/mastering-tabi-settings/#quick-navigation-buttons).\n- [Custom copyright notice](https://welpo.github.io/tabi/blog/mastering-tabi-settings/#copyright).\n- [Custom canonical URLs](https://welpo.github.io/tabi/blog/mastering-tabi-settings/#canonical-url).\n- [Customizable skins](https://welpo.github.io/tabi/blog/customise-tabi/).\n- [Custom shortcodes](https://welpo.github.io/tabi/blog/shortcodes/).\n- [Social media cards](https://welpo.github.io/tabi/blog/mastering-tabi-settings/#social-media-cards).\n- Responsive design.\n- [Projects page](https://welpo.github.io/tabi/projects/).\n- [Archive page](https://welpo.github.io/tabi/archive/).\n- [Pinned posts](https://welpo.github.io/tabi/blog/mastering-tabi-settings/#pinning-posts).\n- [Social links](https://welpo.github.io/tabi/blog/mastering-tabi-settings/#social-media-icons).\n- [Tags](https://welpo.github.io/tabi/blog/mastering-tabi-settings/#tags).\n\n## Development Practices\n\n- **[Conventional Commits](https://www.conventionalcommits.org) & [Gitmoji](https://gitmoji.dev/)**: Commit messages follow standardised formats to enhance readability.\n- **Issue Tracking**: Each bug or new feature has its dedicated ticket, which is linked to any consequent code commits and related PRs or issues.\n- **Comprehensive Commentary**: Tickets are documented with images, videos, and detailed descriptions to facilitate effective communication and problem-solving.\n- **Cross-Referencing**: We link all tickets to the relevant code commits, pull requests, or related issues for complete traceability.\n\n## Project Evolution\n\n**tabi** was originally designed for my personal website, [osc.garden](https://osc.garden). Despite its origins for personal use, best practices were implemented from the outset to ensure quality and maintainability. The theme has since grown to attract a vibrant community of contributors on GitHub.\n\n## Start Your Writing Journey with tabi\n\nYou have something to say. Perhaps it's about how linguists haven't agreed on a [definition of \"word\"](https://en.wikipedia.org/wiki/Word) yet, or about your journey exploring the different [flamenco palos](https://en.wikipedia.org/wiki/Palo_(flamenco)), or about how you managed to solve a bug in a popular open-source project.\n\n**tabi** provides the ideal foundation for your writing space, letting you focus on your words while Zola and tabi take care of the technical side. Step into the world of blogging with a system that makes each post a joy to write and to read. Your voice has value—share it with the world.\n"
  },
  {
    "path": "content/projects/zutsu/index.ca.md",
    "content": "+++\ntitle = \"zutsu\"\ndescription = \"Una aplicació minimalista i privada de gestió de tasques.\"\nweight = 32\n\n[taxonomies]\ntags = [\"interactiu\", \"productivitat\", \"web app\", \"web\", \"JavaScript\"]\n\n[extra]\nlocal_image = \"projects/zutsu/zutsu_logo.webp\"\ncanonical_url = \"https://osc.garden/ca/projects/zutsu/\"\nsocial_media_card = \"social_cards/projects_zutsu.jpg\"\n+++\n\n{% wide_container() %}\n<video controls src=\"https://cdn.jsdelivr.net/gh/welpo/zutsu/assets/ずつ_demo.mov\" title=\"demostració de zutsu\"></video>\n{% end %}\n\n#### [Prova-la ara](https://zutsu.osc.garden) • [GitHub](https://github.com/welpo/zutsu) • [Article](https://osc.garden/ca/blog/zutsu-offline-task-planner-web-app/) {.centered-text}\n\n<ruby>ず<rt>zu</rt>つ<rt>tsu</rt></ruby> és una aplicació web de gestió de tasques dissenyada per ajudar-te a centrar-te en una tasca cada vegada. El nom ve de <ruby>一つ<rt>hitotsu</rt>ずつ<rt>zutsu</rt></ruby>, que significa «d'un en un» en <ruby>日本語<rt>japonès</rt></ruby>.\n\n## Per què?\n\nVolia substituir els esdeveniments de calendari inflexibles per a les sessions d'estudi per alguna cosa senzilla i adaptable. Sense aplicacions de tercers, sense sincronització al núvol, només un espai centrat en a la gestió de tasques.\n\n## Funcionalitats\n\n### Principals\n\n- Gestió de tasques amb possibilitat de reordenar-les arrossegant\n- Temporitzador amb durada personalitzable per tasca\n- Privada i offline —sense comptes, seguiment ni emmagatzematge al servidor\n- Importació/exportació de llistes de tasques (JSON)\n\n### Utilitats\n\n- Temporitzador Pomodoro\n- Calendari d'activitat (vista de 30 dies)\n- Comptador i cronòmetre\n- Espai per prendre notes\n- Selectors aleatoris\n\n### Qualitat de vida\n\n- Tema fosc i clar\n- Notificacions del navegador i so\n- Dreceres de teclat\n- Disseny responsive\n\n[![targeta social de zutsu](social_cards/projects_zutsu.jpg)](https://zutsu.osc.garden)\n"
  },
  {
    "path": "content/projects/zutsu/index.es.md",
    "content": "+++\ntitle = \"zutsu\"\ndescription = \"Una aplicación minimalista y privada de gestión de tareas.\"\nweight = 32\n\n[taxonomies]\ntags = [\"interactivo\", \"productividad\", \"web app\", \"web\", \"JavaScript\"]\n\n[extra]\nlocal_image = \"projects/zutsu/zutsu_logo.webp\"\ncanonical_url = \"https://osc.garden/es/projects/zutsu/\"\nsocial_media_card = \"social_cards/projects_zutsu.jpg\"\n+++\n\n{% wide_container() %}\n<video controls src=\"https://cdn.jsdelivr.net/gh/welpo/zutsu/assets/ずつ_demo.mov\" title=\"demostración de zutsu\"></video>\n{% end %}\n\n#### [Pruébala ahora](https://zutsu.osc.garden) • [GitHub](https://github.com/welpo/zutsu) • [Artículo](https://osc.garden/es/blog/zutsu-offline-task-planner-web-app/) {.centered-text}\n\n<ruby>ず<rt>zu</rt>つ<rt>tsu</rt></ruby> es una aplicación web de gestión de tareas diseñada para ayudarte a centrarte en una tarea a la vez. El nombre viene de <ruby>一つ<rt>hitotsu</rt>ずつ<rt>zutsu</rt></ruby>, que significa «uno por uno» en <ruby>日本語<rt>japonés</rt></ruby>.\n\n## ¿Por qué?\n\nQuería sustituir la inflexibilidad del calendario para planificar las sesiones de estudio por algo simple y adaptable. Sin aplicaciones de terceros, sin sincronización en la nube —solo un espacio centrado en la gestión de tareas.\n\n## Funcionalidades\n\n### Principales\n\n- Gestión de tareas con posibilidad de reordenarlas arrastrando y soltando\n- Temporizador con duración personalizable por tarea\n- Privada y offline —sin cuentas, seguimiento ni almacenamiento en servidor\n- Importación/exportación de listas de tareas (JSON)\n\n### Utilidades\n\n- Temporizador Pomodoro\n- Calendario de actividad (vista de 30 días)\n- Contador y cronómetro\n- Espacio para tomar notas\n- Selectores aleatorios\n\n### Calidad de vida\n\n- Tema oscuro y claro\n- Notificaciones del navegador y sonido\n- Atajos de teclado\n- Diseño responsive\n\n[![tarjeta social de zutsu](social_cards/projects_zutsu.jpg)](https://zutsu.osc.garden)\n"
  },
  {
    "path": "content/projects/zutsu/index.md",
    "content": "+++\ntitle = \"zutsu\"\ndescription = \"A private minimalist task management app.\"\nweight = 32\n\n[taxonomies]\ntags = [\"interactive\", \"productivity\", \"web app\", \"web\", \"JavaScript\"]\n\n[extra]\nlocal_image = \"projects/zutsu/zutsu_logo.webp\"\ncanonical_url = \"https://osc.garden/projects/zutsu/\"\nsocial_media_card = \"social_cards/projects_zutsu.jpg\"\n+++\n\n{% wide_container() %}\n<video controls src=\"https://cdn.jsdelivr.net/gh/welpo/zutsu/assets/ずつ_demo.mov\" title=\"zutsu demo\"></video>\n{% end %}\n\n#### [Try it now](https://zutsu.osc.garden) • [GitHub](https://github.com/welpo/zutsu) • [Blog post](https://osc.garden/blog/zutsu-offline-task-planner-web-app/) {.centered-text}\n\n<ruby>ず<rt>zu</rt>つ<rt>tsu</rt></ruby> is a task management web app designed to help you focus on one task at a time. The name comes from <ruby>一つ<rt>hitotsu</rt>ずつ<rt>zutsu</rt></ruby> which means \"one at a time\" in <ruby>日本語<rt>Japanese</rt></ruby>.\n\n## Why?\n\nI wanted to replace inflexible calendar events for study sessions with something simple and adaptable. No third-party apps, no cloud sync —just a focused space for task management.\n\n## Features\n\n### Core\n\n- Task management with drag-and-drop reordering\n- Timer with customizable duration for each task\n- Private & offline—no accounts, tracking, or server storage\n- Import/export task lists (JSON)\n\n### Utilities\n\n- Pomodoro timer\n- Activity calendar (30-day view)\n- Counter & stopwatch\n- Note-taking space\n- Random choice makers\n\n### Quality of life\n\n- Dark and light theme support\n- Browser and sound notifications\n- Keyboard shortcuts\n- Responsive design\n\n[![zutsu social media card](social_cards/projects_zutsu.jpg)](https://zutsu.osc.garden)\n"
  },
  {
    "path": "i18n/ar.toml",
    "content": "# Hello, the Arabic language has many pronouns and words, and each word indicates a different meaning,\n# unlike the English language, in which, on the other hand, the word can refer to a person and a group.\n# This translation is for individual use, if you are a company or organization, I have put a comment in\n# front of each translation that refers to the person and the other word that refers to the organization\n\nlanguage_name = \"العربية\"  # Shown in language picker for multi-language sites.\ndate_locale = \"ar_SA\"  # The locale code for time and date formatting.\nfull_stop = \".\"  # Used at the end of a sentence.\n\n# Menu items.\n# Should match the names in config.extra.menu and config.extra.footer_menu.\nblog = \"التدوينات\"\narchive = \"الأرشيف\"\ntags = \"الوسوم\"\nprojects = \"مشاريعي\"  #Change this to \"المشاريع\" if the site is for an organization.\nabout = \"عني\"  #Change this to \"عنا\" or \"من نحن\" if the site is for an organization.\ncontact = \"تواصل معي\"  #Change this to \"تواصل معنا\" if the site is for an organization.\nprivacy = \"الخصوصية\"\nsite_statistics = \"إحصائيات المدونة\"\nsitemap = \"خريطة المدونة\"\n\n# Search.\nsearch = \"بحث\"\nsearch_icon_title = \"أنقر هنا أو اضغط $SHORTCUT لفتح البحث\"  # $SHORTCUT will be replaced with the actual keyboard shortcut.\nclear_search = \"مسح البحث\"\nzero_results = \"لم يتم العثور على نتائج\"  #No search results.\none_results = \"تم العثور على نتيجة واحدة\"  # One search result.\ntwo_results = \"تم العثور على نتيجتين\"  # Two search results.\nfew_results = \"تم العثور على $NUMBER نتائج\"  # for 3 to 10 search results.\nmany_results = \"تم العثور على $NUMBER نتيجة\"  # 11 or more search results.\n\n# Navigation.\nskip_to_content = \"الإنتقال إلى المحتوى\"\npinned = \"مثبتة\"\njump_to_posts = \"الإنتقال إلى التدوينات\"\nread_more = \"إقرأ المزيد\"\none_posts = \"تدوينة واحدة\"  #One blog post.\ntwo_posts = \"تدوينتين\"  #Two blog posts.\nfew_posts = \"$NUMBER تدوينات\"  #3 to 10 blog posts.\nmany_posts = \"$NUMBER تدوينة\"  #11 or more blog posts.\nprev = \"السابق\"  # As in \"Previous\" page.\nnext = \"التالي\"  # As in \"Next\" page.\nof = \"من\"  # E.g. Page 1 \"of\" 3\nall_posts = \"جميع التدوينات\"\nall_tags = \"جميع الوسوم\"\nall_projects = \"جميع المشاريع\"\nfeatured_projects = \"المشاريع المميزة\"\nlanguage_selection = \"تحديد اللغة\"\ntoggle_mode = \"تبديل الاوضاع $MODE\"  # $MODE will be replaced by a value (or both) below.\ndark = \"داكن\"\nlight = \"فاتح\"\nreset_mode = \"إعادة تعيين الوضع\"\n\n# Quick navigation buttons.\ntoggle_toc = \"إظهار / إخفاء جدول المحتويات\"\ngo_to_top = \"انتقل إلى أعلى الصفحة\"\ngo_to_comments = \"انتقل إلى التعليقات\"\n\n# Post metadata.\nby_author = \"بقلم $AUTHOR\"  # $AUTHOR will be replaced by the author(s).\nauthor_separator = \"، \"  # For multiple authors. Ensure spacing where necessary.\nauthor_conjunction = \" و \"  # For multiple authors. Ensure spacing where necessary.\ndraft = \"مسودة\"\nzero_min_read = \"الوقت المتوقع للقراءة: أقل من دقيقة\"  #Less than one minute read.\none_min_read = \"الوقت المتوقع للقراءة: دقيقة واحدة\"  #One minute read.\ntwo_min_read = \"الوقت المتوقع للقراءة: دقيقتان\"  #Two minutes read.\nfew_min_read = \"الوقت المتوقع للقراءة: $NUMBER دقائق\"  #3 to 10 minutes read.\nmany_min_read = \"الوقت المتوقع للقراءة: $NUMBER دقيقة\"  #11 or more minutes read.\n\nzero_words = \"لا توجد كلمات\"  # No words.\none_words = \"كلمة واحدة\"  # One word.\ntwo_words = \"كلمتين\"  # Two words.\nfew_words = \"$NUMBER كلمات\"  # 3 to 10 words.\nmany_words = \"$NUMBER كلمة\"  # 11 or more words.\n\nlast_updated_on = \"تم التحديث في $DATE\"\nsee_changes = \"الإطلاع على التغييرات\"\n\n# Post body.\ntable_of_contents = \"جدول المحتويات\"\nload_comments = \"إظهار التعليقات\"\n\n# Copy code block button.\ncopied = \"تم النسخ!\"\ncopy_code_to_clipboard = \"نسخ الشِفرة إلى الحافظة\"\n\n# iine appreciation button.\nlike_this_post = \"أعجبني\"\n\n# Footer: Powered by Zola and tabi.\npowered_by = \"مُشَغل بواسطة\"\nand = \"و\"\nsite_source = \"شِفرة الموقع\"\n\n# 404 error.\n# https://welpo.github.io/tabi/404.html\npage_missing = \"الصفحة غير موجودة\"\ntranslation_missing = \"أو لم يتم ترجمتها إلى اللغة التي تطلبها\"\ncheck_url = \"تحقق من أن الرابط صحيح أو\"\ngo_home = \"انتقل إلى الصفحة الرئيسية\"\n\n# For multilingual quote shortcode.\n# https://welpo.github.io/tabi/blog/shortcodes/#multilingual-quotes\nshow_original_quote = \"إظهار الإقتباس الأصلي\"\nshow_translation = \"إظهار الترجمة\"\nopen_quotation_mark = '\"'\nclose_quotation_mark = '\"'\n\n# Translations for stylised Atom feed.\n# https://welpo.github.io/tabi/atom.xml\n# Must contain \"About Feeds\"; it will become a link.\nabout_feeds = \"الخلاصة تجدها هنا، تُسمى أيضًا Atom. اشترك عبر نسخ الرابط إلى برنامج قارئ الخُلاصات الذي تستخدمه. قم بزيارة About Feeds لمعرفة المزيد والبدء. انه مجانا.\"\nvisit_the_site = \"زيارة الموقع\"\nrecent_posts = \"المنشورات الأخيرة\"\n"
  },
  {
    "path": "i18n/ca.toml",
    "content": "language_name = \"Català\"  # Shown in language picker for multi-language sites.\ndate_locale = \"ca_ES\"\nfull_stop =\".\"  # Used at the end of a sentence.\n\n# Menu items.\n# Should match the names in config.extra.menu and config.extra.footer_menu.\nblog = \"blog\"\narchive = \"arxiu\"\ntags = \"etiquetes\"\nprojects = \"projectes\"\nabout = \"sobre mi\"\ncontact = \"contacte\"\nprivacy = \"política de privadesa\"\nsite_statistics = \"estadístiques del lloc\"\nsitemap = \"mapa del lloc\"\n\n# Search.\nsearch = \"Cercar\"\nsearch_icon_title = \"Fes clic o prem $SHORTCUT per obrir la cerca\"  # $SHORTCUT will be replaced with the actual keyboard shortcut.\nclear_search = \"Esborrar cerca\"  # Title of the X icon next to search input.\nzero_results = \"No s'han trobat resultats\"\none_results = \"$NUMBER resultat\"  # \"1 result\"\nmany_results = \"$NUMBER resultats\"  # \"3 results\"\n\n# Navigation.\nskip_to_content = \"Saltar al contingut\"\npinned = \"Fixada\"\njump_to_posts = \"Saltar als articles\"\nread_more = \"Llegir més\"\none_posts = \"$NUMBER entrada\"\nmany_posts = \"$NUMBER entrades\"\nprev = \"Anterior\"  # As in \"Previous\" page.\nnext = \"Següent\"  # As in \"Next\" page.\nof = \"de\"  # E.g. Page 1 \"of\" 3\nall_posts = \"Totes les entrades\"\nall_tags = \"Totes les etiquetes\"\nall_projects = \"Tots els projectes\"\nfeatured_projects = \"Projectes destacats\"\nlanguage_selection = \"Selecció d'idioma\"\ntoggle_mode = \"Canvia el mode $MODE\"  # $MODE will be replaced by a value (or both) below.\ndark = \"obscur\"\nlight = \"clar\"\nreset_mode = \"Restableix el mode al valor predeterminat\"\n\n# Quick navigation buttons.\ntoggle_toc = \"Mostrar/ocultar la taula de continguts\"\ngo_to_top = \"Vés a l'inici de la pàgina\"\ngo_to_comments = \"Vés a la secció de comentaris\"\n\n# Post metadata.\nby_author = \"Per $AUTHOR\"  # $AUTHOR will be replaced by the author(s).\nauthor_separator = \", \"  # For multiple authors. Ensure spacing where necessary.\nauthor_conjunction = \" i \"  # For multiple authors. Ensure spacing where necessary.\ndraft = \"ESBORRANY\"\nzero_min_read = \"<1 min de lectura\"\none_min_read = \"$NUMBER min de lectura\"\nmany_min_read = \"$NUMBER mins de lectura\"\nzero_words = \"Cap paraula\"\none_words = \"$NUMBER paraula\"\nmany_words = \"$NUMBER paraules\"\nlast_updated_on = \"Actualitzat el $DATE\"\nsee_changes = \"Veure canvis\"\n\n# Post body.\ntable_of_contents = \"Taula de contingut\"\nload_comments = \"Carregar comentaris\"\n\n# Copy code block button.\ncopied = \"Copiat!\"\ncopy_code_to_clipboard = \"Copia el codi al porta-retalls\"\n\n# iine appreciation button.\nlike_this_post = \"M'agrada aquesta publicació\"\n\n# Footer: Powered by Zola and tabi.\npowered_by = \"Propulsat per\"\nand = \"i\"\nsite_source = \"Codi del lloc\"\n\n# 404 error.\n# https://welpo.github.io/tabi/404.html\npage_missing = \"La pàgina que has sol·licitat sembla que no existeix\"\ntranslation_missing = \"o encara no s'ha traduït al teu idioma\"\ncheck_url = \"Comprova l'URL per detectar errors o\"\ngo_home = \"torna a la pàgina d'inici\"\n\n# For multilingual quote shortcode.\n# https://welpo.github.io/tabi/blog/shortcodes/#multilingual-quotes\nshow_original_quote = \"Mostra la cita original\"\nshow_translation = \"Mostra la traducció\"\nopen_quotation_mark = \"«\"\nclose_quotation_mark = \"»\"\n\n# Translations for stylised Atom feed.\n# https://welpo.github.io/tabi/atom.xml\n# Must contain \"About Feeds\"; it will become a link.\nabout_feeds = \"Aquest és un canal web, també conegut com a canal Atom. Subscriu-te copiant l'URL de la barra d'adreces al teu lector de notícies. Visita About Feeds per aprendre més i començar. És gratuït.\"\nvisit_the_site = \"Visita la web\"\nrecent_posts = \"Publicacions recents\"\n"
  },
  {
    "path": "i18n/de.toml",
    "content": "# This has been machine translated.\n# If you would like to help correct errors or improve the translation,\n# please open an issue or submit a pull request.\n# https://github.com/welpo/tabi\nlanguage_name = \"Deutsch\"  # Shown in language picker for multi-language sites.\ndate_locale = \"de_DE\"\nfull_stop = \".\"  #  Used at the end of a sentence.\n\n# Menu items.\n# Should match the names in config.extra.menu and config.extra.footer_menu.\nblog = \"blog\"\narchive = \"archiv\"\ntags = \"tags\"\nprojects = \"projekte\"\nabout = \"über\"\ncontact = \"kontakt\"\nprivacy = \"datenschutzrichtlinie\"\nsite_statistics = \"seitenstatistiken\"\nsitemap = \"seitenübersicht\"\n\n# Search.\nsearch = \"Suchen\"\nsearch_icon_title = \"Klicken oder $SHORTCUT drücken, um die Suche zu öffnen\"  # $SHORTCUT will be replaced with the actual keyboard shortcut.\nclear_search = \"Suche löschen\"  # Title of the X icon next to search input.\nzero_results = \"Keine Ergebnisse gefunden\"\none_results = \"$NUMBER Ergebnis\"  # \"1 result\"\nmany_results = \"$NUMBER Ergebnisse\"  # \"3 results\"\n\n# Navigation.\nskip_to_content = \"Zum Inhalt springen\"\npinned = \"Angeheftet\"\njump_to_posts = \"Zu den Beiträgen springen\"\nread_more = \"Weiterlesen\"\none_posts = \"$NUMBER Beitrag\"\nmany_posts = \"$NUMBER Beiträge\"\nprev = \"Vorherig\"  # As in \"Previous\" page.\nnext = \"Nächst\"  # As in \"Next\" page.\nof = \"von\"  # E.g. Page 1 \"of\" 3\nall_posts = \"Alle Beiträge\"\nall_tags = \"Alle Tags\"\nall_projects = \"Alle Projekte\"\nfeatured_projects = \"Empfohlene Projekte\"\nlanguage_selection = \"Sprachauswahl\"\ntoggle_mode = \"Wechsle in den $MODE Modus\"  # $MODE will be replaced by a value (or both) below.\ndark = \"dunkel\"\nlight = \"hell\"\nreset_mode = \"Modus auf Standard zurücksetzen\"\n\n# Quick navigation buttons.\ntoggle_toc = \"Inhaltsverzeichnis ein-/ausblenden\"\ngo_to_top = \"Zum Seitenanfang\"\ngo_to_comments = \"Zum Kommentarbereich\"\n\n# Post metadata.\nby_author = \"Von $AUTHOR\"  # $AUTHOR will be replaced by the author(s).\nauthor_separator = \", \"  # For multiple authors. Ensure spacing where necessary.\nauthor_conjunction = \" und \"  # For multiple authors. Ensure spacing where necessary.\ndraft = \"ENTWURF\"\nzero_min_read = \"weniger als 1 Min. Lesezeit\"\none_min_read = \"$NUMBER Min. Lesezeit\"\nmany_min_read = \"$NUMBER Min. Lesezeit\"\nzero_words = \"Keine Wörter\"\none_words = \"$NUMBER Wort\"\nmany_words = \"$NUMBER Wörter\"\nlast_updated_on = \"Aktualisiert am $DATE\"\nsee_changes = \"Änderungen anzeigen\"\n\n# Post body.\ntable_of_contents = \"Inhaltsverzeichnis\"\nload_comments = \"Kommentare laden\"\n\n# Copy code block button.\ncopied = \"Kopiert!\"\ncopy_code_to_clipboard = \"Code in die Zwischenablage kopieren\"\n\n# iine appreciation button.\nlike_this_post = \"Dieser Beitrag gefällt mir\"\n\n# Footer.\npowered_by = \"Angetrieben von\"\nand = \"und\"\nsite_source = \"Quellcode der Seite\"\n\n# 404 error.\n# https://welpo.github.io/tabi/404.html\npage_missing = \"Die Seite, die du angefordert hast, scheint nicht zu existieren\"\ntranslation_missing = \"oder wurde noch nicht in deine Sprache übersetzt\"\ncheck_url = \"Überprüfe die URL auf Fehler oder\"\ngo_home = \"zurück zur Startseite gehen\"\n\n# For multilingual quote shortcode.\n# https://welpo.github.io/tabi/blog/shortcodes/#multilingual-quotes\nshow_original_quote = \"Originalzitat anzeigen\"\nshow_translation = \"Übersetzung anzeigen\"\nopen_quotation_mark = \"„\"\nclose_quotation_mark = \"“\"\n\n# Translations for stylised Atom feed.\n# https://welpo.github.io/tabi/atom.xml\n# Must contain \"About Feeds\"; it will become a link.\nabout_feeds = \"Dies ist ein Web-Feed, auch bekannt als Atom-Feed. Abonnieren Sie, indem Sie die URL aus der Adressleiste in Ihren Nachrichtenleser kopieren. Besuchen Sie About Feeds, um mehr zu erfahren und loszulegen. Es ist kostenlos.\"\nvisit_the_site = \"Besuchen Sie die Webseite\"\nrecent_posts = \"Aktuelle Beiträge\"\n"
  },
  {
    "path": "i18n/en.toml",
    "content": "language_name = \"English\"  # Shown in language picker for multi-language sites.\ndate_locale = \"en_GB\"\nfull_stop = \".\"  # Used at the end of a sentence.\n\n# Menu items.\n# Should match the names in config.extra.menu and config.extra.footer_menu.\nblog = \"blog\"\narchive = \"archive\"\ntags = \"tags\"\nprojects = \"projects\"\nabout = \"about\"\ncontact = \"contact\"\nprivacy = \"privacy policy\"\nsite_statistics = \"site statistics\"\nsitemap = \"sitemap\"\n\n# Search.\nsearch = \"Search\"\nsearch_icon_title = \"Click or press $SHORTCUT to open search\"  # $SHORTCUT will be replaced with the actual keyboard shortcut.\nclear_search = \"Clear search\"  # Title of the X icon next to search input.\nzero_results = \"No results\"\none_results = \"$NUMBER result\"  # \"1 result\"\nmany_results = \"$NUMBER results\"  # \"3 results\"\n\n# Navigation.\nskip_to_content = \"Skip to content\"\npinned = \"Pinned\"\njump_to_posts = \"Jump to posts\"\nread_more = \"Read more\"\none_posts = \"$NUMBER post\"\nmany_posts = \"$NUMBER posts\"  # \"3 posts\"\nprev = \"Prev\"  # As in \"Previous\" page.\nnext = \"Next\"  # As in \"Next\" page.\nof = \"of\"  # E.g. Page 1 \"of\" 3\nall_posts = \"All posts\"\nall_tags = \"All tags\"\nall_projects = \"All projects\"\nfeatured_projects = \"Featured projects\"\nlanguage_selection = \"Language selection\"\ntoggle_mode = \"Toggle $MODE mode\"  # $MODE will be replaced by a value (or both) below.\ndark = \"dark\"\nlight = \"light\"\nreset_mode = \"Reset mode to default\"\n\n# Quick navigation buttons.\ntoggle_toc = \"Toggle Table of Contents\"\ngo_to_top = \"Go to the top of the page\"\ngo_to_comments = \"Go to the comments section\"\n\n# Post metadata.\nby_author = \"By $AUTHOR\"  # $AUTHOR will be replaced by the author(s).\nauthor_separator = \", \"  # For multiple authors. Ensure spacing where necessary.\nauthor_conjunction = \" and \"  # For multiple authors. Ensure spacing where necessary.\ndraft = \"DRAFT\"\nzero_min_read = \"<1 min read\"\none_min_read = \"$NUMBER min read\"\nmany_min_read = \"$NUMBER min read\"\nzero_words = \"No words\"\none_words = \"$NUMBER word\"\nmany_words = \"$NUMBER words\"\nlast_updated_on = \"Updated on $DATE\"\nsee_changes = \"See changes\"\n\n# Post body.\ntable_of_contents = \"Table of Contents\"\nload_comments = \"Load comments\"\n\n# Copy code block button.\ncopied = \"Copied!\"\ncopy_code_to_clipboard = \"Copy code to clipboard\"\n\n# iine appreciation button.\nlike_this_post = \"Like this post\"\n\n# Footer: Powered by Zola and tabi.\npowered_by = \"Powered by\"\nand = \"&\"\nsite_source = \"Site source\"\n\n# 404 error.\n# https://welpo.github.io/tabi/404.html\npage_missing = \"The page you've requested seems to be missing\"\ntranslation_missing = \"or hasn't been translated into your language yet\"\ncheck_url = \"Check the URL for errors or\"\ngo_home = \"go back to the homepage\"\n\n# For multilingual quote shortcode.\n# https://welpo.github.io/tabi/blog/shortcodes/#multilingual-quotes\nshow_original_quote = \"Show original quote\"\nshow_translation = \"Show translation\"\nopen_quotation_mark = \"“\"\nclose_quotation_mark = \"”\"\n\n# Translations for stylised Atom feed.\n# https://welpo.github.io/tabi/atom.xml\n# Must contain \"About Feeds\"; it will become a link.\nabout_feeds = \"This is a web feed, also known as an Atom feed. Subscribe by copying the URL from the address bar into your newsreader. Visit About Feeds to learn more and get started. It's free.\"\nvisit_the_site = \"Visit website\"\nrecent_posts = \"Recent posts\"\n"
  },
  {
    "path": "i18n/es.toml",
    "content": "language_name = \"Español\"  # Shown in language picker for multi-language sites.\ndate_locale = \"es_ES\"\nfull_stop = \".\"  # Used at the end of a sentence.\n\n# Menu items.\n# Should match the names in config.extra.menu and config.extra.footer_menu.\nblog = \"blog\"\narchive = \"archivo\"\ntags = \"etiquetas\"\nprojects = \"proyectos\"\nabout = \"sobre mí\"\ncontact = \"contacto\"\nprivacy = \"política de privacidad\"\nsite_statistics = \"estadísticas del sitio\"\nsitemap = \"mapa del sitio\"\n\n# Search.\nsearch = \"Buscar\"\nsearch_icon_title = \"Haz clic o usa $SHORTCUT para abrir la búsqueda\"  # $SHORTCUT will be replaced with the actual keyboard shortcut.\nclear_search = \"Borrar búsqueda\"  # Title of the X icon next to search input.\nzero_results = \"No hay resultados\"\none_results = \"$NUMBER resultado\"\nmany_results = \"$NUMBER resultados\"\n\n# Navigation.\nskip_to_content = \"Saltar al contenido\"\npinned = \"Fijada\"\njump_to_posts = \"Saltar a las entradas\"\nread_more = \"Leer más\"\none_posts = \"$NUMBER entrada\"\nmany_posts = \"$NUMBER entradas\"\nprev = \"Anterior\"  # As in \"Previous\" page.\nnext = \"Siguiente\"  # As in \"Next\" page.\nof = \"de\"  # E.g. Page 1 \"of\" 3\nall_posts = \"Todas las entradas\"\nall_tags = \"Todas las etiquetas\"\nall_projects = \"Todos los proyectos\"\nfeatured_projects = \"Proyectos destacados\"\nlanguage_selection = \"Selección de idioma\"\ntoggle_mode = \"Cambiar a modo $MODE\"  # $MODE will be replaced by a value (or both) below.\ndark = \"oscuro\"\nlight = \"claro\"\nreset_mode = \"Restablecer modo a configuración predeterminada\"\n\n# Quick navigation buttons.\ntoggle_toc = \"Mostrar/ocultar la tabla de contenidos\"\ngo_to_top = \"Ir al inicio de la página\"\ngo_to_comments = \"Ir a la sección de comentarios\"\n\n# Post metadata.\nby_author = \"Por $AUTHOR\"  # $AUTHOR will be replaced by the author(s).\nauthor_separator = \", \"  # For multiple authors. Ensure spacing where necessary.\nauthor_conjunction = \" y \"  # For multiple authors. Ensure spacing where necessary.\ndraft = \"BORRADOR\"\nzero_min_read = \"<1 min de lectura\"\none_min_read = \"$NUMBER min de lectura\"\nmany_min_read = \"$NUMBER mins de lectura\"\nzero_words = \"Cero palabras\"\none_words = \"$NUMBER palabra\"\nmany_words = \"$NUMBER palabras\"\nlast_updated_on = \"Actualizado el $DATE\"\nsee_changes = \"Ver cambios\"\n\n# Post body.\ntable_of_contents = \"Tabla de contenido\"\nload_comments = \"Cargar comentarios\"\n\n# Copy code block button.\ncopied = \"Copiado!\"\ncopy_code_to_clipboard = \"Copiar código al portapapeles\"\n\n# iine appreciation button.\nlike_this_post = \"Me gusta esta publicación\"\n\n# Footer: Powered by Zola and tabi.\npowered_by = \"Impulsado por\"\nand = \"y\"\nsite_source = \"Código del sitio\"\n\n# 404 error.\n# https://welpo.github.io/tabi/404.html\npage_missing = \"La página que has solicitado parece no existir\"\ntranslation_missing = \"o aún no se ha traducido a tu idioma\"\ncheck_url = \"Revisa la URL en busca de errores o\"\ngo_home = \"regresa a la página de inicio\"\n\n# For multilingual quote shortcode.\n# https://welpo.github.io/tabi/blog/shortcodes/#multilingual-quotes\nshow_original_quote = \"Mostrar cita original\"\nshow_translation = \"Mostrar traducción\"\nopen_quotation_mark = \"«\"\nclose_quotation_mark = \"»\"\n\n# Translations for stylised Atom feed.\n# https://welpo.github.io/tabi/atom.xml\n# Must contain \"About Feeds\"; it will become a link.\nabout_feeds = \"Esta es una fuente web, también conocida como fuente Atom. Suscríbete copiando la URL de la barra de direcciones en tu lector de noticias. Visita About Feeds para aprender más y empezar. Es gratis. \"\nvisit_the_site = \"Visita la web\"\nrecent_posts = \"Publicaciones recientes\"\n"
  },
  {
    "path": "i18n/et.toml",
    "content": "language_name = \"Eesti\"  # Shown in language picker for multi-language sites.\ndate_locale = \"et_EE\"\nfull_stop = \".\"  # Used at the end of a sentence.\n\n# Menu items.\n# Should match the names in config.extra.menu and config.extra.footer_menu.\nblog = \"blogi\"\narchive = \"arhiiv\"\ntags = \"sildid\"\nprojects = \"projektid\"\nabout = \"minust\"\ncontact = \"kontakt\"\nprivacy = \"privaatsuspoliitika\"\nsite_statistics = \"saidi statistika\"\nsitemap = \"saidikaart\"\n\n# Search.\nsearch = \"Otsing\"\nsearch_icon_title = \"Otsingu avamiseks klõpsake või vajutage $SHORTCUT\"  # $SHORTCUT will be replaced with the actual keyboard shortcut.\nclear_search = \"Tühjenda otsing\"  # Title of the X icon next to search input.\nzero_results = \"Tulemusi pole\"\none_results = \"$NUMBER tulemus\"  # \"1 result\"\nmany_results = \"$NUMBER tulemust\"  # \"3 results\"\n\n# Navigation.\nskip_to_content = \"Hüppa sisu juurde\"\npinned = \"Kinnitatud\"\njump_to_posts = \"Hüppa postitusteni\"\nread_more = \"Loe edasi\"\none_posts = \"$NUMBER postitus\"\nmany_posts = \"$NUMBER postitust\"  # \"3 posts\"\nprev = \"Eelmine\"  # As in \"Previous\" page.\nnext = \"Järgmine\"  # As in \"Next\" page.\nof = \"/\"  # E.g. Page 1 \"of\" 3\nall_posts = \"Kõik postitused\"\nall_tags = \"Kõik sildid\"\nall_projects = \"Kõik projektid\"\nfeatured_projects = \"Esiletõstetud projektid\"\nlanguage_selection = \"Keele valik\"\ntoggle_mode = \"Lülita $MODE režiim\"  # $MODE will be replaced by a value (or both) below.\ndark = \"tume\"\nlight = \"hele\"\nreset_mode = \"Lähtesta vaikerežiim\"\n\n# Quick navigation buttons.\ntoggle_toc = \"Sisukorra kuvamine/peitmine\"\ngo_to_top = \"Mine lehe ülaossa\"\ngo_to_comments = \"Mine kommentaaride juurde\"\n\n# Post metadata.\nby_author = \"Autor: $AUTHOR\"  # $AUTHOR will be replaced by the author(s).\nauthor_separator = \", \"  # For multiple authors. Ensure spacing where necessary.\nauthor_conjunction = \" ja \"  # For multiple authors. Ensure spacing where necessary.\ndraft = \"MUSTAND\"\nzero_min_read = \"Lugemisaeg <1 min\"\none_min_read = \"Lugemisaeg $NUMBER min\"\nmany_min_read = \"Lugemisaeg $NUMBER min\"\nzero_words = \"Sõnu pole\"\none_words = \"$NUMBER sõna\"\nmany_words = \"$NUMBER sõna\"\nlast_updated_on = \"Uuendatud $DATE\"\nsee_changes = \"Vaata muudatusi\"\n\n# Post body.\ntable_of_contents = \"Sisukord\"\nload_comments = \"Lae kommentaarid\"\n\n# Copy code block button.\ncopied = \"Kopeeritud!\"\ncopy_code_to_clipboard = \"Kopeeri kood lõikelauale\"\n\n# iine appreciation button.\nlike_this_post = \"Mulle meeldib see postitus\"\n\n# Footer: Powered by Zola and tabi.\npowered_by = \"Toetab\"\nand = \"ja\"\nsite_source = \"Saidi lähtekood\"\n\n# 404 error.\n# https://welpo.github.io/tabi/404.html\npage_missing = \"Tundub, et otsitud lehte ei leitud\"\ntranslation_missing = \"või pole seda veel teie keelde tõlgitud\"\ncheck_url = \"Kontrollige URL-i vigade suhtes või\"\ngo_home = \"minge tagasi avalehele\"\n\n# For multilingual quote shortcode.\n# https://welpo.github.io/tabi/blog/shortcodes/#multilingual-quotes\nshow_original_quote = \"Näita originaaltsitaati\"\nshow_translation = \"Näita tõlget\"\nopen_quotation_mark = \"“\"\nclose_quotation_mark = \"”\"\n\n# Translations for stylised Atom feed.\n# https://welpo.github.io/tabi/atom.xml\n# Must contain \"About Feeds\"; it will become a link.\nabout_feeds = \"See on veebivoog, tuntud ka kui Atom-voog. Tellimiseks kopeerige URL aadressiribalt oma uudistelugerisse. Külastage About Feeds lehte, et õppida ja alustada. See on tasuta.\"\nvisit_the_site = \"Külasta veebisaiti\"\nrecent_posts = \"Viimased postitused\"\n"
  },
  {
    "path": "i18n/fa.toml",
    "content": "language_name = \"فارسی\"  # Shown in language picker for multi-language sites.\ndate_locale = \"fa_IR\"\nfull_stop = \".\"  # Used at the end of a sentence.\n\n# Menu items.\n# Should match the names in config.extra.menu and config.extra.footer_menu.\nblog = \"وبلاگ\"\narchive = \"آرشیو\"\ntags = \"برچسب‌ها\"\nprojects = \"پروژه‌ها\"\nabout = \"درباره\"\ncontact = \"تماس\"\nprivacy = \"سیاست حفظ حریم خصوصی\"\nsite_statistics = \"آمار سایت\"\nsitemap = \"نقشه سایت\"\n\n# Search.\nsearch = \"جستجو\"\nsearch_icon_title = \"برای باز کردن جستجو کلیک کنید یا $SHORTCUT را فشار دهید\"   # $SHORTCUT will be replaced with the actual keyboard shortcut.\nclear_search = \"پاک کردن جستجو\"  # Title of the X icon next to search input.\nzero_results = \"بدون نتیجه\"\none_results = \"$NUMBER نتیجه\"   # \"1 result\"\nmany_results = \"$NUMBER نتیجه\"  # \"3 results\"\n\n# Navigation.\nskip_to_content = \"پرش به محتوا\"\npinned = \"سنجاق‌شده\"\njump_to_posts = \"پرش به نوشته‌ها\"\nread_more = \"ادامه مطلب\"\none_posts = \"$NUMBER مطلب\"\nmany_posts = \"$NUMBER مطلب\"  # \"3 posts\"\nprev = \"قبلی\"   # As in \"Previous\" page.\nnext = \"بعدی\"   # As in \"Next\" page.\nof = \"از\"  # E.g. Page 1 \"of\" 3\nall_posts = \"همه مطالب\"\nall_tags = \"همه برچسب‌ها\"\nall_projects = \"همه پروژه‌ها\"\nfeatured_projects = \"پروژه‌های برجسته\"\nlanguage_selection = \"انتخاب زبان\"\ntoggle_mode = \"تغییر حالت $MODE\"  # $MODE will be replaced by a value (or both) below.\ndark = \"تیره\"\nlight = \"روشن\"\nreset_mode = \"بازنشانی حالت به حالت پیش‌فرض\"\n\n\n# Quick navigation buttons.\ntoggle_toc = \"تغییر جدول محتوا\"\ngo_to_top = \"رفتن به بالای صفحه\"\ngo_to_comments = \"رفتن به بخش نظرات\"\n\n# Post metadata.\nby_author = \"توسط $AUTHOR\"  # $AUTHOR will be replaced by the author(s).\nauthor_separator = \"، \"  # For multiple authors. Ensure spacing where necessary.\nauthor_conjunction = \" و \"  # For multiple authors. Ensure spacing where necessary.\ndraft = \"پیش‌نویس\"\nzero_min_read = \"خواندن کمتر از ۱ دقیقه\"\none_min_read = \"خواندن $NUMBER دقیقه\"\nmany_min_read = \"خواندن $NUMBER دقیقه\"\nzero_words = \"هیچ کلمه‌ای\"\none_words = \"$NUMBER کلمه\"\nmany_words = \"$NUMBER کلمه\"\nlast_updated_on = \"آخرین به‌روزرسانی در $DATE\"\nsee_changes = \"مشاهده تغییرات\"\n\n# Post body.\ntable_of_contents = \"فهرست مطالب\"\nload_comments = \"بارگذاری نظرات\"\n\n# Copy code block button.\ncopied = \"کپی شد!\"\ncopy_code_to_clipboard = \"کپی کد به کلیپ‌بورد\"\n\n# iine appreciation button.\nlike_this_post = \"این مقاله را دوست دارم\"\n\n# Footer: Powered by Zola and tabi.\npowered_by = \"قدرت گرفته از\"\nand = \"و\"\nsite_source = \"منبع سایت\"\n\n# 404 error.\n# https://welpo.github.io/tabi/404.html\npage_missing = \"صفحه‌ای که درخواست کرده‌اید به نظر می‌رسد از دست رفته است\"\ntranslation_missing = \"یا هنوز به زبان شما ترجمه نشده است\"\ncheck_url = \"آدرس اینترنتی را بررسی کنید و یا\"\ngo_home = \"به صفحه اصلی برگردید\"\n\n# For multilingual quote shortcode.\n# https://welpo.github.io/tabi/blog/shortcodes/#multilingual-quotes\nshow_original_quote = \"نقل قول اصلی را نشان بده\"\nshow_translation = \"ترجمه را نشان بده\"\nopen_quotation_mark = \"«\"\nclose_quotation_mark = \"»\"\n\n# Translations for stylised Atom feed.\n# https://welpo.github.io/tabi/atom.xml\n# Must contain \"About Feeds\"; it will become a link.\nabout_feeds = \"این یک فید وب است، همچنین به عنوان یک فید Atom شناخته می‌شود. با کپی کردن URL از نوار آدرس به خواننده اخباری خود مشترک شوید. برای یادگیری بیشتر و شروع به کار بروید به About Feeds. این رایگان است.\"\nvisit_the_site = \"بازدید از وب‌سایت\"\nrecent_posts = \"مطالب اخیر\"\n"
  },
  {
    "path": "i18n/fi.toml",
    "content": "language_name = \"suomi\"  # Shown in language picker for multi-language sites.\ndate_locale = \"fi_FI\"\nfull_stop = \".\"  # Used at the end of a sentence.\n\n# Menu items.\n# Should match the names in config.extra.menu and config.extra.footer_menu.\nblog = \"blogi\"\narchive = \"arkisto\"\ntags = \"tunnisteet\"\nprojects = \"projektit\"\ndiagrams = \"kaaviot\"\nabout = \"minusta\"\ncontact = \"yhteystiedot\"\nprivacy = \"tietosuojakäytäntö\"\nsite_statistics = \"tilastot\"\nsitemap = \"sivustokartta\"\n\n# Search.\nsearch = \"Hae\"\nsearch_icon_title = \"Paina $SHORTCUT avataksesi haun\"  # $SHORTCUT will be replaced with the actual keyboard shortcut.\nclear_search = \"Tyhjennä haku\"  # Title of the X icon next to search input.\nzero_results = \"Ei tuloksia\"\none_results = \"$NUMBER tulos\"  # \"1 result\"\nmany_results = \"$NUMBER tulosta\"  # \"3 results\"\n\n# Navigation.\nskip_to_content = \"Siirry sisältöön\"\n# Some will prefer artikkeli (=article) instead of julkaisu (=publication), but this is more general\npinned = \"Kiinnitetty\"\njump_to_posts = \"Siirry julkaisuihin\"\nread_more = \"Lue lisää\"\none_posts = \"$NUMBER julkaisu\"\nmany_posts = \"$NUMBER julkaisua\"  # \"3 posts\"\nprev = \"Edellinen\"  # As in \"Previous\" page.\nnext = \"Seuraava\"  # As in \"Next\" page.\nof = \"/\"  # E.g. Page 1 \"of\" 3\nall_posts = \"Kaikki julkaisut\"\nall_tags = \"Kaikki tunnisteet\"\nall_projects = \"Kaikki projektit\"\nfeatured_projects = \"Esittelyprojektit\"\nlanguage_selection = \"Kielivalinta\"\ntoggle_mode = \"Vaihda $MODE\"  # $MODE will be replaced by a value (or both) below. Literally \"Change $MODE\"\n# The declension assumes that both values are used. e.g. \"Change dark mode\" wouldn't work and would need the \"to\" preposition\n# It also omits the \"mode\" word: \"Change dark/light mode\" instead of \"Change dark mode/light mode\"\ndark = \"tumma\"\nlight = \"vaalea tila\"\nreset_mode = \"Palauta oletustilaan\"\n\n# Quick navigation buttons.\ntoggle_toc = \"Sisällysluettelo päälle/pois\"\ngo_to_top = \"Siirry sivun alkuun\"\ngo_to_comments = \"Siirry kommenttiosioon\"\n\n# Post metadata.\nby_author = \"Kirjoittanut $AUTHOR\"  # $AUTHOR will be replaced by the author(s). Literally \"Written by $AUTHOR\"\nauthor_separator = \", \"  # For multiple authors. Ensure spacing where necessary.\nauthor_conjunction = \" ja \"  # For multiple authors. Ensure spacing where necessary.\ndraft = \"LUONNOS\"\nzero_min_read = \"<1 min. lukuaika\"\none_min_read = \"$NUMBER min. lukuaika\"\nmany_min_read = \"$NUMBER min. lukuaika\"\nzero_words = \"Ei sanoja\"\none_words = \"$NUMBER sana\"\nmany_words = \"$NUMBER sanaa\"\nlast_updated_on = \"Päivitetty $DATE\"\nsee_changes = \"Katso muutokset\"\n\n# Post body.\ntable_of_contents = \"Sisällysluettelo\"\nload_comments = \"Lataa kommentit\"\n\n# Copy code block button.\ncopied = \"Kopioitu!\"\ncopy_code_to_clipboard = \"Kopioi koodi leikepöydälle\"\n\n# iine appreciation button.\nlike_this_post = \"Pidän tästä kirjoituksesta\"\n\n# Footer: Powered by Zola and tabi.\npowered_by = \"Pyörii ohjelmilla\" # \"Works using the software\"\nand = \"&\"\nsite_source = \"Sivuston lähdekoodi\"\n\n# 404 error.\n# https://welpo.github.io/tabi/404.html\npage_missing = \"Pyytämääsi sivua ei löydy\"\ntranslation_missing = \"tai sitä ei ole vielä käännetty kielellesi\"\ncheck_url = \"Tarkista URL-osoitteesta mahdolliset virheet tai\"\ngo_home = \"palaa etusivulle\"\n\n# For multilingual quote shortcode.\n# https://welpo.github.io/tabi/blog/shortcodes/#multilingual-quotes\nshow_original_quote = \"Näytä alkuperäinen lainaus\"\nshow_translation = \"Näytä käännös\"\nopen_quotation_mark = \"“\"\nclose_quotation_mark = \"”\"\n\n# Translations for stylised Atom feed.\n# https://welpo.github.io/tabi/atom.xml\n# Must contain \"About Feeds\"; it will become a link.\nabout_feeds = \"Tämä on verkkosyöte, joka tunnetaan myös nimellä Atom-syöte. Tilaa kopioimalla URL-osoite osoitepalkista uutislukijaasi. Käy sivulla About Feeds saadaksesi lisätietoja ja aloittaakeksi. Se on ilmaista.\"\nvisit_the_site = \"Vieraile sivustolla\"\nrecent_posts = \"Viimeisimmät julkaisut\"\n"
  },
  {
    "path": "i18n/fr.toml",
    "content": "language_name = \"Français\"  # Shown in language picker for multi-language sites.\ndate_locale = \"fr_FR\"\nfull_stop = \".\"  # Used at the end of a sentence.\n\n# Menu items.\n# Should match the names in config.extra.menu and config.extra.footer_menu.\nblog = \"blog\"\narchive = \"archives\"\ntags = \"étiquettes\"\nprojects = \"projets\"\nabout = \"à propos\"\ncontact = \"contact\"\nprivacy = \"politique de confidentialité\"\nsite_statistics = \"statistiques du site\"\nsitemap = \"plan du site\"\n\n# Search.\nsearch = \"Rechercher\"\nsearch_icon_title = \"Cliquez ou appuyez sur $SHORTCUT pour ouvrir la recherche\"  # $SHORTCUT will be replaced with the actual keyboard shortcut.\nclear_search = \"Effacer la recherche\"  # Title of the X icon next to search input.\nzero_results = \"Aucun résultat trouvé\"\none_results = \"$NUMBER résultat\"  # \"1 result\"\nmany_results = \"$NUMBER résultats\"  # \"3 results\"\n\n# Navigation.\nskip_to_content = \"Passer au contenu\"\npinned = \"Épinglé\"\njump_to_posts = \"Aller aux articles\"\nread_more = \"Lire plus\"\none_posts = \"$NUMBER article\"\nmany_posts = \"$NUMBER articles\"\nprev = \"Précédent\"  # As in \"Previous\" page.\nnext = \"Suivant\"  # As in \"Next\" page.\nof = \"sur\"  # E.g. Page 1 \"of\" 3\nall_posts = \"Tous les articles\"\nall_tags = \"Toutes les étiquettes\"\nall_projects = \"Tous les projets\"\nfeatured_projects = \"Projets à la une\"\nlanguage_selection = \"Sélection de la langue\"\ntoggle_mode = \"Basculer en mode $MODE\"  # $MODE will be replaced by a value (or both) below.\ndark = \"sombre\"\nlight = \"clair\"\nreset_mode = \"Réinitialiser le mode aux valeurs par défaut\"\n\n# Quick navigation buttons.\ntoggle_toc = \"Afficher/Masquer la table des matières\"\ngo_to_top = \"Aller en haut de la page\"\ngo_to_comments = \"Aller à la section des commentaires\"\n\n# Post metadata.\nby_author = \"Par $AUTHOR\"  # $AUTHOR will be replaced by the author(s).\nauthor_separator = \", \"  # For multiple authors. Ensure spacing where necessary.\nauthor_conjunction = \" et \"  # For multiple authors. Ensure spacing where necessary.\ndraft = \"BROUILLON\"\nzero_min_read = \"<1 min de lecture\"\none_min_read = \"$NUMBER min de lecture\"\nmany_min_read = \"$NUMBER min de lecture\"\nzero_words = \"aucun mot\"\none_words = \"$NUMBER mot\"\nmany_words = \"$NUMBER mots\"\nlast_updated_on = \"Mis à jour le $DATE\"\nsee_changes = \"Voir les modifications\"\n\n# Post body.\ntable_of_contents = \"Table des matières\"\nload_comments = \"Afficher les commentaires\"\n\n# Copy code block button.\ncopied = \"Copié !\"\ncopy_code_to_clipboard = \"Copier le code dans le presse-papier\"\n\n# iine appreciation button.\nlike_this_post = \"J'aime cet article\"\n\n# Footer: Powered by Zola and tabi.\npowered_by = \"Propulsé par\"\nand = \"et\"\nsite_source = \"Code du site\"\n\n# 404 error.\n# https://welpo.github.io/tabi/404.html\npage_missing = \"La page que vous avez demandée semble être manquante\"\ntranslation_missing = \"ou n'a pas encore été traduite dans votre langue\"\ncheck_url = \"Vérifiez que l'URL ne contient pas d'erreur ou\"\ngo_home = \"retournez à la page d'accueil\"\n\n# For multilingual quote shortcode.\n# https://welpo.github.io/tabi/blog/shortcodes/#multilingual-quotes\nshow_original_quote = \"Afficher la citation originale\"\nshow_translation = \"Afficher la traduction\"\nopen_quotation_mark = \"«\"\nclose_quotation_mark = \"»\"\n\n# Translations for stylised Atom feed.\n# https://welpo.github.io/tabi/atom.xml\n# Must contain \"About Feeds\"; it will become a link.\nabout_feeds = \"Ceci est un flux web, également connu sous le nom de flux Atom. Abonnez-vous en copiant l'URL de la barre d'adresse dans votre lecteur de nouvelles. Visitez About Feeds pour en savoir plus et commencez. C'est gratuit.\"\nvisit_the_site = \"Visitez le site web\"\nrecent_posts = \"Articles récents\"\n"
  },
  {
    "path": "i18n/hi.toml",
    "content": "# This has been machine translated.\n# If you would like to help correct errors or improve the translation, please open an issue or submit a pull request.\nlanguage_name = \"हिंदी\"  # Shown in language picker for multi-language sites.\ndate_locale = \"hi_IN\"\nfull_stop =\"।\"  # Used at the end of a sentence.\n\n# Menu items.\n# Should match the names in config.extra.menu and config.extra.footer_menu.\nblog = \"ब्लॉग\"\narchive = \"संग्रह\"\ntags = \"टैग\"\nprojects = \"परियोजनाएं\"\nabout = \"बारे में\"\ncontact = \"संपर्क\"\nprivacy = \"गोपनीयता नीति\"\nsite_statistics = \"साइट सांख्यिकी\"\nsitemap = \"साइटमैप\"\n\n# Search.\nsearch = \"खोजें\"\nsearch_icon_title = \"$SHORTCUT दबाएँ या क्लिक करें खोज खोलने के लिए\"  # $SHORTCUT will be replaced with the actual keyboard shortcut.\nclear_search = \"खोज साफ करें\"  # Title of the X icon next to search input.\nzero_results = \"कोई परिणाम नहीं मिला\"\none_results = \"$NUMBER परिणाम\"  # \"1 result\"\nmany_results = \"$NUMBER परिणाम\"  # \"3 results\"\n\n# Navigation.\nskip_to_content = \"सामग्री पर जाएं\"\npinned = \"पिन किया गया\"\njump_to_posts = \"पोस्ट पर जाएं\"\nread_more = \"और पढ़ें\"\none_posts = \"$NUMBER पोस्ट\"\nmany_posts = \"$NUMBER पोस्ट्स\"\nprev = \"पिछला\"  # As in \"Previous\" page.\nnext = \"अगला\"  # As in \"Next\" page.\nof = \"का\"  # E.g. Page 1 \"of\" 3\nall_posts = \"सभी पोस्ट्स\"\nall_tags = \"सभी टैग्स\"\nall_projects = \"सभी प्रोजेक्ट\"\nfeatured_projects = \"विशेष प्रोजेक्ट\"\nlanguage_selection = \"भाषा चयन\"\ntoggle_mode = \"$MODE मोड में टॉगल करें\"  # $MODE will be replaced by a value (or both) below.\ndark = \"अंधेरा\"\nlight = \"रोशनी\"\nreset_mode = \"मोड को साइट डिफ़ॉल्ट पर रीसेट करें\"\n\n# Quick navigation buttons.\ntoggle_toc = \"विषय-सूची टॉगल करें\"\ngo_to_top = \"पृष्ठ के शीर्ष पर जाएं\"\ngo_to_comments = \"टिप्पणी अनुभाग में जाएं\"\n\n# Post metadata.\nby_author = \"$AUTHOR द्वारा\"  # $AUTHOR will be replaced by the author(s).\nauthor_separator = \", \"  # For multiple authors. Ensure spacing where necessary.\nauthor_conjunction = \" और \"  # For multiple authors. Ensure spacing where necessary.\ndraft = \"मसौदा\"\nzero_min_read = \"कम से कम 1 मिनट पठन समय\"\none_min_read = \"$NUMBER मिनट पठन समय\"\nmany_min_read = \"$NUMBER मिनट पठन समय\"\nzero_words = \"कोई शब्द नहीं\"\none_words = \"$NUMBER शब्द\"\nmany_words = \"$NUMBER शब्द\"\nlast_updated_on = \"$DATE को अपडेट किया गया\"\nsee_changes = \"बदलाव देखें\"\n\n# Post body.\ntable_of_contents = \"विषय सूची\"\nload_comments = \"कमेंट्स लोड करें\"\n\n# Copy code block button.\ncopied = \"कॉपी किया गया!\"\ncopy_code_to_clipboard = \"कोड क्लिपबोर्ड में कॉपी करें\"\n\n# iine appreciation button.\nlike_this_post = \"मुझे यह पोस्ट पसंद है\"\n\n# Footer: Powered by Zola and tabi.\npowered_by = \"चालित द्वारा\"\nand = \"और\"\nsite_source = \"साइट का स्रोत\"\n\n# 404 error.\n# https://welpo.github.io/tabi/404.html\npage_missing = \"आपके द्वारा अनुरोधित पृष्ठ मौजूद नहीं है\"\ntranslation_missing = \"या फिर आपकी भाषा में अनुवाद नहीं किया गया है\"\ncheck_url = \"URL में त्रुटियों की जांच करें या\"\ngo_home = \"मुख्य पृष्ठ पर वापस जाएं\"\n\n# For multilingual quote shortcode.\n# https://welpo.github.io/tabi/blog/shortcodes/#multilingual-quotes\nshow_original_quote = \"मौलिक उद्धरण दिखाएं\"\nshow_translation = \"अनुवाद दिखाएं\"\nopen_quotation_mark = \"“\"\nclose_quotation_mark = \"”\"\n\n# Translations for stylised Atom feed.\n# https://welpo.github.io/tabi/atom.xml\n# Must contain \"About Feeds\"; it will become a link.\nabout_feeds = \"यह एक वेब फ़ीड है, जिसे एटम फ़ीड भी कहा जाता है। अपने न्यूज़रीडर में एड्रेस बार से URL कॉपी करके सब्सक्राइब करें। About Feeds पर जाकर और अधिक जानकारी प्राप्त करें और शुरू करें। यह सेवा मुफ़्त है।\"\nvisit_the_site = \"वेबसाइट पर जाएं\"\nrecent_posts = \"हाल की पोस्ट्स\"\n"
  },
  {
    "path": "i18n/it.toml",
    "content": "language_name = \"Italiano\"  # Shown in language picker for multi-language sites.\ndate_locale = \"it_IT\"\nfull_stop = \".\"  # Used at the end of a sentence.\n\n# Menu items.\n# Should match the names in config.extra.menu and config.extra.footer_menu.\nblog = \"blog\"\narchive = \"archivio\"\ntags = \"tag\"\nprojects = \"progetti\"\nabout = \"chi sono\"\ncontact = \"contatto\"\nprivacy = \"politica sulla privacy\"\nsite_statistics = \"statistiche del sito\"\nsitemap = \"mappa del sito\"\n\n# Search.\nsearch = \"Cerca\"\nsearch_icon_title = \"Clicca o premi $SHORTCUT per aprire la ricerca\"  # $SHORTCUT will be replaced with the actual keyboard shortcut.\nclear_search = \"Cancella ricerca\"  # Title of the X icon next to search input.\nzero_results = \"Nessun risultato trovato\"\none_results = \"$NUMBER risultato\"\nmany_results = \"$NUMBER risultati\"\n\n# Navigation.\nskip_to_content = \"Salta al contenuto\"\npinned = \"In evidenza\"\njump_to_posts = \"Vai ai post\"\nread_more = \"Leggi di più\"\none_posts = \"$NUMBER post\"\nmany_posts = \"$NUMBER post\"\nprev = \"Precedente\"  # As in \"Previous\" page.\nnext = \"Successivo\"  # As in \"Next\" page.\nof = \"di\"  # E.g. Page 1 \"of\" 3\nall_posts = \"Tutti i post\"\nall_tags = \"Tutti i tag\"\nall_projects = \"Tutti i progetti\"\nfeatured_projects = \"Progetti in evidenza\"\nlanguage_selection = \"Selezione della lingua\"\ntoggle_mode = \"Passa alla modalità $MODE\"  # $MODE will be replaced by a value (or both) below.\ndark = \"scuro\"\nlight = \"chiaro\"\nreset_mode = \"Reimposta la modalità alle impostazioni predefinite\"\n\n# Quick navigation buttons.\ntoggle_toc = \"Attiva/Disattiva indice\"\ngo_to_top = \"Vai in cima alla pagina\"\ngo_to_comments = \"Vai alla sezione commenti\"\n\n# Post metadata.\nby_author = \"Di $AUTHOR\"  # $AUTHOR will be replaced by the author(s).\nauthor_separator = \", \"  # For multiple authors. Ensure spacing where necessary.\nauthor_conjunction = \" e \"  # For multiple authors. Ensure spacing where necessary.\ndraft = \"BOZZA\"\nzero_min_read = \"<1 min di lettura\"\none_min_read = \"$NUMBER min di lettura\"\nmany_min_read = \"$NUMBER min di lettura\"\nzero_words = \"Nessuna parola\"\none_words = \"$NUMBER parola\"\nmany_words = \"$NUMBER parole\"\nlast_updated_on = \"Aggiornato il $DATE\"\nsee_changes = \"Vedi modifiche\"\n\n# Post body.\ntable_of_contents = \"Indice\"\nload_comments = \"Carica commenti\"\n\n# Copy code block button.\ncopied = \"Copiato!\"\ncopy_code_to_clipboard = \"Copia codice negli appunti\"\n\n# iine appreciation button.\nlike_this_post = \"Mi piace questo post\"\n\n# Footer: Powered by Zola and tabi.\npowered_by = \"Alimentato da\"\nand = \"e\"\nsite_source = \"Codice del sito\"\n\n# 404 error.\npage_missing = \"La pagina che hai richiesto sembra non esistere\"\ntranslation_missing = \"o non è ancora stata tradotta nella tua lingua\"\ncheck_url = \"Controlla l'URL per errori o\"\ngo_home = \"torna alla homepage\"\n\n# For multilingual quote shortcode.\n# https://welpo.github.io/tabi/blog/shortcodes/#multilingual-quotes\nshow_original_quote = \"Mostra citazione originale\"\nshow_translation = \"Mostra traduzione\"\nopen_quotation_mark = \"«\"\nclose_quotation_mark = \"»\"\n\n# Translations for stylised Atom feed.\n# https://welpo.github.io/tabi/atom.xml\n# Must contain \"About Feeds\"; it will become a link.\nabout_feeds = \"Questo è un feed web, noto anche come feed Atom. Iscriviti copiando l'URL dalla barra degli indirizzi nel tuo lettore di notizie. Visita About Feeds per saperne di più e iniziare. È gratuito.\"\nvisit_the_site = \"Visita il sito web\"\nrecent_posts = \"Post recenti\"\n"
  },
  {
    "path": "i18n/ja.toml",
    "content": "# This has been machine translated.\n# If you would like to help correct errors or improve the translation,\n# please open an issue or submit a pull request.\n# https://github.com/welpo/tabi\nlanguage_name = \"日本語\"  # Shown in language picker for multi-language sites.\ndate_locale = \"ja_JP\"\nfull_stop = \"。\"  # Used at the end of a sentence.\n\n# Menu items.\n# Should match the names in config.extra.menu and config.extra.footer_menu.\nblog = \"ブログ\"\narchive = \"アーカイブ\"\ntags = \"タグ\"\nprojects = \"プロジェクト\"\nabout = \"自己紹介\"\ncontact = \"お問い合わせ\"\nprivacy = \"プライバシーポリシー\"\nsite_statistics = \"サイト統計\"\nsitemap = \"サイトマップ\"\n\n# Search.\nsearch = \"検索\"\nsearch_icon_title = \"$SHORTCUTを押すか、クリックして検索を開く\"  # $SHORTCUT will be replaced with the actual keyboard shortcut.\nclear_search = \"検索をクリア\"  # Title of the X icon next to search input.\nzero_results = \"結果が見つかりません\"\none_results = \"$NUMBER 結果\"  # \"1 result\"\nmany_results = \"$NUMBER 結果\"  # \"3 results\"\n\n# Navigation.\nskip_to_content = \"コンテンツにスキップ\"\npinned = \"固定\"\njump_to_posts = \"投稿へジャンプ\"\nread_more = \"続きを読む\"\none_posts = \"$NUMBER 投稿\"\nmany_posts = \"$NUMBER 投稿\"\nprev = \"前\"  # As in \"Previous\" page.\nnext = \"次\"  # As in \"Next\" page.\nof = \"中\"  # E.g. Page 1 \"of\" 3\nall_posts = \"すべての投稿\"\nall_tags = \"すべてのタグ\"\nall_projects = \"全てのプロジェクト\"\nfeatured_projects = \"注目のプロジェクト\"\nlanguage_selection = \"言語選択\"\ntoggle_mode = \"$MODE モードに切り替え\"  # $MODE will be replaced by a value (or both) below.\ndark = \"暗い\"\nlight = \"明るい\"\nreset_mode = \"モードをサイトのデフォルトにリセット\"\n\n# Quick navigation buttons.\ntoggle_toc = \"目次を切り替え\"\ngo_to_top = \"ページの先頭へ\"\ngo_to_comments = \"コメントセクションへ\"\n\n# Post metadata.\nby_author = \"$AUTHOR によって\"  # $AUTHOR will be replaced by the author(s).\nauthor_separator = \"、\"  # For multiple authors. Ensure spacing where necessary.\nauthor_conjunction = \"と\"  # For multiple authors. Ensure spacing where necessary.\ndraft = \"ドラフト\"\nzero_min_read = \"1分未満の読了時間\"\none_min_read = \"$NUMBER 分の読了時間\"\nmany_min_read = \"$NUMBER 分の読了時間\"\nzero_words = \"単語数ゼロ\"\none_words = \"$NUMBER 単語\"\nmany_words = \"$NUMBER 単語\"\nlast_updated_on = \"$DATE に更新\"\nsee_changes = \"変更を見る\"\n\n# Post body.\ntable_of_contents = \"目次\"\nload_comments = \"コメントを読む\"\n\n# Copy code block button.\ncopied = \"コピーしました！\"\ncopy_code_to_clipboard = \"コードをクリップボードにコピー\"\n\n# iine appreciation button.\nlike_this_post = \"いいね！\"\n\n# Footer: Powered by Zola and tabi.\npowered_by = \"Powered by\"\nand = \"と\"\nsite_source = \"サイトのソースコード\"\n\n# 404 error.\n# https://welpo.github.io/tabi/404.html\npage_missing = \"お探しのページは存在しません\"\ntranslation_missing = \"または、まだあなたの言語に翻訳されていません\"\ncheck_url = \"URLが正しいか確認してください、または\"\ngo_home = \"ホームページに戻る\"\n\n# For multilingual quote shortcode.\n# https://welpo.github.io/tabi/blog/shortcodes/#multilingual-quotes\nshow_original_quote = \"オリジナルの引用を見る\"\nshow_translation = \"翻訳を見る\"\nopen_quotation_mark = \"「\"\nclose_quotation_mark = \"」\"\n\n# Translations for stylised Atom feed.\n# https://welpo.github.io/tabi/atom.xml\n# Must contain \"About Feeds\"; it will become a link.\nabout_feeds = \"これはウェブフィードです、また、Atomフィードとしても知られています。URLをアドレスバーからニュースリーダーにコピーして登録してください。About Feedsを訪れて、詳細を知り、始めてください。無料です。\"\nvisit_the_site = \"ウェブサイトを訪れる\"\nrecent_posts = \"最近の投稿\"\n"
  },
  {
    "path": "i18n/ko.toml",
    "content": "# This has been machine translated.\n# If you would like to help correct errors or improve the translation,\n# please open an issue or submit a pull request.\n# https://github.com/welpo/tabi\nlanguage_name = \"한국어\"  # Shown in language picker for multi-language sites.\ndate_locale = \"ko_KR\"\nfull_stop = \".\"  # Used at the end of a sentence.\n\n# Menu items.\n# Should match the names in config.extra.menu and config.extra.footer_menu.\nblog = \"블로그\"\narchive = \"아카이브\"\ntags = \"태그\"\nprojects = \"프로젝트\"\nabout = \"소개\"\ncontact = \"연락처\"\nprivacy = \"개인정보 처리방침\"\nsite_statistics = \"사이트 통계\"\nsitemap = \"사이트맵\"\n\n# Search.\nsearch = \"검색\"\nsearch_icon_title = \"클릭하거나 $SHORTCUT을 눌러 검색 열기\"  # $SHORTCUT will be replaced with the actual keyboard shortcut.\nclear_search = \"검색 지우기\"  # Title of the X icon next to search input.\nzero_results = \"결과 없음\"\none_results = \"$NUMBER 결과\"  # \"1 result\"\nmany_results = \"$NUMBER 결과\"  # \"3 results\"\n\n# Navigation.\nskip_to_content = \"콘텐츠로 건너뛰기\"\npinned = \"고정됨\"\njump_to_posts = \"게시물로 이동\"\nread_more = \"더 읽기\"\none_posts = \"$NUMBER 게시물\"\nmany_posts = \"$NUMBER 게시물\"\nprev = \"이전\"  # As in \"Previous\" page.\nnext = \"다음\"  # As in \"Next\" page.\nof = \"중\"  # E.g. Page 1 \"of\" 3\nall_posts = \"모든 게시물\"\nall_tags = \"모든 태그\"\nall_projects = \"모든 프로젝트\"\nfeatured_projects = \"추천 프로젝트\"\nlanguage_selection = \"언어 선택\"\ntoggle_mode = \"$MODE 모드로 전환\"  # $MODE will be replaced by a value (or both) below.\ndark = \"어두운\"\nlight = \"밝은\"\nreset_mode = \"모드를 사이트 기본값으로 재설정\"\n\n# Quick navigation buttons.\ntoggle_toc = \"목차 토글\"\ngo_to_top = \"페이지 상단으로\"\ngo_to_comments = \"댓글 섹션으로\"\n\n# Post metadata.\nby_author = \"$AUTHOR 작성\"  # $AUTHOR will be replaced by the author(s).\nauthor_separator = \"、\"  # For multiple authors. Ensure spacing where necessary.\nauthor_conjunction = \"와\"  # For multiple authors. Ensure spacing where necessary.\ndraft = \"임시 저장\"\nzero_min_read = \"1분 미만 읽기\"\none_min_read = \"$NUMBER 분 읽기\"\nmany_min_read = \"$NUMBER 분 읽기\"\nzero_words = \"단어 없음\"\none_words = \"$NUMBER 단어\"\nmany_words = \"$NUMBER 단어\"\nlast_updated_on = \"$DATE에 업데이트됨\"\nsee_changes = \"변경사항 보기\"\n\n# Post body.\ntable_of_contents = \"목차\"\nload_comments = \"댓글 불러오기\"\n\n# Copy code block button.\ncopied = \"복사됨!\"\ncopy_code_to_clipboard = \"코드를 클립보드에 복사\"\n\n# iine appreciation button.\nlike_this_post = \"이 글이 좋아요\"\n\n# Footer: Powered by Zola and tabi.\npowered_by = \"제공됨\"\nand = \"&\"\nsite_source = \"사이트 소스\"\n\n# 404 error.\n# https://welpo.github.io/tabi/404.html\npage_missing = \"요청하신 페이지가 없습니다\"\ntranslation_missing = \"또는 아직 해당 언어로 번역되지 않았습니다\"\ncheck_url = \"URL을 확인하거나\"\ngo_home = \"홈페이지로 돌아가기\"\n\n# For multilingual quote shortcode.\n# https://welpo.github.io/tabi/blog/shortcodes/#multilingual-quotes\nshow_original_quote = \"원문 인용 보기\"\nshow_translation = \"번역 보기\"\nopen_quotation_mark = \"“\"\nclose_quotation_mark = \"”\"\n\n# Translations for stylised Atom feed.\n# https://welpo.github.io/tabi/atom.xml\n# Must contain \"About Feeds\"; it will become a link.\nabout_feeds = \"이것은 웹 피드이며, Atom 피드로도 알려져 있습니다. 주소 표시 줄의 URL을 뉴스 리더로 복사하여 구독하세요. About Feeds를 방문하여 자세히 알아보고 시작하세요. 무료입니다.\"\nvisit_the_site = \"웹사이트 방문하기\"\nrecent_posts = \"최근 게시물\"\n"
  },
  {
    "path": "i18n/nl.toml",
    "content": "language_name = \"Dutch\"  # Shown in language picker for multi-language sites.\ndate_locale = \"nl_NL\" #nl_BE can also be used for Flemish\nfull_stop = \".\"  # Used at the end of a sentence.\n\n# Menu items.\n# Should match the names in config.extra.menu and config.extra.footer_menu.\nblog = \"blog\"\narchive = \"archief\"\ntags = \"labels\"\nprojects = \"projecten\"\nabout = \"over ons\"\ncontact = \"contact\"\nprivacy = \"privacybeleid\"\nsite_statistics = \"website statistieken\"\nsitemap = \"sitemap\"\n\n# Search.\nsearch = \"Zoeken\"\nsearch_icon_title = \"Klik of druk op $SHORTCUT om de zoekopdracht te openen\"  # $SHORTCUT will be replaced with the actual keyboard shortcut.\nclear_search = \"Verwijder zoekopdracht\"  # Title of the X icon next to search input.\nzero_results = \"Geen resultaten\"\none_results = \"$NUMBER resultaat\"  # \"1 result\"\nmany_results = \"$NUMBER resultaten\"  # \"3 results\"\n\n# Navigation.\nskip_to_content = \"Naar inhoud springen\"\npinned = \"Vastgezet\"\njump_to_posts = \"Naar berichten springen\"\nread_more = \"Lees meer\"\none_posts = \"$NUMBER bericht\" # \"1 post\"\nmany_posts = \"$NUMBER berichten\"  # \"3 posts\"\nprev = \"Vorige\"  # As in \"Previous\" page. - Vorige pagina\nnext = \"Volgende\"  # As in \"Next\" page. - Volgende pagina\nof = \"van\"  # E.g. Page 1 \"of\" 3\nall_posts = \"Alle berichten\"\nall_tags = \"Alle labels\"\nall_projects = \"Alle projecten\"\nfeatured_projects = \"Aanbevolen projecten\"\nlanguage_selection = \"Selecteer taal\"\ntoggle_mode = \"Schakel de $MODE-modus in\"  # $MODE will be replaced by a value (or both) below.\ndark = \"donker\"\nlight = \"licht\"\nreset_mode = \"Reset de modus naar de standaardwaarde\"\n\n# Quick navigation buttons.\ntoggle_toc = \"Schakel inhoudsopgave in of uit\"\ngo_to_top = \"Ga naar de bovenkant van de pagina\"\ngo_to_comments = \"Ga naar opmerkingen\"\n\n# Post metadata.\nby_author = \"Door $AUTHOR\"  # $AUTHOR will be replaced by the author(s).\nauthor_separator = \", \"  # For multiple authors. Ensure spacing where necessary.\nauthor_conjunction = \" en \"  # For multiple authors. Ensure spacing where necessary.\ndraft = \"VOORLOPIGE VERSIE\"\nzero_min_read = \"<1 minuut lezen\"\none_min_read = \"$NUMBER minuut lezen\"\nmany_min_read = \"$NUMBER minuten lezen\"\nzero_words = \"Geen woorden\"\none_words = \"$NUMBER woord\"\nmany_words = \"$NUMBER woorden\"\nlast_updated_on = \"Bijgewerkt op $DATE\"\nsee_changes = \"Zie wijzigingen\"\n\n# Post body.\ntable_of_contents = \"Inhoudsopgave\"\nload_comments = \"Laad opmerkingen\"\n\n# Copy code block button.\ncopied = \"Gekopieerd!\"\ncopy_code_to_clipboard = \"Kopieer code naar klembord\"\n\n# iine appreciation button.\nlike_this_post = \"Vind ik leuk\"\n\n# Footer: Powered by Zola and tabi.\npowered_by = \"Aangedreven door\"\nand = \"&\"\nsite_source = \"Sitebron\"\n\n# 404 error.\n# https://welpo.github.io/tabi/404.html\npage_missing = \"Het lijkt erop dat de door u opgevraagde pagina ontbreekt\"\ntranslation_missing = \"of het is nog niet in jouw taal vertaald\"\ncheck_url = \"Controleer de URL op fouten of\"\ngo_home = \"ga terug naar de startpagina\"\n\n# For multilingual quote shortcode.\n# https://welpo.github.io/tabi/blog/shortcodes/#multilingual-quotes\nshow_original_quote = \"Toon origineel citaat\"\nshow_translation = \"Vertaling tonen\"\nopen_quotation_mark = \"“\"\nclose_quotation_mark = \"”\"\n\n# Translations for stylised Atom feed.\n# https://welpo.github.io/tabi/atom.xml\n# Must contain \"About Feeds\"; it will become a link.\nabout_feeds = \"Dit is een webfeed, ook wel Atom-feed genoemd. Abonneer u door de URL uit de adresbalk naar uw nieuwslezer te kopiëren. Ga naar About Feeds voor meer informatie en om aan de slag te gaan. Het is gratis.\"\nvisit_the_site = \"Bezoek website\"\nrecent_posts = \"Recente berichten\"\n"
  },
  {
    "path": "i18n/or.toml",
    "content": "language_name = \"ଓଡ଼ିଆ\"  # Shown in language picker for multi-language sites.\ndate_locale = \"or_IN\"\nfull_stop = \"।\"  # Used at the end of a sentence.\n\n# Menu items.\n# Should match the names in config.extra.menu and config.extra.footer_menu.\nblog = \"ବ୍ଲଗ\"\narchive = \"ଆର୍କାଇଭ\"\ntags = \"ଟ୍ୟାଗଗୁଡ଼ିକ\"\nprojects = \"ପ୍ରକଳ୍ପଗୁଡ଼ିକ\"\nabout = \"ବାବଦରେ\"\ncontact = \"ଯୋଗାଯୋଗ\"\nprivacy = \"ଗୋପନୀୟତା ନୀତି\"\nsite_statistics = \"ସାଇଟ ପରିସଂଖ୍ୟାନ\"\nsitemap = \"ସାଇଟମ୍ୟାପ\"\n\n# Search.\nsearch = \"ଖୋଜନ୍ତୁ\"\nsearch_icon_title = \"ଖୋଜିବାକୁ $SHORTCUT କୁ ଚିପନ୍ତୁ କିମ୍ବା କ୍ଲିକ କରନ୍ତୁ\"  # $SHORTCUT will be replaced with the actual keyboard shortcut.\nclear_search = \"ଖୋଜା ସଫା କରନ୍ତୁ\"  # Title of the X icon next to search input.\nzero_results = \"କିଛି ମିଳିଲା ନାହିଁ\"\none_results = \"$NUMBER ପରିଣାମ\"  # \"1 result\"\nmany_results = \"$NUMBER ପରିଣାମଗୁଡ଼ିକ\"  # \"3 results\"\n\n# Navigation.\nskip_to_content = \"ବିଷୟବସ୍ତୁକୁ ଯାଆନ୍ତୁ\"\npinned = \"ପିନ୍ କରାଯାଇଛି\"\njump_to_posts = \"ପୋଷ୍ଟକୁ ଯାଆନ୍ତୁ\"\nread_more = \"ଆହୁରି ପଢ଼ନ୍ତୁ\"\none_posts = \"$NUMBER ପୋଷ୍ଟ\"\nmany_posts = \"$NUMBER ପୋଷ୍ଟଗୁଡ଼ିକ\"  # \"3 posts\"\nprev = \"ପୂର୍ବ\"  # As in \"Previous\" page.\nnext = \"ପର\"  # As in \"Next\" page.\nof = \"ର\"  # E.g. Page 1 \"of\" 3\nall_posts = \"ସମସ୍ତ ପୋଷ୍ଟଗୁଡ଼ିକ\"\nall_tags = \"ସମସ୍ତ ଟ୍ୟାଗଗୁଡ଼ିକ\"\nall_projects = \"ସମସ୍ତ ପ୍ରକଳ୍ପଗୁଡ଼ିକ\"\nfeatured_projects = \"ବିଶେଷ ପ୍ରକଳ୍ପଗୁଡ଼ିକ\"\nlanguage_selection = \"ଭାଷା ଚୟନ\"\ntoggle_mode = \"$MODE ଟଗଲ କରନ୍ତୁ\"  # $MODE will be replaced by a value (or both) below.\ndark = \"ଅନ୍ଧାର\"\nlight = \"ଆଲୋକ\"\nreset_mode = \"ସାଇଟର ମୂଳ ମୋଡକୁ ଆଣନ୍ତୁ\"\n\n# Quick navigation buttons.\ntoggle_toc = \"ସୂଚୀପତ୍ର ଟଗଲ କରନ୍ତୁ\"\ngo_to_top = \"ପୃଷ୍ଠାର ଶୀର୍ଷକୁ ଯାଆନ୍ତୁ\"\ngo_to_comments = \"ମତାମତ ବିଭାଗକୁ ଯାଆନ୍ତୁ\"\n\n# Post metadata.\nby_author = \"$AUTHOR ଙ୍କ ଦ୍ୱାରା\"  # $AUTHOR will be replaced by the author(s).\nauthor_separator = \", \"  # For multiple authors. Ensure spacing where necessary.\nauthor_conjunction = \" ଏବଂ \"  # For multiple authors. Ensure spacing where necessary.\ndraft = \"ଡ୍ରାଫ୍ଟ\"\nzero_min_read = \"<୧ ମିନିଟ ପଢ଼ିବା ସମୟ\"\none_min_read = \"$NUMBER ମିନିଟ ପଢ଼ିବା ସମୟ\"\nmany_min_read = \"$NUMBER ମିନିଟ ପଢ଼ିବା ସମୟ\"\nzero_words = \"କୌଣସି ଶବ୍ଦ ନାହିଁ \"\none_words = \"$NUMBER ଶବ୍ଦ\"\nmany_words = \"$NUMBER ଶବ୍ଦଗୁଡ଼ିକ\"\nlast_updated_on = \"$DATE ଦିନ ଅଦ୍ୟତନ ହୋଇଥିଲା\"\nsee_changes = \"ପରିବର୍ତ୍ତନଗୁଡ଼ିକୁ ଦେଖନ୍ତୁ\"\n\n# Post body.\ntable_of_contents = \"ସୂଚୀପତ୍ର\"\nload_comments = \"ମତାମତ ଲୋଡ କରନ୍ତୁ\"\n\n# Copy code block button.\ncopied = \"କପି ହେଲା!\"\ncopy_code_to_clipboard = \"କ୍ଲିପବୋର୍ଡକୁ କପି କରନ୍ତୁ\"\n\n# iine appreciation button.\nlike_this_post = \"ମୋର ଏହି ପୋସ୍ଟ ଭଲ ଲାଗେ\"\n\n# Footer: Powered by Zola and tabi.\npowered_by = \"ଚାଳିତ ଦ୍ୱାରା\"\nand = \"ଏବଂ\"\nsite_source = \"ସାଇଟ ଉତ୍ସ\"\n\n# 404 error.\n# https://welpo.github.io/tabi/404.html\npage_missing = \"ଆପଣ ଯେଉଁ ପୃଷ୍ଠାଟିକୁ ଅନୁରୋଧ କରିଥିଲେ ତାହା ନାହିଁ\"\ntranslation_missing = \"କିମ୍ବା ଆପଣଙ୍କର ଭାଷାକୁ ଏପର୍ଯ୍ୟନ୍ତ ଅନୁବାଦିତ ହୋଇନାହିଁ\"\ncheck_url = \"ତୃଟିଗୁଡ଼ିକ ପାଇଁ URL କୁ ଦେଖନ୍ତୁ କିମ୍ବା\"\ngo_home = \"ମୂଳ ପୃଷ୍ଠାକୁ ଫେରିଯାଆନ୍ତୁ\"\n\n# For multilingual quote shortcode.\n# https://welpo.github.io/tabi/blog/shortcodes/#multilingual-quotes\nshow_original_quote = \"ମୂଳ ଉଦ୍ଧୃତ ଦେଖାନ୍ତୁ\"\nshow_translation = \"ଅନୁବାଦ ଦେଖାନ୍ତୁ\"\nopen_quotation_mark = \"“\"\nclose_quotation_mark = \"”\"\n\n# Translations for stylised Atom feed.\n# https://welpo.github.io/tabi/atom.xml\n# Must contain \"About Feeds\"; it will become a link.\nabout_feeds = \"ଏହା ଏକ ୱେବ୍ ଫିଡ୍, ଯାହାକୁ ଆଟମ୍ ଫିଡ୍ ମଧ୍ୟ କୁହାଯାଏ। ଠିକଣା ବାରରୁ URL କୁ ଆପଣଙ୍କ ନ୍ୟୁଜ୍ ରିଡରରେ କପି କରି ସବ୍ସ୍କ୍ରାଇବ୍ କରନ୍ତୁ। ଅଧିକ ଜାଣିବା ଏବଂ ଆରମ୍ଭ କରିବାକୁ About Feeds ଯାଆନ୍ତୁ। ଏହା ମାଗଣା ଅଟେ ।\"\nvisit_the_site = \"ୱେବସାଇଟକୁ ଯାଆନ୍ତୁ\"\nrecent_posts = \"ସାମ୍ପ୍ରତିକ ପୋଷ୍ଟଗୁଡିକ\"\n"
  },
  {
    "path": "i18n/pt-BR.toml",
    "content": "language_name = \"Português\"  # Shown in language picker for multi-language sites.\ndate_locale = \"pt_BR\"\nfull_stop = \".\"  # Used at the end of a sentence.\n\n# Menu items.\n# Should match the names in config.extra.menu and config.extra.footer_menu.\nblog = \"blog\"\narchive = \"arquivo\"\ntags = \"marcadores\"\nprojects = \"projetos\"\nabout = \"sobre\"\ncontact = \"contato\"\nprivacy = \"política de privacidade\"\nsite_statistics = \"estatísticas do site\"\nsitemap = \"mapa do site\"\n\n# Search.\nsearch = \"Pesquisar\"\nsearch_icon_title = \"Clique ou pressione $SHORTCUT para abrir a pesquisa\"  # $SHORTCUT will be replaced with the actual keyboard shortcut.\nclear_search = \"Limpar pesquisa\"  # Title of the X icon next to search input.\nzero_results = \"Nenhum resultado encontrado\"\none_results = \"$NUMBER resultado\"  # \"1 result\"\nmany_results = \"$NUMBER resultados\"  # \"3 results\"\n\n# Navigation.\nskip_to_content = \"Pular para o conteúdo\"\npinned = \"Fixado\"\njump_to_posts = \"Ir para as publicações\"\nread_more = \"Ler mais\"\none_posts = \"$NUMBER publicação\"\nmany_posts = \"$NUMBER publicações\"\nprev = \"Anterior\"  # As in \"Previous\" page.\nnext = \"Próximo\"  # As in \"Next\" page.\nof = \"de\"  # E.g. Page 1 \"of\" 3\nall_posts = \"Todas as publicações\"\nall_tags = \"Todas os marcadores\"\nall_projects = \"Todos os projetos\"\nfeatured_projects = \"Projetos em destaque\"\nlanguage_selection = \"Seleção de idioma\"\ntoggle_mode = \"Alternar para o modo $MODE\"  # $MODE will be replaced by a value (or both) below.\ndark = \"escuro\"\nlight = \"claro\"\nreset_mode = \"Resetar modo para o padrão\"\n\n# Quick navigation buttons.\ntoggle_toc = \"Alternar Sumário\"\ngo_to_top = \"Ir para o topo da página\"\ngo_to_comments = \"Ir para a seção de comentários\"\n\n# Post metadata.\nby_author = \"Por $AUTHOR\"  # $AUTHOR will be replaced by the author(s).\nauthor_separator = \", \"  # For multiple authors. Ensure spacing where necessary.\nauthor_conjunction = \" e \"  # For multiple authors. Ensure spacing where necessary.\ndraft = \"RASCUNHO\"\nzero_min_read = \"<1 min de leitura\"\none_min_read = \"$NUMBER min de leitura\"\nmany_min_read = \"$NUMBER min de leitura\"\nzero_words = \"Nenhuma palavra\"\none_words = \"$NUMBER palavra\"\nmany_words = \"$NUMBER palavras\"\nlast_updated_on = \"Atualizado em $DATE\"\nsee_changes = \"Ver alterações\"\n\n# Post body.\ntable_of_contents = \"Sumário\"\nload_comments = \"Carregar comentários\"\n\n# Copy code block button.\ncopied = \"Copiado!\"\ncopy_code_to_clipboard = \"Copiar código para a área de transferência\"\n\n# iine appreciation button.\nlike_this_post = \"Curtir esta publicação\"\n\n# Footer: Powered by Zola and tabi.\npowered_by = \"Desenvolvido com\"\nand = \"&\"\nsite_source = \"Código-fonte do site\"\n\n# 404 error.\n# https://welpo.github.io/tabi/404.html\npage_missing = \"A página que solicitou parece não existir\"\ntranslation_missing = \"ou ainda não foi traduzida para o seu idioma\"\ncheck_url = \"Verifique se o endereço está correto ou\"\ngo_home = \"volte para a página inicial\"\n\n# For multilingual quote shortcode.\n# https://welpo.github.io/tabi/blog/shortcodes/#multilingual-quotes\nshow_original_quote = \"Mostrar citação original\"\nshow_translation = \"Mostrar tradução\"\nopen_quotation_mark = \"“\"\nclose_quotation_mark = \"”\"\n\n# Translations for stylised Atom feed.\n# https://welpo.github.io/tabi/atom.xml\n# Must contain \"About Feeds\"; it will become a link.\nabout_feeds = \"Isto é um feed web, também conhecido como feed Atom. Inscreva-se copiando o URL da barra de endereços para o seu leitor de notícias. Visite About Feeds para aprender mais e começar. É grátis.\"\nvisit_the_site = \"Visitar site\"\nrecent_posts = \"Publicações recentes\"\n"
  },
  {
    "path": "i18n/pt-PT.toml",
    "content": "language_name = \"Português\"  # Shown in language picker for multi-language sites.\ndate_locale = \"pt_PT\"\nfull_stop = \".\"  # Used at the end of a sentence.\n\n# Menu items.\n# Should match the names in config.extra.menu and config.extra.footer_menu.\nblog = \"blog\"\narchive = \"arquivo\"\ntags = \"etiquetas\"\nprojects = \"projetos\"\nabout = \"sobre\"\ncontact = \"contacto\"\nprivacy = \"política de privacidade\"\nsite_statistics = \"estatísticas do site\"\nsitemap = \"mapa do site\"\n\n# Search.\nsearch = \"Pesquisar\"\nsearch_icon_title = \"Clique ou pressione $SHORTCUT para abrir a pesquisa\"  # $SHORTCUT will be replaced with the actual keyboard shortcut.\nclear_search = \"Limpar pesquisa\"  # Title of the X icon next to search input.\nzero_results = \"Nenhum resultado encontrado\"\none_results = \"$NUMBER resultado\"  # \"1 result\"\nmany_results = \"$NUMBER resultados\"  # \"3 results\"\n\n# Navigation.\nskip_to_content = \"Pular para o conteúdo\"\npinned = \"Fixado\"\njump_to_posts = \"Ir para as publicações\"\nread_more = \"Ler mais\"\none_posts = \"$NUMBER publicação\"\nmany_posts = \"$NUMBER publicações\"\nprev = \"Anterior\"  # As in \"Previous\" page.\nnext = \"Seguinte\"  # As in \"Next\" page.\nof = \"de\"  # E.g. Page 1 \"of\" 3\nall_posts = \"Todas as publicações\"\nall_tags = \"Todas as etiquetas\"\nall_projects = \"Todos os projetos\"\nfeatured_projects = \"Projetos em destaque\"\nlanguage_selection = \"Seleção de idioma\"\ntoggle_mode = \"Alternar para o modo $MODE\"  # $MODE will be replaced by a value (or both) below.\ndark = \"escuro\"\nlight = \"claro\"\nreset_mode = \"Repor o modo para o padrão\"\n\n# Quick navigation buttons.\ntoggle_toc = \"Alternar Índice\"\ngo_to_top = \"Ir para o topo da página\"\ngo_to_comments = \"Ir para a seção de comentários\"\n\n# Post metadata.\nby_author = \"Por $AUTHOR\"  # $AUTHOR will be replaced by the author(s).\nauthor_separator = \", \"  # For multiple authors. Ensure spacing where necessary.\nauthor_conjunction = \" e \"  # For multiple authors. Ensure spacing where necessary.\ndraft = \"RASCUNHO\"\nzero_min_read = \"<1 min de leitura\"\none_min_read = \"$NUMBER min de leitura\"\nmany_min_read = \"$NUMBER mins de leitura\"\nzero_words = \"Nenhuma palavra\"\none_words = \"$NUMBER palavra\"\nmany_words = \"$NUMBER palavras\"\nlast_updated_on = \"Atualizado em $DATE\"\nsee_changes = \"Ver alterações\"\n\n# Post body.\ntable_of_contents = \"Índice de conteúdo\"\nload_comments = \"Carregar comentários\"\n\n# Copy code block button.\ncopied = \"Copiado!\"\ncopy_code_to_clipboard = \"Copiar código para a área de transferência\"\n\n# iine appreciation button.\nlike_this_post = \"Gosto desta publicação\"\n\n# Footer: Powered by Zola and tabi.\npowered_by = \"Impulsionado por\"\nand = \"e\"\nsite_source = \"Código do site\"\n\n# 404 error.\n# https://welpo.github.io/tabi/404.html\npage_missing = \"A página que solicitou não existe\"\ntranslation_missing = \"ou ainda não foi traduzida para o seu idioma\"\ncheck_url = \"Verifique se o URL está correto ou\"\ngo_home = \"volte à página inicial\"\n\n# For multilingual quote shortcode.\n# https://welpo.github.io/tabi/blog/shortcodes/#multilingual-quotes\nshow_original_quote = \"Mostrar citação original\"\nshow_translation = \"Mostrar tradução\"\nopen_quotation_mark = \"«\"\nclose_quotation_mark = \"»\"\n\n# Translations for stylised Atom feed.\n# https://welpo.github.io/tabi/atom.xml\n# Must contain \"About Feeds\"; it will become a link.\nabout_feeds = \"Isto é um feed web, também conhecido como feed Atom. Subscreva copiando o URL da barra de endereços para o seu leitor de notícias. Visite About Feeds para aprender mais e começar. É grátis.\"\nvisit_the_site = \"Visite o website\"\nrecent_posts = \"Publicações recentes\"\n"
  },
  {
    "path": "i18n/ru.toml",
    "content": "language_name = \"Русский\"  # Shown in language picker for multi-language sites.\ndate_locale = \"ru_RU\"\nfull-stop = \".\"  # Used at the end of a sentence.\n\n# Note on pluralization prefixes:\n# - few_: for numbers ending in 2-4, except 12-14, in genitive singular.\n# - many_: for all others, including 5-9, 0, and teens (11-14), in genitive plural.\n\n# Menu items.\n# Should match the names in config.extra.menu and config.extra.footer_menu.\nblog = \"блог\"\narchive = \"архив\"\ntags = \"теги\"\nprojects = \"проекты\"\nabout = \"обо мне\"\ncontact = \"контакт\"\nprivacy = \"политика конфиденциальности\"\nsite_statistics = \"статистика сайта\"\nsitemap = \"карта сайта\"\n\n# Search.\nsearch = \"Поиск\"\nsearch_icon_title = \"Нажмите или используйте $SHORTCUT для открытия поиска\"  # $SHORTCUT will be replaced with the actual keyboard shortcut.\nclear_search = \"Очистить поиск\"  # Title of the X icon next to search input.\nzero_results = \"Нет результатов\"\none_results = \"$NUMBER результат\"\nfew_results = \"$NUMBER результата\"  # 2, 3, 4 but not 12-14\nmany_results = \"$NUMBER результатов\"  # 5-9, 0, 11-14, and others\n\n# Navigation.\nskip_to_content = \"Перейти к содержанию\"\npinned = \"Закреплено\"\njump_to_posts = \"Перейти к записям\"\nread_more = \"Читать далее\"\npost = \"пост\"\none_posts = \"$NUMBER пост\"\nfew_posts = \"$NUMBER поста\"  # 2, 3, 4 but not 12-14\nmany_posts = \"$NUMBER постов\"  # 5-9, 0, 11-14, and others\nprev = \"Пред.\"  # As in \"Previous\" page.\nnext = \"След.\"  # As in \"Next\" page.\nof = \"из\"  # E.g. Page 1 \"of\" 3\nall_posts = \"Все посты\"\nall_tags = \"Все теги\"\nall_projects = \"Все проекты\"\nfeatured_projects = \"Избранные проекты\"\nlanguage_selection = \"Выбор языка\"\ntoggle_mode = \"Переключить на режим $MODE\"  # $MODE will be replaced by a value (or both) below.\ndark = \"тёмный\"\nlight = \"светлый\"\nreset_mode = \"Сбросить режим к стандартным настройкам сайта\"\n\n# Quick navigation buttons.\ntoggle_toc = \"Показать/Скрыть оглавление\"\ngo_to_top = \"Перейти в начало страницы\"\ngo_to_comments = \"Перейти к комментариям\"\n\n# Post metadata.\nby_author = \"От $AUTHOR\"  # $AUTHOR will be replaced by the author(s).\nauthor_separator = \", \"  # For multiple authors. Ensure spacing where necessary.\nauthor_conjunction = \" и \"  # For multiple authors. Ensure spacing where necessary.\ndraft = \"ЧЕРНОВИК\"\nzero_min_read = \"<1 мин чтения\"\none_min_read = \"$NUMBER мин чтения\"\nfew_min_read = \"$NUMBER мин чтения\"  # 2, 3, 4 but not 12-14\nmany_min_read = \"$NUMBER мин чтения\"  # 5-9, 0, 11-14, and others\nzero_words = \"Нет слов\"\none_words = \"$NUMBER слово\"\nfew_words = \"$NUMBER слова\"  # 2, 3, 4 but not 12-14\nmany_words = \"$NUMBER слов\"  # 5-9, 0, 11-14, and others\nlast_updated_on = \"Обновлено $DATE\"\nsee_changes = \"Смотреть изменения\"\n\n# Post body.\ntable_of_contents = \"Содержание\"\nload_comments = \"Загрузить комментарии\"\n\n# Copy code block button.\ncopied = \"Скопировано!\"\ncopy_code_to_clipboard = \"Скопировать код в буфер обмена\"\n\n# iine appreciation button.\nlike_this_post = \"Мне нравится эта статья\"\n\n# Footer: Powered by Zola and tabi.\npowered_by = \"Под управлением\"\nand = \"&\"\nsite_source = \"Исходный код сайта\"\n\n# 404 error.\n# https://welpo.github.io/tabi/404.html\npage_missing = \"Запрашиваемая вами страница отсутствует\"\ntranslation_missing = \"или еще не переведена на ваш язык\"\ncheck_url = \"Проверьте URL на ошибки или\"\ngo_home = \"вернитесь на главную страницу\"\n\n# For multilingual quote shortcode.\n# https://welpo.github.io/tabi/blog/shortcodes/#multilingual-quotes\nshow_original_quote = \"Показать оригинальную цитату\"\nshow_translation = \"Показать перевод\"\nopen_quotation_mark = \"«\"\nclose_quotation_mark = \"»\"\n\n# Translations for stylised Atom feed.\n# https://welpo.github.io/tabi/atom.xml\n# Must contain \"About Feeds\"; it will become a link.\nabout_feeds = \"Это веб-фид, известный также как Atom-фид. Чтобы подписаться, скопируйте URL из адресной строки в ваш RSS-ридер. Посетите About Feeds для дополнительной информации и начала использования.\"\nvisit_the_site = \"Посетить веб-сайт\"\nrecent_posts = \"Последние посты\"\n"
  },
  {
    "path": "i18n/uk.toml",
    "content": "# This has been machine translated.\n# If you would like to help correct errors or improve the translation,\n# please open an issue or submit a pull request.\n# https://github.com/welpo/tabi\n\n# Note on pluralization prefixes:\n# - few_: for numbers ending in 2-4, except 12-14, in genitive singular.\n# - many_: for all others, including 5-9, 0, and teens (11-14), in genitive plural.\n\nlanguage_name = \"Українська\"  # Shown in language picker for multi-language sites.\ndate_locale = \"uk_UA\"\nfull-stop = \".\"  # Used at the end of a sentence.\n\n# Menu items.\n# Should match the names in config.extra.menu and config.extra.footer_menu.\nblog = \"блог\"\narchive = \"архів\"\ntags = \"теги\"\nprojects = \"проєкти\"\nabout = \"про мене\"\ncontact = \"контакт\"\nprivacy = \"політика конфіденційності\"\nsite_statistics = \"статистика сайту\"\nsitemap = \"карта сайту\"\n\n# Search.\nsearch = \"Пошук\"\nsearch_icon_title = \"Натисніть або використовуйте $SHORTCUT, щоб відкрити пошук\"  # $SHORTCUT will be replaced with the actual keyboard shortcut.\nclear_search = \"Очистити пошук\"  # Title of the X icon next to search input.\nzero_results = \"Немає результатів\"\none_results = \"$NUMBER результат\"\nfew_results = \"$NUMBER результати\"\nmany_results = \"$NUMBER результатів\"\n\n# Navigation.\nskip_to_content = \"Перейти до вмісту\"\npinned = \"Закріплено\"\njump_to_posts = \"Перейти до дописів\"\nread_more = \"Читати далі\"\none_posts = \"$NUMBER пост\"\nfew_posts = \"$NUMBER пости\"  # 2, 3, 4 but not 12-14\nmany_posts = \"$NUMBER постів\"  # 5-9, 0, 11-14, and others\nprev = \"Попер.\"  # As in \"Previous\" page.\nnext = \"Наст.\"  # As in \"Next\" page.\nof = \"з\"  # E.g. Page 1 \"of\" 3\nall_posts = \"Всі пости\"\nall_tags = \"Всі теги\"\nall_projects = \"Усі проекти\"\nfeatured_projects = \"Обрані проекти\"\nlanguage_selection = \"Вибір мови\"\ntoggle_mode = \"Перемкнути в режим $MODE\"  # $MODE will be replaced by a value (or both) below.\ndark = \"темний\"\nlight = \"світлий\"\nreset_mode = \"Скинути режим до типових налаштувань сайту\"\n\n# Quick navigation buttons.\ntoggle_toc = \"Показати/Сховати зміст\"\ngo_to_top = \"Перейти на початок сторінки\"\ngo_to_comments = \"Перейти до коментарів\"\n\n# Post metadata.\nby_author = \"Від $AUTHOR\"  # $AUTHOR will be replaced by the author(s).\nauthor_separator = \", \"  # For multiple authors. Ensure spacing where necessary.\nauthor_conjunction = \" і \"  # For multiple authors. Ensure spacing where necessary.\ndraft = \"ЧЕРНЕТКА\"\nzero_min_read = \"<1 хв. читання\"\none_min_read = \"$NUMBER хв. читання\"\nfew_min_read = \"$NUMBER хв. читання\"  # 2, 3, 4 but not 12-14\nmany_min_read = \"$NUMBER хв. читання\"  # 5-9, 0, 11-14, and others\nzero_words = \"Без слів\"\none_words = \"$NUMBER слово\"\nfew_words = \"$NUMBER слова\"  # 2, 3, 4 but not 12-14\nmany_words = \"$NUMBER слів\"  # 5-9, 0, 11-14, and others\nlast_updated_on = \"Оновлено $DATE\"\nsee_changes = \"Переглянути зміни\"\n\n# Post body.\ntable_of_contents = \"Зміст\"\nload_comments = \"Завантажити коментарі\"\n\n# Copy code block button.\ncopied = \"Скопійовано!\"\ncopy_code_to_clipboard = \"Копіювати код у буфер обміну\"\n\n# iine appreciation button.\nlike_this_post = \"Мені подобається ця стаття\"\n\n# Footer: Powered by Zola and tabi.\npowered_by = \"Під управлінням\"\nand = \"та\"\nsite_source = \"Вихідний код сайту\"\n\n# 404 error.\n# https://welpo.github.io/tabi/404.html\npage_missing = \"Сторінка, яку ви запитуєте, відсутня\"\ntranslation_missing = \"або ще не перекладена на вашу мову\"\ncheck_url = \"Перевірте URL на помилки або\"\ngo_home = \"поверніться на головну сторінку\"\n\n# For multilingual quote shortcode.\n# https://welpo.github.io/tabi/blog/shortcodes/#multilingual-quotes\nshow_original_quote = \"Показати оригінальну цитату\"\nshow_translation = \"Показати переклад\"\nopen_quotation_mark = \"«\"\nclose_quotation_mark = \"»\"\n\n# Translations for stylised Atom feed.\n# https://welpo.github.io/tabi/atom.xml\n# Must contain \"About Feeds\"; it will become a link.\nabout_feeds = \"Це веб-канал, також відомий як Atom-канал. Щоб підписатися, скопіюйте URL з адресного рядка у ваш RSS-читач. Відвідайте About Feeds для додаткової інформації та початку використання.\"\nvisit_the_site = \"Відвідати веб-сайт\"\nrecent_posts = \"Останні пости\"\n"
  },
  {
    "path": "i18n/zh-Hans.toml",
    "content": "language_name = \"简体中文\"  # Shown in language picker for multi-language sites.\ndate_locale = \"zh_CN\"\nfull_stop = \"。\"  # Used at the end of a sentence.\n\n# Menu items.\n# Should match the names in config.extra.menu and config.extra.footer_menu.\nblog = \"博客\"\narchive = \"归档\"\ntags = \"标签\"\nprojects = \"项目\"\nabout = \"关于\"\ncontact = \"联系方式\"  # Machine translated.\nprivacy = \"隐私政策\"  # Machine translated.\nsite_statistics = \"网站统计\"  # Machine translated.\nsitemap = \"站点地图\"  # Machine translated.\n\n# Search.\nsearch = \"搜索\"\nsearch_icon_title = \"点击或按 $SHORTCUT 开启搜索\"  # $SHORTCUT will be replaced with the actual keyboard shortcut.\nclear_search = \"清除搜索\"  # Title of the X icon next to search input.\nzero_results = \"没有找到结果\"\none_results = \"$NUMBER 个结果\"\nmany_results = \"$NUMBER 个结果\"\n\n# Navigation.\nskip_to_content = \"跳到内容\"\npinned = \"置顶\"\njump_to_posts = \"跳转到文章\"\nread_more = \"阅读全文\"\none_posts = \"$NUMBER 篇文章\"\nmany_posts = \"$NUMBER 篇文章\"\nprev = \"上一页\"  # As in \"Previous\" page.\nnext = \"下一页\"  # As in \"Next\" page.\nof = \"/\"  # E.g. Page 1 \"of\" 3\nall_posts = \"所有文章\"\nall_tags = \"所有标签\"\nall_projects = \"所有项目\"  # Machine translated.\nfeatured_projects = \"特色项目\"  # Machine translated.\nlanguage_selection = \"语言选择\"  # Machine translated.\ntoggle_mode = \"切换到$MODE模式\"  # $MODE will be replaced by a value (or both) below. Machine translated.\ndark = \"暗\"  # Machine translated.\nlight = \"亮\"  # Machine translated.\nreset_mode = \"将模式重置为网站默认值\"  # Machine translated.\n\n# Quick navigation buttons.\ntoggle_toc = \"切换目录\"  # Machine translated.\ngo_to_top = \"返回页面顶部\"  # Machine translated.\ngo_to_comments = \"转到评论区\"  # Machine translated.\n\n# Post metadata.\nby_author = \"由$AUTHOR\"  # $AUTHOR will be replaced by the author(s).\nauthor_separator = \"、\"  # For multiple authors. Ensure spacing where necessary.\nauthor_conjunction = \"和\"  # For multiple authors. Ensure spacing where necessary.\ndraft = \"草稿\"\nzero_min_read = \"少于1分钟阅读\"\none_min_read = \"$NUMBER 分钟阅读\"\nmany_min_read = \"$NUMBER 分钟阅读\"\nzero_words = \"没有字\"\none_words = \"$NUMBER 字\"\nmany_words = \"$NUMBER 字\"\nlast_updated_on = \"更新于 $DATE\"\nsee_changes = \"修改纪录\"\n\n# Post body.\ntable_of_contents = \"目录\"\nload_comments = \"载入留言\"\n\n# Copy code block button.\ncopied = \"已复制!\"  # Machine translated.\ncopy_code_to_clipboard = \"复制代码到剪贴板\"  # Machine translated.\n\n# iine appreciation button.\nlike_this_post = \"喜欢这篇文章\"\n\n# Footer: Powered by Zola and tabi.\npowered_by = \"网站基于\"\nand = \"和\"\nsite_source = \"查看原始码\"\n\n# 404 error.\n# https://welpo.github.io/tabi/404.html\npage_missing = \"您请求的页面似乎不存在\"\ntranslation_missing = \"或尚未翻译成您的语言\"\ncheck_url = \"请检查网址是否有误或\"\ngo_home = \"返回首页\"\n\n# For multilingual quote shortcode.\n# https://welpo.github.io/tabi/blog/shortcodes/#multilingual-quotes\nshow_original_quote = \"显示原文\"\nshow_translation = \"显示译文\"\nopen_quotation_mark = \"「\"\nclose_quotation_mark = \"」\"\n\n# Translations for stylised Atom feed.\n# https://welpo.github.io/tabi/atom.xml\n# Must contain \"About Feeds\"; it will become a link.\nabout_feeds = \"这是Web Feed，又称为Atom Feed，把现在的网址复制到新闻阅读器即可订阅本站文章。造访「About Feeds」来了解更多资讯。\"\nvisit_the_site = \"造访网站\"\nrecent_posts = \"近期文章\"\n"
  },
  {
    "path": "i18n/zh-Hant.toml",
    "content": "language_name = \"繁體中文\"  # Shown in language picker for multi-language sites.\ndate_locale = \"zh_TW\"\nfull_stop = \"。\"  # Used at the end of a sentence.\n\n# Menu items.\n# Should match the names in config.extra.menu and config.extra.footer_menu.\nblog = \"網誌\"\narchive = \"歸檔\"\ntags = \"標籤\"\nprojects = \"專案\"\nabout = \"關於\"\ncontact = \"聯絡方式\"  # Machine translated.\nprivacy = \"隱私政策\"  # Machine translated.\nsite_statistics = \"網站統計\"  # Machine translated.\nsitemap = \"網站地圖\"  # Machine translated.\n\n# Search.\nsearch = \"搜尋\"\nsearch_icon_title = \"點擊或按 $SHORTCUT 來開啟搜尋\"  # $SHORTCUT will be replaced with the actual keyboard shortcut.\nclear_search = \"清除搜尋\"  # Title of the X icon next to search input.\nzero_results = \"沒有找到結果\"\none_results = \"$NUMBER 個結果\"\nmany_results = \"$NUMBER 個結果\"\n\n# Navigation.\nskip_to_content = \"跳到內容\"\npinned = \"釘選\"\njump_to_posts = \"跳轉到文章\"\nread_more = \"閱讀全文\"\none_posts = \"$NUMBER 篇文章\"\nmany_posts = \"$NUMBER 篇文章\"\nprev = \"上一頁\"  # As in \"Previous\" page.\nnext = \"下一頁\"  # As in \"Next\" page.\nof = \"/\"  # E.g. Page 1 \"of\" 3\nall_posts = \"所有文章\"\nall_tags = \"所有標籤\"\nall_projects = \"所有項目\"  # Machine translated.\nfeatured_projects = \"精選項目\"  # Machine translated.\nlanguage_selection = \"語言選擇\"  # Machine translated.\ntoggle_mode = \"切換到$MODE模式\"  # $MODE will be replaced by a value (or both) below. Machine translated.\ndark = \"暗\"  # Machine translated.\nlight = \"亮\"  # Machine translated.\nreset_mode = \"將模式重置為網站默認值\"  # Machine translated.\n\n# Quick navigation buttons.\ntoggle_toc = \"切換目錄\"  # Machine translated.\ngo_to_top = \"返回頁面頂部\"  # Machine translated.\ngo_to_comments = \"轉到評論區\"  # Machine translated.\n\n# Post metadata.\nby_author = \"由$AUTHOR\"  # $AUTHOR will be replaced by the author(s).\nauthor_separator = \"、\"  # For multiple authors. Ensure spacing where necessary.\nauthor_conjunction = \"和\"  # For multiple authors. Ensure spacing where necessary.\ndraft = \"草稿\"\nzero_min_read = \"少於1分鐘閱讀\"\none_min_read = \"$NUMBER 分鐘閱讀\"\nmany_min_read = \"$NUMBER 分鐘閱讀\"\nzero_words = \"沒有字\"\none_words = \"$NUMBER 字\"\nmany_words = \"$NUMBER 字\"\nlast_updated_on = \"更新於 $DATE\"\nsee_changes = \"修改紀錄\"\n\n# Post body.\ntable_of_contents = \"目錄\"\nload_comments = \"載入留言\"\n\n# Copy code block button.\ncopied = \"已复制!\"  # Machine translated.\ncopy_code_to_clipboard = \"复制代码到剪贴板\"  # Machine translated.\n\n# iine appreciation button.\nlike_this_post = \"喜歡這篇文章\"\n\n# Footer: Powered by Zola and tabi.\npowered_by = \"網站基於\"\nand = \"和\"\nsite_source = \"查看原始碼\"\n\n# 404 error.\n# https://welpo.github.io/tabi/404.html\npage_missing = \"您請求的頁面不存在\"\ntranslation_missing = \"或尚未被翻譯成繁體中文\"\ncheck_url = \"請檢查網址是否有誤或\"\ngo_home = \"返回首頁\"\n\n# For multilingual quote shortcode.\n# https://welpo.github.io/tabi/blog/shortcodes/#multilingual-quotes\nshow_original_quote = \"顯示原文\"\nshow_translation = \"顯示翻譯\"\nopen_quotation_mark = \"「\"\nclose_quotation_mark = \"」\"\n\n# Translations for stylised Atom feed.\n# https://welpo.github.io/tabi/atom.xml\n# Must contain \"About Feeds\"; it will become a link.\nabout_feeds = \"這是Web Feed，又稱為Atom Feed，把現在的網址複製到新聞閱讀器即可訂閱本站文章。造訪「About Feeds」來了解更多資訊。\"\nvisit_the_site = \"造訪網站\"\nrecent_posts = \"近期文章\"\n"
  },
  {
    "path": "sass/main.scss",
    "content": "@use 'parts/_admonitions.scss';\n@use 'parts/_archive.scss';\n@use 'parts/_aside.scss';\n@use 'parts/_cards.scss';\n@use 'parts/_code.scss';\n@use 'parts/_comments.scss';\n@use 'parts/_footer.scss';\n@use 'parts/_header-anchor.scss';\n@use 'parts/_header.scss';\n@use 'parts/_home-banner.scss';\n@use 'parts/_iine.scss';\n@use 'parts/_image-hover.scss';\n@use 'parts/_image-toggler.scss';\n@use 'parts/_image.scss';\n@use 'parts/_misc.scss';\n@use 'parts/_multilingual_quote.scss';\n@use 'parts/_pagination.scss';\n@use 'parts/_posts_list.scss';\n@use 'parts/_quick_navigation_buttons.scss';\n@use 'parts/_search.scss';\n@use 'parts/_spoiler.scss';\n@use 'parts/_syntax_theme.scss';\n@use 'parts/_table.scss';\n@use 'parts/_tags.scss';\n@use 'parts/_theme-switch.scss';\n@use 'parts/_webmention.scss';\n@use 'parts/_zola-error.scss';\n\n@font-face {\n    src: local('Inter'),\n        url('fonts/Inter4.woff2') format(\"woff2\");\n    /* Copyright 2016 The Inter Project Authors (https://github.com/rsms/inter). Licensed under the SIL Open Font License, Version 1.1. More information available at: http://scripts.sil.org/OFL */\n    font-family: 'Inter';\n    font-display: swap;\n}\n\n@font-face {\n    src: local('Source Serif'),\n        url('fonts/SourceSerif4Variable-Roman.ttf.woff2') format(\"woff2\");\n    /* Copyright 2014 - 2023 Adobe (http://www.adobe.com/), with Reserved Font Name ‘Source’.adobe.com/). Licensed under the SIL Open Font License, Version 1.1. More information available at: http://scripts.sil.org/OFL */\n    font-family: 'Source Serif';\n    font-display: swap;\n}\n\n@font-face {\n    src: local('Cascadia Code'),\n        url('fonts/CascadiaCode-SemiLight.woff2') format(\"woff2\");\n    /* Copyright 2019 - Present, Microsoft Corporation, with Reserved Font Name 'Cascadia Code'. Licensed under the SIL Open Font License, Version 1.1. More information available at: http://scripts.sil.org/OFL */\n    font-family: 'Cascadia Code';\n    font-display: swap;\n}\n\n@mixin theme-variables($theme) {\n    @if $theme =='light' {\n        --background-color: white;\n        --bg-0: #f0f0f0;\n        --bg-1: #e7e7e7;\n        --bg-2: #fefefe;\n        --bg-3: #d8dcdd;\n        --hover-color: white;\n        --primary-color: #087E96;\n        --divider-color: #d7d7d7;\n        --text-color: #222226;\n        --text-color-high-contrast: #313333;\n        --meta-color: #5b5b65;\n        --codeblock-bg: #26232e;\n        --codeblock-highlight: #383444;\n        --theme-switcher-svg: url(\"data:image/svg+xml,%3Csvg xmlns='http://www.w3.org/2000/svg' viewBox='0 0 512 512'%3E%3Cpath d='M283.211 512c78.962 0 151.079-35.925 198.857-94.792 7.068-8.708-.639-21.43-11.562-19.35-124.203 23.654-238.262-71.576-238.262-196.954 0-72.222 38.662-138.635 101.498-174.394 9.686-5.512 7.25-20.197-3.756-22.23A258.156 258.156 0 0 0 283.211 0c-141.309 0-256 114.511-256 256 0 141.309 114.511 256 256 256z'/%3E%3C/svg%3E%0A\");\n        color-scheme: light;\n    }\n\n    @else {\n        --background-color: #1f1f1f;\n        --bg-0: #2f2f2f;\n        --bg-1: #3c3c3c;\n        --bg-2: #171717;\n        --bg-3: #535555;\n        --hover-color: black;\n        --primary-color: #91e0ee;\n        --divider-color: #4a4a4a;\n        --text-color: #D4D4D4;\n        --text-color-high-contrast: #eceeef;\n        --meta-color: #B0B0B0;\n        --codeblock-bg: #19181e;\n        --codeblock-highlight: #282834;\n        --theme-switcher-svg: url(\"data:image/svg+xml,%3Csvg xmlns='http://www.w3.org/2000/svg' viewBox='0 96 960 960' %3E%3Cpath d='M480 776q-83 0-141.5-58.5T280 576q0-83 58.5-141.5T480 376q83 0 141.5 58.5T680 576q0 83-58.5 141.5T480 776ZM80 616q-17 0-28.5-11.5T40 576q0-17 11.5-28.5T80 536h80q17 0 28.5 11.5T200 576q0 17-11.5 28.5T160 616H80Zm720 0q-17 0-28.5-11.5T760 576q0-17 11.5-28.5T800 536h80q17 0 28.5 11.5T920 576q0 17-11.5 28.5T880 616h-80ZM480 296q-17 0-28.5-11.5T440 256v-80q0-17 11.5-28.5T480 136q17 0 28.5 11.5T520 176v80q0 17-11.5 28.5T480 296Zm0 720q-17 0-28.5-11.5T440 976v-80q0-17 11.5-28.5T480 856q17 0 28.5 11.5T520 896v80q0 17-11.5 28.5T480 1016ZM226 378l-43-42q-12-11-11.5-28t11.5-29q12-12 29-12t28 12l42 43q11 12 11 28t-11 28q-11 12-27.5 11.5T226 378Zm494 495-42-43q-11-12-11-28.5t11-27.5q11-12 27.5-11.5T734 774l43 42q12 11 11.5 28T777 873q-12 12-29 12t-28-12Zm-42-495q-12-11-11.5-27.5T678 322l42-43q11-12 28-11.5t29 11.5q12 12 12 29t-12 28l-43 42q-12 11-28 11t-28-11ZM183 873q-12-12-12-29t12-28l43-42q12-11 28.5-11t27.5 11q12 11 11.5 27.5T282 830l-42 43q-11 12-28 11.5T183 873Z'/%3E%3C/svg%3E\");\n        color-scheme: dark;\n    }\n}\n\n:root {\n    @include theme-variables('light');\n\n    --max-layout-width: 1000px;\n    --normal-layout-width: 600px;\n    --medium-layout-width: 400px;\n    --small-layout-width: 200px;\n    --paragraph-spacing: max(2.3vmin, 24px);\n\n    --sans-serif-font: 'Inter', Helvetica, Arial, sans-serif;\n    --serif-font: 'Source Serif', 'Georgia', serif;\n    --code-font: 'Cascadia Code';\n\n    scrollbar-color: var(--primary-color) transparent;\n    accent-color: var(--primary-color);\n\n    line-height: 190%;\n    font-family: var(--sans-serif-font);\n}\n\n[data-theme='dark'] {\n    @include theme-variables('dark');\n\n    .invertible-image {\n        filter: invert(.88);\n    }\n\n    .dimmable-image {\n        filter: brightness(.8) contrast(1.2);\n    }\n}\n\n@media (prefers-color-scheme: dark) {\n    :root:not([data-theme='light']) {\n        @include theme-variables('dark');\n\n        .invertible-image {\n            filter: invert(.88);\n        }\n\n        .dimmable-image {\n            filter: brightness(.8) contrast(1.2);\n        }\n    }\n}\n\nhtml {\n    background-color: var(--background-color);\n    color: var(--text-color);\n    line-height: 1.6em;\n    text-rendering: optimizeLegibility;\n}\n\nbody {\n    display: flex;\n    flex-direction: column;\n    margin-inline: 5vmin;\n    margin-block: 0;\n    min-height: 100vh;\n}\n\n.content {\n    word-wrap: break-word;\n    margin: 0 auto;\n    margin-top: 6vmin;\n    margin-block-end: 4rem;\n    width: 100%;\n    max-width: var(--max-layout-width);\n}\n\n.use-sans-serif {\n    --serif-font: var(--sans-serif-font);\n}\n\narticle {\n    $base-margin: 6rem;\n    position: relative;\n    margin: 0 auto;\n\n    max-width: calc(var(--max-layout-width) - 2*$base-margin);\n\n    p,\n    li,\n    details,\n    summary {\n        font-family: var(--serif-font);\n    }\n\n    strong {\n        font-weight: 620;\n    }\n\n    .full-width {\n        margin-inline-start: -$base-margin;\n        margin-inline-end: -$base-margin;\n        max-width: calc(100% + 2*$base-margin);\n    }\n\n    li {\n        p:not(:last-child) {\n            margin-block-end: 0;\n        }\n\n        p + :last-child {\n            margin-block-end: var(--paragraph-spacing);\n        }\n    }\n}\n\n.section-title {\n    display: flex;\n    margin: 0;\n    margin-top: -0.15em;\n    color: var(--text-color-high-contrast);\n    font-weight: 550;\n    font-size: 2.2em;\n    line-height: 1.2em;\n}\n\n.last-updated {\n    margin-top: -5vmin;\n}\n\nh1,\nh2,\nh3,\nh4,\nh5,\nh6 {\n    display: block;\n    position: relative;\n    margin: 0;\n}\n\nh1 {\n    margin-top: 0.67em;\n    font-weight: 550;\n    font-size: 1.62rem;\n}\n\nh2 {\n    margin-top: 0.5em;\n    font-weight: 550;\n    font-size: 1.4rem;\n}\n\nh3 {\n    margin-top: 0.3em;\n    font-weight: 550;\n    font-size: 1.2rem;\n}\n\nh4 {\n    margin-top: 0.83em;\n    font-weight: 550;\n    font-size: 1rem;\n}\n\nh5 {\n    margin-top: 0.83em;\n    font-weight: normal;\n    font-size: 1rem;\n}\n\np {\n    margin-top: 0.4rem;\n    margin-block-end: var(--paragraph-spacing);\n    font-size: 1em;\n    line-height: 2rem;\n}\n\nstrong {\n    font-weight: 580;\n}\n\n.centered-text {\n    text-align: center;\n}\n\nvideo {\n    max-width: 100%;\n}\n\n.center-content {\n    display: flex;\n    flex-direction: column;\n    justify-content: center;\n    align-items: center;\n    margin: 0;\n    width: 100%;\n    height: 100vh;\n    text-align: center;\n}\n\n.subheader {\n    margin-block-end: 2rem;\n}\n\n.mobile-only {\n    display: none;\n}\n\n@media only screen and (max-width: 1000px) {\n    .content {\n        max-width: var(--normal-layout-width);\n    }\n\n    body {\n        margin: 0 32px;\n    }\n\n    article .full-width {\n        display: block;\n        margin-inline-start: 0;\n        margin-inline-end: 0;\n        max-width: none;\n        overflow-x: auto;\n    }\n\n    .mobile-only {\n        display: block;\n    }\n}\n\n@media only screen and (max-width: 600px) {\n    .content {\n        margin-top: 0rem;\n        max-width: var(--medium-layout-width);\n    }\n\n    article {\n        margin-top: 1.3rem;\n    }\n\n    body {\n        margin-inline: 16px;\n        margin-block: 0;\n    }\n}\n\n@media only screen and (max-width: 300px) {\n    .content {\n        max-width: var(--small-layout-width);\n    }\n}\n\n@media all and (min-width: 600px) {\n    html {\n        font-size: 16.5px;\n    }\n}\n\n@media all and (min-width: 960px) {\n    html {\n        font-size: 20px;\n    }\n}\n"
  },
  {
    "path": "sass/parts/_admonitions.scss",
    "content": "@mixin admonition-type($type) {\n    border-color: var(--admonition-#{$type}-border);\n    background-color: var(--admonition-#{$type}-bg);\n\n    > .admonition-content > p > code {\n        background-color: var(--admonition-#{$type}-code);\n    }\n\n    a {\n        border-bottom: 1px solid var(--admonition-#{$type}-border);\n        color: var(--admonition-#{$type}-border);\n\n        &:hover {\n            background-color: var(--admonition-#{$type}-border);\n            color: var(--hover-color);\n        }\n    }\n\n    .admonition-icon {\n        background-color: var(--admonition-#{$type}-border);\n    }\n}\n\n:root {\n    /* Note */\n    --admonition-note-border: #5b6167;\n    --admonition-note-bg: #f2f4f7;\n    --admonition-note-code: #e1e3ed;\n\n    /* Tip */\n    --admonition-tip-border: #03970f;\n    --admonition-tip-bg: #f0fdf0;\n    --admonition-tip-code: #d3edc5;\n\n    /* Info */\n    --admonition-info-border: #15a2b2;\n    --admonition-info-bg: #f5fbff;\n    --admonition-info-code: #d5e2f2;\n\n    /* Warning */\n    --admonition-warning-border: #eea719;\n    --admonition-warning-bg: #fff8e6;\n    --admonition-warning-code: #feee96;\n\n    /* Danger */\n    --admonition-danger-border: #d8292e;\n    --admonition-danger-bg: #ffebed;\n    --admonition-danger-code: #fcc1c5;\n}\n\n@mixin dark-theme-variables {\n    /* Note */\n    --admonition-note-border: #d0d1d4;\n    --admonition-note-bg: #3d3e40;\n    --admonition-note-code: #495057;\n\n    /* Tip */\n    --admonition-tip-border: #01b010;\n    --admonition-tip-bg: #013100;\n    --admonition-tip-code: #005f00;\n\n    /* Info */\n    --admonition-info-border: #50a9d5;\n    --admonition-info-bg: #193C47;\n    --admonition-info-code: #245e70;\n\n    /* Warning */\n    --admonition-warning-border: #e19d0a;\n    --admonition-warning-bg: #4f3a01;\n    --admonition-warning-code: #8c6b00;\n\n    /* Danger */\n    --admonition-danger-border: #e74f54;\n    --admonition-danger-bg: #4c1012;\n    --admonition-danger-code: #8c2e00;\n}\n\n[data-theme='dark'] {\n    @include dark-theme-variables;\n}\n\n@media (prefers-color-scheme: dark) {\n    :root:not([data-theme='light']) {\n        @include dark-theme-variables;\n    }\n}\n\n.admonition {\n    display: flex;\n    align-items: flex-start;\n    margin-block: 1em;\n    border-radius: 10px;\n    border-inline-start: 6px solid;\n    padding: 0.8rem;\n    color: var(--text-color-high-contrast);\n    font-family: var(--sans-serif-font);\n\n    p {\n        margin-inline-start: -1.75rem;\n        margin-block-end: 0;\n        font-family: inherit;\n    }\n\n    a {\n        code {\n            color: inherit;\n        }\n    }\n}\n\n.admonition-content {\n    flex: 1;\n    strong {\n        font-weight: 580;\n    }\n}\n\n.admonition-icon {\n    display: flex;\n    align-items: center;\n    margin: 0.3rem;\n    background-size: contain;\n    background-repeat: no-repeat;\n    aspect-ratio: 1/1;\n    width: 1.5rem;\n}\n\n.admonition-title {\n    opacity: 0.92;\n    font-weight: bold;\n    font-size: 0.82rem;\n}\n\n\n.admonition-icon-note {\n    -webkit-mask: url(\"data:image/svg+xml,%3Csvg xmlns='http://www.w3.org/2000/svg' viewBox='0 -960 960 960' %3E%3Cpath d='M440-280h80v-240h-80v240Zm40-320q17 0 28.5-11.5T520-640q0-17-11.5-28.5T480-680q-17 0-28.5 11.5T440-640q0 17 11.5 28.5T480-600Zm0 520q-83 0-156-31.5T197-197q-54-54-85.5-127T80-480q0-83 31.5-156T197-763q54-54 127-85.5T480-880q83 0 156 31.5T763-763q54 54 85.5 127T880-480q0 83-31.5 156T763-197q-54 54-127 85.5T480-80Zm0-80q134 0 227-93t93-227q0-134-93-227t-227-93q-134 0-227 93t-93 227q0 134 93 227t227 93Zm0-320Z'/%3E%3C/svg%3E\");\n}\n\n.admonition-icon-tip {\n    -webkit-mask: url(\"data:image/svg+xml,%3Csvg xmlns='http://www.w3.org/2000/svg' viewBox='0 -960 960 960' %3E%3Cpath d='M480-78.258q-33.718 0-56.974-22.166-23.256-22.167-23.59-55.885h161.128q-.334 33.718-23.59 55.885Q513.718-78.258 480-78.258ZM318.257-210.515v-67.588h323.486v67.588H318.257Zm7.846-121.128q-67.692-42.487-106.896-109.134-39.205-66.648-39.205-147.479 0-123.769 88.149-211.884 88.149-88.115 211.967-88.115 123.817 0 211.849 88.115 88.031 88.115 88.031 211.884 0 80.831-38.999 147.479-39 66.647-107.102 109.134H326.103Zm21.927-67.588h264.351q46.311-32 73.17-81.681 26.859-49.68 26.859-107.144 0-96.918-68-164.765-68-67.846-164.564-67.846t-164.41 67.713q-67.846 67.712-67.846 164.725 0 57.52 26.859 107.259t73.581 81.739Zm131.97 0Z'/%3E%3C/svg%3E\");\n}\n\n.admonition-icon-info {\n    -webkit-mask: url(\"data:image/svg+xml,%3Csvg xmlns='http://www.w3.org/2000/svg' viewBox='0 -960 960 960' %3E%3Cpath d='M479.789-288Q495-288 505.5-298.289q10.5-10.29 10.5-25.5Q516-339 505.711-349.5q-10.29-10.5-25.5-10.5Q465-360 454.5-349.711q-10.5 10.29-10.5 25.5Q444-309 454.289-298.5q10.29 10.5 25.5 10.5ZM444-432h72v-240h-72v240Zm36.276 336Q401-96 331-126q-70-30-122.5-82.5T126-330.958q-30-69.959-30-149.5Q96-560 126-629.5t82.5-122Q261-804 330.958-834q69.959-30 149.5-30Q560-864 629.5-834t122 82.5Q804-699 834-629.276q30 69.725 30 149Q864-401 834-331q-30 70-82.5 122.5T629.276-126q-69.725 30-149 30ZM480-168q130 0 221-91t91-221q0-130-91-221t-221-91q-130 0-221 91t-91 221q0 130 91 221t221 91Zm0-312Z'/%3E%3C/svg%3E\");\n}\n\n.admonition-icon-warning {\n    -webkit-mask: url(\"data:image/svg+xml,%3Csvg xmlns='http://www.w3.org/2000/svg' viewBox='0 -960 960 960' %3E%3Cpath d='M109-120q-11 0-20-5.5T75-140q-5-9-5.5-19.5T75-180l370-640q6-10 15.5-15t19.5-5q10 0 19.5 5t15.5 15l370 640q6 10 5.5 20.5T885-140q-5 9-14 14.5t-20 5.5H109Zm69-80h604L480-720 178-200Zm302-40q17 0 28.5-11.5T520-280q0-17-11.5-28.5T480-320q-17 0-28.5 11.5T440-280q0 17 11.5 28.5T480-240Zm0-120q17 0 28.5-11.5T520-400v-120q0-17-11.5-28.5T480-560q-17 0-28.5 11.5T440-520v120q0 17 11.5 28.5T480-360Zm0-100Z'/%3E%3C/svg%3E\");\n}\n\n.admonition-icon-danger {\n    -webkit-mask: url(\"data:image/svg+xml,%3Csvg xmlns='http://www.w3.org/2000/svg' viewBox='0 -960 960 960' %3E%3Cpath d='M239.256-400q0 58.091 27.975 108.995t76.13 81.237q-5.616-8.513-8.487-18.398-2.872-9.885-2.872-19.167 1.333-26.436 12.153-50.307 10.821-23.872 31.41-43.461L480-443.921l105.819 102.82q18.923 19.311 29.885 43.321 10.961 24.011 12.294 50.447 0 9.282-2.872 19.167-2.871 9.885-7.82 18.398 47.488-30.333 75.796-81.237Q721.41-341.909 721.41-400q0-47.622-19.258-93.169-19.259-45.547-53.998-82.549-19.951 13.41-42.202 19.859Q583.7-549.41 561-549.41q-62.448 0-105.108-38.039-42.661-38.038-51.225-98.628v-9.744q-39.385 31.949-69.898 67.68-30.513 35.73-51.987 74.166t-32.5 77.464Q239.256-437.483 239.256-400ZM480-349.539l-57.436 56.436q-12.154 11.821-17.731 26.029-5.577 14.208-5.577 29.074 0 32.769 23.498 55.757 23.497 22.987 57.246 22.987 33.432 0 57.421-22.906 23.989-22.906 23.989-55.561 0-16.162-6.116-30.162-6.116-13.999-17.454-25.154l-57.84-56.5Zm-11.002-469.022V-708q0 38.637 26.832 64.819 26.831 26.183 65.17 26.183 15.609 0 30.818-5.923 15.208-5.923 28.131-17.718l22.615-24.102q67.564 44.128 106.999 114.917 39.435 70.79 39.435 150.156 0 128.206-89.846 218.103Q609.307-91.668 480-91.668q-129.027 0-218.68-89.652-89.652-89.653-89.652-218.68 0-119.178 79.371-232.447t217.959-186.114Z'/%3E%3C/svg%3E\");\n}\n\n.admonition.note { @include admonition-type('note'); }\n.admonition.tip { @include admonition-type('tip'); }\n.admonition.info { @include admonition-type('info'); }\n.admonition.warning { @include admonition-type('warning'); }\n.admonition.danger { @include admonition-type('danger'); }\n"
  },
  {
    "path": "sass/parts/_archive.scss",
    "content": ".archive {\n    margin-block-start: 4vmin;\n\n    .listing-title {\n        margin-block-end: 1rem;\n        font-size: 1.5rem;\n    }\n\n    .listing-item {\n        display: flex;\n        gap: 1rem;\n        margin-block-end: 0.5rem;\n        padding-inline: 1rem;\n        padding-block: 0.2rem;\n\n        .post-time {\n            padding-inline-start: 1vmin;\n            min-inline-size: 5rem;\n            text-align: start;\n\n            .date {\n                color: var(--meta-color);\n                white-space: nowrap;\n            }\n        }\n    }\n\n    ul {\n        margin: 0;\n        padding: 0;\n        list-style: none;\n    }\n\n    li {\n        margin-bottom: 1.3rem;\n    }\n}\n"
  },
  {
    "path": "sass/parts/_aside.scss",
    "content": "aside {\n    margin-block-end: var(--paragraph-spacing);\n    border-radius: 4px;\n    background: var(--bg-0);\n    padding-block: 0.8rem;\n    padding-inline: 1rem;\n    color: var(--meta-color);\n    font-size: 0.9rem;\n\n    article & p {\n        margin: 0;\n        font-family: var(--sans-serif-font);\n    }\n\n    @media only screen and (min-width: 1300px) {\n        position: absolute;\n        inset-inline-start: -14rem;\n        margin-block: 0;\n        border-radius: 0;\n        background: none;\n        padding: 0;\n        width: 12rem;\n        text-align: end;\n\n        &[data-position=\"right\"] {\n            inset-inline-start: auto;\n            inset-inline-end: -14rem;\n            text-align: start;\n        }\n    }\n}\n"
  },
  {
    "path": "sass/parts/_cards.scss",
    "content": ".filter-controls {\n    display: flex;\n    flex-wrap: wrap;\n    justify-content: center;\n    align-items: center;\n    gap: 12px;\n    margin-top: 1.2rem;\n    margin-bottom: -1rem;\n    padding: 0;\n    list-style: none;\n\n    #all-projects-filter {\n        display: none;\n    }\n\n    .taxonomy-item {\n        margin: 0;\n\n        a {\n            display: inline-block;\n            box-shadow: rgba(0, 0, 0, 0.1) 0px 1px 3px 0px, rgba(0, 0, 0, 0.06) 0px 1px 2px 0px;\n            border-radius: 1rem;\n            background: var(--bg-2);\n            padding: 0 16px;\n            color: var(--text-color);\n            font-size: 0.8rem;\n            text-decoration: none;\n\n            &:hover {\n                background: var(--primary-color);\n                color: var(--hover-color);\n            }\n\n            &.active {\n                background: var(--primary-color);\n                color: var(--hover-color);\n            }\n        }\n    }\n}\n\n.cards {\n    display: grid;\n    grid-template-rows: auto;\n    grid-template-columns: repeat(auto-fill, minmax(300px, 1fr));\n    gap: 24px;\n    margin-top: 4vmin;\n    padding-block: 12px;\n\n    .card {\n        box-shadow: rgba(50, 50, 93, 0.25) 0px 2px 5px -1px, rgba(0, 0, 0, 0.3) 0px 1px 3px -1px;\n        border-radius: 1rem;\n        background: var(--bg-2);\n        min-height: 100px;\n        overflow: hidden;\n\n        &:hover {\n            background-color: var(--primary-color);\n        }\n\n        &:hover .card-description {\n            color: var(--hover-color);\n        }\n\n        .card-info {\n            padding-inline: 24px;\n            padding-block-start: 0;\n            padding-block-end: 24px;\n            text-align: center;\n        }\n        .card-image {\n            $margin: 1.6rem;\n            margin: $margin;\n            margin-bottom: $margin / 1.5;\n            width: calc(100% - $margin * 2);\n            height: auto;\n        }\n\n        .card-image-placeholder {\n            width: 100%;\n            height: 12px;\n        }\n\n        .card-description {\n            margin-top: 0.5em;\n            overflow: hidden;\n            color: var(--text-color);\n        }\n    }\n}\n\n@media all and (max-width: 720px) {\n    .cards {\n        gap: 18px;\n    }\n\n    .filter-controls {\n        gap: 8px;\n        margin: 18px 0;\n\n        .taxonomy-item a {\n            padding: 4px 12px;\n        }\n    }\n}\n"
  },
  {
    "path": "sass/parts/_code.scss",
    "content": "code {\n    -webkit-text-size-adjust: 100%;\n    border-radius: 5px;\n    background-color: var(--bg-1);\n    padding-inline: 0.2em;\n    padding-block: 0.1em;\n    font-size: 0.9rem;\n    font-family: var(--code-font);\n\n    mark {\n        display: block;\n        filter: brightness(110%);\n        background-color: var(--codeblock-highlight);\n        color: inherit;\n    }\n\n    /* Legacy table support for Zola versions older than 0.22.0 */\n    table {\n        margin: 0rem;\n        border-collapse: collapse;\n        border-spacing: 0rem;\n        width: 100%;\n        text-align: start;\n\n        td, th, tr {\n            border: none;\n            padding: 0rem;\n        }\n\n        tbody td:first-child {\n            opacity: 50%;\n            padding-inline-end: 0.8rem;\n            width: 1px;\n            user-select: none;\n            text-align: end;\n        }\n\n        tbody tr:nth-child(even) {\n            background-color: inherit;\n        }\n    }\n\n    /* Span-based layout (Giallo; Zola v0.22.0 and above) */\n    .giallo-l {\n        float: left;\n        clear: left;\n        width: 100%;\n        line-height: 1.2;\n    }\n\n    .giallo-ln {\n        display: inline-block;\n        opacity: 0.5;\n        padding-inline-end: 0.8rem;\n        min-width: 3ch;\n        color: inherit;\n        user-select: none;\n        text-align: end;\n    }\n}\n\na:hover code {\n    background-color: inherit;\n    padding-block: 0;\n}\n\npre {\n    display: block;\n    position: relative;\n    border-radius: 5px;\n    padding-inline: 1rem;\n    padding-block-start: 2.4rem;\n    padding-block-end: 1rem;\n    overflow: hidden;\n    overflow-x: auto;\n    line-height: 1.4;\n\n    code,\n    code td {\n        font-size: 0.8rem;  // Fits ~77 characters.\n    }\n\n    // Supports both native Zola way of adding src, and old shortcode-based logic.\n    // See: https://github.com/welpo/tabi/pull/489\n    code .source-path,\n    code[data-name]::after {\n        display: block;\n        position: absolute;\n        top: 0;\n        inset-inline-end: 1.3rem;\n        padding-top: 0.3rem;\n        padding-inline-end: 1.3rem;\n        max-width: calc(100% - 14em);\n        height: 0.9rem;\n        overflow: hidden;\n        color: var(--hover-color);\n        font-size: 0.65rem;\n        text-align: end;\n        text-overflow: ellipsis;\n        white-space: nowrap;\n    }\n\n    code[data-name]::after {\n        content: attr(data-name);\n    }\n\n    code {\n        display: block;\n        border: 0rem;\n        border-radius: 5px;\n        background-color: transparent;\n        padding: 0rem;\n        overflow-x: auto;\n        color: inherit;\n        white-space: pre;\n\n        &::before {\n            display: block;\n            position: absolute;\n            top: 0;\n            inset-inline-start: 0;\n            background-color: var(--primary-color);\n            padding: 0.3rem;\n            padding-inline-start: 1rem;\n            width: calc(100% - 1.3rem);\n            height: 0.9rem;\n            content: attr(data-lang);\n            color: var(--hover-color);\n            font-size: 0.65rem;\n            text-align: start;\n            text-transform: uppercase;\n        }\n    }\n}\n\n// Default to LTR codeblocks.\ncode, pre {\n    direction: ltr;\n}\n\n// Allows RTL codeblocks if `force_codeblock_ltr` is set to false.\nhtml[data-code-direction=\"inherit\"] {\n    code, pre {\n        direction: inherit;\n    }\n    .giallo-l {\n        float: right;\n        clear: right;\n    }\n}\n\n.copy-code {\n    -webkit-mask: url(\"data:image/svg+xml,%3Csvg xmlns='http://www.w3.org/2000/svg' viewBox='0 -960 960 960' %3E%3Cpath d='M217.002-67.694q-37.732 0-64.02-26.288-26.287-26.287-26.287-64.019V-707.69h77.999v549.689q0 4.615 3.846 8.462 3.846 3.846 8.462 3.846h451.689v77.999H217.002Zm175.999-175.999q-37.733 0-64.02-26.287T302.694-334v-463.383q0-37.732 26.287-64.02 26.287-26.287 64.02-26.287h365.383q37.732 0 64.019 26.287 26.288 26.288 26.288 64.02V-334q0 37.733-26.288 64.02-26.287 26.287-64.019 26.287H393.001Zm0-77.998h365.383q4.615 0 8.462-3.847 3.846-3.846 3.846-8.462v-463.383q0-4.616-3.846-8.462-3.847-3.846-8.462-3.846H393.001q-4.616 0-8.462 3.846-3.847 3.846-3.847 8.462V-334q0 4.616 3.847 8.462 3.846 3.847 8.462 3.847Zm-12.309 0v-488V-321.691Z'/%3E%3C/svg%3E\");\n    position: absolute;\n    top: 0.3rem;\n    align-self: center;\n    z-index: 1;\n    cursor: pointer;\n    inset-inline-end: 0.7rem;\n    background: var(--hover-color);\n    background-size: contain;\n    width: 0.9rem;\n    height: 0.9rem;\n    color: white;\n}\n\n.copy-code.checked {\n    -webkit-mask: url(\"data:image/svg+xml,%3Csvg xmlns='http://www.w3.org/2000/svg' viewBox='0 -960 960 960' %3E%3Cpath d='M395-253 194-455l83-83 118 117 288-287 83 84-371 371Z'/%3E%3C/svg%3E\");\n    width: 1rem;\n    height: 1rem;\n}\n\n.copy-code.error {\n    -webkit-mask: url(\"data:image/svg+xml,%3Csvg xmlns='http://www.w3.org/2000/svg' viewBox='0 -960 960 960' %3E%3Cpath d='M479.386-248Q509-248 529-267.386q20-19.386 20-49T529.614-366.5q-19.386-20.5-49-20.5T431-366.886q-20 20.114-20 49.728t19.386 49.386q19.386 19.772 49 19.772ZM416-431h128v-265H416v265Zm64.276 381q-88.916 0-167.743-33.104-78.828-33.103-137.577-91.852-58.749-58.749-91.852-137.535Q50-391.277 50-480.458q0-89.438 33.162-167.491 33.163-78.053 92.175-136.942 59.011-58.889 137.533-91.999Q391.393-910 480.458-910q89.428 0 167.518 33.093T784.94-784.94q58.874 58.874 91.967 137.215Q910-569.385 910-480.192q0 89.192-33.11 167.518-33.11 78.326-91.999 137.337-58.889 59.012-137.167 92.174Q569.447-50 480.276-50Z'/%3E%3C/svg%3E\");\n}\n"
  },
  {
    "path": "sass/parts/_comments.scss",
    "content": ".utterances-frame {\n    width: 100%;\n}\n\n.comments {\n    margin-top: 2rem;\n    border-top: var(--divider-color) solid 0.5px;\n    border-bottom: var(--divider-color) solid 0.5px;\n    padding-top: 2rem;\n    padding-bottom: 2rem;\n\n    iframe {\n        margin: 0;\n        border: none;\n        aspect-ratio: inherit;\n        width: 100%;\n        max-width: 100%;\n    }\n\n    .load-comments-button {\n        display: block;\n        cursor: pointer;\n        margin: 0.5em auto;\n        border: none;\n        background: none;\n        padding-block: 0.5em;\n        padding-inline: 1em;\n        color: inherit;\n        font-size: 0.95rem;\n        font-family: var(--sans-serif-font);\n        text-decoration: none;\n    }\n}\n"
  },
  {
    "path": "sass/parts/_footer.scss",
    "content": "footer {\n    margin-top: auto;\n    margin-block-end: 1.4rem;\n    color: var(--meta-color);\n    font-size: 0.88rem;\n    font-family: var(--post-font-family);\n    text-align: center;\n\n    .nav-links {\n        color: var(--primary-color);\n    }\n\n    p {\n        margin: 0;\n    }\n}\n\nfooter section {\n    display: flex;\n    flex-direction: column;\n    align-items: center;\n    gap: 0rem;\n}\n\nfooter nav {\n    display: flex;\n    margin: 0 0rem;\n}\n\n.socials {\n    display: flex;\n    flex-grow: 0;\n    flex-wrap: wrap;\n    justify-content: center;\n    align-items: flex-end;\n\n    svg {\n        max-height: 15px;\n    }\n\n    ul {\n        gap: 5px;\n    }\n}\n\n.social {\n    display: flex;\n    justify-content: center;\n    align-items: center;\n    background-image: unset;\n    padding: 0.5vmin;\n}\n\n.social > img {\n    aspect-ratio: 1/1;\n    width: 1.5rem;\n    height: auto;\n    color: #000000;\n}\n\n.social {\n    &:hover {\n        & > img {\n            filter: invert(1);\n        }\n    }\n}\n\n@mixin dark-theme-social {\n    .social {\n        &:hover {\n            & > img {\n                filter: invert(0);\n            }\n        }\n\n        & > img {\n            filter: invert(1);\n        }\n    }\n}\n\n[data-theme=\"dark\"] {\n    @include dark-theme-social;\n}\n\n@media (prefers-color-scheme: dark) {\n    :root:not([data-theme=\"light\"]) {\n        @include dark-theme-social;\n    }\n}\n"
  },
  {
    "path": "sass/parts/_header-anchor.scss",
    "content": ".header-anchor {\n    display: inline-flex;\n    position: absolute;\n    justify-content: center;\n    align-items: center;\n    opacity: 0;\n    margin-inline-start: -2rem;\n    padding-inline-end: 0.3rem;\n    width: 1.9rem;\n    height: 100%;\n    user-select: none;\n\n    @media (max-width: 500px) {\n        display: none;\n    }\n}\n\n.link-icon {\n    -webkit-mask: url(\"data:image/svg+xml,%3Csvg xmlns='http://www.w3.org/2000/svg' viewBox='0 0 24 24'%3E%3Cpath d='M14.78 3.653a3.936 3.936 0 1 1 5.567 5.567l-3.627 3.627a3.936 3.936 0 0 1-5.88-.353.75.75 0 0 0-1.18.928 5.436 5.436 0 0 0 8.12.486l3.628-3.628a5.436 5.436 0 1 0-7.688-7.688l-3 3a.75.75 0 0 0 1.06 1.061l3-3Z'%3E%3C/path%3E%3Cpath d='M7.28 11.153a3.936 3.936 0 0 1 5.88.353.75.75 0 0 0 1.18-.928 5.436 5.436 0 0 0-8.12-.486L2.592 13.72a5.436 5.436 0 1 0 7.688 7.688l3-3a.75.75 0 1 0-1.06-1.06l-3 3a3.936 3.936 0 0 1-5.567-5.568l3.627-3.627Z'%3E%3C/path%3E%3C/svg%3E\");\n    align-self: center;\n    cursor: pointer;\n    background: var(--text-color);\n    width: 1rem;\n    height: 1rem;\n}\n\nh1, h2, h3, h4, h5, h6 {\n    &:hover .header-anchor {\n        opacity: 1;\n    }\n\n    .header-anchor:hover {\n        background-color: transparent;\n    }\n}\n"
  },
  {
    "path": "sass/parts/_header.scss",
    "content": "header {\n    width: 100%;\n    font-family: 'Inter Subset', var(--sans-serif-font);\n}\n\n.page-header {\n    margin-block: 4rem 1rem;\n    font-size: 3em;\n    line-height: 100%;\n    font-family: var(--header-font);\n}\n\n.navbar {\n    display: flex;\n    flex-direction: row;\n    flex-wrap: wrap;\n    justify-content: space-between;\n    align-items: center;\n    margin: 0 auto;\n    padding-block: 1em;\n    max-width: var(--max-layout-width);\n}\n\n.nav-navs {\n    display: flex;\n    flex-wrap: wrap;\n    align-items: center;\n\n    ul {\n        display: flex;\n        flex-wrap: inherit;\n        justify-content: inherit;\n        align-items: inherit;\n        gap: 1px;\n        margin: 0;\n        padding: 0;\n        list-style: none;\n    }\n}\n\n.menu-icons-container {\n    display: flex;\n    align-items: center;\n    margin-left: auto;\n}\n\n.menu-icons-group {\n    gap: 1px;\n    margin: 0;\n    padding: 0;\n}\n\n.nav-links {\n    justify-content: right;\n    padding: 0.66rem;\n    color: var(--text-color);\n    font-weight: 340;\n    font-size: 1em;\n    line-height: 2.5;\n    text-decoration: none;\n}\n\n.home-title {\n    margin-inline-start: -0.12rem;\n    border: none;\n    padding: 0.12rem;\n    color: var(--primary-color);\n    font-weight: 450;\n    font-size: 1.7em;\n    text-decoration: none;\n}\n\n.meta {\n    padding: 0;\n    padding-top: 0.7vmin;\n    padding-bottom: 3vmin;\n    color: var(--meta-color);\n    font-weight: 300;\n    font-size: 0.8rem;\n    line-height: 1.4rem;\n    letter-spacing: -0.4px;\n\n    a {\n        color: var(--meta-color);\n        font-weight: inherit;\n        text-decoration: none;\n        text-decoration-color: none;\n    }\n\n    ul,\n    li {\n        display: inline-block;\n        margin-inline-end: 0.2rem;\n        font-family: var(--sans-serif-font);\n        list-style-type: none;\n    }\n\n    .tag {\n        margin-inline-end: 0;\n    }\n}\n\n.separator {\n    margin-inline-end: 0.2rem;\n    user-select: none;\n}\n\n.language-switcher {\n    display: flex;\n    justify-content: center;\n    align-items: center;\n    margin-inline-start: 0.5rem;\n    margin-inline-end: 0.5rem;\n\n    .language-switcher-icon {\n        -webkit-mask: url(\"data:image/svg+xml,%3Csvg xmlns='http://www.w3.org/2000/svg' viewBox='0 0 24 24'%3E%3Cpath fill='none' stroke='%23000' stroke-width='1.8' d='M1 12a11 11 90 0 0 22 0 11 11 90 0 0-22 0m1-4h20M2 16h20M11 1a21 21 90 0 0 0 22m2-22a21 21 90 0 1 0 22'/%3E%3C/svg%3E%0A\");\n        position: relative;\n        align-self: center;\n        cursor: pointer;\n        background: var(--text-color);\n        width: 1rem;\n        height: 1rem;\n\n        &:hover {\n            background: var(--meta-color);\n        }\n    }\n\n    .language-switcher-icon-with-code {\n        margin-inline-end: 0.3rem;\n        width: 0.7rem;\n        height: 0.7rem;\n    }\n}\n\n.language-switcher-icon-code {\n    position: absolute;\n    top: -0.15rem;\n    z-index: 10;\n    inset-inline-start: 0.7rem;\n    width: 100%;\n    height: 100%;\n    color: var(--text-color);\n    font-size: 0.5rem;\n    text-transform: uppercase;\n\n    &:hover {\n        color: var(--meta-color);\n    }\n}\n\n.dropdown {\n    display: inline-block;\n    position: relative;\n    z-index: 1;\n    font-size: 0.8rem;\n\n    &:hover .dropdown-content,\n    &:focus-within .dropdown-content {\n        display: block;\n    }\n\n    summary {\n        list-style: none;\n\n        &::-webkit-details-marker {\n            display: none;\n        }\n    }\n\n    .dropdown-content {\n        display: none;\n        position: absolute;\n        left: 50%;\n        transform: translateX(-50%);\n        z-index: 1;\n        background: var(--background-color);\n        padding-inline-start: 0.5rem;\n        padding-inline-end: 0.5rem;\n        text-align: center;\n        white-space: nowrap;\n\n        a {\n            display: block;\n        }\n    }\n}\n\n@media only screen and (max-width: 1000px) {\n    .navbar {\n        max-width: var(--normal-layout-width);\n    }\n\n    .nav-navs {\n        display: flex;\n        justify-content: center;\n    }\n\n    .menu-icons-container {\n        margin-left: 0;\n    }\n}\n\n@media only screen and (max-width: 600px) {\n    .nav-navs {\n        margin-top: 0.8rem;\n    }\n\n    .navbar {\n        flex-direction: column;\n        align-items: center;\n    }\n}\n\n@media only screen and (max-width: 300px) {\n    .navbar {\n        max-width: var(--small-layout-width);\n    }\n}\n"
  },
  {
    "path": "sass/parts/_home-banner.scss",
    "content": "#banner-container-home {\n    display: flex;\n    justify-content: space-between;\n    align-items: flex-start;\n    margin: 0.2rem auto;\n    width: 100%;\n\n    @media only screen and (max-width: 600px) {\n        display: block;\n        margin-block-end: 2rem;\n    }\n\n    #home-banner-text {\n        flex: 1;\n        margin-block-end: 1.5rem;\n        color: var(--primary-color);\n        font-size: 1.875rem;\n        line-height: 3rem;\n\n        li {\n            font-size: 1rem;\n        }\n\n        #home-banner-header {\n            margin: 0;\n            margin-block-end: 1rem;\n            font-weight: 550;\n            font-size: 2.8rem;\n\n            @media only screen and (max-width: 600px) {\n                margin-block-end: 0;\n                font-size: 2.2rem;\n            }\n        }\n\n        #banner-home-subtitle {\n            color: var(--text-color);\n            font-weight: 250;\n            line-height: 1.75rem;\n\n            p {\n                font-size: 1rem;\n            }\n\n            a {\n                font-weight: 400;\n            }\n        }\n\n        @media only screen and (max-width: 600px) {\n            width: 100%;\n        }\n    }\n\n    #image-container-home {\n        position: relative;\n        margin: auto 0;\n        padding-inline-start: 2rem;\n        max-width: 11rem;\n        overflow: hidden;\n        text-align: center;\n\n        #banner-home-img {\n            border: none;\n            aspect-ratio: 1 / 1;\n            width: 100%;\n            height: 100%;\n            object-fit: cover;\n\n            @media only screen and (max-width: 600px) {\n                max-width: 12rem;\n                max-height: 12rem;\n            }\n        }\n\n        @media only screen and (max-width: 600px) {\n            padding-inline-start: 0;\n            width: 100%;\n            max-width: none;\n        }\n    }\n}\n"
  },
  {
    "path": "sass/parts/_iine.scss",
    "content": ".iine-button {\n    display: inline-flex;\n    align-items: center;\n    gap: 5px;\n    transition: background-color 0.2s ease, color 0.2s ease, border-color 0.2s ease;\n    cursor: pointer;\n    border: none;\n    background: transparent;\n    color: inherit;\n    font-family: var(--sans-serif-font);\n    -webkit-tap-highlight-color: transparent;\n    appearance: none;\n    margin: 0;\n    padding: 0;\n    font-size: inherit;\n    line-height: inherit;\n\n    .icon {\n        display: inline-flex;\n        align-items: center;\n    }\n\n    .counter {\n        margin-left: .2rem;\n        font-size: 0.8rem;\n    }\n\n    svg {\n        width: 1em;\n        height: 1em;\n    }\n}\n\n.iine-auto-buttons {\n    margin-top: 2rem;\n    padding: 1rem 0;\n}\n"
  },
  {
    "path": "sass/parts/_image-hover.scss",
    "content": ".image-hover-container {\n    position: relative;\n    width: 100%;\n\n    .image-default {\n        display: inline;\n    }\n\n    .image-hovered {\n        display: none;\n    }\n\n    &:hover {\n        .image-default {\n            display: none;\n        }\n\n        .image-hovered {\n            display: inline;\n        }\n    }\n}\n"
  },
  {
    "path": "sass/parts/_image-toggler.scss",
    "content": ".image-label {\n    cursor: pointer;\n}\n\n.image-toggled {\n    position: absolute;\n    top: 0;\n    left: 0;\n    visibility: hidden;\n}\n\n.image-toggler-toggle {\n    display: none;\n}\n\n.image-toggler-toggle:checked ~ .image-label .image-toggled {\n    position: static;\n    visibility: visible;\n}\n\n.image-toggler-toggle:checked ~ .image-label .image-default {\n    position: absolute;\n    visibility: hidden;\n}\n"
  },
  {
    "path": "sass/parts/_image.scss",
    "content": "figure {\n    display: inline-block;\n    box-sizing: border-box;\n    margin: 0;\n    max-width: 100%;\n    height: auto;\n}\n\nfigcaption {\n    color: var(--meta-color);\n    font-size: 0.72rem;\n    font-family: var(--serif-font);\n    text-align: center;\n}\n\nimg {\n    display: block;\n    margin: 0 auto;\n    max-width: 100%;\n    height: auto;\n}\n\nimg.inline {\n    display: inline;\n    vertical-align: middle;\n}\n\nfigure h4 {\n    margin: 0;\n    margin-block-end: 1em;\n    font-size: 1rem;\n}\n\nfigure h4::before {\n    content: \"↳ \";\n}\n\n.img-dark {\n    display: none;\n    &.inline{\n        display: none;\n    }\n}\n\n.img-light.inline {\n    display: inline;\n}\n\n[data-theme=\"dark\"] {\n    .img-dark {\n        display: block;\n        &.inline {\n            display: inline;\n        }\n    }\n\n    .img-light {\n        display: none;\n    }\n}\n"
  },
  {
    "path": "sass/parts/_misc.scss",
    "content": "kbd {\n    border: 1px solid var(--divider-color);\n    border-radius: 5px;\n    background-color: var(--bg-0);\n    padding: 0.1rem 0.3rem;\n    font-size: 0.8rem;\n}\n\n.draft-label {\n    margin-inline-end: 0.3rem;\n    background-color: var(--primary-color);\n    padding-inline: 4px;\n    padding-block: 2px;\n    color: var(--hover-color);\n}\n\n.article-title {\n    display: block;\n    margin: 0;\n    color: var(--text-color-high-contrast);\n    font-weight: 550;\n    font-size: 2rem;\n    line-height: 3rem;\n}\n\niframe {\n    display: block;\n    margin-inline-start: 15%;\n    margin-inline-end: 15%;\n    margin-block-end: 3vmin;\n    border: none;\n    aspect-ratio: 16/9;\n    width: 100vmin;\n    max-width: 70%;\n}\n\nul {\n    margin-top: 0;\n}\n\n.toc-container {\n    margin-block-end: 4vmin;\n}\n\n.padding-top {\n    padding-top: 4vmin;\n}\n\n.title-container {\n    padding-bottom: 8px;\n    .social {\n        margin-inline-start: 0.5rem;\n    }\n}\n\n.bottom-divider {\n    border-bottom: var(--divider-color) solid 0.5px;\n}\n\n::-moz-selection {\n    background: var(--primary-color);\n    color: var(--hover-color);\n    text-shadow: none;\n}\n\n::selection {\n    background: var(--primary-color);\n    color: var(--hover-color);\n}\n\n.nav.tags {\n    display: inline-block;\n}\n\nblockquote {\n    margin: 0;\n    border-inline-start: 0.3rem solid var(--primary-color);\n    padding-inline-start: 1em;\n}\n\na {\n    position: relative;\n    color: var(--primary-color);\n    font-weight: inherit;\n    text-decoration: inherit;\n}\n\n// External link styles with `external_links_class = \"external\"`.\nmain {\n    --external-link-icon: url(\"data:image/svg+xml,%3Csvg xmlns='http://www.w3.org/2000/svg' viewBox='0 0 24 24'%3E%3Cg fill='none' stroke='currentColor' stroke-linecap='round' stroke-linejoin='round' stroke-width='2'%3E%3Cpath d='M11 5h-6v14h14v-6'/%3E%3Cpath d='M13 11l7 -7'/%3E%3Cpath d='M21 3h-6M21 3v6'/%3E%3C/g%3E%3C/svg%3E\");\n\n    a.external:not(:has(img, svg, video, picture, figure))::after {\n        display: inline-block;\n        vertical-align: -0.05em;\n        margin-inline-start: 0.1em;\n        background-color: currentColor;\n        width: 0.8em;\n        height: 0.8em;\n        content: '';\n        -webkit-mask-image: var(--external-link-icon);\n        -webkit-mask-size: 100% 100%;\n    }\n\n    &:dir(rtl) a.external:not(:has(img, svg, video, picture, figure))::after {\n        transform: rotate(-90deg);\n    }\n\n    .meta a.external:not(:has(img, svg, video, picture, figure))::after {\n        background-color: var(--meta-color);\n    }\n\n    a.external:not(:has(img, svg, video, picture, figure)):hover::after {\n        background-color: var(--hover-color);\n    }\n}\n\na:hover {\n    background-color: var(--primary-color);\n    color: var(--hover-color);\n}\n\na:hover rt {\n    color: var(--text-color);\n}\n\na:not(.no-hover-padding):hover::before {\n    display: inline-block;\n    position: absolute;\n    z-index: -1;\n    inset-block-end: 0;\n    inset-block-start: 0;\n    inset-inline-end: -0.15em;\n    inset-inline-start: -0.15em;\n    background-color: var(--primary-color);\n    max-inline-size: 105%;\n    content: \"\";\n}\n\n@media screen and (max-width: 600px) {\n    .list > ul {\n        margin: 0;\n        padding: 0;\n    }\n}\n\nhr {\n    margin: 3.5rem 0 1rem;\n    border: none;\n    background-color: var(--divider-color);\n    height: 1px;\n}\n\n.footnotes-list,\n.footnotes {\n    text-align: start;\n}\n\n.footnote-reference {\n    font-size: 0.7rem;\n    font-family: var(--serif-font);\n}\n\n.footnote-definition {\n    margin-block-end: 0.6rem;\n\n    sup {\n        margin-inline-end: 0.15rem;\n        font-size: 0.75rem;\n        font-family: var(--serif-font);\n    }\n\n    p {\n        display: inline;\n    }\n}\n\n.footnote-backlink {\n    margin-inline-start: 0.2rem;\n    font-size: 0.8rem;\n}\n\n.footnotes-list a[href^=\"#fr-\"],\n.footnotes a[href^=\"#fr-\"] {\n    font-size: 0.8rem;\n}\n\n.footnotes code {\n    font-size: 0.8rem;\n}\n\n.references p {\n    margin-inline-start: 2.4rem;\n    text-indent: -2.4rem;\n}\n\n.info-box {\n    margin-top: 1rem;\n    margin-block-end: 1rem;\n    border: 1px solid var(--primary-color);\n    border-radius: 10px;\n    border-inline-start-width: 0.3rem;\n    padding: 1rem;\n    text-align: center;\n}\n\n#page-content {\n    margin-top: 4vmin;\n}\n\n.hidden {\n    display: none;\n    visibility: hidden;\n}\n\n.visually-hidden {\n    clip: rect(0 0 0 0);\n    position: absolute;\n    margin: -1px;\n    border: 0;\n    padding: 0;\n    width: 1px;\n    height: 1px;\n    overflow: hidden;\n    white-space: nowrap;\n}\n\ndetails summary {\n    cursor: pointer;\n}\n\n.interactive-icon {\n    cursor: pointer;\n\n    path {\n        fill: var(--text-color);\n    }\n\n    :hover path {\n        fill: var(--meta-color);\n    }\n}\n\n.article-navigation {\n    display: flex;\n    margin-block-start: 2rem;\n    border-block-start: var(--divider-color) solid 0.5px;\n    padding-block-start: 2rem;\n\n    div:first-child {\n        flex: 1;\n        text-align: start;\n    }\n\n    div:last-child {\n        flex: 1;\n        text-align: end;\n    }\n\n    div p {\n        color: var(--meta-color);\n        font-weight: 300;\n        line-height: 1.2rem;\n        font-family: var(--sans-serif-font);\n        letter-spacing: -0.4px;\n    }\n\n    @media (max-width: 600px) {\n        flex-direction: column;\n\n        div {\n            text-align: center !important;\n        }\n    }\n}\n\n:dir(rtl) .arrow {\n    display: inline-block;\n    transform: scaleX(-1);\n}\n\n// This for the arrows that point to a corner, (e.g. '↗', '↘', '↙', '↖')\n:dir(rtl) .arrow-corner {\n    display: inline-block;\n    transform: rotate(-90deg);\n}\n\n.mermaid p {\n    font-family: var(--sans-serif-font) !important;\n}\n\n.mermaid .node .label {\n    max-width: none !important;\n}\n\n// For the `force_text_direction` shortcode.\n[data-force-text-direction=\"ltr\"] {\n    direction: ltr;\n    unicode-bidi: bidi-override;\n}\n\n[data-force-text-direction=\"rtl\"] {\n    direction: rtl;\n    unicode-bidi: bidi-override;\n}\n\n[data-force-text-direction=\"ltr\"] *,\n[data-force-text-direction=\"rtl\"] * {\n    direction: inherit;\n}\n\n.title-with-jump {\n    display: flex;\n    justify-content: space-between;\n    align-items: center;\n}\n\n.title-with-jump h1 {\n    flex: 1;\n}\n\n.jump-link {\n    flex-shrink: 0;\n    font-size: 0.9rem;\n}\n\n@media (max-width: 500px) {\n    .title-with-jump {\n        flex-direction: column;\n    }\n}\n\n#skip-link {\n    position: absolute;\n    top: -40px;\n    left: 0;\n    transform: translateY(-100%);\n    opacity: 0;\n    z-index: 9999;\n    transition: all 0.1s ease;\n    border-radius: 0 0 5px 0;\n    background-color: var(--primary-color);\n    padding: 4px 8px;\n    color: var(--hover-color);\n    font-weight: 500;\n    font-size: 0.9rem;\n    text-decoration: none;\n}\n\n#skip-link:focus {\n    top: 0;\n    transform: translateY(0);\n    opacity: 1;\n    outline: 2px solid var(--text-color);\n    outline-offset: 2px;\n}\n"
  },
  {
    "path": "sass/parts/_multilingual_quote.scss",
    "content": ".quote-container {\n    border: none;\n}\n\n.quote-toggle {\n    display: none;\n}\n\n.quote-label {\n    display: none;\n    cursor: pointer;\n    border-radius: 5px;\n    color: var(--meta-color);\n    font-size: 0.75rem;\n    font-family: var(--sans-serif-font);\n    text-align: center;\n    text-decoration: none;\n}\n\n.quote-toggle:not(:checked) ~ .quote .translated .quote-label-original,\n.quote-toggle:checked ~ .quote .original .quote-label-translate {\n    display: inline;\n}\n\n.original {\n    display: none;\n}\n\n.quote-toggle:checked ~ .quote .original {\n    display: block;\n}\n\n.quote-toggle:checked ~ .quote .translated {\n    display: none;\n}\n"
  },
  {
    "path": "sass/parts/_pagination.scss",
    "content": ".pagination {\n    display: flex;\n    justify-content: space-between;\n    align-items: center;\n    margin-top: 2rem;\n    padding: 0;\n    font-size: 1em;\n    list-style: none;\n\n    .page-item .disabled {\n        opacity: 0.5;\n        pointer-events: none;\n    }\n\n    .page-numbers {\n        color: var(--meta-color);\n        font-size: 0.9rem;\n    }\n}\n"
  },
  {
    "path": "sass/parts/_posts_list.scss",
    "content": ".bloglist-container {\n    display: grid;\n    grid-template-columns: 1fr 8fr;\n}\n\n\n$padding: 2.5rem;\n\n.bloglist-meta {\n    display: flex;\n    align-items: flex-start;\n    background-color: var(--navbar-color);\n    padding-block: $padding;\n    min-width: 13.5rem;\n\n    .thumbnail-image {\n        margin: 0;\n        margin-inline: auto;  // Centred by default.\n        max-width: 70%;\n    }\n\n    li.date + li.post-thumbnail .thumbnail-image {\n        // Styles for the thumbnail when there's a date above (either date or updated).\n        margin-inline: 0;  // Since metadata is left aligned, the image looks weird when centred.\n        margin-block-start: 0.7rem;\n    }\n\n    ul {\n        margin-inline-end: 0.7rem;\n        padding: 0;\n        color: var(--meta-color);\n        font-weight: 300;\n        font-size: 0.9rem;\n\n        li {\n            list-style-type: none;\n            white-space: nowrap;\n        }\n\n        li.draft-label {\n            width: fit-content;\n            line-height: 1.2rem;\n        }\n    }\n}\n\n.bloglist-content {\n    display: flex;\n    position: relative;\n    align-items: flex-start;\n    background-color: var(--navbar-color);\n    padding: $padding 0;\n\n    .pinned-label {\n        display: flex;\n        position: absolute;\n        top: 0.8rem;\n        align-items: center;\n        gap: 0.3rem;\n        color: var(--meta-color);\n        font-weight: 300;\n        font-size: 0.8rem;\n    }\n\n    .pinned-label svg {\n        width: 0.8rem;\n        height: 0.8rem;\n    }\n\n    div {\n        flex: 1;\n\n        .bloglist-title {\n            margin: 0;\n            font-weight: bold;\n            font-size: 1.2em;\n\n            a {\n                color: var(--text-color-high-contrast);\n                font-weight: 550;\n\n                &:hover {\n                    color: var(--hover-color);\n                }\n            }\n        }\n\n        .bloglist-tags {\n            margin-top: 0.1rem;\n\n            .tag {\n                display: inline-block;\n                margin-inline-end: 0.7rem;\n                font-weight: 400;\n                font-size: 0.75rem;\n                text-transform: uppercase;\n            }\n        }\n\n        .description p {\n            margin: 0.5rem 0 1rem;\n            color: var(--text-color);\n            font-weight: 250;\n            font-size: 0.9rem;\n            line-height: 1.5rem;\n        }\n    }\n}\n\n.all-posts {\n    font-weight: 350;\n    font-size: 1.3rem;\n}\n\n#all-projects {\n    margin-top: 2rem;\n}\n\n.posts-first #featured-projects {\n    margin-top: 4rem;\n}\n\n.projects-first #posts-list {\n    margin-top: 4rem;\n}\n\n@media only screen and (max-width: 1100px) {\n    .bloglist-container {\n        display: block;\n    }\n\n    .pinned-label svg {\n        margin-bottom: -2px;\n    }\n\n    .bloglist-meta {\n        border-bottom: 0;\n        padding-block: 2rem;\n\n        ul {\n            margin-block-end: 0;\n            width: 100%;\n\n            li {\n                display: inline;\n                margin-inline-end: 0.3rem;\n            }\n        }\n\n        .post-thumbnail {\n            display: none;\n        }\n    }\n\n    .bloglist-content {\n        flex-direction: column;\n        align-items: flex-start;\n        padding: 0;\n        padding-bottom: 2rem;\n\n\n        .pinned-label {\n            position: static;\n            margin: 0;\n            margin-top: -1.9rem;\n        }\n\n        div {\n            width: 100%;\n        }\n    }\n}\n"
  },
  {
    "path": "sass/parts/_quick_navigation_buttons.scss",
    "content": "#button-container {\n    display: flex;\n    position: fixed;\n    right: 2rem;\n    bottom: 2rem;\n    flex-direction: column;\n    gap: 0.6rem;\n    z-index: 2; // Above \"copy-code\" buttons. Important for the ToC.\n\n    #toc-button,\n    #comments-button,\n    #top-button {\n        display: flex;\n        justify-content: center;\n        align-items: center;\n        z-index: 2;\n        cursor: pointer;\n        border: none;\n        border-radius: 50%;\n        background-color: var(--bg-1);\n        padding: 0.4rem;\n        width: 1rem;\n        height: 1rem;\n        text-align: center;\n\n        &:hover {\n            background-color: var(--bg-3);\n\n            svg {\n                fill: var(--primary-color);\n            }\n\n            &::before {\n                background-color: transparent;\n            }\n        }\n\n        svg {\n            fill: var(--text-color);\n            width: 1rem;\n            height: 1rem;\n        }\n    }\n\n    #toc-floating-container {\n\n        $padding-vertical: 0.7rem;\n        $padding-horizontal: 1rem;\n\n        #toc-button {\n            position: relative;\n            z-index: 2;\n        }\n\n        .toc-container {\n            margin: 0;\n            margin-top: $padding-vertical;\n            max-width: 80vw;\n        }\n\n        .toc-content {\n            display: none;\n            position: absolute;\n            right: 0;\n            bottom: 100%;\n            z-index: 2;\n            margin-block-end: $padding-vertical;\n            box-shadow: rgba(0, 0, 0, 0.15) 1.95px 1.95px 2.6px;\n            border: 1px solid var(--divider-color);\n            border-radius: 5px;\n            background-color: var(--background-color);\n            padding-inline-end: $padding-horizontal;\n            max-height: 70vh;\n            overflow-y: auto;\n            font-size: 0.8rem;\n            text-align: start;\n            white-space: nowrap; // Prevents wrapping, allowing content to define width.\n\n            ul {\n                padding-inline-start: $padding-horizontal;\n                list-style: none;\n            }\n        }\n\n        .toggle {\n            display: none;\n\n            &:checked {\n\n                + #toc-button + .toc-content {\n                    display: block;\n                }\n\n                + #toc-button svg {\n                    fill: var(--primary-color); // Show the ToC icon as toggled.\n                }\n            }\n        }\n\n    }\n}\n\n@media (max-width: 700px) {\n    #button-container {\n        display: none !important;\n    }\n}\n\n@media print {\n    #button-container {\n        display: none;\n    }\n}\n"
  },
  {
    "path": "sass/parts/_search.scss",
    "content": "$icon-size: 1.3rem;\n\n#searchModal {\n    background: color-mix(in srgb, var(--primary-color) 5%, transparent);\n    text-align: start;\n\n    #searchContainer {\n        padding: 1rem;\n    }\n\n    #searchBar {\n        display: flex;\n        position: relative;\n        justify-content: center;\n        align-items: center;\n        box-sizing: border-box;\n        padding: 1rem;\n\n        .search-icon {\n            position: absolute;\n            inset-inline-start: 1rem;\n            width: $icon-size;\n            height: $icon-size;\n\n            path {\n                fill: var(--text-color);\n            }\n        }\n\n        .close-icon {\n            display: none;\n            position: absolute;\n            right: $icon-size;\n            margin-inline-start: 1rem;\n            margin-inline-end: 0.5rem;\n            width: $icon-size;\n            height: $icon-size;\n        }\n\n        #searchInput {\n            flex: 1;\n            border: 1px solid var(--divider-color);\n            border-radius: 20px;\n            background-color: var(--input-background-color);\n            padding-inline: 3rem 1rem;\n            padding-block: 0.75rem;\n            width: calc(100% - 2rem);\n            color: var(--text-color);\n            font-size: 1rem;\n        }\n    }\n\n    #results-container {\n        display: none;\n        border-top: var(--divider-color) solid 0.5px;\n        border-bottom-right-radius: 1rem;\n        border-bottom-left-radius: 1rem;\n        overflow: hidden;\n\n        #results-info {\n            padding: 0.5rem;\n            color: var(--meta-color);\n            font-size: 0.8rem;\n            text-align: center;\n        }\n\n        #results {\n            display: flex;\n            flex-direction: column;\n            max-height: 50vh;\n            overflow-y: auto;\n\n            b {\n                font-weight: 590;\n            }\n\n            a {\n                display: block;\n\n                &:hover {\n                    background-color: inherit;\n                }\n            }\n\n            > div {\n                cursor: pointer;\n                padding-inline: 1rem;\n                padding-block: 0.5rem;\n\n                &[aria-selected=\"true\"] {\n                    background-color: var(--primary-color);\n                    color: var(--hover-color);\n\n                    a,\n                    span {\n                        color: inherit;\n                    }\n                }\n            }\n\n            span:first-child {\n                display: block;\n                color: var(--primary-color);\n                font-weight: 590;\n            }\n\n            span:nth-child(2) {\n                color: var(--text-color);\n            }\n        }\n\n    }\n}\n\n.search-icon {\n    display: block;\n    position: relative;\n    align-self: center;\n    margin-inline-start: 1rem;\n    margin-inline-end: 0.5rem;\n    width: $icon-size;\n    height: $icon-size;\n}\n\n.search-modal {\n    -webkit-backdrop-filter: blur(8px);\n    display: none;\n    position: fixed;\n    top: 0;\n    left: 0;\n    z-index: 1000;\n    backdrop-filter: blur(8px);\n    background-color: rgba(0, 0, 0, 0.1);\n    width: 100%;\n    height: 100%;\n    overflow: auto;\n\n    #modal-content {\n        position: relative;\n        margin: 8% auto;\n        border: var(--divider-color) solid 0.5px;\n        border-radius: 1rem;\n        background-color: var(--background-color);\n        width: 80%;\n        max-width: 28rem;\n    }\n}\n\n@media only screen and (max-width: 600px) {\n    .search-modal {\n        #modal-content {\n            top: 3.5rem;\n            width: 92%;\n\n            #results {\n                max-height: 70vh;\n            }\n        }\n    }\n}\n"
  },
  {
    "path": "sass/parts/_spoiler.scss",
    "content": ".spoiler-toggle {\n    display: none; // Hide the checkbox.\n}\n\n.spoiler-content {\n    display: inline-block; // Allow content to only take up its own width.\n    cursor: help; // Indicate interactive element.\n\n    .spoiler-hidden {\n        -webkit-user-select: none;\n        -moz-user-select: none;\n        -ms-user-select: none;\n        filter: blur(6px);\n        user-select: none;\n\n        a {\n            pointer-events: none; // Make links unclickable.\n        }\n    }\n}\n\n.spoiler-toggle:checked + .spoiler-content {\n    .spoiler-hidden {\n        filter: none;\n        user-select: auto;\n\n        a {\n            pointer-events: auto; // Enable clicking on links when revealed.\n        }\n    }\n}\n\n.spoiler-container.fixed-blur {\n    .spoiler-content:before {\n        display: inline-block; // Block display within the inline flow.\n        filter: blur(6px);\n        content: 'SPOILER'; // Display the word \"SPOILER\".\n    }\n\n    .spoiler-content .spoiler-hidden {\n        display: none; // Completely hide the actual content.\n    }\n\n    .spoiler-toggle:checked + .spoiler-content {\n        &:before {\n            content: none; // Hide the word \"SPOILER\".\n        }\n\n        .spoiler-hidden {\n            display: inline; // Reveal the actual content.\n        }\n    }\n}\n"
  },
  {
    "path": "sass/parts/_syntax_theme.scss",
    "content": "/*\n * based on \"Catppuccin Frappe\" https://github.com/catppuccin/catppuccin\n */\n\n:root {\n    --rosewater: #f2d5cf;\n    --flamingo: #eebebe;\n    --pink: #f4b8e4;\n    --mauve: #ca9ee6;\n    --red: #e78284;\n    --maroon: #ea999c;\n    --peach: #ef9f76;\n    --yellow: #e5c890;\n    --green: #a6d189;\n    --teal: #81c8be;\n    --sky: #99d1db;\n    --blue: #8caaee;\n    --lavender: #b4befe;\n    --text: #cdd6f4;\n    --overlay0: #737994;\n}\n\n.z-code {\n    background-color: var(--codeblock-bg);\n    color: var(--text);\n}\n\n.z-comment {\n    color: var(--overlay0);\n    font-style: italic;\n}\n\n.z-string {\n    color: var(--green);\n\n    &.z-regexp {\n        color: var(--sky);\n    }\n}\n\n.z-constant.z-numeric,\n.z-string.z-regexp {\n    color: var(--sky);\n}\n\n.z-constant {\n    &.z-character.z-escape {\n        color: var(--sky);\n    }\n\n    &.z-language {\n        color: var(--lavender);\n    }\n}\n\n.z-support.z-function.z-builtin.z-variable.z-other.z-constant {\n    color: var(--sky);\n}\n\n.z-keyword {\n    color: var(--red);\n\n    &.z-control {\n\n        &.z-loop,\n        &.z-conditional,\n        &.z-c + + {\n            color: var(--mauve);\n        }\n\n        &.z-return,\n        &.z-flow.z-return {\n            color: var(--pink);\n        }\n    }\n}\n\n.z-support.z-type.z-exception {\n    color: var(--sky);\n}\n\n.z-keyword.z-operator {\n    color: var(--teal);\n}\n\n.z-punctuation {\n    &.z-accessor {\n        color: var(--teal);\n    }\n\n    &.z-section {\n        color: var(--text);\n    }\n}\n\n.z-keyword.z-control.z-import.z-include {\n    color: var(--peach);\n}\n\n.z-storage {\n    color: var(--red);\n\n    &.z-type {\n        color: var(--yellow);\n    }\n\n    &.z-modifier {\n        color: var(--red);\n    }\n}\n\n.z-entity.z-name.z-namespace,\n.z-meta.z-path,\n.z-storage.z-type.z-class {\n    color: var(--rosewater);\n}\n\n.z-entity.z-name.z-label {\n    color: var(--blue);\n}\n\n.z-keyword.z-declaration.z-class {\n    color: var(--red);\n}\n\n.z-entity.z-name.z-class,\n.z-meta.z-toc-list.z-full-identifier {\n    color: var(--teal);\n}\n\n.z-entity {\n    &.z-other.z-inherited-class {\n        color: var(--teal);\n    }\n\n    &.z-name.z-function {\n        color: var(--blue);\n        font-style: italic;\n    }\n}\n\n.z-variable.z-function {\n    color: var(--blue);\n    font-style: italic;\n}\n\n.z-entity.z-name.z-function.z-preprocessor,\n.z-keyword.z-control.z-import {\n    color: var(--red);\n}\n\n.z-entity.z-name.z-function {\n\n    &.z-constructor,\n    &.z-destructor {\n        color: var(--lavender);\n    }\n}\n\n.z-variable.z-parameter.z-function {\n    color: var(--rosewater);\n}\n\n.z-keyword.z-declaration.z-function {\n    color: var(--maroon);\n}\n\n.z-support {\n    &.z-function {\n        color: var(--teal);\n    }\n\n    &.z-constant {\n        color: var(--blue);\n    }\n\n    &.z-type,\n    &.z-class {\n        color: var(--blue);\n        font-style: italic;\n    }\n}\n\n.z-variable {\n    &.z-function {\n        color: var(--blue);\n    }\n\n    &.z-parameter {\n        color: var(--rosewater);\n    }\n\n    &.z-other {\n        color: var(--text);\n\n        &.z-member {\n            color: var(--rosewater);\n        }\n    }\n\n    &.z-language {\n        color: var(--peach);\n    }\n}\n\n.z-entity {\n    &.z-name.z-tag {\n        color: var(--sky);\n    }\n\n    &.z-other.z-attribute-name {\n        color: var(--mauve);\n        font-style: italic;\n    }\n}\n\n.z-punctuation.z-definition.z-tag {\n    color: var(--maroon);\n}\n\n.z-markup.z-underline.z-link.z-markdown {\n    color: var(--rosewater);\n    font-style: underline;\n    font-style: italic;\n}\n\n.z-comment.z-block.z-markdown,\n.z-meta.z-code-fence {\n    color: var(--peach);\n    font-style: italic;\n}\n\n.z-markup.z-raw {\n\n    &.z-code-fence,\n    &.z-inline {\n        color: var(--peach);\n        font-style: italic;\n    }\n}\n\n.z-punctuation.z-definition.z-heading,\n.z-entity.z-name.z-section {\n    color: var(--blue);\n}\n\n.z-markup {\n    &.z-italic {\n        color: var(--maroon);\n        font-style: italic;\n    }\n\n    &.z-bold {\n        color: var(--maroon);\n        font-weight: bold;\n    }\n}\n\n.z-constant.z-character.z-escape,\n.z-source.z-shell.z-bash .z-meta.z-function.z-shell .z-meta.z-compound.z-shell .z-meta.z-function-call.z-identifier.z-shell {\n    color: var(--pink);\n}\n\n.z-variable.z-language.z-shell {\n    color: var(--red);\n}\n\n.z-source.z-lua .z-meta.z-function.z-lua .z-meta.z-block.z-lua .z-meta.z-mapping {\n    &.z-value.z-lua .z-meta.z-mapping.z-key.z-lua .z-string.z-unquoted.z-key.z-lua {\n        color: var(--lavender);\n    }\n\n    &.z-key.z-lua .z-string.z-unquoted.z-key.z-lua {\n        color: var(--flamingo);\n    }\n}\n\n.z-entity.z-name.z-constant.z-java {\n    color: var(--peach);\n}\n\n.z-support {\n    &.z-type.z-property-name.z-css {\n        color: var(--flamingo);\n    }\n\n    &.z-constant.z-property-value.z-css {\n        color: var(--text);\n    }\n}\n\n.z-constant.z-numeric.z-suffix.z-css,\n.z-keyword.z-other.z-unit.z-css,\n.z-variable.z-other.z-custom-property.z-name.z-css,\n.z-support.z-type.z-custom-property.z-name.z-css,\n.z-punctuation.z-definition.z-custom-property.z-css {\n    color: var(--peach);\n}\n\n.z-entity.z-name.z-tag.z-css {\n    color: var(--lavender);\n}\n\n.z-variable.z-other.z-sass {\n    color: var(--peach);\n}\n\n.z-invalid {\n    background-color: var(--red);\n    color: var(--text);\n\n    &.z-deprecated {\n        background-color: var(--mauve);\n        color: var(--text);\n    }\n}\n\n.z-meta.z-diff {\n    color: --OVERLAY0;\n\n    &.z-header {\n        color: --OVERLAY0;\n    }\n}\n\n.z-markup {\n    &.z-deleted {\n        color: var(--red);\n    }\n\n    &.z-inserted {\n        color: var(--green);\n    }\n\n    &.z-changed {\n        color: var(--yellow);\n    }\n}\n\n.z-message.z-error {\n    color: var(--red);\n}\n"
  },
  {
    "path": "sass/parts/_table.scss",
    "content": "table {\n    margin: 1rem auto;\n    border-style: hidden !important;\n    border-radius: 5px;\n    border-collapse: collapse;\n    border-spacing: 0;\n    overflow: hidden;\n    font: inherit;\n    text-align: center;\n\n    th,\n    td {\n        border: 1px solid var(--bg-1);\n        padding-inline: 13px;\n        padding-block: 6px;\n        font-size: large;\n    }\n\n    thead tr {\n        background-color: var(--primary-color);\n        color: var(--hover-color);\n\n        code {\n            background-color: transparent;\n        }\n    }\n\n    tbody {\n        tr:nth-child(even) {\n            background-color: var(--bg-0);\n        }\n    }\n\n    details,\n    summary {\n        font-family: inherit !important;\n    }\n}\n"
  },
  {
    "path": "sass/parts/_tags.scss",
    "content": "#tag-cloud {\n    margin-top: 4vmin;\n\n    ul {\n        margin: 0;\n        padding: 0;\n        list-style: none;\n    }\n\n    .tags-item {\n        margin-block-end: 1rem;\n    }\n}\n\n@mixin column-count($count) {\n    -webkit-column-count: $count;\n    -moz-column-count: $count;\n    column-count: $count;\n}\n\n.two-columns ul {\n    @include column-count(2);\n}\n\n.three-columns ul {\n    @include column-count(3);\n}\n\n@media (max-width: 1000px) {\n    .three-columns ul {\n        @include column-count(2);\n    }\n}\n\n@media (max-width: 600px) {\n\n    .two-columns ul,\n    .three-columns ul {\n        @include column-count(1);\n    }\n}\n"
  },
  {
    "path": "sass/parts/_theme-switch.scss",
    "content": ".theme-switcher {\n    -webkit-mask: var(--theme-switcher-svg);\n    position: relative;\n    align-self: center;\n    cursor: pointer;\n    margin-inline-start: 0.5rem;\n    background: var(--text-color);\n    width: 1rem;\n    height: 1rem;\n    &:hover {\n        background: var(--meta-color);\n    }\n}\n\n.theme-switcher-wrapper {\n    position: relative;\n}\n\n.theme-resetter {\n    -webkit-mask: url('data:image/svg+xml,%3Csvg xmlns=\"http://www.w3.org/2000/svg\" viewBox=\"0 -960 960 960\" %3E%3Cpath d=\"M295.87-193.869v-78.001h291.152q43.63 0 72.369-33.424 28.739-33.423 28.739-79.271t-28.739-79.391Q630.652-497.5 587.022-497.5H343.913l87.478 87.478-55.652 55.153L193.869-536.5l181.87-181.631 55.652 55.653-87.478 86.978h243.109q75.435 0 127.272 56.522 51.837 56.521 51.837 134.174 0 77.652-51.837 134.293-51.837 56.642-127.272 56.642H295.87Z\"/%3E%3C/svg%3E');\n    position: absolute;\n    visibility: hidden;\n    opacity: 0;\n    transition: opacity 0.3s ease, visibility 0.3s ease;\n    transition-delay: 0.5s;\n    cursor: pointer;\n    inset-block-start: -0.6rem;\n    inset-inline-end: -0.6rem;\n    background: var(--text-color);\n    width: 0.8rem;\n    height: 0.8rem;\n}\n\n.theme-switcher-wrapper:hover .theme-resetter.has-custom-theme {\n    visibility: visible;\n    opacity: 1;\n    transition: opacity 0.1s ease, visibility 0.1s ease;\n    transition-delay: 0s;\n    &:hover {\n        background: var(--meta-color);\n    }\n}\n"
  },
  {
    "path": "sass/parts/_webmention.scss",
    "content": "#webmentions {\n    position: relative;\n    z-index: 100;\n    margin: 0;\n    background-color: var(--background-color);\n    color: var(--text-color);\n    line-height: 1.2em;\n\n    h2 {\n        margin-bottom: 1.5em;\n        font-size: 1.1em;\n    }\n\n    h3 {\n        display: flex;\n        align-items: center;\n        font-size: 0.9em;\n\n        svg {\n            margin-inline-end: 0.2rem;\n        }\n\n        .svg-icon,\n        span {\n            margin-inline-end: .3rem;\n        }\n    }\n\n    ol {\n        padding: 0;\n    }\n\n    li,\n    p {\n        font-family: inherit;\n    }\n\n\n.likes {\n        display: flex;\n        flex-wrap: wrap;\n        margin-top: 0.5rem;\n        padding: 0;\n        list-style: none;\n\n        li {\n            position: relative;\n            transition: transform 0.8s ease-out, z-index 0s linear 0.4s;\n            margin-bottom: .375rem;\n            margin-inline-start: -.75rem;\n\n            &:first-child {\n                margin-inline-start: 0;\n            }\n\n            &:hover {\n                transform: scale(1.3) translateY(-4px);\n                z-index: 10;\n                transition: transform 0.05s ease-out, z-index 0s linear 0s;\n            }\n\n            img {\n                display: block;\n                border: 2px solid var(--background-color, white);\n                border-radius: 50%;\n                aspect-ratio: 1/1;\n                width: 2.5rem;\n                height: 2.5rem;\n                object-fit: cover;\n            }\n        }\n    }\n\n    .comment {\n        margin-bottom: 1rem;\n        border-radius: 10px;\n        background: var(--bg-0);\n        padding: 1rem;\n        overflow: hidden;\n        font-size: 80%;\n\n        div {\n            display: flex;\n            flex-wrap: nowrap;\n            justify-content: space-between;\n            align-items: center;\n        }\n\n        p {\n            margin-bottom: 0;\n            line-height: 1.5em;\n        }\n\n        .p-author {\n            font-style: bold;\n            font-size: 1.3em;\n        }\n\n        .u-url {\n            font-style: italic;\n            text-decoration: underline;\n        }\n\n        .u-author {\n            display: flex;\n            align-items: center;\n\n            img {\n                display: block;\n                margin-inline-end: .625rem;\n                width: 2rem;\n                max-width: 100%;\n                height: 2rem;\n            }\n        }\n    }\n\n    form {\n        input {\n            flex: 1;\n            border: 1px solid var(--divider-color);\n            border-radius: 20px 0px 0px 20px;\n            background-color: var(--input-background-color);\n            padding-inline: 1rem 1rem;\n            padding-block: .75rem;\n            width: calc(60% - 2rem);\n            color: var(--text-color);\n            font-size: 1rem;\n        }\n\n        button {\n            flex: 1;\n            border: 1px solid var(--divider-color);\n            border-radius: 0px 20px 20px 0px;\n            background-color: var(--input-background-color);\n            padding-inline: 0.7rem 0.7rem;\n            padding-block: .75rem;\n            width: 7rem;\n            color: var(--text-color);\n            font-size: 1rem;\n        }\n\n        button:hover {\n            cursor: pointer;\n            background-color: var(--primary-color);\n            color: var(--hover-color);\n        }\n    }\n}\n"
  },
  {
    "path": "sass/parts/_zola-error.scss",
    "content": "// Styles Zola error messages. See https://github.com/welpo/tabi/pull/359\nbody > div:last-child > div:last-child[style]:not([class]):not([id]) {\n    position: fixed;\n    top: 50%;\n    left: 50%;\n    transform: translate(-50%, -50%);\n    z-index: 9999;\n    box-shadow: rgba(50, 50, 93, 0.25) 0px 50px 100px -20px, rgba(0, 0, 0, 0.3) 0px 30px 60px -30px;\n    border: 2px solid var(--admonition-danger-border);\n    border-radius: 5px;\n    background-color: var(--admonition-danger-bg);\n    padding: 15px;\n    width: fit-content;\n    max-width: 80%;\n}\n\nbody > div:last-child > div:last-child[style]:not([class]):not([id]) > p[style]:first-child {\n    margin: 0;\n    color: var(--admonition-danger-border);\n    font-weight: bold;\n}\n\nbody > div:last-child > div:last-child[style]:not([class]):not([id]) > pre[style]:last-child {\n    margin-block-end: 0;\n    background-color: var(--admonition-danger-code);\n    padding: 10px;\n    overflow-x: auto;\n}\n"
  },
  {
    "path": "sass/skins/blue.scss",
    "content": "@mixin theme-variables($theme) {\n    @if $theme =='light' {\n        --primary-color: #3271E7; // Contrast ratio: 4.51:1\n    }\n    @else if $theme == 'dark' {\n        --primary-color: #6cacff; // Contrast ratio: 7.05:1\n    }\n}\n\n:root {\n    @include theme-variables('light');\n}\n\n[data-theme='dark'] {\n    @include theme-variables('dark');\n}\n\n@media (prefers-color-scheme: dark) {\n    :root:not([data-theme='light']) {\n        @include theme-variables('dark');\n    }\n}\n"
  },
  {
    "path": "sass/skins/evangelion.scss",
    "content": "@mixin theme-variables($theme) {\n    @if $theme =='light' {\n        // Evangelion Unit-02.\n        --primary-color: #d12e36;  // Contrast ratio: 5.05:1\n    }\n    @else if $theme == 'dark' {\n        // Evangelion Unit-01.\n        --primary-color: #c09bd9;  // Contrast ratio: 7.01:1\n    }\n}\n\n:root {\n    @include theme-variables('light');\n}\n\n[data-theme='dark'] {\n    @include theme-variables('dark');\n}\n\n@media (prefers-color-scheme: dark) {\n    :root:not([data-theme='light']) {\n        @include theme-variables('dark');\n    }\n}\n"
  },
  {
    "path": "sass/skins/indigo_ingot.scss",
    "content": "@mixin theme-variables($theme) {\n    @if $theme =='light' {\n        --primary-color: #1460bd; // Contrast ratio: 6.1:1\n    }\n    @else if $theme == 'dark' {\n        --primary-color: #e6c212; // Contrast ratio: 9.48:1\n    }\n}\n\n:root {\n    @include theme-variables('light');\n}\n\n[data-theme='dark'] {\n    @include theme-variables('dark');\n}\n\n@media (prefers-color-scheme: dark) {\n    :root:not([data-theme='light']) {\n        @include theme-variables('dark');\n    }\n}\n"
  },
  {
    "path": "sass/skins/lavender.scss",
    "content": "@mixin theme-variables($theme) {\n    @if $theme =='light' {\n        --primary-color: #9055d8;  // Contrast ratio: 4.69:1\n    }\n    @else if $theme == 'dark' {\n        --primary-color: #cba2e8;  // Contrast ratio: 7.74:1\n    }\n}\n\n:root {\n    @include theme-variables('light');\n}\n\n[data-theme='dark'] {\n    @include theme-variables('dark');\n}\n\n@media (prefers-color-scheme: dark) {\n    :root:not([data-theme='light']) {\n        @include theme-variables('dark');\n    }\n}\n"
  },
  {
    "path": "sass/skins/lowcontrast_orange.scss",
    "content": "// WARNING! This skin, in light theme, may not provide optimal contrast for readability\n// and might not be suitable for users with certain types of visual impairment.\n// Furthermore, low contrast will affect your Google Lighthouse rating.\n// For more information on web accessibility: https://www.w3.org/WAI/WCAG21/Understanding/contrast-minimum.html\n@mixin theme-variables($theme) {\n    @if $theme =='light' {\n        --primary-color: #f56a00; // Contrast ratio: 3.02:1. Not very accessible.\n    }\n    @else if $theme == 'dark' {\n        --primary-color: #ec984f; // Contrast ratio: 7.19:1. Accessible.\n    }\n}\n\n:root {\n    @include theme-variables('light');\n}\n\n[data-theme='dark'] {\n    @include theme-variables('dark');\n}\n\n@media (prefers-color-scheme: dark) {\n    :root:not([data-theme='light']) {\n        @include theme-variables('dark');\n    }\n}\n"
  },
  {
    "path": "sass/skins/lowcontrast_peach.scss",
    "content": "// WARNING! This skin, in light theme, may not provide optimal contrast for readability\n// and might not be suitable for users with certain types of visual impairment.\n// Furthermore, low contrast will affect your Google Lighthouse rating.\n// For more information on web accessibility: https://www.w3.org/WAI/WCAG21/Understanding/contrast-minimum.html\n@mixin theme-variables($theme) {\n    @if $theme =='light' {\n        --primary-color: #ffa057; // Contrast ratio: 2.01:1. Not very accessible.\n    }\n    @else if $theme == 'dark' {\n        --primary-color: #ffab7f; // Contrast ratio: 8.93:1. Accessible.\n    }\n}\n\n:root {\n    @include theme-variables('light');\n}\n\n[data-theme='dark'] {\n    @include theme-variables('dark');\n}\n\n@media (prefers-color-scheme: dark) {\n    :root:not([data-theme='light']) {\n        @include theme-variables('dark');\n    }\n}\n"
  },
  {
    "path": "sass/skins/lowcontrast_pink.scss",
    "content": "// WARNING! This skin, in light theme, may not provide optimal contrast for readability\n// and might not be suitable for users with certain types of visual impairment.\n// Furthermore, low contrast will affect your Google Lighthouse rating.\n// For more information on web accessibility: https://www.w3.org/WAI/WCAG21/Understanding/contrast-minimum.html\n@mixin theme-variables($theme) {\n    @if $theme =='light' {\n        --primary-color: #ee59d2; // Contrast ratio: 3:1. Not very accessible.\n    }\n    @else if $theme == 'dark' {\n        --primary-color: #f49ee9; // Contrast ratio: 9.87:1. Accessible.\n    }\n}\n\n:root {\n    @include theme-variables('light');\n}\n\n[data-theme='dark'] {\n    @include theme-variables('dark');\n}\n\n@media (prefers-color-scheme: dark) {\n    :root:not([data-theme='light']) {\n        @include theme-variables('dark');\n    }\n}\n"
  },
  {
    "path": "sass/skins/mint.scss",
    "content": "@mixin theme-variables($theme) {\n    @if $theme =='light' {\n        --primary-color: #00804d;  // Contrast ratio: 5:1\n    }\n    @else if $theme == 'dark' {\n        --primary-color: #00b86e;  // Contrast ratio: 6.34:1\n    }\n}\n\n:root {\n    @include theme-variables('light');\n}\n\n[data-theme='dark'] {\n    @include theme-variables('dark');\n}\n\n@media (prefers-color-scheme: dark) {\n    :root:not([data-theme='light']) {\n        @include theme-variables('dark');\n    }\n}\n"
  },
  {
    "path": "sass/skins/monochrome.scss",
    "content": "@mixin theme-variables($theme) {\n    @if $theme =='light' {\n        --primary-color: #727272;  // Contrast ratio: 4.81:1\n    }\n    @else if $theme == 'dark' {\n        --primary-color: #b3b3b3;  // Contrast ratio: 7.86:1\n    }\n}\n\n:root {\n    @include theme-variables('light');\n}\n\n[data-theme='dark'] {\n    @include theme-variables('dark');\n}\n\n@media (prefers-color-scheme: dark) {\n    :root:not([data-theme='light']) {\n        @include theme-variables('dark');\n    }\n}\n"
  },
  {
    "path": "sass/skins/red.scss",
    "content": "@mixin theme-variables($theme) {\n    @if $theme =='light' {\n        --primary-color: #ca4963; // Contrast ratio: 4.52:1.\n    }\n    @else if $theme == 'dark' {\n        --primary-color: #ea535f; // Contrast ratio: 4.63:1.\n    }\n}\n\n:root {\n    @include theme-variables('light');\n}\n\n[data-theme='dark'] {\n    @include theme-variables('dark');\n}\n\n@media (prefers-color-scheme: dark) {\n    :root:not([data-theme='light']) {\n        @include theme-variables('dark');\n    }\n}\n"
  },
  {
    "path": "sass/skins/sakura.scss",
    "content": "@mixin theme-variables($theme) {\n    @if $theme =='light' {\n        --primary-color: #D33C5C;  // Contrast ratio: 4.61:1\n    }\n    @else if $theme == 'dark' {\n        --primary-color: #fabed2;  // Contrast ratio: 10.48:1\n    }\n}\n\n:root {\n    @include theme-variables('light');\n}\n\n[data-theme='dark'] {\n    @include theme-variables('dark');\n}\n\n@media (prefers-color-scheme: dark) {\n    :root:not([data-theme='light']) {\n        @include theme-variables('dark');\n    }\n}\n"
  },
  {
    "path": "sass/skins/teal.scss",
    "content": "// When creating your own skin, you can use https://webaim.org/resources/contrastchecker/\n// to verify the accessibility and readability of your colourscheme.\n// The default light background is #fff and the dark background is #1f1f1f.\n\n// This defines theme-specific variables.\n@mixin theme-variables($theme) {\n    @if $theme =='light' {\n        // Light theme colours.\n        --primary-color: #087e96; // Contrast ratio: 4.73:1\n    }\n    @else if $theme == 'dark' {\n        // Dark theme colours.\n        --primary-color: #91e0ee;  // Contrast ratio: 11.06:1\n    }\n}\n\n// Apply light theme variables by default.\n:root {\n    @include theme-variables('light');\n}\n\n// Apply dark theme variables when dark theme is explicitly set.\n[data-theme='dark'] {\n    @include theme-variables('dark');\n}\n\n// Apply dark theme variables when user's system prefers dark mode\n// and the theme is not explicitly set to light.\n@media (prefers-color-scheme: dark) {\n    :root:not([data-theme='light']) {\n        @include theme-variables('dark');\n    }\n}\n"
  },
  {
    "path": "scripts/upgrade-deps",
    "content": "#!/usr/bin/env bash\nset -eu\n\nMERMAID_DIR=\"static/js\"\nMERMAID_FILE=\"mermaid.min.js\"\nMERMAID_PATH=\"${MERMAID_DIR}/${MERMAID_FILE}\"\nKATEX_JS_DIR=\"static/js\"\nKATEX_CSS_DIR=\"static\"\nKATEX_FONTS_DIR=\"static/fonts/KaTeX\"\nKATEX_JS_FILE=\"katex.min.js\"\nKATEX_CSS_FILE=\"katex.min.css\"\nKATEX_MHCHEM_FILE=\"mhchem.min.js\"\nKATEX_JS_PATH=\"${KATEX_JS_DIR}/${KATEX_JS_FILE}\"\nKATEX_CSS_PATH=\"${KATEX_CSS_DIR}/${KATEX_CSS_FILE}\"\nKATEX_MHCHEM_PATH=\"${KATEX_JS_DIR}/${KATEX_MHCHEM_FILE}\"\nUGLIFY_ITERATIONS=5\nCURL_RETRIES=3\n\ncleanup() {\n    rm -rf \"$TEMP_DIR\"\n}\n\nexit_with_message() {\n    echo \"$1\" >&2\n    exit 1\n}\n\nprint_usage() {\n    echo \"Usage: $0 [options]\"\n    echo \"Options:\"\n    echo \"  --mermaid   Upgrade Mermaid.js\"\n    echo \"  --katex     Upgrade KaTeX\"\n    echo \"  --all       Upgrade all dependencies (default)\"\n    echo \"  --help      Display this help message\"\n}\n\ncheck_dependency() {\n    if ! command -v \"$1\" &> /dev/null; then\n        exit_with_message \"$1 is required but not installed.\"\n    fi\n}\n\ncurl_with_retry() {\n    local url=\"$1\"\n    local output=\"$2\"\n    local retries=\"$CURL_RETRIES\"\n    local wait_time=5\n\n    while [ $retries -gt 0 ]; do\n        if curl -L \"$url\" -o \"$output\"; then\n            return 0\n        else\n            echo \"Curl failed. Retrying in $wait_time seconds…\"\n            sleep $wait_time\n            retries=$((retries - 1))\n            wait_time=$((wait_time * 2))\n        fi\n    done\n\n    echo \"Failed to download after $CURL_RETRIES attempts.\" >&2\n    return 1\n}\n\nget_latest_version_jsdelivr() {\n    local package=\"$1\"\n    local temp_file=\"${TEMP_DIR}/jsdelivr_response.json\"\n    if curl_with_retry \"https://data.jsdelivr.com/v1/package/npm/${package}\" \"$temp_file\"; then\n        jq -r '.tags.latest' \"$temp_file\"\n    else\n        return 1\n    fi\n}\n\nget_latest_version_github() {\n    local repo=\"$1\"\n    local temp_file=\"${TEMP_DIR}/github_response.json\"\n    if curl_with_retry \"https://api.github.com/repos/${repo}/releases/latest\" \"$temp_file\"; then\n        jq -r '.tag_name' \"$temp_file\" | sed -E 's/^v?//'\n    else\n        return 1\n    fi\n}\n\nget_local_mermaid_version() {\n    local version\n    # Old format.\n    version=$(grep -o '[A-Za-z]\\+t=\"[0-9]\\+\\.[0-9]\\+\\.[0-9]\\+\"' \"$MERMAID_PATH\" | grep -o '[0-9]\\+\\.[0-9]\\+\\.[0-9]\\+' || true)\n\n    # New format.\n    if [ -z \"$version\" ]; then\n        version=$(grep -o 'version:\"[0-9]\\+\\.[0-9]\\+\\.[0-9]\\+\"' \"$MERMAID_PATH\" | grep -o '[0-9]\\+\\.[0-9]\\+\\.[0-9]\\+' || true)\n    fi\n\n    if [ -z \"$version\" ]; then\n        exit_with_message \"Could not detect local Mermaid.js version\"\n    fi\n    echo \"$version\"\n}\n\n\nget_local_katex_version() {\n    local version\n    version=$(sed -n 's/.*version:\"\\([^\"]*\\)\".*/\\1/p' \"$KATEX_JS_PATH\" | head -n 1)\n    if [ -z \"$version\" ]; then\n        echo \"Could not detect local KaTeX version\" >&2\n        return 1\n    fi\n    echo \"$version\"\n}\n\ncompare_md5() {\n    local new_file=\"$1\"\n    local current_file=\"$2\"\n    local new_md5\n    new_md5=$(md5sum \"$new_file\" | awk '{ print $1 }')\n\n    if [ -f \"$current_file\" ]; then\n        local current_md5\n        current_md5=$(md5sum \"$current_file\" | awk '{ print $1 }')\n        if [ \"$new_md5\" = \"$current_md5\" ]; then\n            echo \"same\"\n        else\n            echo \"different\"\n        fi\n    else\n        echo \"new\"\n    fi\n}\n\nuglify_file() {\n    local file=\"$1\"\n    local iterations=\"$2\"\n\n    for i in $(seq 1 \"$iterations\"); do\n        echo \"Running UglifyJS iteration $i\"\n        uglifyjs --compress --mangle -- \"$file\" > \"${file}.tmp\"\n        mv \"${file}.tmp\" \"$file\"\n    done\n}\n\ngenerate_commit_message() {\n    local template=\"$1\"\n    local version=\"$2\"\n    echo \"${template//\\{VERSION\\}/${version}}\"\n}\n\n\nsafe_file_manipulation() {\n    local file=\"$1\"\n    local manipulation_command=\"$2\"\n    local temp_file=\"${file}.tmp\"\n    awk \"${manipulation_command}\" \"$file\" > \"$temp_file\"\n    mv \"$temp_file\" \"$file\"\n}\n\nappend_autorender_extension() {\n    local file=\"$1\"\n    # Auto-render Extension (https://katex.org/docs/autorender) with a slight modification to add `$` inline delimiters.\n    local extension_code='\n,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}},$={};function r(e){var _=$[e];if(void 0!==_)return _.exports;var n=$[e]={exports:{}};return t[e](n,n.exports,r),n.exports}r.n=function(e){var t=e&&e.__esModule?function(){return e.default}:function(){return e};return r.d(t,{a:t}),t},r.d=function(e,t){for(var $ in t)r.o(t,$)&&!r.o(e,$)&&Object.defineProperty(e,$,{enumerable:!0,get:t[$]})},r.o=function(e,t){return Object.prototype.hasOwnProperty.call(e,t)};var _,n,a,i,o,s,l,h,m={};return r.d(m,{default:function(){return h}}),_=r(771),n=r.n(_),a=function(e,t,$){for(var r=$,_=0,n=e.length;r<t.length;){var a=t[r];if(_<=0&&t.slice(r,r+n)===e)return r;\"\\\\\"===a?r++:\"{\"===a?_++:\"}\"===a&&_--,r++}return -1},i=/^\\\\begin{/,o=function(e,t){for(var $,r=[],_=RegExp(\"(\"+t.map(function(e){return e.left.replace(/[-/\\\\^$*+?.()|[\\]{}]/g,\"\\\\$&\")}).join(\"|\")+\")\");-1!==($=e.search(_));){$>0&&(r.push({type:\"text\",data:e.slice(0,$)}),e=e.slice($));var n=t.findIndex(function(t){return e.startsWith(t.left)});if(-1===($=a(t[n].right,e,t[n].left.length)))break;var o=e.slice(0,$+t[n].right.length),s=i.test(o)?o:e.slice(t[n].left.length,$);r.push({type:\"math\",data:s,rawData:o,display:t[n].display}),e=e.slice($+t[n].right.length)}return\"\"!==e&&r.push({type:\"text\",data:e}),r},s=function(e,t){var $=o(e,t.delimiters);if(1===$.length&&\"text\"===$[0].type)return null;for(var r=document.createDocumentFragment(),_=0;_<$.length;_++)if(\"text\"===$[_].type)r.appendChild(document.createTextNode($[_].data));else{var a=document.createElement(\"span\"),i=$[_].data;t.displayMode=$[_].display;try{t.preProcess&&(i=t.preProcess(i)),n().render(i,a,t)}catch(s){if(!(s instanceof n().ParseError))throw s;t.errorCallback(\"KaTeX auto-render: Failed to parse `\"+$[_].data+\"` with \",s),r.appendChild(document.createTextNode($[_].rawData));continue}r.appendChild(a)}return r},l=function e(t,$){for(var r=0;r<t.childNodes.length;r++){var _=t.childNodes[r];if(3===_.nodeType){for(var n=_.textContent,a=_.nextSibling,i=0;a&&a.nodeType===Node.TEXT_NODE;)n+=a.textContent,a=a.nextSibling,i++;var o=s(n,$);if(o){for(var l=0;l<i;l++)_.nextSibling.remove();r+=o.childNodes.length-1,t.replaceChild(o,_)}else r+=i}else 1===_.nodeType&&function(){var t=\" \"+_.className+\" \";-1===$.ignoredTags.indexOf(_.nodeName.toLowerCase())&&$.ignoredClasses.every(function(e){return -1===t.indexOf(\" \"+e+\" \")})&&e(_,$)}()}},h=function(e,t){if(!e)throw Error(\"No element provided to render\");var $={};for(var r in t)t.hasOwnProperty(r)&&($[r]=t[r]);$.delimiters=$.delimiters||[{left:\"$$\",right:\"$$\",display:!0},{left:\"$\",right:\"$\",display:!1},{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}],$.ignoredTags=$.ignoredTags||[\"script\",\"noscript\",\"style\",\"textarea\",\"pre\",\"code\",\"option\"],$.ignoredClasses=$.ignoredClasses||[],$.errorCallback=$.errorCallback||console.error,$.macros=$.macros||{},l(e,$)},m=m.default}()}),document.addEventListener(\"DOMContentLoaded\",function(){renderMathInElement(document.body)})\n'\n    safe_file_manipulation \"$file\" '{gsub(/;$/,\"\"); print $0}'\n    echo \"$extension_code\" >> \"$file\"\n    echo \";\" >> \"$file\"\n}\n\nmodify_katex_css() {\n    local file=\"$1\"\n    safe_file_manipulation \"$file\" '{gsub(/url\\(fonts\\/KaTeX/, \"url(fonts/KaTeX/KaTeX\"); print $0}'\n}\n\nupgrade_mermaid() {\n    echo\n    echo \"Starting Mermaid.js update…\"\n    if [ ! -d \"$MERMAID_DIR\" ]; then\n        exit_with_message \"Directory ${MERMAID_DIR} does not exist. Are you running this script from the root of the repository?\"\n    fi\n\n    local commit_msg_template\n    commit_msg_template=$(cat << EOM\n⬆️ chore(deps): upgrade mermaid to v{VERSION}\n\nChangelog: https://github.com/mermaid-js/mermaid/releases/tag/mermaid%40{VERSION}\n\nSource: https://cdn.jsdelivr.net/npm/mermaid@{VERSION}/dist/mermaid.min.js\nEOM\n)\n\n    local latest_version\n    latest_version=$(get_latest_version_jsdelivr \"mermaid\" || get_latest_version_github \"mermaid-js/mermaid\")\n    if [ -z \"$latest_version\" ]; then\n        exit_with_message \"Unable to determine the latest Mermaid.js version.\"\n    fi\n\n    local local_version\n    local_version=$(get_local_mermaid_version)\n    echo \"Latest Mermaid.js version: ${latest_version}\"\n    echo \"Local Mermaid.js version: ${local_version}\"\n    if [ \"$latest_version\" = \"$local_version\" ]; then\n        echo \"Mermaid.js is already up to date. Skipping update.\"\n        return 0\n    fi\n\n    local download_url\n    download_url=\"https://cdn.jsdelivr.net/npm/mermaid@${latest_version}/dist/mermaid.min.js\"\n    if ! curl_with_retry \"${download_url}\" \"${TEMP_DIR}/${MERMAID_FILE}\"; then\n        exit_with_message \"Failed to download Mermaid.js\"\n    fi\n\n    uglify_file \"${TEMP_DIR}/${MERMAID_FILE}\" \"$UGLIFY_ITERATIONS\"\n    local comparison_result\n    comparison_result=$(compare_md5 \"${TEMP_DIR}/${MERMAID_FILE}\" \"${MERMAID_PATH}\")\n\n    case \"$comparison_result\" in\n        \"same\")\n            echo \"Mermaid: New version is the same as current version. No update needed.\"\n            return 0\n            ;;\n        \"different\")\n            echo \"Mermaid: New version differs from current version. Proceeding with update.\"\n            mv \"${TEMP_DIR}/${MERMAID_FILE}\" \"${MERMAID_PATH}\"\n            ;;\n        \"new\")\n            echo \"Mermaid: Creating new file: ${MERMAID_PATH}\"\n            mv \"${TEMP_DIR}/${MERMAID_FILE}\" \"${MERMAID_PATH}\"\n            ;;\n    esac\n\n    echo \"Mermaid.js updated and minified successfully!\"\n    echo \"Preparing to commit changes…\"\n    git add \"${MERMAID_PATH}\"\n    local commit_msg\n    commit_msg=$(generate_commit_message \"$commit_msg_template\" \"$latest_version\")\n    git commit -m \"${commit_msg}\"\n\n    echo \"Most recent commit:\"\n    git log -1\n}\n\nupgrade_katex() {\n    echo\n    echo \"Starting KaTeX update…\"\n    if [ ! -d \"$KATEX_JS_DIR\" ] || [ ! -d \"$KATEX_CSS_DIR\" ]; then\n        exit_with_message \"KaTeX directories do not exist. Are you running this script from the root of the repository?\"\n    fi\n\n    local commit_msg_template\n    commit_msg_template=$(cat << EOM\n⬆️ chore(deps): upgrade KaTeX to v{VERSION}\n\nChangelog: https://github.com/KaTeX/KaTeX/releases/tag/v{VERSION}\n\nSource: https://github.com/KaTeX/KaTeX/releases/download/v{VERSION}/katex.tar.gz\nEOM\n)\n\n    local latest_version\n    latest_version=$(get_latest_version_github \"KaTeX/KaTeX\")\n    local local_version\n    local_version=$(get_local_katex_version)\n    if [ -z \"$local_version\" ]; then\n        exit_with_message \"Unable to determine the local KaTeX version.\"\n    fi\n\n    echo \"Latest KaTeX version: ${latest_version}\"\n    echo \"Local KaTeX version: ${local_version}\"\n    if [ \"$latest_version\" = \"$local_version\" ]; then\n        echo \"KaTeX is already up to date. Skipping update.\"\n        return 0\n    fi\n\n    local download_url=\"https://github.com/KaTeX/KaTeX/releases/download/v${latest_version}/katex.tar.gz\"\n    if ! curl_with_retry \"${download_url}\" \"${TEMP_DIR}/katex.tar.gz\"; then\n        exit_with_message \"Failed to download KaTeX\"\n    fi\n\n    tar -xzf \"${TEMP_DIR}/katex.tar.gz\" -C \"${TEMP_DIR}\"\n\n    # JS.\n    cp \"${TEMP_DIR}/katex/katex.min.js\" \"${TEMP_DIR}/${KATEX_JS_FILE}\"\n    append_autorender_extension \"${TEMP_DIR}/${KATEX_JS_FILE}\"\n    uglify_file \"${TEMP_DIR}/${KATEX_JS_FILE}\" \"$UGLIFY_ITERATIONS\"\n    local js_comparison_result\n    js_comparison_result=$(compare_md5 \"${TEMP_DIR}/${KATEX_JS_FILE}\" \"${KATEX_JS_PATH}\")\n\n    # CSS.\n    cp \"${TEMP_DIR}/katex/katex.min.css\" \"${TEMP_DIR}/${KATEX_CSS_FILE}\"\n    modify_katex_css \"${TEMP_DIR}/${KATEX_CSS_FILE}\"\n    local css_comparison_result\n    css_comparison_result=$(compare_md5 \"${TEMP_DIR}/${KATEX_CSS_FILE}\" \"${KATEX_CSS_PATH}\")\n\n    # mhchem extension.\n    cp \"${TEMP_DIR}/katex/contrib/mhchem.min.js\" \"${TEMP_DIR}/${KATEX_MHCHEM_FILE}\"\n    uglify_file \"${TEMP_DIR}/${KATEX_MHCHEM_FILE}\" \"$UGLIFY_ITERATIONS\"\n    local mhchem_comparison_result\n    mhchem_comparison_result=$(compare_md5 \"${TEMP_DIR}/${KATEX_MHCHEM_FILE}\" \"${KATEX_MHCHEM_PATH}\")\n\n    if [ \"$js_comparison_result\" = \"same\" ] && [ \"$css_comparison_result\" = \"same\" ] && [ \"$mhchem_comparison_result\" = \"same\" ]; then\n        echo \"KaTeX: New version is the same as current version. No update needed.\"\n        return 0\n    fi\n\n    local changes_made\n    changes_made=false\n    if [ \"$js_comparison_result\" != \"same\" ]; then\n        echo \"KaTeX JS: New version differs from current version. Proceeding with update.\"\n        mv \"${TEMP_DIR}/${KATEX_JS_FILE}\" \"${KATEX_JS_PATH}\"\n        changes_made=true\n    fi\n\n    if [ \"$css_comparison_result\" != \"same\" ]; then\n        echo \"KaTeX CSS: New version differs from current version. Proceeding with update.\"\n        mv \"${TEMP_DIR}/${KATEX_CSS_FILE}\" \"${KATEX_CSS_PATH}\"\n        changes_made=true\n    fi\n\n    if [ \"$mhchem_comparison_result\" != \"same\" ]; then\n        echo \"KaTeX mhchem: New version differs from current version. Proceeding with update.\"\n        mv \"${TEMP_DIR}/${KATEX_MHCHEM_FILE}\" \"${KATEX_MHCHEM_PATH}\"\n        changes_made=true\n    fi\n\n    rm -rf \"${KATEX_FONTS_DIR}\"\n    mkdir -p \"${KATEX_FONTS_DIR}\"\n    cp -r \"${TEMP_DIR}/katex/fonts\"/* \"${KATEX_FONTS_DIR}/\"\n\n    if [ \"$changes_made\" = false ]; then\n        echo \"No changes detected in KaTeX files. Skipping commit.\"\n        return 0\n    fi\n\n    echo \"KaTeX updated successfully!\"\n    echo \"Preparing to commit changes…\"\n    git add \"${KATEX_JS_PATH}\" \"${KATEX_CSS_PATH}\" \"${KATEX_MHCHEM_PATH}\" \"${KATEX_FONTS_DIR}\"\n    local commit_msg\n    commit_msg=$(generate_commit_message \"$commit_msg_template\" \"$latest_version\")\n    git commit -m \"${commit_msg}\"\n\n    echo \"Most recent commit:\"\n    git log -1\n}\n\nmain() {\n    local upgrade_mermaid=false\n    local upgrade_katex=false\n    # No args = default to upgrading all dependencies.\n    if [ $# -eq 0 ]; then\n        upgrade_mermaid=true\n        upgrade_katex=true\n    else\n        while [[ $# -gt 0 ]]; do\n            case $1 in\n                --mermaid)\n                    upgrade_mermaid=true\n                    shift\n                    ;;\n                --katex)\n                    upgrade_katex=true\n                    shift\n                    ;;\n                --all)\n                    upgrade_mermaid=true\n                    upgrade_katex=true\n                    shift\n                    ;;\n                --help)\n                    print_usage\n                    exit 0\n                    ;;\n                *)\n                    echo \"Unknown option: $1\"\n                    print_usage\n                    exit 1\n                    ;;\n            esac\n        done\n    fi\n\n    check_dependency \"jq\"\n    check_dependency \"uglifyjs\"\n    check_dependency \"curl\"\n    check_dependency \"git\"\n    check_dependency \"sed\"\n    check_dependency \"awk\"\n    check_dependency \"md5sum\"\n    check_dependency \"tar\"\n    TEMP_DIR=$(mktemp -d)\n    trap cleanup EXIT\n\n\n    if ! git diff --cached --quiet; then\n        exit_with_message \"There are staged changes. Unstage them before running this script.\"\n    fi\n\n    echo \"Updating local repository…\"\n    git fetch origin\n    current_branch=$(git rev-parse --abbrev-ref HEAD)\n    echo \"Current branch: $current_branch\"\n    # Check if the branch exists on the remote\n    if git ls-remote --exit-code --heads origin \"$current_branch\" >/dev/null 2>&1; then\n        # Branch exists on remote, compare with local.\n        local_commit=$(git rev-parse HEAD)\n        remote_commit=$(git rev-parse origin/\"$current_branch\")\n        if [ \"$local_commit\" = \"$remote_commit\" ]; then\n            echo \"Branch is up to date with origin/$current_branch\"\n        elif git merge-base --is-ancestor \"$remote_commit\" \"$local_commit\"; then\n            echo \"Local branch is ahead of origin/$current_branch\"\n        else\n            exit_with_message \"Your local branch is behind origin/$current_branch. Pull the latest changes before running this script.\"\n        fi\n    else\n        echo \"Branch $current_branch does not exist on remote. Assuming it's a new branch.\"\n    fi\n    echo \"Local repository is ready.\"\n\n    if [ \"$upgrade_mermaid\" = true ]; then\n        upgrade_mermaid\n    fi\n\n    if [ \"$upgrade_katex\" = true ]; then\n        upgrade_katex\n    fi\n}\n\nmain \"$@\"\n"
  },
  {
    "path": "static/custom_subset.css",
    "content": "@font-face{font-family:\"Inter Subset\";src:url(data:application/font-woff2;base64,d09GMgABAAAAABzEABMAAAAALzgAABxWAAQAQgAAAAAAAAAAAAAAAAAAAAAAAAAAGoJjG4gGHCo/SFZBUoIJP01WQVKBKAZgP1NUQVSBXCc0AHwvgTYKjFSKdjCvWgE2AiQDVAssAAQgBYlAByAb4ywzo8HGAQTB7wEj+T8ccFME86LTO1iwBhRNtzeaN4zFsemn0TDWYsm9p6RthgWjFSUK/mPvRjzNKtorVwg/tIZur9XWgup3iqPA/pp8tgX/3zgNCQIRRyFYAP91SrXlCElmISKc2uydJNsnxSE5zVPqAmAKmBKDSyoQugDwSOR/9sWcfkokCVRdFdAAwQU+AKR3fkKXJ75737uz758PlFJVYo1Kx4S4GEQgFqsaDc8A3ewaKF/NkbsG+ikMj2d9r3KgBoVTT4ZK2zGKoZbsfsC6W+P/dJZ/QIQrL9ABYHlB6JKKqjS1Jb8+qTqEeuEAybbYgtEMEbra9A1PSApGIS3OdTmCuyE7+58voCmFyk37i/pUf7F7AHO5ZONNFpqFZqWQSLE75pKQo1bAUepvmv//NtNWd1HqPAtUjeEfPzkkJ0WTk6IC7JOi4tGMshpY27O7Ot6xZBizZPxmaRUEqvhrTD/8zRPkHqtUbcoUTZmiDYiITsm0nOstpOWZbNfSliXXCo6DGSQrbiqElMZh/3kBBE0OeG1JCoUQ8t/kkk0QoLIIMBEwF7ASbAdXwJPgY5CGRFHY0MwxYA95CIMIU7BUF5VGgQUIyom4ZfF4Xq2/M1J/kukVlkl5hRxPQ0PrE99857+VwMUOm+rSsvzR60pXNOakUv6Pbe7nocnCfGP43EdW5edhw/i87wBL2aSU7vmwmI+eU+tJo0yin6QEBMzFvkkNkRBgMyQH+6P01Zq06v+2vJ3P3hlHvnNdqfI7y8MFft1PfdcZk/cJiW5oZLC0VdH9O3TP55kyTK6noxR9D8Ns1SrDi5NOG0iyHtSEUaYKDeY2ZZaiBJboiZQ65IlAivCmCuguE1cU5bXyDamuPoWGW7qVHLmkg5iiplBFpeIkp1lVz0CqeJ0FWf24GLURP0v52d6bU8KpS38Shk1Usm8zmqw9x6gryufW89W7vTl58r3lx+AnSD3DuvG8lL6f673zXqt8AuorEz98IKQv5jf8/E18kr43vwVGe3U8eMHM77gVHEqX7vlZ8ZbS62Eeunp8+9XFTzrdyImtW5AfMypgyA1NEChzj8gCC4bf9gWbEBoHQxgPe4aiQqQERhpqNAELbsJevASWoGm0FsbSFDSHmp2SjULAmYFms/IsADMcEotZbIZDDlVUCJyDgzSjWVACoNmGKgnow4aYSTBDA3RuIphcVnnKawz+9naedyA81Qc/yXN4p68Y/MJeBI+PupC7gYyR7bez5bl63EjcaLTNX67nffV234r7tMPNuz7zVrfui3I2t76Ofnbw+Kc1t5gTq36+y9OnF077blxcOO2/tVyw4rv3XPUxN4Jgoy7bYdPrnQMwgmpwF8UPmm7x/2nwJUtM/gZKYKTZTNNmBIfgqI1hAQJAGJva/wF4IQDSw1qkZk1mZI7WgjVzx3I2GEvzkBu644J0FTHNaPr15mp59zMdOYW+9e/hIwGQs4uPzWyway3PA3vFNPU863PIx9408WRAYAu7Hz28zbSaRev9/xGjmy3vcbo12L6MF6/RIiEMjebz9qQyNWWqFRIGmHPN2wQM+aPn90YYU+QrrbpG2nrH0MX+bJckAFJJFJKdlcUhvYqRvRZPaY5UhtaVhGXzmLpPvcDIb9Drqb5XjuJ6s2a8lDOX00EVqwLyNrmrQrtgZkjygd7zbsfOXllT9qok1rJW9E6HQuXikkbEToux310WjUU1IyCRTGdzpS1UMwziyDSmoqq/7S8b7QLtK6vMkp2EaJ7XYeMQwiCisigUvghRV1hoRBEfRdyyCBhgM6veqxERhTNV87RYEjzVZYuTEvzebgWp+L5bQzryLgOZKemX3Ray8W+Xg9yUbGO3J/Ex0KYA8HAB/jIFkDy1IOW7mwJsPHclsbUGseAiityTeCdCQh6fByLlFnUjgLDgLDLSRf2359PWGxKIaffZhS+KBE+IWcu/Vrl3+Ka7TgiObsSce5rsvS4U2c7rlG4++60zabi+omESruWe1VjyNltKbb1X/rLkHQlUwgf16nmwXrfdz43UivfiL4rcDEue7zSDaLfWxp42LX6jRZ/T0j4RcwqmIEGJuQpuccNLpqsbXI6RCVZJXbOxajVqB7HvRgbsxCHxetEYeVc/Ma86WJpLVSe739BVqLNnTHFzfULalBsxNysrMShVhdutVujdp/x2sJrZ+2baxr2yy05+r17hs881CWwZ4IaXfLGoVq7XkawqkjHfHXS0zW0rfYa4cxNzTWLAGxTFXs8TtKPu6yYJer9AdnP4XokhN0D8b8bm31Br+YIHQ0fRS4YG1rZ40dWbNZn6ZsOBPk88d+ikZee2cvOiBplzRf7syyoumvYUbaR3tPZmIT5eoHmeHh0dtpXQ2JrgNk1ezOl2/mBTnJbgFirztGxwcygMntTlmOzMNr/WSSb+zMNwv7Ksawy1HGj1LUZdRdaphcU5+ip6NDlufVnRbvVM703UpPG5yNyi9aYFvjuo8WTJ7v4T26qn5xZbT7G0bpv0V7dEV8i0fdDs+ZxE3GoAW6/skqrWO2WTvbexM6baDeLaFtQIfHGqOh7QU1UbgTcAM2SP/A/bItPTCfsxkdbvnMOF23u8zgwcSePsLJwsTnYNHZWNzegAJY5qip5m9B3tUajdqclvxtQcuaF11ZeMjaZrLmsmjbakFvsE19kMP3t68dqH4qZevW0eEzuvtSS+zM3M+/86EjlxiU/rROPspUcSm8Eb70WvLxyNTGoz2zDHZxk5ekwBuqhopWyG7xx6OOF90DW4X1lGFBcYLwaLp00mDd7EPLGyFqzcMqtrq7d/BwClXEuB/7WU67Pk+1PxXSZpnbC3JOIiAWnty9rZdnGG8fd8vibW3PRi/ezvJ6dtqWy9XTLFexc/c5p5Xjt9A/BwYck3/xDKPY8q2lpY0Q3/TVDKhWnP0vPmnJFSw/aulP+h1bbZw7cqaBuQl6183GMGQjJzvx6ZuCN6ZHVvZmuHrk0REncHrnFecfXsoYWEze5zKvu9RtbJdb5df3YmvSirPbHIpV9Duswp7zXOy66eMbyEdjwW7aw0w2PvWORJnw5OrRLibTq1dj1XFNHcctvuNTyil9OPv6NoT+Q9x6bWKA3Je86Vs4ty7E0Nhj5UAc/RXYsFY6uyoYPc2aJ9QL1z13jwN50Kn/7YYdj6TnLHOUZQo1uQR8pTHYfAdiAFEJ2F1/KPmdf0Fy1r61vUWEcCHGUJAw4+R9QSqYYBiQMLgD5zjpKzPk+RmiUE/JDs1CyNnt8g0kZqrr/QfVkh7VT8b/94r0T+SrlAeTg0oe6mfueB7x+c1T794O6HNuoeBTAGXvVJ9GZ5KEs0ik36+gicl+0dq/EaSr+Oe2WmJy/8IDRYKUWysmKkafXQ/I0FelJlHEWJohXbfpp1uXDePEs6SkIbnIisVseTdBg4ZLiubXkyQwuY+i3QrRuV5FMoXhw5nPTzTq+FpBMCJ9iTqN1JI0RZr0EkKty1MAPSdzb4+/axQHqcx0oX6UQp+b1qzVWvYKgVh0n1a5vkx1ZpF3IL7R/+ilaZvQusRk2KeRmvXj1QTNXtZQeqMsuhq6baeK3FX8Ur657qbURDg0fDP2xZ5W3o7e5oa2mssaP63nk8QuqdKimRjt9e3duYHtyFtRePFs1Vg7XI49v9scsv9InrIkbbWfR67u6d2aEFg4O3T0TEwInVnlDlqhpwANTblkbQllmtiqJMmd6DdR5hMB6e7PWrHD2GX525usP5BxW8si1PKY+G6neTlVr110MNimVYaTgjdE9mS8/wsU36QDaQZ/NIORQ6hlAcDnxO98h+QKCW5908q5RpUKhyQSeyTn8YqTyhgATx0+u6KIks8qISknLHzfrQK1fmbwxRw3kc+IBML+w4l6T4jrY5Ms4Ux8N2ktRSVIiiJp10Z5rCUc4nDUbDHn/EZavMdOf6ytkw40kSRQzWPMqk4QkVncJhg8oBDSqC+KFRIziItm3d1JOsbhtoTQK+LodWbxZMZ7Zljk4nRXvSwmy0ag5GaWELSFyGog6JDrRjJTrQruR7cdTeqEoSQhDi8qI+T9YKSaoEXcKapIjRAoOr0BxIhltOHwqDLLrzO3n8oo4Eh+4QUcaeQLv7hCGOaNjBgEXlP7OIo4zxcT3f7UcYwxd8tnF1FLJyKp3NRgh5NNVzwwqbt8FDQE7JvtnZLHvVjqt3SJGUGVg6XHRO3F1v6l2FUruZO+fRCB+bTFk0CDI+p5V365850RgyYMEs0lozvaC+gXhytKt++OmzcREsBUFMsIqgFYVWLxLDiPSezoOm0CZYaVbrb4ADHL7POuGX59H5mHTHm6DSjCpw4NBWXcvfHj0PEYrj3o3c5XvVG4skdq0ceEITvbiyKxdVfeRM07c732CV/4vKzFyJZsPegpqmq8dlnMBxFbfYVEX1In7EShAQRFFW9nh6r4X3GHhZ59l2S6JRp9ViU7HaQUFg7091/9gs8pxORU2RvPZABKV9DyWlL8MJG5UMS2TgVrShA+OtbWXv85yKlKscAU2dqVQbFMMQiiqgrMoE3Srndw9assqDAwE1TvWeymSLzIMpGv6w2qK5NK/ja8FM3HJ0OGIxDMrtugOjkWOS4Sy7frlfO5Tr1QyDW3jUN19k93uFkPu2qzf7FfwWbbVq20rHMOu/RA82TdSUu5IbzlFunSQrWzyD/b/6e7tNHfnMKROyR+ZqoWJm8/r5+x4+5j5kT4rgoI3tNwU9knb9QfHUcP7T+qlyqKaUU4lSJEJnN3kE1dkF9+wGsSo6Ir3wizw3IIJN8h7g3LBhdSq3J4/KHYgspNyKBndh8gRvRw+5zuChlWBb/e489pdzLh/8fwmBvCDQOD5tUbNW+9YjV/beuEMyDhLEC8tVSap6+3iD7G1nTxdnzRByM4vrdZd1YDiKIKTW7MHpzRmU5G5xPG4GhUFJ4JxkVcOpFN6htHsEa+L5WGI6y2nacF4DZ8/StD9hAxNd6EIKnegu8w88LiCnNbiahpjC9BhBVIo1kmMb1XI7x1uNcqFazp+qQK3To+c601J4fXThUpYFQRobPBQhKl1AE0F09aTXOKhw8G5HrkOpMlY8rjS6sa6UTBZI2ZPrhNi9oXODueLqqmefu9wG3wC1NLbMeVOKJP+hsrLP2XHQslLaIn36ffi9nxotHD1zmN69o5i2fnDiPRArHBwa8AO+lTbX7Fp+/4IYAytNbt8Fca+Aj1IXx5OO6zwQbU6VrpMVGt61U9F699HbrrD895VB1397X9/ftTlysTn3gFVPI3dnJzTEO0az515vkH+sl9Rd/Spoq7qCYbn3GhfJ4z74jK5a5SVDrETo7PSZiqJtMS9G0HRqpI8HthgvdlX0gDqq4AzYICjOsNLW3NjI73dxnNPrtJopouyBoNvu8gUjXmtZ6UmGCWNciPtD7L5e23ZZkkvHq/md2YjX5a/i9qRjkSOxHGuwFcuxTFz4ukJT3h1Hu+yKjJuhVI9OX6YgEkg0eXzouFir5SF7J0mc5AnRIHVDkOkaOBhrrR6UTqh/7BPoNPoFd233NvEjjcXrgmhwnsawV2Ct/pKTVEmGjEpuLOE4t2v708IjuWdu3Uj0e1kyvNxuuV9o8wJtMdK6Mw6SvBqyfJ2376KJ/DFfO4FvPH4CPBBQ+UBn+ePGjmf86cVHHxm/bkuWpEomWAxfXjLIqTBVqP4bZ97c9/IwaFsl18Pk9H9uDxUiuflk88aZD+3V/V4WJ7NjZjBoSMrqMQ4K1NYFKmlknM1ugczXBHnxSP3+am9GBIfsxDKopkGRbTLGIz3B/e3LpAk8G9xylvv8OpzPEtu1UR93tgUSJXNg/RywluwfU7aatPft8Uzn+myh4qEeG5Xq9XP3YGEznBWZuDHrGBLMPERM1XgO5ZRA8m0I63QRB43I01eyPgU6KK6QZhn55wP5JtRV7At11RBE1qiuutmFC7V+2s898ZinWOIIg2mJBBVmzlq5bD2vbkFgKAeB8gRJhjOvvTm8jEACw1y0vnweYRgXRCjAKTNFaDOuUjyJWoW8IFDWVBeX84wOCrqD0UTlTpIfxfZkd/sFWku25So6Bm/14l02WCbcpICNhXW10wUECGiEAKFIvQFncL6I5bdXt45Fgn5HnhVECI4HOHv4PmMqdWgD3+b5SCDXg+Ku3O1HZmh+ZXR3d7/cMPE1T+ba9UHg01AiLLRa5diAWzsNCltWu2i8I1s9665T3q1/Zf8pehxTj2GTZs1F/vg8gnvCzcXE8j3dQC1w/XDsS67scOwb3blNXtrk2SRbfw1cOOT+F8+s/OY0prrUZiLmX7eTKAbvfjvhexQe7hppXfWdQ8Nh3+tkV04EQQ4HzIQLSt06EpiXLdOjADOU+BjuTO+SGxi2anAmTy0M40EJFVRQQQUVv7Wegt/Tcd9hA+93/AwmD8klmgR8g/pJ7Uaojd6Rt6KM4HXKuPvwGnAI4LZh9stiF3DSPItKNuCumCV1uRd5H/hWFhmsSQvwJi3EtCAD4wMIBWOoPrQXI8sCN8L1Opa7PQmtval+B+6KRO49NTZ5sotxVh2IgE+BHA8/3S7Y51rDnkUAp5oTc+OR20ofvsxbWqMBJn+2vl+Shy0shV0q3UIU/SWOE7t1475RYU0TPFPiRI8WWlD6Tn+hCqZpJloRNS5WaD0Y+CFZMmdT+Wjz6rLAhi2nO3zg05Zf26StfmiSGiD6jQBoYIUGZKDoqtl/GOniboOmuSyzuqdaV+XHr+J/cQHPmA43TegBW8QQQxJxxGbZfZygtr+eZOcEUcgdzs7tr3TWgN0Blb2bnZOy3+ugrp/pfvfxGmcsKrGyscVhI8kRKRssiQU29W4WbTwHbwzXskGbBoekqamhwUXnyO27g+lVbCUVoDePohES7GlfprxeLO5OO667ePGXfzy6frsVsvvDl3tFsM9G9CRm7QW11zJF+wx6GbwBNsWmPcXJ41Meaxpy5od7uVTV2KG+hgm4oiaRu4wovKBSP0Ziu+ArqmpVtJyK0sZC4q1QHScw8HGyESZdbgemqITNzUESzm1jvXmTvKyNjw7wNjNepIG42KkVJVeVMKZSeVkwx+kZ2Es/c0svWdo9U+SroB2r0IE227dfK0lVYfbo8MqoUXcHhzCAeGib9uvYNRdQXAOEEEYY14uAFVLc0uBgCHETuGDwMBL6AcUlMOuh4WinjLfOiwQXj3LU6PUhYmh78NDi5QKKaw/MiKlhOLqTf2XKtdz13ssYbrpUE3KPFNcAzIiTEYSTe2WqF+njDoqCJy2V4HkVvPI01e0kCU8IIYI4AgjY+F7K8y385PgUk0wmMuhzp2GKwJa0ib+O1esVDL1wcLrZWiFr5J50k0kGp/bOl35DIjFHTeLSoJkANXDe4w3jmYDeAKt1js1bmakTBvJu/UvJbTVbh4Ys0dwbRrTFvtTyWrDNaCTYQLXtvF/DKc/6KnlXtIl+MQDqU2IzrP1J/4rStNOdoXB3IBD0eOHDsZngR3DRf4JepTdtPG4HlpF+rgBBNE2ZI6LY2t091N9ZsolyGDQaQpAqVWeSK2/mvTR9qyCJJFYgjbQw9BpkOKSTNbZ54ML1o7ysuLm/triPnZnYm1/dOK1ev7S0NBe5fBRsC6inmg/J8RImyjpx4NiWIdEtSZ7dMXeC0xlOEDW563n9QZWf2w+w7C3JvYfL7ejalPdcYmgfKp0YHJY3PjX3WegL+7WTdPHESXB3QL0zVXLpBo7i9pv7f16lRx20DhNnFs0ZmY9VTXIarzZ50x8NrEr6z1r9Q1ug10in+Ty6M3wOCyuY1fZEzZwdBeq/RMt9qRqw97yVMlsXA8tQzEQydE2CBExOZ4Sq5snVoVIwdYxtlvJnlc46zD1q4cUcGvWjwcOBwsLyqsJNOhotMdrleGESTLBLoZPKoaWL4412a8Dl4X1BHfyHPnZmOJ/EQwBjv5JWDyjTPhdBlIVjji24uO9EPeuVw4yCmerdUXq/QlZ2uN2UNqrCYCgO8QG3wpxBd7B6nxJzbr4k1xflrYUC70APCpGAm4Z4kBNUOqaPddrbNFQp6zhvyiKrOibKU+yJ+U45I4hIsu5r6mhQoBwxhJpCZljhO1I4bR+nvSpZ8cBx4Be8r+M4JklGusN3NG+oLfda2XXvwjwtTpXpm2LuV8PbjzI8u0ZMi1YxIu28dKmfwqnf0t8Km5vbbnhibxuFRib+lw1tBIu77g6CAjwfHosUzkJB60PQN1dWh8lTfzyZ7YF7V7sslMTmg18bZzg1cCoFbjgbk6C2fdZD8gD5C3kjdsSJggBkDMaBHa3CDJxllVTRcFNMfNqcjSUSCrvH+Q6+8qqWzrqlseZoDweJL5I706DcaVK37QmQgmuzEuzgEAAHLqbpSqgy9VpJk0WyePEo2JM2wdexWVCfStVj9eqvuicaDl/3BkGz92st6c2HHgLugPOed+xGkohHejphz73xgZuXga62ptaGWOXdD93t6nKzKXB3e2DbYiy2WA02ov+FDlzzspkgL7MRGIvb+NJfyqt6fX1oaO3tfWdnqlRH82j7v+2J6DQT4BQkuaCqTP9ivJMjQ+q6hFvk1qQUBp5s4G49VPvZAONfDP8betRPwZOsjBttIgAe5FTmwxKEsEuFJEHViBDvV2SY6Bw6lqrdxJquKrfbxE4tWdO2N05gjCgJHgCX76QkD2OOGgUGdZ7nnzo2e0727nAETcX1c4PAOYo5+66dyL7JK54p1/A8rKqjoytuc+ZpFg1k/yLmd6GWPWvwTF2a3peEHz4jbYGDgXBZS+vgzBxoTvia8lqXfZnM4nwiPSF25v0+rHB1sXrjYFf8V2RbaA5103MXka7j+ZUtMRmlsuSxlDkXrAsKXhaEwFBocHpYPBOG4oQbadkHCFBQQDguLlMVmMyDDCt3kO8jmePu+06G/P0IL5rhJyYIeXc070pxh/79hkqd0gYJQYjSL4rsk71V4NB44V7Z8vLf7vsDKQoKsuy+Ux1gQrAAN1Cw7JMCVyjTlXVL+/63vVOfX/n8l0/t7SAAASxaHDrPvZcRdfoP6zAAn56RLMS9BX+T/00MJvneBzQKAIHv+v3YYyH1x9irg+9KbNVM/KwWfpfnHFy2GrtMNrykWw6On1hgioW0D9kansvWh6de/2MamBiIvyxmmU+ouurADsxHADHLeUxVROA5WW18rKQRKxLdXQET7tSVKaJMZJqBwitbOGZT1xtTHG4f3iYC+aMsMMGTJkwRlvZMU3p0Zxb+mM1sqo+1zMHIPSxE2fEYCxNl4pUjeTODJ98wl/IjxTyyUsAyqJyqLEpY9WeZlM5wliUnK1m2qtnNchj1BMtVNi+xivKTYpWEdZRVFtV5rApPl45fldHVtaTRBUwCDs82VlN1NdQ9D+v5FFhpR2X7XqaxrkwaEaI1guacPlt1tHdXta6l3kjqWhJFhl4j4DhrC1bOVVU9N6fwXCuwTTVQ82diDcNhI8ZMoe7ePLQr7dFf/O7qpuB9Ih2r9DxoVOP8Lm0elSc6+5lGI70x+eYh9AVtZ9nLiGutljSP0y74a0yke4pLrXvJOpf9emHwHmr6rAUO5XpRrPN2gfJqn9LIGN2LxV9mlod8HS+75sVrGFquv/c06ZpHXrxLW3vg3aR3zjeaRa+QAjdhYk2dyiuN8mSxKgs/pf+HrQc=);}\n"
  },
  {
    "path": "static/feed_style.xsl",
    "content": "<?xml version=\"1.0\" encoding=\"utf-8\"?>\n<xsl:stylesheet version=\"3.0\" xmlns:xsl=\"http://www.w3.org/1999/XSL/Transform\"\n  xmlns:atom=\"http://www.w3.org/2005/Atom\" xmlns:tabi=\"https://github.com/welpo/tabi\">\n  <xsl:output method=\"html\" version=\"1.0\" encoding=\"UTF-8\" indent=\"yes\"/>\n  <xsl:template match=\"/\">\n    <html xmlns=\"http://www.w3.org/1999/xhtml\" lang=\"en\">\n      <xsl:attribute name=\"data-theme\">\n        <xsl:value-of select=\"/atom:feed/tabi:metadata/tabi:default_theme\"/>\n      </xsl:attribute>\n      <head>\n        <title>\n          <xsl:value-of select=\"/atom:feed/atom:title\"/> • Feed\n        </title>\n        <meta charset=\"utf-8\"/>\n        <meta http-equiv=\"content-type\" content=\"text/html; charset=utf-8\"/>\n        <meta name=\"viewport\" content=\"width=device-width, initial-scale=1\"/>\n        <xsl:variable name=\"baseUrl\" select=\"/atom:feed/tabi:metadata/tabi:base_url\"/>\n        <link rel=\"stylesheet\" href=\"{$baseUrl}/main.css\"/>\n        <link rel=\"stylesheet\" href=\"{/atom:feed/atom:link[@rel='extra-stylesheet']/@href}\" />\n\n      </head>\n      <body dir=\"auto\">\n        <div class=\"content\">\n          <main>\n            <div class=\"info-box\">\n              <!-- This block replaces the text \"About Feeds\" with a hyperlink in the translated string -->\n              <xsl:choose>\n                <xsl:when test=\"contains(/atom:feed/tabi:metadata/tabi:about_feeds, 'About Feeds')\">\n                  <xsl:value-of select=\"substring-before(/atom:feed/tabi:metadata/tabi:about_feeds, 'About Feeds')\"/>\n                  <a href=\"https://aboutfeeds.com/\" target=\"_blank\">About Feeds</a>\n                  <xsl:value-of select=\"substring-after(/atom:feed/tabi:metadata/tabi:about_feeds, 'About Feeds')\"/>\n                </xsl:when>\n                <xsl:otherwise>\n                  <xsl:value-of select=\"/atom:feed/tabi:metadata/tabi:about_feeds\"/>\n                </xsl:otherwise>\n              </xsl:choose>\n            </div>\n            <section id=\"banner-home-subtitle\">\n              <div class=\"padding-top home-title\">\n                <xsl:value-of select=\"/atom:feed/atom:title\"/>\n              </div>\n              <p>\n                <xsl:value-of select=\"/atom:feed/atom:subtitle\"/>\n              </p>\n              <a class=\"readmore\">\n                <xsl:attribute name=\"href\">\n                  <xsl:value-of select=\"/atom:feed/atom:link[@rel='alternate']/@href\"/>\n                </xsl:attribute>\n                <xsl:value-of select=\"/atom:feed/tabi:metadata/tabi:visit_the_site\" />\n                <xsl:if test=\"/atom:feed/tabi:metadata/tabi:current_section != /atom:feed/atom:title\">\n                  <xsl:text>: </xsl:text>\n                  <xsl:value-of select=\"/atom:feed/tabi:metadata/tabi:current_section\" />\n                </xsl:if>\n                <span class=\"arrow\"> →</span>\n              </a>\n              <p></p>\n            </section>\n            <div class=\"padding-top listing-title bottom-divider\">\n              <h1><xsl:value-of select=\"/atom:feed/tabi:metadata/tabi:recent_posts\" /></h1>\n            </div>\n            <xsl:variable name=\"post_listing_date\" select=\"/atom:feed/tabi:metadata/tabi:post_listing_date\"/>\n            <div class=\"bloglist-container\">\n              <xsl:for-each select=\"/atom:feed/atom:entry\">\n                <section class=\"bloglist-meta bottom-divider\">\n                  <ul>\n                    <xsl:variable name=\"show_date\" select=\"$post_listing_date = 'date' or $post_listing_date = 'both'\"/>\n                    <xsl:variable name=\"show_updated\" select=\"$post_listing_date = 'updated' or $post_listing_date = 'both'\"/>\n\n                    <xsl:if test=\"$show_date\">\n                      <li class=\"date\">\n                        <xsl:value-of select=\"substring(atom:published, 0, 11)\"/>\n                      </li>\n                    </xsl:if>\n\n                    <xsl:if test=\"$show_date and $show_updated\">\n                      <li class=\"mobile-only\">\n                        <xsl:value-of select=\"/atom:feed/tabi:metadata/tabi:separator\"/>\n                      </li>\n                    </xsl:if>\n\n                    <xsl:if test=\"$show_updated\">\n                      <li class=\"date\">\n                        <xsl:variable name=\"update_string\" select=\"/atom:feed/tabi:metadata/tabi:last_updated_on\"/>\n                        <xsl:variable name=\"update_date\" select=\"substring(atom:updated, 0, 11)\"/>\n                        <xsl:value-of select=\"substring-before($update_string, '$DATE')\"/>\n                        <xsl:value-of select=\"$update_date\"/>\n                        <xsl:value-of select=\"substring-after($update_string, '$DATE')\"/>\n                      </li>\n                    </xsl:if>\n                  </ul>\n                </section>\n                <section class=\"bloglist-content bottom-divider\">\n                  <div>\n                    <div class=\"bloglist-title\">\n                      <a>\n                        <xsl:attribute name=\"href\">\n                          <xsl:value-of select=\"atom:link/@href\"/>\n                        </xsl:attribute>\n                        <xsl:value-of select=\"atom:title\"/>\n                      </a>\n                    </div>\n                    <div class=\"description\">\n                      <xsl:value-of select=\"atom:summary\"/>\n                    </div>\n                    <a class=\"readmore\" href=\"\">\n                      <xsl:attribute name=\"href\">\n                        <xsl:value-of select=\"atom:link/@href\"/>\n                      </xsl:attribute>\n                    </a>\n                  </div>\n                </section>\n              </xsl:for-each>\n            </div>\n          </main>\n        </div>\n      </body>\n    </html>\n  </xsl:template>\n</xsl:stylesheet>\n"
  },
  {
    "path": "static/inter_subset_en.css",
    "content": "@font-face{font-family:\"Inter Subset\";src:url(data:application/font-woff2;base64,d09GMgABAAAAAE2AABIAAAAAeTgAAE0RAAQAAAAAAAAAAAAAAAAAAAAAAAAAAAAAGjYbIBwqP0hWQVKGAz9NVkFSgSAGYD9TVEFUgiIAgRAvghAKr1imXTCBomIBNgIkA4I4C4EeAAQgBYoOByAb43IVqls65OpOCGt96qrYUdSJ0erToiibrKWy/z8l0DHEgn8Kii4BmaRaVZvE9JkKzRCrXOj6NuqyHkezR98XbphNsCvFPo9wt0o72VVXLcIIOxY1+3QgBYCw+YFFtoMJEFjyE2Kpw/0qDCYZPSahBL/ppEX2ZK8+e0hMgqYqlBMnV0cdv//p9SM09kmu/z3p9M99u1otkliW9SJkIctCxkIUt8ZksCi2U3HtxCmd4C43YZNSOpOQzvcnDftnCNFoMoy+P2m4DB+3NcFy5yt8IQtZ1s4/te3/zjCgprgguGwPW7fDW3ag3rrsM0C4IRoRIiKNOBIi4kgIw4iAaGySEW1k5Ce/Wb9ny/GXLRv89+73757J5L6PJEkWhEbhWiOqAHXOzy2DrTCEuqtGMQrTZhC2OQOToxde5Y18JpEeo1AQsTEBxWhCEAzCwMDEKI72Q17qfx/Nq+p/m1Z9BYQQGFOulDybQjrH07w57bvHUwS7enM47ynFVN1T7YpTdtkYY4IQQgz/ulWPEEIIKS/hEUIM8REChBiKyCAiEx1sU6rjzmwrZSLefl0v22q921bc1tVxPUcd9eFzs96IecZqGiyl7Sd8POD1YJoEaEgzNK0zs7unkx71PIfTnsT0rMDzcerLIAzJBsxFMF0DqIA9L5Z6PnFRycr/Df2c9WPa7pWeK/3V1j/f+/hp1eRaqdmeZBM3McQgKiIiIHUYBhgQEGsSY1zPf5tm8yN7y3u7HdAkV9npxEsTd0mpZcaf2NRLoxM/VNEB0rvXAfB/qmbtfAW/0V6inMTLoOPIEU59fP3For+iHA64xpAQvUNKfAZJBxKrfQa0DkNI8kGkk9YX+OTLpa8KIXU3gNbPQzqB0q4fL4RYhRSaJnVXFOUVre1pv1bPXjwkRBukVBolU8p+3RPVhvY1zONPF9qRAiHRGvBF6fzqVROSVq0/xgMWAgULDsdNi7/ZX7ATm1oPQhdKlobS2d8TXca0DHxubG+MZRcFxFSpySH114LAIGFQLO4DHgTZdYUx+9lCZhRndZPUBERgaZwJDpxQ9VwhMQF8A3qb2GUne/DC3jvp4Y1aHTYEwKEEqa/M7t7PeBFncD88GgQAVFtgK/p7hhnlMWAwBLGbxizgAxxiL79srzdP95Ejh9eA5MArQf+bKBiGK/b11fOh76/JN+F/fvtLm/4+zogP/risvJD68BKtoD4+J+AH7u1YBf39JyXPRB754T/33R18n0e9KviDV88uDn/vq0Oro19pSgtin4kffuX14jwZ+12Q2vjbZ+LfFf/B4fDC+F/boisT3xgViT/ECNcc+PsvnrITbwqsVfV858NJce+voTz1GcPUA1Nfi+icU49isk9v9Gc/Mv2pFO879atv9i6afZt/rST6iWsmV85f9ZmbH//cvCk5/+qhhucvb7jnXx068znOO9Kb+GNvjK9b11wOfxpGW261fX0+KwHf8T4AOtQfjzqvQnbrv7YEuPy+XcJlYNcZV6Qa3HMGWX3p2hRfcUNhAM2dk60RgD7q0Q/4E3Y2C955dkrrAPLnhqml4LEXb+04AZSNN8D4S8C3vL4/Fig+jx/l4Lrv5Mxy/QVyC2yCevR9x4P5IjDDPr4T4hqfr3s5Cp5/u+zZOA5jeiwDfmBwpP2oqewQ8LQ1YXYIaNvSbaGjrdX3x74BHjWACE0QGGnHRCddGNhuInw8Qzdt1KNnK0qa2WlEIEYGysQLKNiBgg4LoIFtaDrUKoyrGyyDJLyLgm+2xMzUQDIaLYQDyM1HJjyBGRWYGUnA25ZFAgKWTRY8TQ0tFqRJOorXaTIewT4YL8KAqQcz1AYI5PO2AGDrnkIDyNq+fsfusnb2ObLwd+0CgHXnATA2vpE+Wft3bx05AxcqP7kEvGEivQwmYJjC/wrg3Tlnrj/vH+HBwCErBOnOA+2incE4grjYOwime8C1Qu/m4aMc+HWGbsd1NSmSmf2sSra5FLb9vop2QOp5bF/LODAW9Odr+gPjwOBmPGXMZRm1+Ej8PIkKAxacBPiNXiIP60jGJrCecyM8q1ApcAYDo5evYu3m61SKwdwYtqLfWj5XwLmOs3GN6wmMPJDgdFAh5DkvsueKoMwVQ50rga1ZCgc3VvSPvzoitledJgeaz/LY/fm1IcNBRD4eJjCL2axiDet5AD97Wxe48FDKRGZTYTxhIw/uNg/ACF6/to02bldstwsINsK7GNzXXVRyK3O5jerICzOoKFgChbGJaWnp0QBJQWNgjR3X/pmzSAawaRaUXQ6vT1UAKoUR0y+DOh07NQdgK8A63X5qLsC5wDbdWB7pBlzTdgblPGAAjumXYdR0rOwEBjB6ur3sAgaQP9VoDfbUsPC+Qyycr5nlEigAOCxsr46BhsxA4+7Ye5nBBMO61rQvvbPk0JQXZzatjbPsbRgVz51TxIiMTHqMxVKZY5ANoEKXFvlkeqNGeMFYYgwmTwqLXMkIm2QnSGaQjUEY27rIIUNHB+waGwgcr2JYJgUlINx1rCFZg6UxFquRXk1AKB0SbLRNi9oeSJJJRfGhsZXHYqk5qKcyWUZDtiyntFyOtajrDle2ixDDjCIXFZOqEqHMxvzvtNFIX9Bq6m3VpApKHJTapTXSLsiz1hPeC0O4Ivca0rbm3Cm8bqaD3wm9urqY/8oV4WZKCtVZS0WxaPGcJ9v6WTDyUHdt88UWUdRhkaCGN603dqiv3nChjeWIte28pk2/3MkLLP/IOGZ4axLR7cBWam9xczf/uSZi0fZi1vCoCMlOTAW6sQok0rjmXvihS1QqfRoOqqw0HC21sSzVn731SypxxSJpJmKtMQOd1U1uc2MTW9jA9tYI2/StB5+rV3eSPjYbHUtZ99Hj8i+RzG3levRidYlkPAheUIqacOVCzMXR/7wYf+8qDBR3nldkPihg9mXs2Qvz0V5RfY9z+lSImE1fVq8wwK+/4IlIccvCGGjdXSV7ier1G+F2jRpeXN2P7bcHtYubXlA7zGX5dvy0ipM6dhmnXHDoSO955bmBs8VtTxbLLB8s+tnhFuusHUStuDcAOYGudeE6T7ARbY2t+LR35pcXTK0tf25rL/Jbvt3blexSW40XzJ+vMak0/nYQhFflbVdpOvgiC0Q2bGXB5nnD1sXmEd2qpgYvgc7Vfh7GL4tWJ87oZnPGvG1Ms5ZK3lI2uBMVSanPGgTvQCSLTy+VfrnfX1eL2Dx0hFZX17PK2sqGr65ee0ns+CvjuIKw//OXTlFpSRcLwXjGPucqysSGKf9hsSDLlO1SYhktZafx1oMVI5iLr9CRNnouQHnlz66LDzktSgNrRFY4qdMd4c61ej8XRXRpTfBVTZazrrTyzRf29Yrhu6MfOrgeVTpSWcI0rsYy39yau7bZan6b67GgUCF0kzygC17v8DIXwmo2f7u5SvxraJO8lVsOZ8MW9C3Uogimh9blCy1qQ9MJ7USOohuWOX8iekFULX0/dzchu/5lobsJXUS9uc5zZtEsfjt8Kyksi/BHcfNTGfHi8/7L0U/Cd/vzLh1T3Hdyt6uB1x5n0Ns7Gg4uePIMmtn6S/MLOADEYNV4Kil6lBWJKWARP8KoSWkIvS0rNW7kVsJwXCtzWFUaTSbz5d5RqU1+/iIkOcMnWlqqooBVfsThovEvKa/f9IXU8TpSp1GtBUlj8LQxG2+fzFQ/qYuX9HaJwGpqUmApLn3gLq90ik7TRZebNLqKrvzcAMHHGtr4J3m2+d6xTNNXssEsTdHNdcOfkJ+/7PLZNjp6TzeAGLiVwMQXay5+wrMJbuGbn1+e5rSFRCMpA6v5iGm38c8or990hziO/5x1BtcqTBpGJOvnkMgZyQHF7v7iGwx2J7S8W1ImmGSxTpbmTWg2tR1uhNmpaPaP1c+vnz9fw/wGzN7S5Cf6M5J3ODSlK91+xL1zB4O1luCTgtzEo7uNFSvumz7EtNM+sWv+6xWn1dlkzS3I6PilRxOFz9g1e5uANc5wY0nnMbtSvO/ocApsUVBK+8IvdSVFC4JxjYRG1nS+aluF7x5Ya6qCqcCquKLw1YQ2esR5/MuQN6/7KBHt6JHeKhFYTk5dKEIqxREKoraa+1fLj757hclXxaKk6/d4Awma6JxKf5SCSkPJqvxydwDIGnBp33TXTfV30nTbzboJNf1sVvJ58sw97kBte0UGhxI8GV1cSfoB6L6v+SyrGZSXai4xxfstcvXmgdEs9XpqfCcmIjfAKzBMstln1p4pA5NO6+sKJfpn8YqK+0zRkAlnCMran1sSXb3Tn0P0jYqUrOfUuLa+uF68OliRmFM49Z4LzBzXaNZprmyZrmMCYpxh7IZa8c6kdHDQtFT5Tn1j7DQrom80sqBgOLJuZAEbR8OCzohdEQftW1lvjzOAqdI3Rt8LF3fA4E8On4UfO2wLjz28f15xtx4h/hhqGdjuNvCToRvzlcsvLz+/wo+Ajn3l/vP7L4EJznC+eZ9d46GLy46mMMeYoEiyBjvkoeHFLRp4VBdPS5P6/eLUrMhdWTxs/TJvyzgof8awqxlxXWziOt8A7moS2aVjlntcVZrOM2hDuU1cQ2kpftBbDzXxD5BsIaiZMl8iD9eiIyEibKhW7WiLdCf8LjRVTAONxclCIo7uidA1A4hB6wTzwND4BJb8qizHeGg42/SqvE2drOQSJ7OzCIcUHFaiPIF4kMcjTShSQNScpj5VnknP7kLTR9V1WQcpOc0zEnRTGCp7Azm2Jj7wIF/oP1GbGBenTJhN082I5jbr6QA1R3Zcnm06PJRjfFUmZ/MIGJoHzEkLiKFTh/DE0QXEYHWCPIU0weMRDioSElOUHMKhrGzipJIL0ua0XVYWGu/W5UMeVNeF7N4sawi82uTFUBn5bEzF9ZfFxMdXxXq3hUfzQwabaQkYlpOvD6lr5bzAPN4ePHCZZlTRqqyxjB+RNL5ypy/SD+GyuvqXs07BZErgeRAxXjCyI5PpKl3PJvjkb46TDgllj2wXYUMFWLb0Kr/g84Au5991hZwlIGBpHnYjzbwKwditoPK4xs5L+lzCPkVCIquCTRhPyyCOVMYCQqtGCyAGrYOtE6WJKebupkXoK6a1eqm0ghckJpCloOVtLQNYPmFohwATXyv3T7HXuHwJNBAEtrccW/r844EMG9sYb/t89M4fLczNntUKxI0DtF8f0PJ255CtPbUxhsWuinGhQt0pWmZ2rDogpE/0RvemQJpIzxQokm8M/YvIzYjGpBPwhQH6NjGbOKfg1nh+Wl8YQ6+8mESlquwAIcFLFTWn7i8lz0Q3yDOtoq6+dPBNB3V8k16rzqjOJExw0giT1VkZ6VVZFmmc4S+7KhM0dwAz0GlHRM2CrF+9XjCf7cv7zyqsUbvWB9bld5MSE+8XxEjzYtGpSGDS/z9qoYsDNMxYeOqDAeH15f58DDGTPAW2bopWu4YmMdOYUgYBSd8VPaerUDWm6o2MSFbd0ZAi7iA6TQhntDnPphZRk7IVTMJWf59A+HiJukD5VnpEUuV4QCanCR2XBVwWM6rQVT2WWpcQgp/PTEzgcvUOrNVy6RetArD++MlaLUN7GOoc1JnNA4ltNPoB5bT2YG5ZZRJAipE0CUA/Ycw7xhgqOT0cUwBNARp4WMaYzfJQKOxx6H5YBwXaVyhjr0GvTQzN1gLMXP8lszxlfWo6fmW857T1sOXwdJRn1Epg2eeBgV5zr+GC4Zmb3bKXRcw7FnvEz6R+0XGLJZTwtPwNgFfk3DERytSgy5xEXmiBBvDhnRdGc4M+f/JTno91dJhW8enzUPVstLOZo1b42RyZcESt5HSP7lx65LK/QQFc7hh8yI/KIawSP+5P/w50ktcLgsPZ5Rvysb5eXP52fwoPh+UF60Q5xy+yi0Sn2Fx9RsCWhpeP8Eu846BjGlBZWKFFiYZMM/ZCCJR42UZKpAodlOUT6F/YhE/L0eSEFu/cle5DRnFytgd6SzcGxxuTBkwyRIPXooVlx9mJexL5RYOp9tBwmDgKXxORW3TkERtkr1Z5n0BTnnktfgyDcjXZqxn4FTpADCAZLuiKzR82TR2wjGkXpdPlwf41kVFB6urQCJzC7WIfnHrXTMNeu0+8UrqCzNSR+BL83hQF//5dPuDDw/aha1vCZp6f9lot6z6RwiuciskfmskdmhujEGWld0uoYG+OnBMqXeEtuMzj36soy3t0X1LbSq+kBTcwwgNqFVQ6XRVOqadSAxsqwsHJFV0eKNB4tK7FXpAOr+tL4h/IZXhpipL9YRZJC3jjWdz0kbNMkfAMM2OEm8Ufj19g4QsrSiZrGLmCA4+TVK1xVUjfLL+gYGE1iRGpIgUJg/x8s6qRcZN++TsJXG8yjpO/PSCoYDuWQ/bGcwt2Aqfdz8LA3k11zTSbcydIawRkCguPmYZnZo7MYSrF87skNKN9UBlaGbnK490DSlCyN9+T1raFVdEC6w2/OmVomM8IVVNDhrySHywUvOpYEvdAbri/XJYXirreepZMI2+OTvLfRNm0KmTlus3JVPYvt6Poi2t77mLwG3apQuKodW1BKWltwRR5UFJSIw3jejeGhCtfjCfmiLyoQa1EgZQ8kdOtwhwITlSXPc0j+t32Abf+A47OBcrzfPH1BjJ5w39PUHjEqnyD6h4K3g3pSCkIIaDiUzfhtrJWeSVAgvea5mUMTtAzhQfj4ofScpN2M7Z88yHaJ9Ksgs5QdLoF5y42+/Eh14J9f/lPuCFwXiPTACsTTcjqafVqYJ9d51PvcRvlBFSJ0STgBfKQvjhvpIPfRRLViNJilB1QLUZ5nYnn0jk6bd0VABhAg7drgA1u7hoCZtWqKK+5as4RzMN8eQLu5G5qbNx/wIh57StlIGcUcICj9QN6VIB+Ze6lhwUg8N9AHyL+d0SXAcC3PA3s6eQ7cLkZ6TKjsRdZB/cZ18P98ihfNR0wvQHBQurN6GZHzfssFlsSLTMtL1rZWAmsTlj9NY9CWTOoMEcbS5t3tvkz+xWMXYxdmfIcDoGz4Eq1cJb+N9yYPT1HZ1s/b+O8ifmB81V5kAWUBbcXrl/4dZH7ovdL5i+RuFKWNedfWr7OIdoR4ch1KvZ0eY47vXRe6LzdubjwhvMPl80uAjaz6k7weAQgkGFjNDxKIIwEAu6rysDtl/DkFtsiC3aSp5oieFJPVPAt9qRpP1nU8RE19QsRTUQyYb9/2YHXUdsq+IitEcY6ggShQm9rD2RqmtVJp2V248H/lAH8Y67NLDDVZM/GIo4TNZkJAgbYVO/DCZRtL3fd55HtutpQHX+4fsqQ7cxPKWyHUzDbxzaGLXKuUlmCNAVZ1QShXqFIpnkNWZXdbx7X0G2k4LYASXPq+tfArKRsWZXUwCTszl2ntHbCyIsYQ3Mdp4ZeMLwD0uamqVPYYl0+Y59KuxDl6pWSGHFECBTW/5DYGpJe40mLoDij99H3itXq1jq1vOMCFntSdtaekmkLCe7wzg4kQzFGkMbvDVnAO4E+EZYns8+IzUG0IhwgxiUjzp+dz81EmGKksFDgx50RhIaO1XQorUmQtMH79zDwMciYmwLMcszCTCFKmwXdgjCfiUN1Mgut0kDxpFkpDboSYQfJUWHckIsxRuhso9NxKCuYFrbA8XiUx47iZ2f2PBU9W6VWMLsj/3bNiu/2DFrWSaLSc3vrlodjfJDq2wGYCz1EsGFPFREXazCdr78tYUCIPeDtm3KN9RjswQWF+ATXwS9H2tVOT+s5tKXJL6g8s6XL11O0ksDPZ9vQ7T3o8xNN8i83DtSB+gK8KAyJZ9BxI+bQ/atGY81DFlukeYtt6u4Hdw+hAarQ9tCc7/V6xcDDiWXXWwSh91q7xuYCwRXCbsuxQ5Teaey2NiO8bsDZBvIwnRrZYZirqMw6jdvJrKcpkSUQsFqm7Arv97hxaZD9d1VD6PVqgTq5i+wfcDrdw5G6qck364WNk48KRz+5yx0ADVry7klVDsLO8T0II9QrHgRT2S/ksnk3jaXWstMkShjcwKZSLVdqXJGiWNVvxSr6zXgMW64VlshhEBGGXM9BOFEZMcf3VzksIVwRzGnu4hZ9Ld8VyHXhjCsYjywDA0zKAia4TamNCMIWtP1IyDBwD7TZeY+NE3DxwghZZMEmmGo/seBFCHMKtvZTd7WPEMGmeQ9ADr6Pr4h/TTdaAhX6qH8YJ1Oi3WAxr9eCW6UWK6jwMLEN1kbdtSX3DrF90qwjutGNeU9Qgwi7GOVuzKi058bez9SAciWzqL1dfWJ7AilYN464vU3TIATQkyHZCPzw94i8oYCx98wbFgArANN9TPdXC0DYDGH4MrOnvHA4Th6a/j+j7vlGJQa1vF5BwfMo7M6JXUKX7dk3hJgEE3BolL2SWbKhzV1lzK9hDAx0z6sCuItHEfdYZsVaT3ljIWG7+i6i7oNHD/jhGvM4by+8efLgrPx5C8L21b0yZUej9xq74IHZzAPYDidQiqAAQ4YhNzXsqKnnjaS+ucL/9t6c78Z89SWdVKax86vuaGzkLZnXktkMtSyHBXWeq4abLmaP2JkLXwU/3B/UO/+IXO5e12oXO2S7hcU+xfhBTn1bNrtIukau600VCrYfp/VK7eS8HuKH9fVahL8v2hEWNC4PrhvDgxHrxnJCCHXzZtP6Ik0fmWpSqDx0yP6GQLZhQCaHK/7ujD0s2GkMphzb7UHaNNOiB5NOPKAkO3k3dQpXuu0wA4vZMy5kK5tGDgkT2zwx93ulPkwpblJrtMs9YCOpwIhY9ECtIjAZD63eze5xoNNthxmwvGcUOJIx4UM4ruKJK85qeKAtlHkVYswxZUmXnaZU3eUk0DjOYFnDbT8EyWZQosRCJyrmCCyoRivlBFyKejJt27IMifYqPfTJyxfV3vb48/HWy+8CQYG7PGDdsvH82c8PYdMu2gWef7krh5S0WZDtwTlUtHg93MNMH7N9TAImIT3TJzCCAAS+Rgqmgs9Vw2AI4xdA4epd+7+BnS0CZl3izoL1deiEQQhDGHUlMotiZNsYYh91iJ7/OYme2KQ9ZaJ/azsGQUIYe+i/KzhOk3qldNuc0xjcr0fXAeYLBqdyan4VJpTrutXk3rRC86uww640fjfeFsYyP/vMxUVtg6IRXZGI2vwhDy7NoQmmBjdn/w/Wt24j8QqKqelwC0TmcNsajXxpVQS6l0fUQ/QrvIdzNlRuLYdxyEjOu7r9whj3l4wzKbs88DvayMSzJ7nhnhM3GTglEUk7V2DcMcXkEEBfsj6MgmRla4XBrOAGa3q1fNIuQtCVgnpT7pAZCjUw9GbluTvayIzYQ5qVv60Gm1UHWEdTAdzFI5OCE7QOx214CoVeBAi5Z7mCZxOlXbR0xY/ScQQuIUEewB5kj9UgC6sgCUlI7lCsXDqb/sZLTRxFvXXRtzQtwJPCRGG6uoo4wAFdlI7+QiTmjv1LEZ1dOMIM+3OjeF4pbW8Bz/xeJLVQRMxBINZ74f/zG45EbedMXN2bhio6MloBfcKg6KNeye+1+dmgCOPp6fAeiM0QYB9mGHd4aVDi5fmLIpVcLpls17ggKqCHJyiSNU+QeBVZHIOucEny1uY2KXTH30sdixc/P9Vlq8ibradZIsoqxmdhQM2ujtzf5Zerrrn7se5mgDD+3C9MajF2jyKKKESM78MBJQmUG2tBEXId+2IQx88KEVIBYfClBrar7RWLEebvCk20sYJPjYerJZMIbxiEQ/AAeBA0KEFyeleoJsLlYlxniHqjs9qzlrBFpisv6z6dnXvAlDDE6MB8FUWHB8AZXHrMPgu4mxprScrJRz8jiO21ZH5ToSs4nFx5vVIVb5N6b+VTihaafrMWr8bmaGiNuQO7c1jlSkq/BTy1Y3UBaNX5KLhvDZRQlTMYOSgJJ+pWmHJxlZ5FFD5Q4ykpxVSMODnOws9Skjl4XOOZiGR1D0eQNEj+JpkIwpS4/BpNeEPjXD6L8xwWtZLY+CJOTmbg8oM8DLPykwhh34N83qD6lNPVkSLHkFgmCVYy1GqRqKqGReJlYgHnXKlcEfFMLswWwHmSBYtyTQ0W54MQwF24fyCsJruicRCvVGayTxttxn3nf/DA/Gwmo6XiwVduAUFJOdKhaTU2Cn1CRGeZjUNU81OQ+yXrHeNGSodxPsjw/vErm91e6LRebT2p4I8Bqz1p7o1z5lVdIEF39FPm0Ca3moh3ogFU2vpuWmRtXgncXFbiOKbGzW/6owKaJ+WYV1lvHRtZhCZ4vE73Rmt4DHwesBE20fqbDEPu6E8WI8d/vF32rA8U9G4Yg1yzHvbbD/7tK5IIBKZ5ofiCt/3VfFTQsnkCY2IHutQPue/VyrSTF9IHX6oarOU09cI640tVg9Y3vWFBW4688rm3qw56YSMEIIB63z3BoAQggY+9DiaC4smuqMC4pJzMPZzuaiR6jkAHcc9B6mvZg0TunnRuE8DOxKYUbUfT0ssp3ONJ89IT/linKdADa7Rv+EVU7KOyIMkfwtrWmTuPgEoIv5sQvHgVPvZUXWBLU4DlSTmchW3OQD7RdXG1TA+EvoT0WDgRfgknwolwgqby6XDbsv/dTgxPjTYNgpIxPxOoGn6O+CM7jIs94av6kpFOWZ0nPp/9sdF1j4KYKhhWEwirrzWj6RqbCqZztdrqmhGhsAhCwsLiM3eaZFWdOuXpZYIIzIS7uIVMhu+uXcR07YyUlEdv9DAbl0AMXrn4YSJw/1f/8nEp2RmaXuD0OBYogB41QjgwGo8GUtAHalgZxEdhwBaMs7BA7CIAGSc3AR0YOVpFtvfSHxd8+u6WgenrHyUm7461t2P/F/Z9xs72R2+ZrUGhA4guvidCM9TyliAWyFVN4Wm5r4rEhIC7J17mNHRetFyEoOOKsiTI9Jr52PkHV5Q53SwEcW1keKPv0Iz/HqlEWFJxstekslLG0zvU7Yxn4hZjNnXQJlBanevRzF6NW4IuNWyRoK0mLF3k5S2Se7zLJVFVU9tACJtGHDpyZLbkM/5xhTlqXU0GpDamx5ONp3Dn8etKBMp6UpD43Ga4JnSw9CDt8eETK7nsDNPaiXR0YZI0g4rkP7jidLvruTgyyhBCu3v2CQVNtVbM5cpVt9vo7illZNqWHtd+9GY6dKU02KnixGB8UiYzHpxTurJBQi8AQZBoiqLizEe6bjqkB7RwO7Zb+61X6OtCKCgjN8xCfyR+RuxjLKqgB7cYl3f9n1GPLgG8tRVy2u2nN/tz1ln+Mh0iQujOMoZU2iO7qyNgZgvncf9QazHZ7sdA7PAvkrC7vL2zgm7T7L3M4yNWReegCMPNY3tJYvSfxNMomo9GBxA2wVQqIr44OXKqpVHJ8xhJD+ZfdlRan6EoAkiqQXRmckJsy2U6G6DJK7F4Gw/cpLWVkw+0FCx6wUbKl5oon2TxroxkCaAwxdgKzgOyKcAxaW423M5UfIJykAuXChUZ6kY6IUxY+G36HyJcRSXiF0N4ARfGBTGUJ7K0lh13F3cRSDzXoVPGjqiZIsfZYMNOvWfwOibI7qVv1hxaLxsKW9UizSEBX9w6EDvY3910mgQSX5eez//kKRWQRNOc7Zy9i1ZHxAyj95A6830C4I+o2EyKZhQs9D3O5w+gOUgVYcL/RCQrTRHJUigiDLgTBedThKjgyKLE2Q0PRCWBBpI0pXkKEwUnLRRoXLcoganqMoK2WWw2U1PwcMVvhtDGuG4tqEAyMM9jUVzSjeWwZ7m/1CMmczVZ8DWNhUOXVP7f1MXSZtZQ+GU4L7klPKsAHTzYSOvWvQhzK/3+vyqhEWfQiz6JIJXYgAofCcPzH7tsX+HyvqfigyPNU3LCOVZWhDE7FyISNslw+TQiYLTc75mSPTxaZkDZj7Sd0d15RWP9s7IM3lQDnTmTxpt/tBVgYAv6PaH2l0qc4sRmqP+Q4KNLDbRq1Z7P+rbtcNeQ7BIeQOkfGWQ0WE8VEaKPnbNtcAoK9Ta19/C6Opaca2D9gYEqfGUSmlDydrDVQIgoYD7E1ZLs0ngHuBHYDVuj22AoayLMzyg4589L9uz1aV0bOOVr1Otmfta4Jm/PmWdE6TTE3SHEqYVOlHYHRt5/diJtGETFZ7jJk98vBo4iw+tmT+jbXw6yUQj1UBX+DIzhVHOPd7v/IYU/pXFn3p6IhYMBL7kHFv79gJR72OWPan6mt9tuaVMNN0/AhSMckNGcjyy07M7dcufWzvtRvG6xtTfv4F98TaOEra6zlfzKSTy0mNqXQKxSkZusSBtXlthejPdOn5PxGcSHw0ctCQnjHEuuwXgCu+nwE2X+5OXvPByMVThe7m015c49vboTaL6jq5papoZmN+hKVfHhkqozNlffzRYhhuYURWOxI5tKFGvID1FrAFeRSaAinQWuTJ7nYKKwRVUptyxyu60PDGFpDxmHL4fR9FhjhnOi9bI2ihvE9+MuogdVEc9UmG96PvF7ZmQts9sFrg7DCSLOraZkRYlheh2QznikuZZEglQZNQFp8eM7reBXFU2mtraErqZKVTL6LZzJra5pt6Vm6ZDHMDWFlA0ll7RRFFPjccS3clBXJI4zv7MWlSTd0nbkbZxKx/XGTk8PmOGmumVZikBhTdjZk2UKBzCEGbMHKl7eUF6DAWOUkpUsUjVgdVt2ug1SkQNH+ijNSwwmK3mKG4Mbv5C0aTBhjoVWBLb1nytjw2gq672ofoiw25d7b7IStiPn7winbEkMJc7Xm04g72mf54YpAmkQFsagx54odypeVx+ACKQyA0/Ag2SbvHk4WidV4Nqyd1wtfRgtA3iwYsa+JSzMfBNkfhDbwyAUBtB4XHeIQDZs/8pJejtPlD1JUY9Zj1nnt+NJMU0iX3jsgDGQnClEzNwEkVZiDQidP+oegw+viS2NEk0HwZpzCYi6vDcVvBRcF3tZ9g4EItWAV+T8ZpWoLdnUZafRdGARk6dnUj8baD3KzphRcqnu9hBJyETBCereDjJwY0BayyPr7qNnDQNiCvWTg9MnWztBch3uNSGZeuJ4IWmIBw6JmdKBDzpYr6VZFrNZk2ZVTuNa3TQbli7hqvRFU4/RzcBaMhGVGcHz6yIx8UekVts1iISY6ChkcsV8IoaGKzEaB47HHEwFnyQDrdgOsoHJNy5YgNKq8qPEtOmmNxjsN5VNYRiOhSEYghHfmDszroYuGpRX1QL7ndv6iOJ0Wa90zv/Mb2z0vxvjEDglrJRjtGmwQsQW4jE7LehfXw5PqlElMCiphiEneSFrt3YKb+EhoRE0H/ZPTJvbKdoYxY++PZY88v1uWQSKIu324lWxeAorESARpL34sNs66QV9d6tfMHTHwNITXNBkmlBs9NV3ycHCPzREzqwLokTKosBXFSRpVR7zwpO6UgL9wniJmJgCy7xchqZ3xwZXNdQYRKSlJ3F0uDBKkHJOI4mjjjBO2sk1BGm1T9qxupPDX4XxxDU1VagwrzHodk9VUoQoZnCMjsgsTpZ6aEhRBKWLKZAkYSFAhSGKCEJjcggBlgsihCAlpnWjc1sDKCoybZsmpwoRS4lDDns3U1kmAx1ql3pM88ddt986wrTeunKxpl2GF1lNddpQ/JkxyX+pVI7qo9q/R8vZJg9mS0wM+td9ain3YXnYR85x4ubmBiN2bxkutHO/jxxC1ssL27YaifdYc62UTb6rI/Dmlo30y+fcOww2HkPRTRg/CbPfiyfFMokqxYnYOCNLxMk8YvEWed06bccwePP8yZpjYMmp1T9qy2eDKnDPd+4+wISCppWtZ7sPjPOiPK4seou54ruB2AFaXtRLyzGNnf8p89xY/9+YnBxjo5pDCVbCRrRJ3t+ZdHcTi/Ege5W63t+Pfd9NJOcNU11BCKlGILbUdrTcqppjYswjrBbX4j1aJ8djulk0sKRbsp+eqAoB6bit8qYZNYqxrhuGbG0MnePFFBFxWWgbVjEhiC0hBj7/MUVMZdt2ixtt+ww5QwijjVk8EeXvJEP0H56QmNhndm46KiuhAi1OtIhbpqpWLrFUMVnXS92TmN/t6XMf8R2egwR5cUaHGYNJGMvumfX6G+JgzXqy/0D0HZyX70PpDNlGz05PFFvUmSzSm1D6s6WM+1EyNX1b+7mKx1xQ4iiHNIxWXV6oHDJ9vSiaQbWmpWk0C9Ko8IqitU/gO5aWyG+i0e240nvjgnd4yFRityVqPtp9kxysCNJipQylok3qtOOy2VyxuEkVa7ohdWUjy7B5imYLFDCd5fJ0/9qssigXib74bBwESahz2oFh8eGIjjPFjnhWThii7KS8KeNo567WGBIXKqmdojHFzHOl7rSQyCqjstLAyVkuZ95WqlhBuV6MjiGkKJKcLiuVmSzJpI4CiWXxm3pnmMu326HIaDy+d5upcbZtYJzOYFXklbydyaWSGVtFTBfseljFJG+hTlidYmKmq1S9zJTJpdn+0Q0H6+a+vxQ/F4xQhNXQBUXokuT7qFeHGZQ2T0+WqNRhu5v9QuqTAccwgVFsrBC20PpvSkbGSW4G2xyMF9sWFLx5Mz3s3McK18Sc1PiwJAyQAh02UMBY7+BsukNjydCzIqO/ad5nIZ32ZFQzIbbS6d7khvQO9A5UdlA6M5xubb3u8EsAKjD4lA3wJH5Yb1y4cevqSfPTL7LZCMSVK+/iIHPtVqLJ82e7IIwHAfjgwXdggIXr0XhjtYuT+ybzAqgYxmZDoVsGYALoqRxuXgESiJkpIW0R3PWYwCSHYllIODQ+yuRwMu976b+LET8+Bfwr4FQ4VX8KMFQnxz1Ze4xCndFBlZAZ/1ZqZ1oRzqdq82I+Qv/GGDyajC7z/FcKR1rIjoKRyv+34RvnazSQe66/6vZTJ5Mb9/F8KmksCeOIXQfG10cu9LbSR2/OVp+QSBHH1PFrAmJ46GHr0vq+n56zydzVpHKMJuREg11Qb+006fVi5CiNbPGMff0Mr+G7NbFEbNpNeFoY5DqUQDU7Ob9VhyRXZ7r/s0cHbvzhNtmF0gx3OgJWWDejacYcf2BsT85v7dSh9Y5QfpUavzBtn1NbR8PkJfWmoJ9DMQpLZyiattlIZV2jFEnYPeGEiXQaC1QSmMRa4/iT5zMuchtNoq2Sm2vGkGeGlblikaLexA2LM6wpBoVyoepViOfvSO9ZdPonkTfHtW8TDkx1AaSjmKCvKM1YT4LKhjSHBE9l7CAd6UMsQXxnmntgPmClyAfiN/F7V2m5CDJKtYFd5WrVdSfj7eJAz1im88wL1mcNvXWHU3RcfWovyXoo4vl8kHUkESpWxj2BO4GiyDLSHXT3gFBY2pSFRhFgeljl9dID87saJdIWWbZtykQqjRa2acyqXS57nGKJgA9akq8yrKmJpGE98KATNkAZylBuKDdZu52ysdNt3rnTbLRYPa2xS1YQF3i6wmkpIiLpWB5ralMWTz6XRJFkh4WZ7wDvVHrasGHCQm9lIHJnuvuP2bICNRxdAT8LHKhwr5bMwf9rTnjQj/JNUtWqVSy1pE+dWRb4AlWQtVXFcjBwCkRrZphZC4uepGKadYSlOZmvzZecOXOYtEtTTuRWYJsQRFejO1oza00rpnpP1WNNr9RFKY/a4vn+nqYb2OxIqyBNTsRE1gMbtICCA9K1OYebOuoaatsbCEIQgpjpQ1De5W7whrZ8H6uBXzMET7FUT4DaH9GJuYwO9n0gIQoc0FO0TzAQZHu2R1g3WcGBKjxnSs2v63Afzv0fCgH4BGaqVw5AID6tgZoaxTrAWQDt0D6eDgjgH3eHvb9/bA4TmXx8AbwnDG4Nz84hL+aqTlIKwtu6/PF9I7sGyX+J2Cgw4Nf5cCmYiZrUlpKtZFu5WqS0vF3xKR7FNgIBISFq7cjFiIxr8VR5zWzrhcXJyZxgIa9R7jZ7w9yEOLRIp0Ao5noLvB8wKxLQPQyunGrbcMX3XBrumPDxQTi2GLx3TB96ghluoeMB0u2kInQBBgxakv8LFj3L19UGaP1C3D2j68v6WssVoBBcfO1ODLKjCcj3vexhGeew5vYb8BrAvy3M+Kxo+7w25+HzqVNJaNN+N36PO+nYV+9qzD9nc1XVsnDqdNuqA5dY/apKURAu4TMGHe/ozn3bhpY8y26AiElHNuu6hoHdZjizsyGGHHepRTuu04GkB/PDD8B7IGaGIEnOwu8q8AZkSQv/HfflZs5kmPfaXgBb+i+Mg2Abt1J7h91MdJndHtKi1RgsHc1CufWVd07CqzAik9lXreyPxlSwSaCIEVGUxhw8mGCPxXZPJDQtexnRh9Xu9ZSdgMLMj2s27c5rgcRNHCrJjBMPI4wxXoj8dYTesXmDhwNE7qdr9q5tTtRBgroI0UgC92qNOcaF65BwDh4qmAfw7SptHlxTOR6fotxhm9a01p+QLdcQxnINWTByjZpSF1WUCGeRjSFyDcvGNgoDv9CCENU8WMd4Pezn80yated00F6kM8JqLkrF78ufYpvIxpLhNoNG7MNhZpEFMTK4fbsSqWq5Sx/WGQ61pub29NRMG15P2tOgxaITXvUuIJQkWeW7AkHTEtcVw/QkSwUoBtFP5FmONwOkSOC0SHI+9rEkIxjWZIjxgiLE3uxyig2fANQMG6EClelOZlnefXBJuyPQ+taKXT0Z0ynU7rkhLnW4aO8CC2b6mxUHbmiQc7cgHOwaC2i8RogEUWsyesl6ZhIGB9lijkcnnbqOMAJ+DmhBiFroJeZwmmcfYLDcL3Ty8Sd7HoLojlq5a9RSrYGeLtEQASzyWtxMWQmYSOpECHI+4DCGGWKXOuzfJTAMwYEwgAeAIS85NRntc/v+5DZDBVEw2svdfuLmTHu2+VgrMAtSp3mVGS5VSqMqGnbgj1csRxF7vzpxiwPk7VfUUo5zBSGhN3T7AQW6Vux48gGxOm6TNGn+2xQEo0WV8NRWIounFqWojQw6unVfCI3XR+1dava8eK2b6DrD0g8nUq3+QzJQEobElzbXgpExPrtSA9ofR+6U2rWAQVjOzcWVyrGO5VTSaSL2KiSJ9diLEmLG11VHo/cdjNp5g7nHCrC6n8HU6HtLdupVzuG6O9lyJWPLWVUgUY5qqFLyUYUlNYUmqp2mBiE0U48nC/uTcc36mG9pEivwvFXgduureTHYSsczclPkmtMBq6dR3+sm8agDaQvANjgHxmAsmRslI4nLcgSu76xWQeuTJFvKKkVceUujbA7LcWY1hDyC2GX7vSmjlZEIWbEcV9x6OSjmQ9cV7OmFlgqNZevUKrokRElZLsP1DHBazIeGkVirC7LuOnzaOrWKLil3xSyX43oGWC3mU0I4LNd3rKc+SxYpl+N6GlQaII1KuYAj67IKh1gisyiBsnyZePIj42F52BftfY3l7Q+EyOZW22yJLKOUdDp+0lA+KWPZgoksorw8rdaFGUbceLp7v9EVSbbPMaqMLWZMFCLKNxTFoKxLDOT0jfZrh422/wftc1KVscWCiYJH+cZ1BZSWE2MlMylBhqGrMsbhTBQyytdeHofCk2bqA8OQhzx0lKIhmmwdCB1iH1GrtbEZabWLgPBGRkJ+NdKCD7Tbq2cjzw91CprfhqVUddOIhDq2oFRC2pkpGE7oIYt6AX/H7DFx0FsmL9O9LdpiEM62NrKet2pIlsUjpKPTYGg34q79K17QSSGKpreoNNa9gZFcQ5CZd0YxDuPzILtv4Ok4NAOYkcrder3XxVjDz6xuuAW7RlEmme8+vNE7IgjzUWFCaX+h2VKkYIrRMIXFVy8u4/42HARnwUFwEBwcyd9SJl25uRcwqGD8BKHetXe652radZaxGbBURjiDbVMj7nZLx3vBqPLlwuEVpZmVV4RygZOxvx3F9GH23xWNlL7ZzGK12tmB42GsYZ21lhBb9ISjJx3Jg8BtrD7kggIjVJFClx+K3XhrPSyCA9PUnvlrr2G4OxMtmyC8YYsOmBktWS7+0pdpRy0JgZlz8vV6P88ZsYfUXyey+xmktqE9PJNdCSWTceS7uChd02bxkYP+AzIqyx4frfaOTei1GiJKTamsy3AHD5lyIrs+SMf2AIPy91sHq/gVQVYtlCQg3Shm3CvRVEk5T1qBi6Bh1HlWqkybdI1pwxcE4HsZHU4ipsb5lddsIi1oBh0kw1YTJ9T0UYBlSzhT/t6ErSyjs2Bnk0pVUfmoeNjlB5pJuSKJioYN08yVKlINFjtrXCW9O28oMtJOpl1VY5VeTyYg1WOwvykjqTL+lkZLpdOCgIy36q6H/lZK3dZoGTC1V0KViOR/G41eLSz/FEJ+XWayodbpJDfIQwMKUICKHL0MX+41r+N+JqcD3+X4qL9mmmCSjfL32S0dEowttZel2vZCm0f2bn+Vpr8KGz5Y/dim4mppM/xI/ICmwctclLcOeQ9sh5hCylZnwU2e1ym1eIepdFpwN5e9v/WfwfsoX78E6kUoX1F4ZxhGkSrCi+XFSizyvOOS05xQ6gCLNNpMzfuWVUpQYQifNamio7WyH5avK/WavDF1bUeCXP01ury3HNSTCj/iaq3IBoSzY/IK6wWBAJpQOaZCPxCPeIvcT8YbVccKdJN8FLFW18GXgUyZWLbICMV/EruO1NXWtvdKoVPGlztHYmTKRHZBNJZZ+Ao1rMo7l4VzqQkQgP9Zxir7tpHFpbRvruIEXnbU8VwtuZoXbCfScMbJzjR+t1VTMC54gKLSBQvydq2UruWDO5x0VgWzeL7kWFyQ5xlq2emKTcoqt0GLwL4+o7huVEVhizFq8ZCpDSGAEUS+5fRDOb3xG+HYT+QliHZRMiB5AGYHBCxhUpUKqbhNLFmWgSACcZBBBlEb+hRR7P5beuJacJ9kIhc7eh2IgYBdFhTCHjNr94EPRwVaH77POqYYCGJVBQ6tcyvG5hfzWZcdF+ExllcgwAMno9wl8PP+YIsfQFa4HdA8GZQ2w5xbyB6TTqUal82k32ecm37ERvH7k6MgIYk412X3RFqzHRFWv4deHRd+Z72s+o5P+ALhvTHSzln//AviK/V5xZ4Nm/bb9iqK3MEQHkOQZ7KHzbhGjZsc/mg2wmqhWSFqpKfcodajRcHVdlBbQqBg28EqC3Pg/eLqcpecEe6Cy+rc1rp7OHM/UnerQOmYo4EDJYHSJvTFi4GPp3MHMUiCDjqgZO5zSa68Xs7teWR5sZUn8tGC0R48f1yI0cre+woI0tjt6meEUEboiUPgx1n0tZo8zu5DhgY1A2G3UwG/kBCiKJwG2ySoj5pnn0EuwaRZo2ZvGOXYdXVUhZ1uSxQez5YQ3Lu9O5A8OUDenYnS30scE87n4duhmQolcwwHxt/TE2/S6joG9gNaWAbxz+m2dBBzoV98Lwk+4Q+HTI1pUZtZnHnFo0EhbD6+cMdmDKQ5TozGAj4UBhl7TejQo/Xa62+koW36Hlw9q0/wohMSdfzhws9KXhTVNLFcUzBw+DAirzOD5dynMzZpkT2bpvuPNRs909R77KJ3RQmJe7MAghiHSQwsFptJ/ZqrS+cLrgYrOYgupiPZxfMS/j0qgAhJLG9EF9A8LGM43smkd+CQB1llD07ece1UNjJWAIGE9Bq3h0JrCAT7uMmbq5fhIuXuESaLBXiFMlp+Hpt7jFUvKZWyLBHe+VOAsuwFes7P37eBHMhB99Z2wZtbu1qrF6DeigJVYWkvxQyGqFLoZmF1F6lcW4I9uuzCW4y5j5DvYo2vug1IA+ACOEh7MsPuu/qOeTwvlJHMlAr9vPAtaOKDlmEQBqHfxx/1fTUJbnlSeB0nrKHJ5CzdCMdw67Fq356lGEEMNizpCLVKsC347QVbGlRvb9o8DiZvNUuLcnx1jAgI2bcDMH3hoTSaGIKWw1zuBvyHFrv9pgfKhrCDpD7qIRpJliyG2xbSgDH0Y0n7Jl6ZE7rojo37Hr3Cj/USPUuNbqan12ZZuT7s4zKbw9SQNlyrKjWyaFjQVCGtgFs66dWA5baXuBEwNgPcVZYPUp1ji9nlIIZjbGHNaXgjsSrWaDzQqoqntQRGtTooBfM1/aRlAKOvjCFAxi1+r+EmJjxTU8o0wamaaSHDMjVZLMmElhJDbtRAaZRpoLCooChU5Oc5yxGULOgTK7i/LSd1QkpGDENoKGMJ/EnlWbGqkM120MLyCZC5JA2OifJQjcMbKvq7rjYlE/Bv/AKbpfjHHMhK06LQRfYADiWQ0b5VDV5L/8EyATTlNtNEhUbNsi3om/IKfZNSZHFRzNAYjkN/S6Glqwr6Hprf7PlvpqJ37mgybJErleKBKMu8resp23FcmRtmDn5TXGMXSYZWP9IwXjbR/xY9TCHBgKTKHiLOAzCGPw4r4EWhQvC8gPlindRJX04J0u2E2etNXoiom2OByx61icExKtJXd9LxabIcZ8B7rjSWNiqoD0bd0tAvKNMEdqHlOVXS67I3wJDL/1JUfZtq0Q2fL7jj4lYyOfzX5qmAzrAWmiB9ZSxGbJZKt5Fx3TTnDhgXWn7lrYvSObuNNmLZkNVcuJ+iWx/r6dYIcau6MBsYx5MegMRuKt/ZWamUO1LhWn77oPhk1yUZFD00/8qV+cjIyP4ff0AQwkJRRgKrulFef1JOlnBS/6+zc+E4TfXr/dWUdVlb7/7ZSVxIM38pf+qut5Ir70qLF9zCoW7rysLwFJVWXE4XodHoCKs0vqMgLgziaFjaf1gps/XI2QP26kbjJQ+m1IKVPmPh2x6UNhFrXoccAqpZxsl2MyStc/Qhhwg1ysWHOH1u8beYe05wL/TX5rnFRxC7vJU/2AEp8q7VOjTLXXdMNuX62lCo4oT0QfXx6RTXbZHcxmisQaEeZyQ3jK/ioh9kEgk87Qf2tYzogmU/M09C53RGZ+dy9MUdK5kgWWdN4cnwjl5+a5NhqNXgBCKBZ01JyaCoLV3J7Ip793RHbnUoEjae+MWpotrLngsBtbGRvDmYycyKC0nqFggdQdmwjFEy+w9FiT06nuMA3WqLhL3Q/tODaqRdN1Iaa+EKaYpmnSjIIImXtLDPgvV0h2b+Ul48ceNU8a1lnYFBWHk8HWdwfPjcTMoUwUo7otIK7ekRpGe7rsPp8coSJCJrDQR8hA7usK9cMO5ohoWRJCGZ9ZRmmlpbhNOtYau3UXScUsHucBRSvDdZaWPrAG8DGB7G5otjDUVdJ2S5bI64lAbKlqZYz5TLOBTlkLtF2oK7uFooI9EUMlCCFKQgFSti3PgxuQTHUTQ3Pt6PqQAuqPQBTIQcefc9Ta8G/58bmKeMF17u/98kEPVYsaZYZkx+qlvAn2tx25tCvhPds3n/74DTm+UF7Z5LVUWeoS8o2FMo4a+Cple9TIfmVMXBns5vx9Pf5/Onrqqv3hpSAvSDjxRb/a5YhIM9dYqJpSXPS84T7jd3y1lzv4ZtYYZdFS34ZgwIfEvoh7NhE2yC7ersc1LH4T8k2xGemlEYZQUK6z66yn6DxapW6U0Oc5tGra5BQJ4kcEtS1x0QPUfnd9bEEriM7rHrms/9FSZd1+xn0gzJO5nP8sUvw2Rp5FFhZC/y9uDOfUUHUNTWu/I0eZW+NlcPf9BOAJ/05jhLz2ibIfEnYIhaG41VeahAHnp8StubeyoVjULbWCRSxB+NsdEsoyi6mVHo4TJKb1c4UY5TlKNUgaJ+vsmVnr1SfeURat/FXgSGbY0iOHwlDFGcRiM5voB3AySr66OnFNLs49LoboRoBNJ4uOzn8ROddBqK67SLVKTZlLqR5vvP5i+E+by00EsOuKV3UqBVGDIg5nLW3IJUnqzN0/WWd7VOdkkHqdNEzbY0w+y5tt+GAiS5gJrKToRSaXsDBQ+/aXxsTb797Ud1vaEhvmoNRMRNjSYGhB0NofJkmnTlhxw60hIh2boj1QdYr9JZW2rdy4Qz+v9IenC8xP6laIMyj71Qy4w6tSI3WnNiDn2SO86gpqWlXlrW2ttcjv7Pb3lFXq38/vy9LzfZ6ZXYwZ4SSdOe0xdAPSY0gy+Ffzk+eLso8MeyX5fG5e64JpFUD0xOTB433HvzGyiVXafufa9cg3r2s9NtZQqRdeRmG2nwBml3nW51vcqjmwCeOBPOPl4Jwiis3d8oFDRjJJhgH4FS7hrZKU7T1+6d7t7RpugjDkdvt30QYhTGynpQ3o7TKELOby46VZvG2kSkcWq+0BbjcHcbMw280DXksUcnpZnVv6LzXV4rnUCgqgJt3RVJS5bKAvJ895bsksjUMVaqool8US0wDAStmrMgwthzXOAfktS1mU8uddOaCu71sEK0KPNSgaU8k1J6melK01gd+lOy0oOKlgqVI+SoFyNdNc7Jhia5HoRBK3aQygo2XRellwIbIwhr1TSqUfB0XLOmeN/SKWpNMRHCCZcK0DgTY7TjUZ8QSmlkeW5Tf2BQ5zGjbqsibpAgHslYRFAUlYZyzNSXmAw6SdyeV9SaiaJq2EhNaFmJVFK+y6OlHPc89xFB9m2h1J5BHo8KEnl6BpEscXQjI9RCmTU4ivsTqcVJ6fMYYhb4CEbY2nLlB5RJFN6B3wVYJ7cpJif/vIXAX8DjQsVmRuOMJrTUQx0GDGInaSYPPowpfVKuau5CiTSsAv8CfD1wMVwcHQeGvh9nYioE7qhUHJkQVxLIFYkli2XAN5qS0ZLxw0fcnuFArB1r1bZSx89OFtaLGPLF8Ljmc2pPQeAjYTmsyebNvpKU6zE4WsO5PZ5IlcetfoPBr8/1pc6O+10C9XJiHFIGg965enGfJklzozp/Sc8XLLMRbt12VZWq68HtOwwB/X5a70bdnUzgZVzuHdk2PzgtZu/uk3Zsu0CTgrAP2fAyG8uKW27eNOFVdcK2gWzpcgwOZ0ZKlaeyNpGZHspFXnTSMNqx4OwjBfLI/7fjOXQfYvYX9YmytltcahJr6MPnJteB/++XWw9qDkaAHcRcjUEZ3lYt6zDfALauClrYdlhvhVAKPCZGBqQUjiJCmlsVkM/VCYJWIn5YQTd7q/IexnIEoHBTNRFCeePwQc2Eh0zVoFWBdWtDwfGtK0vbU7RgV8wqogj99JnertBFaOkT9RVTUnkPfGOroMLYtlK4gYM/O2I6SRLEJsQkgWWR9qzyztENCXTRb7BPVlyXeERzp2GYCwKCnXE8JwVeEpYzLIFg8LWgw9MIEniYGR4CfjrSMOzXG20G0SlSGHBdbnFhkxS2uNeYS8Vw6sV/Ayd2+f9QXm4Ru11+FTf9J5gS7JRuXdXiTJzcldIEXJUoP67Gg5fFFUKr1fbGUt0VackpM0ArNIOS89bZjeHHfyEsfrsjyZFVvxtKjGjsJTNGmTfWizYzfMnpwcLLBYfqageqrLrRhpNgP48kqvzQ/Ogem3wFASgUM8usK4piNIjM5asWskaPGi5R6KC03D78kDA0y3ux9TYCjsnnreJUEMmuprG9ephNcxxlZNv4vOOZSGZeNSdwhxKCpDymGps9e7DEygx5zd2/OYYk0EBWMSIStBqaXBOVspsaZ1iKghuBDRXNQpbjoewHYeO+COPedR34tZG3bkQ4wdEixSkhUMpYDL7DAg4TMJtZ/DEgQLoOnpZdiuZyydU6CZtxUHbT7F7YQQDrioUDYiPb1FEmhrBgX8RZ9bxq1qf21x1juUf31I4DAaNwXXCwVMnSSHWVWj4GDei8DG/AtQNijSdRf4Q7yZnbPLCk1w9LaEctz8yTVhwmcbPNINK+2E5vuYQ71ZxAwo7k2lKWdyA3j3q9wWPJtn3wGmkoiWFZA0fbpZaSMPx661tGzKw04X4OCFLaY2ByQVUas1r+OQXLQoxubfidvPwXru1B2Nko94LeMDUe82g/F7uSp1JQjDt3WUInEUl6RIBfC+Ol+9fc0gDtS7PFkaR/GjtOFPKSJ6KZ2rW7TboYZEQ7qxCn4Poxc514RUuqtvZWleJ4dQXd3U91TAB0atCSReReNZRIV+HZtYM1HiiqfXurTWCYPnX84OkAYn+/0h9P9eRTCYrtmOBetnS3MTGWUfriQSIJdYPKId9E68EbW+VYQsH/xxo7Xhy/ov5/hekUHWqhRw6aSprX83JLn3fs7MG8v/KYLhwmHSc2VNTOcgGHo9qBjgbbvUnQNFj9/4FKkTZHmwpNKrYwiLswIEA8e2ICEtvJ5zj8yX03unhV7+88DSYE907+SWjZDgLgW55aDjHQLen3sfqRYxQoO1GPFiFmgbmQHcyyXu3gTT4zKas0O0+fNsQa9IXCYRAX8w/w7/td4HFrV84bu80VWP+qm+jrrgcBiDGoiEgOk91AYkInAssDTyun5ghzJ4FX6RqcUIVjOBrG4TA4HA4XRRtDCXnPqumMvW6mCZXnZqiJy2UoGQFLl9ma+V9WdzY0Ljyqj4Cg8pKvlY4H3W0N2dtq+ljIGQwaKiJBXNN6xrhiunvH3wz/v09sF1vGSNo+cJDRYksYzYgXtd3uPX5rgUcTdqiczP36qOJHF1GokxCl4cIJrW6DiYvwUw+lULALrP8+ofErS7FjIqSDr7nXNm8L9v3HNiU0gYD/xrjVzGLs4bArbMmUoliYuXy5ODAan5F/trwoaYTjjXh6GDgpnMqC89VsPptU9EmQ8jKiT1MXe9VK1CvWxyKv8hnPnjNTetvh60G1LLuZqTR9fXGDRUSdclj8V26Xrb+9P56O/guu5WlZnBXzqwSOIf+QGzzQlx1wD43oT4EOZcgvkGnOnkfpRYRF7vQM0snIkr4q5c0e45zF8fkQst3uukjBtdFg3TWjMe6zVqRVZiUuOGzMBwIvLzHOX5ykZplePsluLxOtT+BSazaSxn89027McSpI9rHI7NG6vrjQv4ybmYFGfpYweLbWQ+z8wUMTSNQAwxdfqCpoPzng/5UzP505A+ofNv8pnW17mdI8Vb3GBJpD71mfW5bCSiETMkEAYUoo/KOQte2OE/CWMNg44sLhPnveDjoXSj0Cne0gJyjLSQIHLAFGgAwSkuJgUF2/dAMXvv3TqDx3yaaBqRJBniUqehtIa2vTF9i1Iw14zHHV4JCcY0CZQEh0QET+Uz0Nh4rnKCiFRWATQAARiRKYBWERVIrA1+4M/Ir5FBsaWQxYAzIPMDd+MhRecBbACxJYOkcvEwaeuLwlr/gMzf10A2F/IgsMMczPavZchLkZrv6LNo/trv3jj2FpbV4MuJ/D333lUw1wV4y1fDwxNKNXimrmf3+1vCWpZpjsm3ey2WBQR+p5DuhxFOCCZy9vPTt032+jGLWUUvzpJo5uVFHrYu8D3d/+ccqfcJqAFk4cXyChvPHT3//yqy9aey47LkYcNbFzgV/cddtitSrQaoHKn6eM1mVStrycpqOD4m/WleIjk51y3/agbFPFXOV1TLiXhy4UC9qqupqz2f1/OFzYyHx78J7bOg5ECFi2sjC0Yc1vqT+BemXv9SK2BPtlC47+ggjbM3K5lPlHn1meOMqeVzrNSQZ98bwgX5I/eUnW/nGMTimTpDI/+j1m6HCq1Ie8oVFgMXVtWnUDzQzHhGDuwjn2KKa6T/xM2x0BAgEVAd9+zpf7YghscbHhQAD+HPEXAUlLxqGtxeKIc7cM8V0Rg2Vt56PKfJu6h8nNm29Uv8r8UUSVIaEs4H/6iJxDC/jGNQbQ5jRNTfbRrYb7MTVPxfe5G/4crJUIt8VvhzCILs+4I1iGJXmmeQA8GFvpxCjalspqipyKbMeW7EZtHoDRfRyDt7E6F2BmXoSpGfBHIz6NM/gw9ia0oOYO78PU0Hq5HQaLwA9LqzqBKqWlFNzFZ6K13GmLzlQ8WTeOx8N+Eb74LPwtmNsFwotFWAFeLMFedyOkmdgQPXg32/B2dmNHdK/LXoU/6u/Hxepc5Z18v2/AP0LCIu/F+LEmtaDZ16HJaxHwueaZjs/Sh3pa1i+5T9faf0INr5KDOoyapg2x+JvXIuBAfAaABWANggkLwCBhFThM5X7wsPM+DJhMW1KYil28QRIsxABZsJkEGSJ5kA2rjYcCE1VnR7i0ASosVIMcKLYKVoj0MHKh2hbYYKKnkyOhV2GHx75CHuz2BxyQ6DJGwWEJOOF2FaNh8+lwocBrkA+718MN1Z/HGMjehAIo3oqxsHkIHoisN1kI9YW9dDlcRIPhYkqGSuIcC5eSCeNg4ZS/jYeJc+JrORvNdbVarmjMxNi4ySRewsWlr67dk2okI9dmo9ak88imY62urzTUFioZuyTL0w05uxOLp4Yhy7lKTt7P50pmS2ddH2bxJzHpecOyVW0sX2XYkbFhc6rsQG2YRu86sQCXvQw3tUEyP9mWGfK2UdTgeLXuam+r2bxgwZfFYCQLg+WaEt+FlfMM35Cn+JwTyGabyU5eou1Wc7u3zrhVmkx4yAhDo/JdYq0060TIpaYmzs+0nIsWkZwxd6vWjRfKcFG8Pc4aWSUzWaJWO+JcINY1Mae0wDtkzM2GMhjXDw3OikEFAA==);}\n"
  },
  {
    "path": "static/inter_subset_es.css",
    "content": "@font-face{font-family:\"Inter Subset\";src:url(data:application/font-woff2;base64,d09GMgABAAAAAFP0ABIAAAAAhawAAFODAAQAAAAAAAAAAAAAAAAAAAAAAAAAAAAAGlobIBwqP0hWQVKGZz9NVkFSgSAGYD9TVEFUgiIAgmQvghAKtSSrZzCBsj4BNgIkA4MEC4FEAAQgBYoOByAb8X4Vyu0j0Z0AWlvSeueJItg4wL1hBqIo3awJZP9/PS6OsPjKARvJRoYpqIk11sDeu7AgyaIjMnSwfYe9BikDC3Vykd1XsFnqmWpbLNyEcnoimMzPSkhkYT7RUQUTFgTfYJkelAoJmeeMckMwTJb1E6xWqTOzWRx16XW/RWZpqzHx12zI+Wjh4PJezf/3p/8IjX2S6z+Pc/Xn3id50aalRYplxBlnjW+wKsI30nZFjFVzRlYZ0WIFmiHmrEaCiEgZcERARERERCxRY9QY4ppstrYr26+0782YL6Xvl9b3a8+V0rNZ10tMG57fZs/I4zBgVbqoIKSMRBBUsAALMSmVECZGIgYW3yjGFKNYud3mojnd2kXG/7lVfQpadGd7ZkU13hON711474JY4wN6TqPMNFSJ1rRFGRrooiiKoupD8fvF9sHlcgqGIoglWhSMVlMwJRGs5c/Qz1k/pl1rubtXe/l/H1/1c43XSt+eZBM3McQgKiIiIuAwDAMMCIg1iW1dHz4362XElJqh+gkfaZtaYCjTpg7UYg2ppXVmTQ6nPe7R5KhncTlrEDe2o8cQi34x97n7J/nwNp+u4Ilcp45wfOAAIcmaxObsTXXnVElVGMZACeC/2u/zdM/MX0S5SSQJvWRElIqe++eFQSbCICqUW9GMKnJ7/j83+3uHYunYympKe8kWj1s6ELeU+vV62f5mgGBDNEiQIJIRR8QRcRyRMiIgEkBEQ7CEKGYxlpRmutvnQ+f+cjlCGiikiKg8HHPhWqPuifyxN+6FC5WOzY8UUx5NF00jhRjSkBRAYGFe/99Sk+63NC8j+9o6RUqXXjJXqgyYYxbceQdwdmaVmS2SV8VNvipd0aa6tl0JKK02ZjmtVZhjeQdgUABrPACAAHp/r6rV/ifZu28zRSdc/rz4ZTfwaq7Pueo/QfkEkQ4gJe2ClHxDYuUZwPERtGYhclPghdimXG6s7wOyZz6p9c2ntIG+GNqbPqeqLA++MJtfvWrD7tD+ChGMMWCMwuE42zG23GFt7a/4KF5PRLgBIutexrQMlN83vDGW3RAxRd6kNO3/QBAYAcr+O8ISDijrEtBn1cBVfWTULKTWktUmOTkbDXWeGWNLAhZ5wADPDadJ0TZtviWXmwGZRyIRVQdMZcMeGPz4Q3QZMxEAi5iSPT/fsbpoHiqEk+FM4wK7NwH0nxK7HOA0TIPNeu4LS6jXsd4F0AnAZ0VbFjPvWStpfGoJgCEgKx5PtBKkugw46xVQRA0vOToH9gpbkjp2YyGedbl3FiSpfYrpTn9L6WqPGTg3187uNluarq0Lc52pv6QTfj1Xd/hXM+pcv6/67eKgt/7a6hAGur85vDNipF4gvxbDYWMjGy/tTF8e8R054xbhD51bGcU/HTHKMJaXPe9Z+Ox1Z0lx5e9cFYEXH+sb/clnVgYf+8xZ5wO1P2isZyd+2+i1Puc9X3joEdNPPfb+05+1fWrHeVeR8JUvn+kyeNcfnNnit8IWE18ZaPpA2upfORyT+5ZwG3mffviZT3u2LduDGwM8Z28t2TubCsO32wego2gudOV7lN/78kwUbH20mPE28KYjtsnW4MdHkLV/nDvDz19QQKC6eu6bSYDeHtlv3LtsbRncdv/81xYgffLj2Q7g/LMPfm4dyDtvgLlXgN9+sz0eyL5kn+CBD/3Ili4HX/1xbqMT8O0BNkE9yX8eH7u9CURwDO+LrPknyx9dg0vezb9/cOVuvSSjM3a/7PkI8GdOXgGg+1vsv9hxFTgj7AXyK0D8mG/8/AtAI/1l+nFYI/XTrW+QjzMA6hoOQoI1zrI8fYNV1jmTNm/bgTLJBZaUFSLOSWaIcG6r+MF6PiNknmFewTRimG8JszLDzoMpZllgp202jQXbwbx6FRYhvP+E7WLGE/svC8d9Gj4C9iY4GX7OE8Vr25hnEbN9ZzpuB5k3mJdJQZBT3t4j2FMr1S8jxBwn8DgAnVbLDO9TxxgGCu3/tdaBBVptWgEAzjixqADxxW1nHz2tX8Kt0E337s4AOPAyBszDt+WuA8rGAuXNijf/+X0J4WoeXQEFqIXyEwA38WX4E1zCugKUQIol2I4on3XX3XqyGCuyZr1cFQCcMcURug2IyAOVOQ4REgTiUmoW/0baYGcdNaMzcOCkkysGswS+r9F9ScwaYJiooGkWFEQhCWLk4ir04kE8izf7tB/7w5ODKMTVALPwRPxFzVAbiCiD6YoOflm8IUtlCE/ivatrq8OTRbRrvSgNJjBNBQtLQNMAw9PBh2dACM8ECs+ClG+DSo3n1XOugRVOeKd7BMRVnHb+r35jOhz1kixFDROmHAEdTGRADAma8Bre6IM+6at+7OeW1N/K1iKGhEoIEYurpGnKFlDARCT4UtZ0Cos0wmtMfqaA5A86qo1AVAwZrNiMViNhSd0vC2wMHLQiudj7o3yyg+EHCsigIuhG5guFgoaBDQcufARIIcQTIBGiKIksKcWUvq37ZFmOnKKSUWUVY8bVNTRNaF2ZDO7zAjr6MGJYfPAQ50FBJDLgJ1a0unKKQgAcHqJ08v0ZyQBQG9SgKyCLyiQHwHtDLXqZ1AGICOpRKdmgByKUZ2A1gg7QhK6AZlS2WkAHaEUvWxHoABOCJV7WHwL8ZqXjAKzdwO5zGCswbtdjMpRwCQw6wiBKMF7ZODjMiC72L1pZ1mYuTlICn8DYGJRRMJMDlhwp+EjICwmBnhkVkBJRlCXiNGXTuMgQkIZQIksTr5BiTbAUZfNRmNwrbJy3uCjOBPB0M7dVGB185OJnHQlpREOO1McowkrB6kGIv2UfHWAUG6u02n9HtESzhkAlQIjJJjVEm5u1jCI5vUbZOBXD9c06x0oOi1dOJmkGCfkW9cZ/aWxlDvJMIpG/UKRlgmTKk7TSRLPvh/WGeusrReqfMICKttEoNQszSrMJ4/4oPMmXLW5Ku7JWGG0lAcyW1LOvQ8lbPrN0O2qxmlQjOc2bsiGrq/q6XbqpB036dsZJ1OAjRBW/LO076/pqHc+aeMQlPad9SetuuYsTnOwbY0Wc811wL9iypDktU3PJX8ddyUyfj3vXLbLK0fGC7Eo5S2Ucv+w87ROV8g5YAVSl3Lt/llok58rXoeZltfG0j7AeWju7oIWvUzIjk1OYxiRmpp84e6FvwJTmzhbtgJ+GH8tx/8nd6nvmWG/1uPtQ6hNCBhh3Kbtd+LwlpzHZ/SktRr9wFASPW3dqMhfomKqYvLZnLkpYVbWd0Ts9YjwbM0ckwDedcVNWMTQjgtPevSKhqKs44awGKDU4qLlx0wG7arUbT9WlRMkveLfqWNx2EhdZ12jx4JH9X540O2PYxVLugqwZr98oNalunWp2zQN8AcZMsEzIKiajLT/trQ1ObdmmRmfoyIqxtpIzo6glKoU3KnPn9bpUyN0WgnBBxkyd7gCmlboiAWfkg6B58t4uus4a1aX2PdimNseJXxatlpfSzpsyZ0d7pBrVCmsZpxU1gtWUKhi/Abcnm2JENW9TLNRHbPpHaGZsIk2Pj3ZyrHnCbm/JNz0v3CjT3OHCyg1uw0eTHCMuc2m11DHPPyYGeUR4prDRgqH6xbhje5SadOXTe1KgwzaKlH29+LI9rELwqd2o4JoELcxsNP6Uy2yj0Sn3G/JDhmDmeq/Y276QPG7dP7gFUW7ZyhjzccZj7jWmvMwZDSxK3XEPJUIXkC3fGQ+WZJG2U0/QF92lvfdToeQOzgdYzw2oGmxEcSSabDy8BZ369ITidyYo2rG01KUShNfMnhvdFfdtn1BciZfNN5dlj6S1EazUGy3s+kJcck2v0yz3YEzg2eAyNRZvqwPSO6TA6cmb5nKKzHmbnXzelattIVXcd7o3Om20Wo/cI51abR1oc/3fvUJyTIKJztuX+86jCTl4sL/mvUl8dLnhzCa3JebDcx4/hWXofkI/wQBszKl1Z+Oj+rkR2Cwu8QOcWuVTw5r4yay+W7G9LB27V5Mf5e0tLfaJTK73D1CgvNN9o1T5GgpY70G6Vox2UV697qLWd3c0eRyty4ofQKQMOPn4ZiT7q5Z7qm7nyRxPjMscJFU3cvkjnaPzdLEzNb6/iTZ+PkB2JEQf/VgssBsayLB+KevhQzk3Nw+1eD970ea7u3/BoBnYmKmZ4LQkga58xLMb3ILmv9hvoHP2uVBuAdhQiRxf6WinvHrdTl0yeox/HqeTx/ciE0wzSN7pCYH7VwUobxB4Y3KHu/UFsuNc7pl88Ri0vWmkDu6qUVO01Myu4cNqZ9eC6Ye6/LA/doiPhiS2pbn1ue3by+BuIvgmorZLwlYOrCzhjB9hyz3rY9fORy95umVWG2/Z9I9efTijH9XcOASBf3DmGwuICddi3PPIoUT4PHJi89zP9cREPQjGddKH3tiqdlbjx5o3WWvgGrA+4lHQ1+lHHls22kV9/aqLEiOPdqhu5ckcjnuPFEelwpJbRM27+6tla3AqO2FSqYi/PiXpjoWihOUB6BIaHV2k9hftBTYbwdWr4203td9I4003q8e06lV+wqWlz0+ldlc1ly3CoWV7Y/aXk7aDdk7DBW4DKGyAXrCVI/Yik11QFFe7hcZpxTJFgZ5Bobk7fKcNburecM5UnZ1resopKbvHVhiteEYYd0SUF1WxL4BH9IuMyN3Cq1yhK9miDA0uiRNmn3iXCmyXbIQY0MSK+fVdAiLLPHBDW/LWKr+nxzq/9K32xsA5LrOrPyIrqzeijuYCpyXmOcRT1+Icdjv1NLnhzODErF+89iGE8igc8fjoBYTvqDMi5ujIrP1mE1LZDHNP4HzcLE2AdfET/7/4/9mElAl71xP3nt17Aaxw5ksNw651w1cWTSSyB9ggp7DRw7gaslaelO6duuDJrD8e9GdpuRFefIlHzSIfBxZM+m/vClvi5pi4zX6BqRtI3stbdqxiqVMMa8hbC2cqfSmN++931sCsAgJzdxK07CI/ogTXaCAhmU40x2aMfdpSvBeGpqSDuroEOREXtgZpaAU2ZnIpfDWWLiVwiyeLhJbGXoH1ZHOTNqE0lXhcwCccKeFx44pjiYclEtJYSSKInFH/UCO26jiQbf2wrZp/mCJs+DfWcAJLS9rqHVPJCToslQeMVcWxWKWx0+mGf6NSG0xhAD2jaLJYYN1rFFpOyoqTJAQsfTV8qR7YmAkDcg0uTEYM1sYWJ5LGJBLC4ZLYuMRSHuEIX0A8XpoKUmY0vSzNtjxgyLS531ZNPbCjSB40We/J0Fj4bkvGHSyI5nDUMT5N4VFSak8DPRbLXernS2pbNytILBnEg+VrDDVd/Y8H43sEXVq6zw/lj1y+oeLkhaXveFOCLgHmaFbf3gz2ChUzieCbuYOlMsqLHjqXe4TIPJJUk9KsT90G4d/rsmKujOBBX+3a1yopkw3cIhey6lqvmkSE4ZLYOG5ZEmE0JZ3YVx4DCDpID2zM5GLnpZR6tjL1AJ1pKhvXm1SqMglZSfBWgcYD9Qzg8PhTvRGw8VXFAYlu0EI/Ah2QwZ72iYXnPswlOTlH+8hrMb5keJlIME0HkDccQP4woPHAVqOzG60umpukjl5Og62i6NmCGG0gtUvx2vA6SxUXliErSbhh/MsUpUdh0wj47ECTSUym1tJgHUea0hXKMJUe4xT5GkGgnOA5W+SM6kelEitDj8S6OtU1pZfUuscgtRqfVZtekUEY46UQjlfw09PUfDspvOmOVmeAhjuALSCeREaNXTSsbhLcd4/78OnnYdGs94X3+ZkkxsRDguxrna+4EwnM9v9HPWx+IMSOQSTf75Zfd39DiiVmeJ8Auw5GaVeExLNT2CoGARXmFTWjbZnGXV2ZHsGt/o+ak9qDSZEjGE3LptNyaPGCEjZhV4BvEGJ0lXZxaXUaM758NDCDV49h8cHyHQw1Rt3hoF9OJfj7umCD3LVCD0d3VadOBrasnKnSM/RHYcvIrQIJiGuihx0qHdcfFhWUxwOUEkXPBZjHjFknGcYSMyGwWbBXAWpzHowBJ/cQGPzR+yPwFgpsTFMGHt1Op2i6sx3g0v61OZxNO+vnrOOsGf+n16F3PHJN5Drg0LUaC7v2qyNWwYiMHSsFi5izTsa0+Gma5p2yX0AJT8ncCiQ5y1rWw9gQpmCpYjksCwJSROuzfhH508fXUJ4NtLSYVufjp+lsT/tbG3jaEn+nY2MeoS3ltffvW3jsWKwtActvMSTlnXJIx7gPI2nfQG3h1Zx3wpMKt2Z6+HmmSvcEUCQ4D0mwQSE8dSUpR3E2KdWUHriztuyYNM+HBRuAQHlN2f1IhdE6fciGQOEUbaNEaDBkvm9QQHY9PkUICUP27/NK8/VG84R7gnxU24I5lqRuq3RFz7UoecGppLjBOGlOT7IbLByujMRXMkU5xx4mAcEGzfLHsMSnnvMfwWGpkGADA7/WAGzMIAEhuxCT2Wud3O0Q3axICysODqiMiCRrK0KYuJKVV7oQtLu2UNKmYfuJhrXebANJmosfSiyR3rsrBVJE6DCmqjHU5dK454Yi8+lESfaJ6EyjS6pxZnSJgp/WnksDQ14xL0S11kf2v0Q6VVYgfngvt0oXVk4PrmWEB1aV0MLCNOGUGhotqLYsHJxZ0bMUBvCX9Sv2LwdpiOqH8dJDIoYnlJMQALePnyMZ5aem9V1gK+Tn2el9qXzpKGeOvR88J8EbYohkhx7Fa3QsNcqP708OlleQGBEaEllO9vfjV6BYx/0z9xFSfbxxvMw9geSsPR48bx98atY+sPT401AwtFl1A93pYoC00YJF2dkn43pdMvpmsEuVy7bl0i2GYUXKuhGTEskUKEXeJn5KVdUUqqYH1UT9q0tDQkuMEC2NOpXlbF9ohHoiPvWQKDyguEgcgr6uu+BN994RFR+wnbJ9PXXd5h0JtKSfK//DXNnU68OQ1nppqCxadRM5MaUpmFJMjo+vo2NX3I0m4Qrn44lChSeNrCPKVN5jwnYN9lBwnLbgiZjof9sX3LoH/vvoLcf79qutL3tvXRZcEIHH68kuLm7KOVtrkWNt639PkH3MsXCrhkXBr0QtoWRRCWhO8nbcLu56z1ib4CFrcXrPWFiG/DCLY0wRxR9g7PzqS3SLozuSz6t/E6W3OhZsrWQF41d9oWZRCRgOr43GOBvyct3HwvmyIyC+CztQy3wIWe+V/pkVtt9744+IDd5LUBEEHK0g6oZx7Qbgllbz43e8YCEMVCsxJOAJxCg/nA9qsf8VEs2C0mghCKxQoj3X/blLlu7yCgQMAN3eDAFn38tClxPHXMDG9Zn3Y7Hr2RmZFz8rsHzH78W4Ge8JrrNT5mRaUr3S/JaiZ5YTZ83aTxJYhvonASfczI2ENurkRv5crkhApgIfACwct7eZf08QEcgLhhriiaDQYANIbq7e9QSaQDJ2nFsAZAW2wL2sxqm8/vgZuyCq2wxU6k/utsckGwwJtLIn/jOhgXH3SHw1xpm6Y2w/LWZM0FGlbO4kSZeSEYoA4NdB+eTtAKo9sAMWgNyUh9jEvojfFj+zxBLPF5tN9+D38cP8I4tr6k8L7abdADrxAAC9ctV0peKP2Wv056LM+J9j4GS0Nz5YRmeaX6hns5d+BBHw+Gzcy267ZXhZ5wN4LS75/XO9zxNW/P8H7MVntgcz8L/3DCI5c5vizwUZH4ov2Rda5NQm/j84yfIr3NILZg7tI39KeXrRoL7LkX8caUnNvmQq9GbarTT+MpeZeuIRDqX+uvKf2U/nvObMN3NHnz179u25f+Uhq4HekefJ/7agj7jvohAtS7pMNHtvySel9ti3Sv2lT5cV1M+WCe9++fV/V7y34tE9b6+8VNXKINqOp26q/9fOtY98e7ex4+Hm61pe2/LJvXf3xj4Rdq5sfX3rp/cJn4S5qdwb2v6/H7nZV7YqnpMNAHoKkJAA/vZ5XQYARn6pGEYKmeAty1YGkddZ+gy+AaQ6JIiGTiz+A1/tbNOo/Pp95n4wNLnbGIpDidIWg8+tFg8/CMbVXeaKtvksNrvUd6LhXXUIX75qcdqZSjprU7EgyLrC+T4HzKqHSYxkw+pe4wNsOY4+lvfR9SBUXEkRhQAEkYrAgkDjTW40bgWFc7WvRzBi6pcSzuxUCTDXkKJWZm2FQ/BCpCOcGcE+mm2vriGW+R0pVnRJqpcZkWu957sVGW27Q0XfpWomdROToz9+hcsSZ9CKoPirg17o2KV5YNd8CB+74bfV2d+4e4GwV7VVazymPQOPzATFgWDzPHumP95PILPII0Jzteh1zsQoyPb3MHu/kF63o2eXt1zBjlicYW6SrFiQMrbx8xA/KUwPZGx9rw/cluwrcfE+z3NCk5BsEgdyDGIxN86kMtOJJZNCUTgC/cUIQeX1BiJbIraKZPMwLkFkQ0CaRC/MFSKkPdViQ5i3ZKhOZiCQYnGKomAsk1CxxLm5IhfLOcHsUu77ttSz3oF9cCMeBHaFfWFm3aiZR08nackqP1T+KQx3XlvC6xoReyK2e8fQdIwP4oOKwsCO2YSFlczhyipKZmsfyhkIQiO49ZJUY1qGXaRdQZ7Pc7D1PW3roKM2Tld08e6rnJeb2TspjhJ1P86OwcmN6KGRFumfn3E0gPpMvK2YLTLw3QIn8WhTr697Rin6I1NtILsHPLoFmrBX2n5ay/pMRuium9/9uk0xerd9iBU/5AyOYyVDc5vK3G4eFpbjeKmQ0VdhwJ6iTCsuBpkl+xgNOtGiryyhHEHoKTrLS7OmDIPG3GlPT+pY+kYlz5wZpoZGrFbH+NO6oSs06vmdM7cKyxA17PCCJu387QtHEnCsQpCBMCx9shmCRfYFBRb2aECKu0s3wpjBTWKolVK5KhQYhte8dqR5rx+P40vV/Bo1DiaFsV8n4NiFEoywg6CGgcMIjTQYqOptOSFjFkn0hgjBaoTblpACOjAsRBhpSWmRIFaIrdqg4xXvOPbETZqPwe0RjJn0LXNNP/95qR8iGOPQvlzb1q7EBFf1B2C3PHX5Hf+pNY3IOcXSiGccxAu802Ihr7WC2ySPCZTFmNAH5+cdF9ecu4T+MelLrao0fmCxCBOH4OzNSVWOLtyE6TpQWp8WSiSUZyXiSELNeNKJmKpCCGAsYWcg8PS/ddkgQesLZg05oAQkIzRRJwfM0sTwc9btf6Mi5xYiR1g7Xq5+DwY9Hx4g401kNtX5Jr4he/2ZzgxCMDC0OHNOubShDt5lvL41ix+qThdQ1asOTE3H2gkkAqq/XbMkk38JPHtxlIHpj0wiY67VKlW1L5ijZ77v4R0fPL/FZ4/Ii834+e6mXQAv+fc3CM2dX5RDuyrpaL0J4ZfJP3AIbsJDwG5wKMyuGbfyqWdVYdFqVfxnebLz8nLlFY1Uhilh1TvIW29ZELVotGLVMhjKg6LQSRbCTscKfRfA0zcGDdyZns3+xLa/Fe3OaMO92hjMDrrNAxMTBWIapWEg3qh6aLdTq1RPTGjEbjY2qtno4Gj1u0HreGmtilWrwG2IPUqZW2uN2ZdTfWKhQ6EI0EH7Ak+8rlXE3Xtfdece5h9oJNq23AwkVe1XPYR2g6El9eFDjFL4Q/fs5bDK3nEpCopadJnS3bNm5JyoD1MLu8wWY3OMmCm794Fc9SC9LHFpJ2Me4ng20OievRwo7x05kigLU0JEGc/a+3KRDLSkkqghQgSuVGsodkuqXuz5uiBAnodueQGI9YOklKT8QiGOYyi1kirUw7WwT2WSpKnN5WbVxfRrFtF+98Xvi08N98Av8EgMTMtd4O/+8R13rYIe8ObDvQU0UfugvBu+YgULP9q/kIqQjhAHSDnVYpDgh/9xqqYfeEh1LGrMaa7hecT1Ac7xAuC0RddduK8xsWPIu5dg470a2nA0wpklDAFdXFWJMiSTZTGifdcTjfuRFPrDIWvBZJ/Rdo2CqLCMxJ0NHGcorVy6NuvaSSTSCG8AnM62xayWfwiDow29JeTmBZXlH4KGbal8DT5VzDX87WWFgr5G0YymykRtRnsX6ZZF8xwWzmWOgPuLV4lIA4U6Ot4OwwucJKhYm9TwQIuzSDaz5MTMvvRahduKIRiz5LCn0ifqUX9p4U6KPgt2LXtKvP45T/dY8YUCq0QCdPCFGad0wd0APmx6NhnZ8jBCZzbwdfv5/f1eq/JDG60aabxTcsxnoOv84mauZU8pqgmpdv7SCfbqQsA9dyKAecXwJFcJeNflU/zsCfghYIsbpVAKrJfwJlp836fK5wy+EGGe0LvQnm1CBsYgCRMguUW+/S2X2Z88cjnPonxn+wWGqQG/FyaI887W1RxzOi3v/YJE2F3nXYK6h3j2cfyfDwpnVdJyD3jtfW/qkXjrBoLheV3/31m3nmbvDOTNE8k12VbwSTAoDGY/9dHsTim+jEIoWrwQwTQyAoCTCBnvuGRI0unla8LVbDaZTKgMHVXQ4AaLUJoliH+STJRFn2TWRO/cnRJj3//msYQd8989P1hN4ErzTgpJWWM+jgNtai4hjRf67sH0ml8yuwH89VWhc+qxjIAqyzi8iFyHA7tEsEm0Bzl6Q0alH0UPejKiCoJgVRPtSmthod/yd00m213F743JbJFy6AFm4BROg5vDZgiSaw/reQuh18t5nRPqze5qY7sQCF8ov6r5fnqnwVlhbL2N5czJJYdgNN6yj74M8nbo7ImY1xpdwgj7a8n8nmJXT0dy881Nae52aeCXRZSinea/rt3HojYbO8/g42ICq3wi92HgD+vNsyCrzn3giS7IsTdeLLRxCnbYy4j2eYQePMfdSqNPyxGVM16h6uGSIMMJeGaMAB0J2x4WIHEnZM8bGMGoCbktRnAFsoVcBnMCynpZ7rzneZ/lGbinqi76wjG28zGMgmJVcLWvRUIb1nguBb4GtUpBqGjQpNEqNYF1plSuU8inDOhzMDxbYVU8xjkoY4p5QX4weGBPA2x+zfpJPgto9j4BVqxE/dNxOVlJypDbq0uZZ/W24+7NF1+8s3LJ0hL88ItHgV9inqmlabd2i30ihJeYrVOa/Q4yHzY91H1b+QKJWos1/fharc2lJuuK9cfJIwXsj8WZDUPudU2wmu7IA8GjB22E4O5CI6y4+SSHp21P+2veJrmuqbrmj/xLBdRYzDM0oo053cwWVM5oRumhbBMPg68GdsMOsv4pI8ge+ckOnPlXl4jMxwt+GzwJW/UR+ucPyf7BCQEIDHOE+Ala/20+L5qfJ9An2KIJbUixu1TCHt5PDH9EdW5pBmvubzB/RHWu9EFfmd8RPVS/97QacCLC3UvoByg5DTGcIXX98bC89i9/IuC9fCoqMCMxL92hpyutZH8R8IV644T2RXaINOEpX90NYGtijEEIQ0/vjuKxWJyleuLJbkOwp85TH4+sokKMUpEpKCHsqrJykgDlIXwS4bz4JAEjVedvqkrAjsU8XMeW0bBASLk8EtMTNkkCsfA8eBc8D46HmzTVpd/vsJ3PdBLj4ek2g6hkLC+pqmHrsHj8mH5HhxBXX9PTLmlwRkT2X6/r84GwyiCseAhV3mhF0zMV9hez9fpk3QgTUIIQcW6Xl211xK1oRJnIrhOElwsOSYCq9Ubnm0fHW2cirzFq22rb0VyEnokr7xe9m0/+qcQlCtKhTrD0RQ9QApPTyDkoCo8BKnShUqeTHokFOzEre3ukFwEU8aJ6GND3VEtcn4XfL7tN7Y3dlpa/pOsIPvucvtbG0G3s5fHhRaK8heUzzTw0g5op1sR9paKrIqsMJI8s1MOLc58IGjYnmw7GyHZkpSbJ9oZxWThUlQa6RQAiqoawdur0eNdNXJlYk0E80lRkEpHeVnmQdi5qjKnzlRiH5MiwES6tK1Ks8Z8O2tTvSI9nC6KCVWr6znLJlJlvQYyJAaMgkBPL0oORJ2XGkHVVBVCqVmfezjO4dUVROkidEQJ4YCzN1WlbecoMbfQRCrVm3MW00pyODOdl1W03mSZV3XFvZ1BGRhmB0xpP9CpoavViNluueePGeKNkyXZgzKE+s76U05XS4JjqpOH8jESin4/JLNVGKuMHAQgGUaEwvbPnpYN6QsVJv9WeFN6cqQpCASnzQxr94fyd0dcoqZAt39bv8aOsXRgywLcrkNJptz8ejpb2isedQSFIP69j7GoivF7hwMY3zEf9W5PFZCKMInroFylYvWrraJV91/CEVG2YnpwEL8bAz+/hZ9sIN16ol4KYFVzwgxveL0FeDbSzpoYwmUukzTVPWITkFxJFRUxYcgsG17Gh8Fu6sM/lFGqccFZti5y9ROFkP2hRsSyZ8NDS35QklwHuc4rMzwTvUz8EsWHCngaI5t6LJK5UOx5OTJsXtZ9Gwb6i6H2vkAIRYsBDusILo1PBWwjJUpCJt3z5Ar3DSAc6h8v/WfpDgO+NeZh3DJ3ceu4VRnkghW1bUW/5BIPofR0G4VqoGrIgWGDGFlpPbBZRbPXps0sOVaCAwgerZ8lD/IFz3pz/wra57gwFBNaupbL549ufAtFAwHUPlZ3TZlXQBqN30Lr0awLgGhTUEqcVGduoh880gdYoXvlT+Yk3+yzxZhcSb+nwSznl6QW4YGhK0qymC0ISQT5OUk7Rp5fzwp6KjHVTEhlaA1hfosFFg4vJzE3hKQzMixJUoRKIJJJmUUvyULSifheNk1ENh3boHmSu65jrIZSWF/wu+DA8Z05kwTuEwjNmCVw+eg8Wt7Lb+EuIBHdgHRkcswh+NPWqZdBma8G7zGB1AkPH9ucmTzYrEwk1FmkSgXQUIhxIsQRsfYzsnUX91oRcjaMZAshsRtocirLvmG9szKbBR4TBEEqCj/9oG8CqZaz59ZO/CBFnoRvqywI6CQuDi4d2Osnf54L9OKR10QWEfjnCZTiDdEaMvnVRf7rxPN9cMWcMffPvvX0sRiJHro4BaSacQ86nkb4EQoCC0D2cL4a6xVeB88Dr4IzkK0DQHF6q13P6cs1aXlts0frLxCrzprHSXFVV9kPuOUnaDf5igEhioR2lvWnrnd0zpkEQUiWE40n0z6r3BYZuj52fukIMUSn2NhJJ8gPoRwudnTKneXq621pWVY3ae5ID/Xy8iKgF4GemUiBHhsfOafrhg119+jr18M5blmuj0UCJMnXOXH0K7nft5+XEKPT67oYy3cLnhipllA+/nVe5mYsBNc6YyQAjn2Vv/7c8WFoPXXCFuc6Awx4QSn6A0PEzfXqzEi5WGU/f080niOXd6vEie2Trh1f8/rIgKv3tHbHawSzZvqbbDU3XSszQzA5brqgeWlN0h2P5k0xl41hBVfUyY9tVw0jHXqDWAa4CiqJBuvAd2VzXJlTlC6pidl1gM1YHdEFpj+rHJTunzhRxjBFjalkbRk3qeVHPMMUKbt1QmW64HvX6OLIS7EFeqKNgDluV3Yy7C6Qxnfm0+zyhZzWZYk1BTemAXn3AizobllTZYrG4pqRKNTHyBFrKTta1B0anTDeCYfgGF0JbckkbQVFFlhG7KdymIDAk0xkUEQTN1ubyjp5Ky1rrWNrCUnWyukVRCEMOccovKhIYhEGMMQVQzSVJDgiIUopMcCpduqiI7riBK6G5wj3JyLKYrSYUPwD6DaJQE97YfaBLuBC4noryu2sS/G2FB8J1Mkfwjvua5fj9daCGdOFA/6tKCHDQyP7xUv0oQdN7On3TRJhPCw+ls+YUxqJCvs2NxJXOMDs18LIgKIyFzX6eWG36XDmHYZAhBp6DL2I+o+47KHOZ3Rc33FNK8bOpGcGJCSM0tPm4n4KCOqT2M4odHWjeuUSMQDJkfzTIervX7bspGjHWYzb4ZLwsRTkDy6ZXZPHxxVyB0ySOxbJ6FwjdOO9oAHefEyqPklvmwYGzCYi2h4GB3yXHIV5V3Av8WXYkqkputxJmTzEaik30BiDFQXJpbcDcpkDsYM4pu1h3+oYUYplKYtw3HwRJc2gW93BuxxIPIcIWNM6MrlW0D6JoFox0EMw92w0woS5YEAdKqb05mu6ptk1F69KMZVVmU2m91dYEvSZ90dSi5Lprrz3CCyJFsWxbIso/g8erCRUhzsdzBatQzMejaHhEBm4/pfVgMXJVxwkwqdlPOqBOS/kiy9o5LNoK89NIRDXDBMzC9bAVtwC2hcbgxQ1K6CIBaVLJ87/obMbE/rpWZtZvxa27u94ICxicF1aME3JcE8WE5ThtxzwYXtoIzihRRU9CWXbtFnhE0XathL6PhYhm0HLNwyyahyg68LnL78xz632+dUEosnQkg/elkcSmVKIA2ZeftakX3p1khTwza5bBruOSUOVakK33JTfwU9tyFzEFoy7JNVGRJbGiYqJXRCJKd+pyEQwJy5ciakg807JphjnuH1zRcXPoSGNGYFBpm0CxCnYzjsK2IIo70QyBtjuPO4G6m8BnwvJJq5IqVJnWmkmlQlWDoqVHSRc7HsR91Z1oiSAoXVSCOO6mlHZDFOYojelnEaBNKg0hSMVcRemiClQooJqZJiNzCbEgtq1vvUJR6FC5sfYqr1tedbcx5B7Br7ffaarqObzAZioXoOotPSMj5Wr5hDZqQ9OWplEe+ikhhb5+QCnO2yJ2+ChY1tVtuzh/cwO91cq8NFkMbZTeajusJ37ndPdKycS3GgJvba/GHrzg3nFQfRRFH0H3XRh9PV6WppwRSpqjcyyzsypOsY3e2qRtvfRAB7j5+ItUy2DnybXc2eWzTpOEF7uPF3KhYFl17/m+uWxRViCUZHc+Vz0nWEZE60f14nJK8+CumsqLD24RCsNzAisFHBM1aAgt+vL2uLcrvhoNs1fK2UT0nuhTvUiuX+c3VIyxBoMxpGip2bG6U6KUbfcQTec0CtU6zSxyXdBsMUyXFA5ixkjlPdF8RNc1zTBEe3cvF4lKEmSR29Tl6oYNY4jhgHbLQz8XSBmpzHR6eUKn7VBAk6uKL6K5CHsXCaX/bDFn4sjU4GKOSkHRGPX0SFu3Vyy67SZEb1srdX9U3NWY8n7tPhWDNcR2pKXrKq7KuxXn4mZzYW645j3axYQ+IvxTGUpr4IyPX5wvNMoDRaI7J7UnyuP+R7HE/O3pa44dScix1EOaRsacG5aPGv56L3lBraWZjKuyJQu5NTG98xh9xCrhF9HotnyXWcg5xUJiMnt7BrzXpq7gp/be8wu1FNUKFq2zTstksoXCLlOoNqDU5Z0Mx+cYls8zwHCgpepgZlR4lKvEYCQa+X4cNARtAZpiMLUKHNJ3OrKcG82QpV1KwoMnenNMml2vHHCNfOE6UncLDLMMJUsR+g60FBrrU8Uqyp3CgB0hSRLEdFlGZrIk4oqjpJj/We92Z7R2ElBiRJbX98v5NMcxdD1t6QrPKrvByqaSlqMgygynHdaI5FzUzTa3GF/qSZnHTFqMcccza8HwsQ49by1SN02eeSdYzAuw4Z4cYLhm7eO9BBwowiRUYC5UiPi8/f1rFKXDor41e7zE6rFfpQY+oXBtcSdwhFJKvOacqsv2W6k74ZH13y6zw0rd3dkbVtTp4bkE6jC5WAWfqkdAkd3iN6kNF0wh53W4QpvUjnjcqR7TL10pg02j+XAPqV0fMpBqk+shAq3K9ZqDtY0thp8I6DXO7g66tVa0MBenLUpCscNdEW2ETSKwadc44UCd10lxh23l6vYY7c7cJh9L7xgwzQuqohnrBJKyZfA0D656oTDObHaorCsZ5zXhwIt3Vgcc+ELw3cCL4GT9C0FQW/xzR1TPe3G3d6FCzJSfmO8wqwglfK11rY7QZ+CjS4HwbcEIpwqkx2jmT5TXd+DJ43UqxGnl5W2PnovPTMgwFTSYhfFIs425/kS888184Eq/8pRIzHYsnrrXOxpO+jm8uKL5WcI2Y0+LwuLLVzIh/5bycLdBqxUmdWR4T9r+mJ4vcP+yvkyMw9biVcWw1WkCxUyf/P5eSnBhqXeX2zix4Y13kCVJstqkw6C7NQqaBg7EET8SE7cf06HtIW9g0z4T9ztDZu9Exl2U1uC/J6zvpDybtjOMuSvasvmuUQLtDOYbWKfzQKEaisq2bjyyyaoSG9B9ib2iU2hFSOSGlYRCgWHeh0LUUqKrkIJdqeo1RFIPqaOSWfe/LBrUZ3nbNgyayhBoL8UfSMsQhCBAWcG1aiiw+HJgYx0dQLA64H72oQGklyVOo/fpE2NqIYyMUn1gpVyred590T7fq1m26d67BeszhtarabzraYtZxDoo7OkszLgCDxWb2OHIIJQkUTydIzlNkGh8PeUaacLcjT/330ODmEpIDFB9tsMUhVQaDRMrJeyUy13BETDwQVPwrSDn5WZG1bXIhVHYBWUsg3JD6gn3sKnfSvXdDmNvr9Fsh/qCzq+ZQETg1caqFA7Lteldbr+OKHfkvtyVibaZj/sF8EGlpQMzBkgGqgNeeZz+5PS+Btb01Cr4rzDXEEaPV5C61lwX3pAelOk6FiBbUdgcqEhinskr2opq2gRYBUIt2cdtBSX3pJDkPuNZXiRWU0WhA4O4k2Lq0bZvGQiEmtlDzUH1jh9bs1HrcgdUm14pj9poDIImVTN0M5dWYCqNR3naCAeWKsEQ9y1jcKlB+sulpAiSy7cBC3kWfLYHwx+elbxMxXs15JGKkImABFOhpOYAE/DP6yUwtz66hr8sSs1nfQ6T7BHoBUgfStfACYDKbz8ZlLL0GwVQ6TUh1yJWCBd/8iHgLMv7pRFqwAwgAa3jacDAzvk+PTQ0FSNcFucK4A5hyLlgdIX9Ya7rpsQguApqtdXB4cNteKFMYJ4DnylGuoElZ9PqK2KL3L1MFkPTPawvJdPAwsArtpfZOp1wpKiFvvqU3WrkV8+EspLJXNzpdZpCnHKcVtvMAiGfnSiwppDKEIAWY8joUKxzLf50S7PVGc53AXfsCOcx/C6v4dU9GDTIQAV0iEDNGnmQ1HJHmvoA2rgEts/yf99cjMkKmZAuX+nFoB3Nw5HfS56VoC5ojEvAZQD0mo/De+eVP+64A3eXrAmfal8Y7+NjNvXTR9zZZStCRTNNkrqGZdaBTdyIpjEMQksYgvJ5Wy/0LAuZuKbVBJMWHZmM5xkGtjHj6em1fDDNXeJJ37V9JGMETR+HCyCsdqFucGL3ZrAAFmW+/2M4e1Uo7SB/DkGwr/91ORgccO3Ufp82Ej1Gh5Mi1Sod2dUqlllP3+bArYCH7/O8bnLW8BohlEMFFqeqzRUkFpGYCxufRywidpOEox3DWuqehz7u952CHA4moXAQwZ5lxIqTCaaYqbzyWMQ+MrPGwgJeidC1XD+QQAMUqCs/lUQIozdXWBXnIcEKkuDlGOBrpMiDbwrL7XPl61ZkNbWhqHsPmyBEqWIThQ6sqnVZw8mIhS2CmMNNi1hYlImq+Ryvt2vhx3g7WMrluCR3I9mko5NMCt7VmIWP0/Y1W9giNei0/GbkIVFGsImIAqV1SWGTcXrUw4/xFn9no1E37WUz6UiDFk5FsuY3QENBEBXWnGJjMow5HUKHEa20o5g9A88P4oFiURZp0rgMotUjKMfxvLTtxEmU7Z3G1WoJ0Drshiq0o1HmWdm37aRoSyj3b9y0KmcCxZWivqHe4qxBDy6JkaH+RAsOGKhR8iXpwsrhOJpjHT+RU9oB4aCNbJEwOIkvZEV82LvRwAQDD1uU6mXmusntIZG9gsHmAVfR6/d2XYyUh9qc3kI16507KtFwQtj3zMpmyo7DVITeCEHGBD2ipSQyJS09apEhCP4FCmoMfs7DkZgxLr8nD/2oE/ioCFvhOEzDZrA1SK6ZzwkF4YjwgGNSyMNOvKstXFnqLHafaHqjIL6fVp5mkBWpqWSWa+NDZ8gl5Fwuk1ZoxV+8phTzWmxiCD6vha8s0jUc1u9TCS252yVVotVJCkIuqBFeTMzkIqmkpKSgNLoqdzECa/PqG0Tti0KuQOi5QmhnCWlVufMQpwBrYnF3K5p4LmY2asR4crEzp3ZMoBOWnlpONIF3qqDRrna8KKKxeduLY2pEN1QJ4ctOhZ1EE8a1CrreorHY+9iyVeym6XgqnylXM7WcUTi8kG41FCF5260LSgrNV7Uw1R9ENTU5WfAtSVm1XxfaqkBTBF4N8P4aqSV88zpdvp6s8kRVTUgrfjjwY+5F01ZIpub8IHBXXoKOPXlAx7/napJw9kiDhw+AXwkcht1TB2nQw+BxWe/CTCYZ73yiNAEvU8gT9coYztwdAg2tpGdIfCziKuXx7IJ8F0zXJb2zDkO9xHEka7GkJ4L9isuZ66M7vV4BT+f1zgSYPOolEMbmZolu205Vu5y5PrrT6Ul4OrN3JuDyqA/mOJzbuzDaaM4v2m50Zu+MwVhw6KBHPVqwby9zSERe7uhKRfGcZGSyRKLTu4SjC8ayWKON/CSzzYwj8opHGY2Gla1QNMrGaTKRVzo660mj+TryzrN97NGD6pkMsCxnstgOnshHOroQhuGN7ksaCv984uJeZ+w5/3RmwmSxHTKRD3t0IZ4nG+Vb6Lrx8kdXZhiayWI8pshHPLrgbhZzJEqN3wNmIQ+zIVeOxGi2aSRQzD+mVKrDXG2zd8LwRpIk3pso0AnaO7aUjSm2vT+y/HOQtaqPZkQ0cAi5MtLB8aKRgBaqqlfwTUwVysPeNHo+ycRA8BCfiBUkXf5KKVuRj1OWbp2uU4/bZlf5oCFeL8vuMcm3Xd4H2YakcC9MEhJEV0C6uUGg6cFagA3i9JrNCQ++hY1OtvyCU2eeJps1T2tNbOO4/ADnlBI/dJoWIRCD63I9DVfWjiORAE7Ay+EEHAAnE/lHS0yck30JbvfnnibUwyci/UaT+lVyigO7xENxaJis4S86xVMDwKdsunB2RiYxcVMq5QWFePtOnTZmdzZUUrovk8VabTSnyyGoYX62x8+cA8H04tUsCLeB7epNQoYee2WgKZn85VvcwVGwYJg8mr/4H7hz4OmZDOm8I8RhxFd1SeGujxlLPRle7rJ8s9lmg0bWQ6IdgvtyA+MAWhG5zEYgEM8cqlwVH7hCuqlRT6F4ed3pZtSuKaZRrWKqVtXytrjDX2womfz2KBPOAjplu0ybF4VlSdFMlNSnPadO7yiRqCJlGLd9ByMI6yIvVRrvJ3t5SvRBP3hKvFuVPkKNXZq4rRbW/ZbfNcQDQ88ONANDEf0YBuEbco0F4qMFJ5NUKpLCzkVXRuJQlaXkBV5SdcM0s6WqVDOlXl4vaeN5QxJPu1ZCUesKGUnGISFQGBsVTyoUK2nUVDrNUcCa21MnNqVURW5UgcgAH1yR0Jet6O5Os597JTHHKMxTJjRLrfnkBnloQQHmQJVGbsf3dDK/yiZnIr7nCGI4WDNNcB+HFu76ni4R/FD91rDccKSn2Mc2v0qwP4TZXapxdG1pobzeV8/f5WfBuwxa0KdOx9OQY5i82TPyeB5cm9fLv2ZPsJK/u+56N3wOH7S7x0XQKJ2Fqira4+CcSjPw1o8akSSLTovPVbki9+o8iXU77oS2XYoTwZ1Aj1NFq6mXtkW8JDXrCg5ri1vCa+df0ZRj0z12UhY6/mo/MoewPzM/LleQNfo7uW8J5Afer12FtomjVXPtZN+kS8iebIJHghlycbbMEYO3M/3HG+rrOwfEMAi/37vyhhctsBHVee7wD6ihoxwc4k7Gx8ADlZ/lbKJvA5pfS9J4KQG8bmvgjVYfzZmdSGDVscsDjdcrKj6YEcIzTDJnge7nUqpc3Phjr9sULaJUUWHMFEWOWbfawiGqCTusDOj2nMXzklYU0mktlQlK2DgahhHFkys/pHTDawazdjIvw0nWpmUIMUA4DQ9r2GQmbS1WeFGpBYRkBISsACSMaDBR1pQ7AzZEQX3IBHIE/rUeyt9QF+Ab6mzm3vlmqWhiSABQazV3nMsFr3j35P9XDmAYKANFQC3j9QheRxS2+XiCeS/4qSQChUGeB6bDa+f8I0HnQfWfgl8/9VL7wl1KoQoxIpoGLGIbFDP8NZVSbTuYPHPUIB54as+dHgb/+TcbgxDSlOtDjSVQxa7KGzJXTDHKG+c22E99/saKL3zbXhm5egJEJZIbHKwdAb9VFWQKO/jpzMgfy4fUbvvmhM/k3hOA5qo/9J0OgVbJ7ppd1mMtuglWbqEbFiaQyGWOGXGVEjdYPM80JJ3qSfEyUzXhFYZB8HKip0EdAWG97ri6QoMBeG+sOD18SlwPHNJ0m558h25dn6y3n6PU7qrguApQRZtE7hKSMwdRSIIGLYCymZti1rc0SvMCmj1dbHXP5QJfB97c35nehtwhBEEC2qLfaEOevU8tgH+/ZH9VU/szwhjqSIeYOIMKeMT2YoYhSbDKgmEmlRHCDiW0W5V2nuFO3fM0VoEtvBrpbLvnOc5OXh4KQiwF7PP1Q/0TxDXh/hiWfY++VDRm0DLxEwnmfR/TdI3MAkZYEmZu0hzqIm7EuPhVEhd7464z3bOqL60u3MIJ/0iYvUfh+7XpSLBCmOYKfq0Y6jMdaNCi9tWXFjC4b2mHx8a0+XxhLkMwnw8/OIWipK7LpapKgMUXIMlRhqQ2+wDXJ006qhi6c11mZuOjgeUEupKURv2Fj0CYC0jDksdVr/6e4dL5iSuBpHBzc6g1s3pF84iDoCFCAk2O1xcCNcaSksEBE83927vQJnYR+aHjpyOGzggQZJi2RFob32sEApHeYb21c7iAODzNRJLAJZTl+e9N3RGL771MLkkh8cJfAyqza8mkb75+MGHS6cC5bXy8YuohVyhXkqGoXPaV8I+60Id861XnF5hsR4C1+G39DxiDL+AnuD1qXguSAPghnKGN1mzqhnY0jzfVw5KZlLZADraggyeWYAY2h3ao991q51Ng6h/n38QFUyAUX6LxEI/2nqgNTi/FBFtnYU052aIES+F9J8wJULWbarsD3He5VVzQ2FdnEjCauxrAsmtvAs9thZpTbA0DvqXdJl/pgpItcIyYLnuMZmI1BXKSQBWY0YZMx53/EvEV3FJ9/DaolfF+eW2x2dMGB0mZpUxAyEMsRKmhHfhLFcq0ZDBUlmJjEowaKzU2sFEMctcDc9WN/NPKltQFvpBZD2Hcz+e3rNw16a8QjS4CtSoQGhWg5ZjaLI3l3PqkicD0nUMTlBQrmbgTW3ANXS2xgqDphom5aeiKXFYIzRBjN+QgZSFZJ7EqYZBLRyDDagHKnL5B6o+pmEGQB6a7gyUJlzALyn2Zblb/eIpkTwnozchAoayM5XM4sqpr6FqhqIRki2/xeIrrKyBlHeVc8ewBjkVQb53XDG6n/6TMAR0am2yiQqtu6nDKRmmDuZAULo4KaRrjuQhvBnr6XkIUo9m/O1dJSRs9qjKsECyVZL8rxdCgaSnHDVya5wte+Hhhi23EcLV5tWE8fi9zyTZLIULnRIVlkgSDID40rIK31cMPLzMtD7spjehXKYPZVja6XLG4o1NzScicMAvD0SvRVT32R+fMauxHRiY6z/IyHkBVXUj7TcYwAC20tHEljbriDgni+jsMU9+3tLRczOWdGX9vgjy+uXvWq/22AgwdSL7s8RnGbCdt+7k6tMGM0NKa7al01omRqhz9bC0bGibpcvbGuIpR51V1zPironkPIKcxlR8drVbLuVSoQWExMeuqHLNBMUaza0ohMixx5NOfEBxHUSFJInnNKO88IiVLeH7/e/TccIOm9tcPqyn7XMfAbHReL6SdLVe+Vr/l6jvrpvrQyR/tM+3rGIri4obNaiNUKg1hksa9BCLCIC7j4ggll910mr/AmWy1/hGghFLw0g+s71oXquh4hhxkMBDjcRDv6660DdijFkFK5AuZQW2wegX2/nVrf7C2LKw+jbC0PX8yDjHS3jnT2GVr8Tr97drVvb0zCYhv1XZGCoO3ZzdpHeQVCuWObbRpbHEwcpczWU2gPh1+Q0M/69kfzHLWWK2h6Eq6tvahGc/h2s9KLF56iIns7XIcs+lnMPU9ZdTUHA47ZKLwG47j823ZzbFJH7xzR/uqZi0HDgL27dvotpnx8Rg/TFEVIDoXWnEJRGL5NsPIfWs82waZjJUk1iEZV+6G3GauhcBSZYJ1VVN26OdwTSR6MChbo/NtqiyGxOjOhfuqZy7rIAFBZSHZn8bx8bqlpJKwmaxTqdl5TXdbc6bnucxFmaYwbMhRBQI+dMWH7JtXOVzVsHUkCCjtWko1TTUWZnRZ3Z5oFV23VHByroTLzcxWYv+3Nr0WGDGWgQbv6zjsOYMlQq/7igLJExvkrHJZD7rSMbce1zvG5ZGFI4ZcDBaUIAWTIJXKR3rrmavXncNRNDOUVykALtjt55gg6Wrz3iWG2Q/Yb2/iBOFL717+PyhQuSZOpThDVBX7FXzVilc+9rD2eq/n5i/g4q635Nxy3h3JzHfZq/iVSgmfCQ6odZMdqluTCzby9UYIfZDLnf2+8uVjAeOBIfBZoRlx5AIu2Egb9CqtuW58BfeLcZguWpMcGoSQfSsa+hwBxueVOOAVsAd2gEPqzPvz3VpwmeRYMLwkMcomkrbdTIX966RJqdAaLMYOlZo9o4DaE/BLHPjvQuhfEndu4VVBS+tm7FBf9pSRtE5rwAR25VTLg4HC92GADG/jrOuYjcHDewtxoCiN2XiVvMp/XVw9fEknXtzSX8VaB3rbjeJGBsa21g/ysywUQAd3htT17DOpaOZbgZSAQDgS5sNlRpE005JI924s9qAsyEqUIGwZhZLy0IRQfP67kS9etZ9MdMIwpNcojsGjwtgWt9VKzqUiN0Siuu7BZeK0otr0eoRIGJKwtPvuTHmX+oG8zdgwooyGeIome/4cuBojVxffGqBGHNJb7eCoMDYvZ7PmSqqSRK1Fuj5wWq+6XXO4NQlbHYHdaWWb7Ubkx9eMkK6xFalY3N+KAi0fv/ClfD5xa4dmILBhtloHk8pTph5EqKsQZBL5cY8ePm3tCRKt/OGxDGsrdNERSg1y4aX9HyI9uJHQnxYsUOKxh9W074paYC3U8RvGkuxRXKdT1zUeM3Y2CW3FZqEz77fMhAbIVjf4dG+9VDsSygFqYdxGeri/t7nkivyxtvV7saZFb5FK28zj93b2JBRN4LJjPZOhwz1l1OIjoWcb5eK0lYFADdb2ups23Kmx6MOLM4eC5Z0zPdALe5v5sMMwRZRFTuKtac/wQSFCVrp3cfzoCsUgEXjqtB0HH70wrnli3pFJrMAu3121Kg7unJWYH7PfGVQlx/FUg06loCs06Og2l7C0eQM691a2dBOBotWwbtskK1orSdj1nEu2Q0OjQYhakQ3siXoRhBiZtWAiTEhgO8CzIVhiWz651qO5lZGScRnptqHEkljjAClVzFRVa9S2UESm3HYWBod60LAf4IYGD9lQrzkuQn47srHGSi47Di6sRRbBCNWaAZtRINlfNRfYQI2llyvmpajV3I6pGAdoLX1PhkJIKcNCxpIhv5UnAaPuSx40qR9NRSBjJMtqUz1pGIzORt0o6ISVta8Kw1rQLK1ouRIVVpAyhV046Xnu9BCn9lEatlDAYoNm5hmgIXXJ7IwWUpGeCSNRf7yoOin8GINawSeYWNV74s/lsRPIAXwhcPvukJ0J/Vsn8FTwonqExnffGq1TuaM7OgzCviQHwccs5YLs6GyTRO0m8M3g14A3wP+ScyCYWsWNKgh6rlp1RYHfjJFQENZIcsTtM+T0eHZsWyrd4ef1k0fVR+0r/Hxhp4R2H49OqX6kdBwD9whLh+MsbA2ksPwMBQsHcEOGxYdlErQHmuA9xwbj/Gx/n0C9EcOjcv9j93KteEQl65ZNY8vZVM78ABbWrUZeIepa8OCeBochnNb7UA4vInAxQweG9+XbLsjZe3jx6GYSDHGIu5EMzQ/QrHwpzJcNuEydsH0ws1m6zja6KoRMqMjRmayOxTLpdQLLxlZ4xZ4iqR/JPZVDTyKzXVpyltWL8lqLkGEvXFpsc/V7yr2n2VWCAW3shJoDR9BVzTT3ij2Q0vms9SwrSs/FcAKcF0t3SuJmDBfjlxoQ5uqCcCKmwCYSO89VL2BWHcB460Yvhnmm9DS7jPYcs4TMBkpXczAb+KXzkp2boMNWw05DEk87d7hJCjOPUd7dBDQh1QuQi7MKmt0ZsmxvlxL+i1u24xSoRalBI9OknUV1uzs/oMjzXo8ePek41KXWtXvHFXyfEnsc106AfwhLTwoQDF+CBFIWIGHi5AgII3LX1LjQxhW6uQlSILiuUcaxRK6xHKO0S8Xwold8Lzyva+TD0nq70Gb9S9z5WX9QshKa/cV2a7TiUAhcYZKRbDcocqC8sEGo1eqBcLC3WVizSxxQC03C1gzbl7ZmG74Ye9nyECe6vsaQeKIJl8woob5xqhAwQ+WCLhW8vXp6SavS6NkUOrAS7I+RQlEy87YVi/4AXuzIpuf7N1UVNqki5ComNqcnZ8sUGuyWdg5lcu190QvtDxFwSjbSDfYlmR4qGjsmZ9m8iqVOtI1NOJRMw8wdIUUHFguidFQxFb10pszJKF4pPLgVDWlnQcpgj4TfbupKVVbLdqoCNFWVNH0LqbqJTdvFuo/IIthEvnijATyqRrrlMppjaFFSSSGqSCw5hLYOGUikLa3MbSSAuzYB61kLKeBopU6DVuSX7bR6I8v3UV01iU8tbBkNrEHxTDTOk6y5bjNrlckW19jo0710xYaorHmUcyqVUYxSYTUXt5GGsSSNgF+zfGJNp+GgR7rRwGldsKbah62xZtXzVF3cjoI4anUYTDnOdzKIY/i5mT2maxPXGhbpQ8TdEy6XvzTecADeuEBLMjYUuMZqsZAkDO95xW9fjeNkh7zfAEGSFUjoVVryvp57QeZoN5Mb298vLX0402kiWOVe9iUtNzQv31L/KIZjlQnIy+4PyUAFpnHfTOE9wvLVwUxYG2HciXR+m97BJKzHC3mhK6iV+m1rPBofIEj6NVBPwZ12s817TY1RR+dRhWyltUqam1O5eYBOxkxFQvZ1XZGJiy8XZuZYoE7t+ysrBLqpoGW+2keY7Km2s6M9U6M5iiMY4CKHepMsUZqQcrPNYzIaBka3+6r1PpyalFNNjefmOj9b3n9N+f+KYw16KMlMFBlqMj/Q1z7omuIvzM2G7rP5Y5Rlvbdaf3kSODylY2hosiPrBC0DS/6fKxdYY7qF16Q6JwxiJQIEWTezdwQk50cxxzELp75hyx/zfu9FsCiEmf8w37TNA0Dr+6YBRnpzW3x0zjhMgdLr9GjlZxS4YXsxxjr1k+v8YEb9Qj9lSqFcRZ6UPwYi4uSCSx8OwaxqH4qweagn4P5HXS+fORcAIF0HJR6JIaxaxMd1PLBj8Kq86fs5+bg+puN6XBGuwe0fFz2u8kMJ3Aff5fpJ3+1vX/ZWFxJbtyaSF8ATQpMqffCjPrV7noXGbS1lcm3rKcAlo6fQwbBNKbFoq8KBdd3tPi7ZBtoolQy/lOZY/zf2lf6jLhn9kct6SIUnrCBVTel7VW8FBRqEa2EOzsChcNYrzgwuBY0ddZ2x7qRbUGlmWsVzhLRd3Ptvj9GUu5A2bgWr8o/rJ4FfWfKrwYrf26Ei58CcvhVyHIMJ7qTIrNo4HCymx4/+zRLfnD4iNO1i6cjIPN5TB4XjPVHWvd3Ruf1FFi3Q2IuctDsOSl3oIWTqFERp2LBCs7fSw0TnBqcpZAyD8+9XNB6lWJwoD0n6sZk3alezePvfhpRVXq93fq7WQKwiLgl69ZmCb725HbOQLxenbemWeMl0w7gZzLbK6WnkpvRUBszXrHwmKemTIBUnCVeV1X62OdIt1GYSJ7m0c7p2QW/ZJheQ43TT1MWkeb3cIBLCbtqde70wbB7qHEpnQjXgh4SmfFDIbRJZxjy3HeDIoDXiGJvQnwVdSqL/trHIWKvD5HYMSsw3ZqgrvKbPUrExKvv22vtX+jDLTeUkcgaLivfV8uZsP61LJBXcLHNp+lkIjNeP4VeaSaI21088az+OlD6C33Wgx2MbHY33oR9PiMR0QtNWu+p33aBDZQQpvW7wGcTOIQR6O89OnHFvcWB2MU1aS00ofHF9lrz0SjhIhvUALpv0hJK7e6zFGltOnSqvWyXx4PXWKFvhRPWjKjxPPWi0lUnfYV/35zXyr3FT753cNc/iaD0zZXa3rae2i545QYJNWM6GqZ/DeceIgyU3g58paKRj27JXB8FXVBik7hYch/BSARi5OK/47fYw2l4+QiHtcMzb0qaoJ/cKEpNOXNEirHg7vpRexT0nSVEf5C/t3adwvmRPBgNibM2didbHmo00Q1XvohQYayBfWAeKuJswUvGBfJkxkJM/MeBdl07hpKwHb24xcNTBmJhgTvGUNuplipX6idYrRSZ1XyU87kelNUOWk/pp3GAaJPvdpxiakloOcXww6lm+76BNHNClAz1eVgiMju/NRsvv1PTLst7WQZ9Mp3Ip/+jY7qJ1cnK8UsxKLFbGJ3uFElezPR2P5AVelVegqSPHtntOtmh8z1zwaTjBCntsXh6EiDK/SIE4nJKWeL+KGFqELEBclZNJHa2jst3yv3ebExsXsek22KYTV93ZnJmPY6n7jXlW8EsL/EPLUgoh8M+bMjvuJpn3AgE6SJgCJvDLZcVI8y/AvulpK+/D3i8BviVM4gEAF23BFsCXavERn4FZQBggBApbC/6s/i4HwAX+/qfUJE7Eawe1KTw0EhL8GrAfp3Aj4Ee/8AK1mnKwLVKgGYOA6GQprlg5B88Z19AjAkiggAADZAnz8Bfx/hrkAB+nQGUR4xNUQuLAMUguIm3AryiCGT+Vh+0Is7C9Hw3YpJMkcqjMtN8h1rA7LvZlfptHjtyEXOP1P6jV/2usHt914qtez5eWXe0D8VrwIsmnVXFLNVAORmiVrVuojP/DF0E21jQjSPaicc80ncrKWp4tmoz8VCBIC/iENQXMsfacdY+OJuJnvaaBpMM0DM6lwa0K60BXffthZyCBnPnm6qujFeL30FHB1WUQ5gZg2bIpmJ4UOlXjdwBTbyTuBTmeis54Kn5+a3HE61PSiKqDwk4Ft6+1f95HR4LenFVEQ3TKCKIq7XHhzAOkkYpV2w7BAjM3tb8/m7xG20QJ2j1QvuI8bL+NzYAbPdqryf8YLWetZbc8dSg3oWp4Vo13FtscKwyWivqCtjCZ8an/g1x+J02b26yuhx/NMmdEf4z5dHiUDvh0+j/4p9IHvOuetyGSLGaBeWAqEtH5PsuXv6kEMrFsnif2YJxeQ71vw+8l1IvljZTq90hJjikGnGQgsfCkWNUME6Q4WNdtBkxLZKfuf1AgjpkIO5HJPDhqJ8EKLbPfOz/OMmfEXpgikA3S35GeP8kDcPNn42eZ7jyVKMFZl2/SJ3zId035Ufb7AItB+7wYZ+ZuUXbtle1+WcHCAmrwQhivHrsCp8QfT/rUHI2tyqNtKRYuHAQTQ/pNYy9atiZxMLTrWmhOfWuS+pKZ8KX8VD2jUpGC/jSh9UBhZn2QFwO+YdynmIvOmr7tNKaTL1DsvcLIWqdwFZ2Byf7+uU66O3p8lh3Z7HlHQQ9epyxo0Xnkt0AothTJldmaLNL95RxnJEFtPjnQYPo/FhCRJCjye+ZZ51zGdF6inftzz+uHV8L5TAd2BEFHt9Jdw4yMydfIZTwCsqScZT4BNSHBjnaHvrl8RCe+/Pb9xI8P9EiQReBh4DOgpzMpvFoPXj9VoiwowubGnv917zJP7h6oXjUoNy8+yxJ/ArVuvE2/kX3NmKxYigEANn4N08n9As5nGOQtxx+iUERUsdh8MrIdP1RmdKux2ibPgNH3CB+VlzmeC70kBWS1PaQMbC0JafvGmXhNTiFnaTX/k/jZeJlyw1n19gTGdDzL3YC2Tka9u2Jz4aJU4mh24hR6pYRrBFt78UO07+Kx92H00lN0E+ytT2hu+8eio78yXxobx3E0V6VMpflnHbkWz+Pn6JTto0ii6cBrWUEGIgQKkgwJldcgtuO0vIGb1Ex8TLnhVdLcFnmNbtl1qHxerde2uddzBA6WmEHCseurOZJvYc48jCuYAYijIGO8GbCZot5su2hMfhYGweSLNFMCS8dHN6NCtINwCNTwcDLswvNcChLA9PwBAZK2d98ZgBAc7EwYAfYlAu2uJxJ5nyIKC/zY0HPBxwZRgAhpYiApSGQiISBpKkkDkXqQDKBoc6tGpw4kC2jakeSAQItJLSDpeFIHEO0h9YCiC4wG0KckDyz6lzQChXJJE2BpGWkGKkmkBRhsJa1A4gEkAiZPJxOAwrvIRID4fTIJQP6ZTAYCnyNTgMReMhWQ/KMxDbQUpqNlMAOtg5moosyaLxjakENePe+YED5YwRGhSVCoVmlmTZuhs3Cyn3RZ5B6g5HaS6VdDC6P/3KWY660ppBu3SG+5rUJTKVxeyfR8p2hiPTM/ow5IBnUZrmvPSWVevkE7V0/gW7JYeYFbMUtZ7m/hZmUfyVCvxVXTdahpqUMZFHZ39VHDnTVlYsGyKJdGz32w7CxZ0oqqmUhxEjMnbo+S4tFdJMn2IPW0YCnEOcGFAq+YEYlNmUe7K9TF9assen1FldDwIioU+h54SPuKy1nhraxUJOhVWi2snokUhIVef7+J3DdcKNvt4BbSirzIF/Dv7mRHbn1fdhYWeK7M5SmkSVG2F+VRYQ8ZAAAA);}\n"
  },
  {
    "path": "static/isso.css",
    "content": "/* ========================================================================== */\n/* Generic styling                                                            */\n/* ========================================================================== */\n#isso-thread * {\n    /* Reset */\n    -webkit-box-sizing: border-box;\n    -moz-box-sizing: border-box;\n    box-sizing: border-box;\n}\n\n/* ========================================================================== */\n/* Thread heading area                                                        */\n/* ========================================================================== */\n#isso-thread {\n    margin: 0 auto;\n    padding: 0;\n    width: 100%;\n    color: var(--text-color);\n    font-size: 0.9em;\n    font-family: var(--sans-serif-font);\n}\n\nh4.isso-thread-heading {\n    padding-bottom: 0.2em;\n    color: var(--text-color);\n    font-size: 1.2rem;\n}\n\n.isso-feedlink {\n    float: inline-end;\n    padding-inline-start: 1em;\n}\n\n.isso-feedlink a {\n    vertical-align: bottom;\n    font-size: 0.8em;\n}\n\n/* ========================================================================== */\n/* Comments                                                                   */\n/* ========================================================================== */\n\n.isso-comment {\n    margin: 0 auto;\n    max-width: 68em;\n}\n\n.isso-preview .isso-comment {\n    margin: 0;\n    padding-top: 0;\n}\n\n.isso-comment:not(:first-of-type),\n.isso-follow-up .isso-comment {\n    margin-block-end: 0.5em;\n    border-top: 1px solid var(--divider-color);\n}\n\n.isso-avatar {\n    display: block;\n    float: inline-start;\n    margin: 0.95em 0.95em 0;\n}\n\n.isso-avatar svg {\n    border: 1px solid var(--divider-color);\n    border-radius: 3px;\n    width: 100%;\n    max-width: 48px;\n    height: 100%;\n    max-height: 48px;\n}\n\n.isso-text-wrapper {\n    display: block;\n    padding: 0.3em;\n}\n\n.isso-follow-up {\n    padding-inline-start: calc(7% + 20px);\n}\n\n.isso-comment-footer {\n    font-size: 0.95em;\n}\n\n.isso-comment-header {\n    font-size: 0.85em;\n}\n\n.isso-comment-header a {\n    text-decoration: none;\n}\n\n/* Only for comment header, spacer between up-/downvote should have no padding */\n.isso-comment-header .isso-spacer {\n    padding-inline: 6px;\n}\n\n.isso-spacer,\n.isso-permalink,\n.isso-note,\n.isso-parent {\n    color: var(--meta-color);\n    font-weight: normal;\n    text-shadow: none;\n}\n\n.isso-permalink:hover,\n.isso-note:hover,\n.isso-parent:hover {\n    color: var(--hover-color);\n}\n\n.isso-note {\n    float: inline-end;\n}\n\n.isso-author {\n    color: var(--text-color);\n    font-weight: 500;\n}\n\n.isso-page-author-suffix {\n    color: var(--text-color-high-contrast);\n    font-weight: bold;\n}\n\n/* Style author comments and replies */\n.isso-is-page-author>.isso-text-wrapper {\n    background-color: var(--bg-1);\n}\n\n.isso-textarea,\n.isso-preview {\n    background-color: var(--bg-2);\n    padding: 10px;\n    width: 100%;\n    color: var(--text-color);\n    font-size: 0.8em;\n    font-family: var(--sans-serif-font);\n}\n\n.isso-text p {\n    margin-top: -0.4em;\n}\n\n.isso-text p:last-child {\n    margin-block-end: 0.2em;\n}\n\n.isso-text h1,\n.isso-text h2,\n.isso-text h3,\n.isso-text h4,\n.isso-text h5,\n.isso-text h6 {\n    font-weight: bold;\n    font-size: 130%;\n}\n\n.isso-comment-footer {\n    clear: left;\n    color: var(--meta-color);\n    font-size: 0.80em;\n}\n\n.isso-feedlink,\n.isso-comment-footer a {\n    margin: 0.4em;\n    padding: 0.1em;\n    font-weight: bold;\n    text-decoration: none;\n}\n\n.isso-comment-footer .isso-votes {\n    color: var(--meta-color);\n}\n\n.isso-upvote svg,\n.isso-downvote svg {\n    position: relative;\n    top: .2em;\n}\n\n.isso-upvote:hover svg,\n.isso-downvote:hover svg {\n    fill: var(--hover-color);\n}\n\n/* Reply postbox under existing comment */\n.isso-comment .isso-postbox {\n    margin-top: 0.8em;\n}\n\n.isso-comment.isso-no-votes>*>.isso-comment-footer .isso-votes {\n    display: none;\n}\n\n/* ========================================================================== */\n/* Postbox                                                                    */\n/* ========================================================================== */\n.isso-postbox {\n    clear: right;\n    margin: 0 auto 2em;\n}\n\n.isso-form-wrapper {\n    display: flex;\n    flex-direction: column;\n}\n\n.isso-textarea,\n.isso-preview {\n    margin-top: 0.2em;\n    border: 1px solid var(--divider-color);\n    border-radius: 5px;\n    width: 100%;\n}\n\n.isso-textarea {\n    outline: 0;\n    width: 100%;\n    resize: none;\n}\n\n.isso-form-wrapper input[type=checkbox] {\n    position: relative;\n    bottom: 1px;\n    vertical-align: middle;\n    margin-inline-end: 0;\n}\n\n.isso-notification-section {\n    display: none;\n    padding-top: .3em;\n    padding-bottom: 10px;\n    font-size: 0.90em;\n}\n\n.isso-auth-section {\n    display: flex;\n    flex-direction: row;\n}\n\n.isso-input-wrapper,\n.isso-post-action {\n    display: flex;\n    flex-direction: column;\n    justify-content: flex-end;\n    align-items: center;\n    margin: 0 auto;\n    max-width: 35%;\n    font-size: 0.8em;\n    font-family: var(--sans-serif-font);\n    text-align: center;\n}\n\n.isso-input-wrapper {\n    margin-inline-end: 0.5em;\n}\n\n.isso-input-wrapper input,\n.isso-post-action input {\n    margin-top: auto;\n}\n\n.isso-input-wrapper label {\n    display: inline-block;\n    margin-top: auto;\n    height: auto;\n    line-height: 1.4em;\n}\n\n.isso-input-wrapper input {\n    border: 1px solid var(--divider-color);\n    border-radius: 5px;\n    background-color: var(--bg-2);\n    padding: 0.3em;\n    width: 100%;\n    color: var(--text-color);\n    line-height: 1.2em;\n    font-family: var(--sans-serif-font);\n}\n\n.isso-post-action input {\n    cursor: pointer;\n    margin: 0.1em;\n    border: none;\n    border-radius: 5px;\n    background-color: var(--primary-color);\n    padding-inline: 1em;\n    padding-block: 0.6em;\n    color: var(--background-color);\n    font-size: 0.8rem;\n}\n\n.isso-post-action {\n    display: block;\n    align-self: flex-end;\n    margin: 0 auto;\n}\n\n.isso-post-action>input:hover {\n    opacity: 0.8;\n}\n\n/* ========================================================================== */\n/* Postbox (preview mode)                                                     */\n/* ========================================================================== */\n.isso-preview,\n.isso-post-action input[name=\"edit\"],\n.isso-postbox.isso-preview-mode>.isso-form-wrapper input[name=\"preview\"],\n.isso-postbox.isso-preview-mode>.isso-form-wrapper .isso-textarea {\n    display: none;\n}\n\n.isso-postbox.isso-preview-mode>.isso-form-wrapper .isso-preview {\n    display: block;\n}\n\n.isso-postbox.isso-preview-mode>.isso-form-wrapper input[name=\"edit\"] {\n    display: inline;\n}\n\n.isso-preview {\n    background: repeating-linear-gradient(-45deg,\n            var(--bg-0),\n            var(--bg-0) 10px,\n            var(--bg-2) 10px,\n            var(--bg-2) 20px);\n    background-color: var(--bg-0);\n}\n\n/* ========================================================================== */\n/* Animations                                                                 */\n/* ========================================================================== */\n\n/* \"target\" means the comment that's being linked to, for example:\n * https://example.com/blog/example/#isso-15\n */\n.isso-target {\n    animation: isso-target-fade 5s ease-out;\n}\n\n@keyframes isso-target-fade {\n    0% {\n        background-color: var(--divider-color)\n    }\n}\n\n/* ========================================================================== */\n/* Media queries                                                              */\n/* ========================================================================== */\n@media screen and (max-width:600px) {\n    .isso-auth-section {\n        flex-direction: column;\n        text-align: center;\n    }\n\n    .isso-input-wrapper {\n        display: block;\n        margin: 0 0 .4em;\n        max-width: 100%;\n    }\n\n    .isso-input-wrapper input {\n        width: 100%;\n    }\n\n    .isso-post-action {\n        margin: 0.4em auto;\n        width: 60%;\n    }\n}\n"
  },
  {
    "path": "static/js/codeBlockNameLinks.js",
    "content": "document.addEventListener(\"DOMContentLoaded\", function() {\n    // Convert URLs in data-name to links.\n    document.querySelectorAll('code[data-name]').forEach(function(code) {\n        const name = code.getAttribute('data-name');\n        if (name.startsWith('http')) {\n            const link = document.createElement('a');\n            link.href = name;\n            link.className = 'source-path';\n            link.textContent = name;\n            code.insertBefore(link, code.firstChild);\n            // Remove data-name to avoid overlap with Zola's native display.\n            code.removeAttribute('data-name');\n            code.parentElement?.removeAttribute('data-name');\n        }\n    });\n\n    // Legacy support for old shortcode. https://github.com/welpo/tabi/pull/489\n    document.querySelectorAll('.code-source').forEach(function(marker) {\n        const sourceUrl = marker.getAttribute('data-source');\n        const nextPre = marker.nextElementSibling;\n        if (nextPre?.tagName === 'PRE') {\n            const code = nextPre.querySelector('code');\n            if (code) {\n                if (sourceUrl.startsWith('http')) {\n                    const link = document.createElement('a');\n                    link.href = sourceUrl;\n                    link.className = 'source-path';\n                    link.textContent = sourceUrl;\n                    code.insertBefore(link, code.firstChild);\n                } else {\n                    code.setAttribute('data-name', sourceUrl);\n                }\n            }\n        }\n    });\n});\n"
  },
  {
    "path": "static/js/copyCodeToClipboard.js",
    "content": "const copiedText = document.getElementById('copy-success').textContent;\nconst initCopyText = document.getElementById('copy-init').textContent;\n\nconst changeIcon = (copyDiv, className) => {\n    copyDiv.classList.add(className);\n    copyDiv.setAttribute('aria-label', copiedText);\n    setTimeout(() => {\n        copyDiv.classList.remove(className);\n        copyDiv.setAttribute('aria-label', initCopyText);\n    }, 2500);\n};\n\nconst addCopyEventListenerToDiv = (copyDiv, block) => {\n    copyDiv.addEventListener('click', () => copyCodeAndChangeIcon(copyDiv, block));\n};\n\nconst copyCodeAndChangeIcon = async (copyDiv, block) => {\n    const code = block.querySelector('table')\n        ? getTableCode(block)\n        : getNonTableCode(block);\n    try {\n        await navigator.clipboard.writeText(code);\n        changeIcon(copyDiv, 'checked');\n    } catch (error) {\n        changeIcon(copyDiv, 'error');\n    }\n};\n\nconst getNonTableCode = (block) => {\n    return [...block.querySelectorAll('code')]\n        .map((code) => {\n            const clone = code.cloneNode(true);\n            clone.querySelectorAll('.giallo-ln').forEach(el => el.remove());\n            return clone.textContent;\n        })\n        .join('');\n};\n\nconst getTableCode = (block) => {\n    return [...block.querySelectorAll('tr')]\n        .map((row) => row.querySelector('td:last-child')?.innerText ?? '')\n        .join('');\n};\n\ndocument.querySelectorAll('pre:not(.mermaid)').forEach((block) => {\n    const copyDiv = document.createElement('div');\n    copyDiv.setAttribute('role', 'button');\n    copyDiv.setAttribute('aria-label', initCopyText);\n    copyDiv.setAttribute('title', initCopyText);\n    copyDiv.className = 'copy-code';\n    block.prepend(copyDiv);\n    addCopyEventListenerToDiv(copyDiv, block);\n});\n"
  },
  {
    "path": "static/js/decodeMail.js",
    "content": "(function () {\n    'use strict';\n\n    // Utility function: Base64 Decoding.\n    function decodeBase64(encodedString) {\n        try {\n            // Can't use atob() directly because it doesn't support non-ascii characters.\n            // And non-ascii characters are allowed in email addresses and domains.\n            // See https://en.wikipedia.org/wiki/Email_address#Internationalization\n            // Code below adapted from Jackie Han: https://stackoverflow.com/a/64752311\n            const byteString = atob(encodedString);\n\n            // Convert byteString to an array of char codes.\n            const charCodes = [...byteString].map((char) => char.charCodeAt(0));\n\n            // Use TypedArray.prototype.set() to copy the char codes into a Uint8Array.\n            const bytes = new Uint8Array(charCodes.length);\n            bytes.set(charCodes);\n\n            const decoder = new TextDecoder('utf-8');\n            return decoder.decode(bytes);\n        } catch (e) {\n            console.error('Failed to decode Base64 string: ', e);\n            return null;\n        }\n    }\n\n    // Utility function: Update href of an element with a decoded email.\n    function updateEmailHref(element) {\n        const encodedEmail = element.getAttribute('data-encoded-email');\n        const decodedEmail = decodeBase64(encodedEmail);\n\n        if (decodedEmail) {\n            element.setAttribute('href', `mailto:${decodedEmail}`);\n        } else {\n            // If the decoding fails, hide the email link.\n            element.style.display = 'none';\n        }\n    }\n\n    // Fetch and process email elements with the \"data-encoded-email\" attribute.\n    const encodedEmailElements = document.querySelectorAll('[data-encoded-email]');\n    encodedEmailElements.forEach(updateEmailHref);\n})();\n"
  },
  {
    "path": "static/js/filterCards.js",
    "content": "document.addEventListener('DOMContentLoaded', () => {\n    const cards = document.querySelectorAll('.card');\n    const filterLinks = document.querySelectorAll('.filter-controls a');\n    const allProjectsFilter = document.querySelector('#all-projects-filter');\n    if (!cards.length || !filterLinks.length) return;\n    allProjectsFilter.style.display = 'block';\n\n    // Create a Map for O(1) lookups of links by filter value.\n    const linkMap = new Map(\n        Array.from(filterLinks).map(link => [link.dataset.filter, link])\n    );\n\n    // Pre-process cards data for faster filtering.\n    const cardData = Array.from(cards).map(card => ({\n        element: card,\n        tags: card.dataset.tags?.toLowerCase().split(',').filter(Boolean) ?? []\n    }));\n\n    function getTagSlugFromUrl(url) {\n        return url.split('/').filter(Boolean).pop();\n    }\n\n    function getFilterFromHash() {\n        if (!window.location.hash) return 'all';\n        const hash = decodeURIComponent(window.location.hash.slice(1));\n        const matchingLink = Array.from(filterLinks).find(link =>\n            getTagSlugFromUrl(link.getAttribute('href')) === hash\n        );\n        return matchingLink?.dataset.filter ?? 'all';\n    }\n\n    function setActiveFilter(filterValue, updateHash = true) {\n        if (updateHash) {\n            if (filterValue === 'all') {\n                history.pushState(null, '', window.location.pathname);\n            } else {\n                const activeLink = linkMap.get(filterValue);\n                if (activeLink) {\n                    const tagSlug = getTagSlugFromUrl(activeLink.getAttribute('href'));\n                    history.pushState(null, '', `#${tagSlug}`);\n                }\n            }\n        }\n        const isAll = filterValue === 'all';\n        const display = isAll ? '' : 'none';\n        const ariaHidden = isAll ? 'false' : 'true';\n        requestAnimationFrame(() => {\n            filterLinks.forEach(link => {\n                const isActive = link.dataset.filter === filterValue;\n                link.classList.toggle('active', isActive);\n                link.setAttribute('aria-pressed', isActive);\n            });\n            if (isAll) {\n                cardData.forEach(({ element }) => {\n                    element.style.display = display;\n                    element.setAttribute('aria-hidden', ariaHidden);\n                });\n            } else {\n                cardData.forEach(({ element, tags }) => {\n                    const shouldShow = tags.includes(filterValue);\n                    element.style.display = shouldShow ? '' : 'none';\n                    element.setAttribute('aria-hidden', !shouldShow);\n                });\n            }\n        });\n    }\n\n    const filterContainer = filterLinks[0].parentElement.parentElement;\n    filterContainer.addEventListener('click', e => {\n        const link = e.target.closest('a');\n        if (!link) return;\n        e.preventDefault();\n        const filterValue = link.dataset.filter;\n        if (filterValue) setActiveFilter(filterValue);\n    });\n\n    filterContainer.addEventListener('keydown', e => {\n        const link = e.target.closest('a');\n        if (!link) return;\n        if (e.key === ' ' || e.key === 'Enter') {\n            e.preventDefault();\n            link.click();\n        }\n    });\n\n    filterLinks.forEach(link => {\n        link.setAttribute('role', 'button');\n        link.setAttribute('aria-pressed', link.classList.contains('active'));\n    });\n\n    window.addEventListener('popstate', () => {\n        setActiveFilter(getFilterFromHash(), false);\n    });\n\n    const initialFilter = getFilterFromHash();\n    if (initialFilter !== 'all') {\n        setActiveFilter(initialFilter, false);\n    }\n});\n"
  },
  {
    "path": "static/js/footnoteBacklinks.js",
    "content": "// Assign unique IDs to the footnote references based on their hashes.\nfunction assignReferenceIds() {\n    const references = document.querySelectorAll('.footnote-reference');\n    for (const ref of references) {\n        ref.id = `ref:${ref.children[0].hash.substring(1)}`;\n    }\n}\n\n// Create backlinks for each footnote definition if a corresponding reference exists.\nfunction createFootnoteBacklinks() {\n    const footnotes = document.querySelectorAll('.footnote-definition');\n    for (const footnote of footnotes) {\n        const backlinkId = `ref:${footnote.id}`;\n\n        // Skip if there's no corresponding reference in the text (i.e. the footnote doesn't reference anything).\n        if (!document.getElementById(backlinkId)) continue;\n\n        const backlink = document.createElement('a');\n        backlink.href = `#${backlinkId}`;\n        backlink.className = 'footnote-backlink';\n        backlink.textContent = '↩';\n        footnote.lastElementChild.appendChild(backlink);\n    }\n}\n\n// Initialise the handlers for the footnote references and definitions.\nfunction initFootnotes() {\n    assignReferenceIds();\n    createFootnoteBacklinks();\n}\n\n// Wait for the window to load, then execute the main function.\nwindow.addEventListener('load', initFootnotes);\n"
  },
  {
    "path": "static/js/giscus.js",
    "content": "function setGiscusTheme(newTheme) {\n    // Get the giscus iframe.\n    const frame = document.querySelector('iframe.giscus-frame');\n\n    if (frame) {\n        // If the iframe exists, send a message to set the theme.\n        frame.contentWindow.postMessage(\n            { giscus: { setConfig: { theme: newTheme } } },\n            'https://giscus.app'\n        );\n    }\n}\n\n// Function to initialize Giscus. This function is run when the window loads.\nfunction initGiscus() {\n    // Get the div that will contain the comments.\n    const commentsDiv = document.querySelector('.comments');\n    if (commentsDiv) {\n        // Get the various settings from data attributes on the div.\n        const repo = commentsDiv.getAttribute('data-repo');\n        const repoId = commentsDiv.getAttribute('data-repo-id');\n        const category = commentsDiv.getAttribute('data-category');\n        const categoryId = commentsDiv.getAttribute('data-category-id');\n        const strictTitleMatching = commentsDiv.getAttribute('data-strict');\n        const term = commentsDiv.getAttribute('data-term');\n        const reactionsEnabled = commentsDiv.getAttribute('data-reactions-enabled');\n        const inputPosition = commentsDiv.getAttribute('data-input-position');\n        const lightTheme = commentsDiv.getAttribute('data-light-theme');\n        const darkTheme = commentsDiv.getAttribute('data-dark-theme');\n        const lang = commentsDiv.getAttribute('data-lang');\n        const lazyLoading = commentsDiv.getAttribute('data-lazy-loading');\n\n        // Create a new script tag that will load the Giscus script.\n        const script = document.createElement('script');\n        script.src = 'https://giscus.app/client.js';\n        script.async = true;\n\n        // Set the various settings as data attributes on the script tag.\n        script.setAttribute('data-repo', repo);\n        script.setAttribute('data-repo-id', repoId);\n        script.setAttribute('data-category', category);\n        script.setAttribute('data-category-id', categoryId);\n        script.setAttribute('data-term', term);\n        script.setAttribute('data-strict', strictTitleMatching);\n        script.setAttribute('data-reactions-enabled', reactionsEnabled);\n        script.setAttribute('data-emit-metadata', '0');\n        script.setAttribute('data-input-position', inputPosition);\n        script.setAttribute('data-lang', lang);\n        script.setAttribute('crossorigin', 'anonymous');\n\n        // Set the mapping if it is provided.\n        const mapping = commentsDiv.getAttribute('data-mapping');\n        if (mapping) {\n            script.setAttribute('data-mapping', mapping);\n        }\n\n        // Choose the correct theme based on the current theme of the document.\n        const currentTheme =\n            document.documentElement.getAttribute('data-theme') || 'light';\n        const selectedTheme = currentTheme === 'dark' ? darkTheme : lightTheme;\n        script.setAttribute('data-theme', selectedTheme);\n\n        // Set the loading attribute if lazy loading is enabled.\n        if (lazyLoading === 'true') {\n            script.setAttribute('data-loading', 'lazy');\n        }\n\n        // Add the script tag to the div.\n        commentsDiv.appendChild(script);\n\n        // Listen for theme changes and update the Giscus theme when they occur.\n        window.addEventListener('themeChanged', (event) => {\n            const selectedTheme =\n                event.detail.theme === 'dark' ? darkTheme : lightTheme;\n            setGiscusTheme(selectedTheme);\n        });\n    }\n}\n\n// Initialize Giscus.\ninitGiscus();\n"
  },
  {
    "path": "static/js/hyvortalk.js",
    "content": "function initHyvorTalk() {\n    // Get the div that will contain the comments.\n    const commentsDiv = document.querySelector('.comments');\n    if (commentsDiv) {\n        // Get the various settings from data attributes on the div.\n        const websiteId = commentsDiv.getAttribute('data-website-id');\n        const pageId = commentsDiv.getAttribute('data-page-id');\n        const pageLanguage = commentsDiv.getAttribute('data-page-language');\n        const loading = commentsDiv.getAttribute('data-loading');\n        const pageAuthor = commentsDiv.getAttribute('data-page-author');\n\n        // Create a new script tag that will load the Hyvor Talk script.\n        const script = document.createElement('script');\n        script.src = 'https://talk.hyvor.com/embed/embed.js';\n        script.async = true;\n        script.type = 'module';\n        document.head.appendChild(script);\n\n        // Create a new Hyvor Talk comments tag.\n        const comments = document.createElement('hyvor-talk-comments');\n        comments.setAttribute('website-id', websiteId);\n        comments.setAttribute('page-id', pageId);\n        comments.setAttribute('page-language', pageLanguage);\n        comments.setAttribute('loading', loading);\n        comments.setAttribute('page-author', pageAuthor);\n\n        // Choose the correct theme based on the current theme of the document.\n        const currentTheme =\n            document.documentElement.getAttribute('data-theme') || 'light';\n        comments.setAttribute('colors', currentTheme);\n\n        // Add the Hyvor Talk comments tag to the div.\n        commentsDiv.appendChild(comments);\n\n        // Listen for theme changes and update the Hyvor Talk theme when they occur.\n        window.addEventListener('themeChanged', (event) => {\n            const selectedTheme = event.detail.theme;\n            comments.setAttribute('colors', selectedTheme);\n        });\n    }\n}\n\n// Initialize HyvorTalk.\ninitHyvorTalk();\n"
  },
  {
    "path": "static/js/initializePlausible.js",
    "content": "// Initialise Plausible analytics queue for the new script format (v3.1.0+).\n// This sets up a queue system that collects tracking calls before the main\n// Plausible script loads, ensuring no events are lost.\nwindow.plausible = window.plausible || function () {\n    (plausible.q = plausible.q || []).push(arguments);\n};\n// Initialise the Plausible configuration object.\nplausible.init = plausible.init || function (config) {\n    plausible.o = config || {};\n};\n// Set up Plausible with default configuration.\nplausible.init();\n"
  },
  {
    "path": "static/js/initializeTheme.js",
    "content": "(function () {\n    // Get the default theme from the HTML data-theme attribute.\n    const defaultTheme = document.documentElement.getAttribute('data-theme');\n\n    // Set the data-default-theme attribute only if defaultTheme is not null.\n    if (defaultTheme) {\n        document.documentElement.setAttribute('data-default-theme', defaultTheme);\n    }\n\n    // Attempt to retrieve the current theme from the browser's local storage.\n    const storedTheme = localStorage.getItem('theme');\n\n    if (storedTheme) {\n        document.documentElement.setAttribute('data-theme', storedTheme);\n    } else if (defaultTheme) {\n        document.documentElement.setAttribute('data-theme', defaultTheme);\n    } else {\n        // If no theme is found in local storage and no default theme is set, use user's system preference.\n        const isSystemDark = window.matchMedia('(prefers-color-scheme: dark)').matches;\n        document.documentElement.setAttribute(\n            'data-theme',\n            isSystemDark ? 'dark' : 'light'\n        );\n    }\n})();\n"
  },
  {
    "path": "static/js/isso.js",
    "content": "// Function to initialise Isso.\nfunction initIsso() {\n    // Get the div that will contain the comments.\n    const commentsDiv = document.querySelector('.comments');\n    if (commentsDiv) {\n        // Get the lazy-loading setting from the div.\n        const lazyLoading = commentsDiv.getAttribute('data-lazy-loading') === 'true';\n\n        // If lazy-loading is enabled, create an Intersection Observer and use it.\n        if (lazyLoading) {\n            const observer = new IntersectionObserver((entries) => {\n                // Loop over the entries.\n                entries.forEach((entry) => {\n                    // If the element is in the viewport, initialize Isso.\n                    if (entry.isIntersecting) {\n                        loadIsso(commentsDiv);\n                        // Once the Isso is loaded, we don't need to observe the element anymore.\n                        observer.unobserve(commentsDiv);\n                    }\n                });\n            });\n\n            // Start observing the comments div.\n            observer.observe(commentsDiv);\n        } else {\n            // If lazy-loading is not enabled, initialise Isso immediately.\n            loadIsso(commentsDiv);\n        }\n    }\n}\n\n// Function to load Isso.\nfunction loadIsso(commentsDiv) {\n    // Get the various settings from data attributes on the div.\n    const endpointUrl = commentsDiv.getAttribute('data-endpoint-url');\n    const pageId = commentsDiv.getAttribute('data-isso-id');\n    const title = commentsDiv.getAttribute('data-title');\n    const lang = commentsDiv.getAttribute('data-page-language');\n    const maxCommentsTop = commentsDiv.getAttribute('data-max-comments-top');\n    const maxCommentsNested = commentsDiv.getAttribute('data-max-comments-nested');\n    const avatar = commentsDiv.getAttribute('data-avatar');\n    const voting = commentsDiv.getAttribute('data-voting');\n    const hashes = commentsDiv.getAttribute('data-page-author-hashes');\n\n    // Create a new script tag that will load the Isso script.\n    const script = document.createElement('script');\n    script.src = endpointUrl + 'js/embed.min.js';\n    script.async = true;\n\n    // Set the various settings as data attributes on the script tag.\n    script.setAttribute('data-isso', endpointUrl);\n    script.setAttribute('data-isso-lang', lang);\n    script.setAttribute('data-isso-max-comments-top', maxCommentsTop);\n    script.setAttribute('data-isso-max-comments-nested', maxCommentsNested);\n    script.setAttribute('data-isso-avatar', avatar);\n    script.setAttribute('data-isso-vote', voting);\n    script.setAttribute('data-isso-page-author-hashes', hashes);\n    script.setAttribute('data-isso-css', 'false');\n\n    // Set the id and data-isso-id of the Isso thread.\n    const section = document.createElement('section');\n    section.id = 'isso-thread';\n    section.setAttribute('data-isso-id', pageId);\n    section.setAttribute('data-title', title);\n    commentsDiv.appendChild(section);\n\n    // Add the script tag to the div.\n    commentsDiv.appendChild(script);\n\n    // Create a link tag for the Isso CSS.\n    const link = document.createElement('link');\n    link.rel = 'stylesheet';\n    link.type = 'text/css';\n    link.href = '/isso.min.css';\n\n    // Add the CSS link tag to the head of the document.\n    document.head.appendChild(link);\n}\n\n// Initialize Isso.\ninitIsso();\n"
  },
  {
    "path": "static/js/loadComments.js",
    "content": "// Wait for the full HTML document to be parsed and ready.\ndocument.addEventListener('DOMContentLoaded', () => {\n    // Retrieve the button element.\n    const loadCommentsButton = document.querySelector('#load-comments');\n\n    // If the button exists…\n    if (loadCommentsButton) {\n        // Add a \"click\" event listener to the button.\n        loadCommentsButton.addEventListener('click', () => {\n            // Create a new \"script\" HTML element.\n            const script = document.createElement('script');\n\n            // Set the source of the script to the URL in the button's \"data-script-src\" attribute.\n            script.src = loadCommentsButton.dataset.scriptSrc;\n\n            // Load asynchronously.\n            script.async = true;\n\n            // Add the script element to the end of the document body, which causes the script to start loading and executing.\n            document.body.appendChild(script);\n\n            // Hide the button after it's clicked.\n            loadCommentsButton.style.display = 'none';\n        });\n    }\n});\n"
  },
  {
    "path": "static/js/lunr/lunr.da.js",
    "content": "/*!\n * Lunr languages, `Danish` language\n * https://github.com/MihaiValentin/lunr-languages\n *\n * Copyright 2014, Mihai Valentin\n * http://www.mozilla.org/MPL/\n */\n/*!\n * based on\n * Snowball JavaScript Library v0.3\n * http://code.google.com/p/urim/\n * http://snowball.tartarus.org/\n *\n * Copyright 2010, Oleg Mazko\n * http://www.mozilla.org/MPL/\n */\n\n/**\n * export the module via AMD, CommonJS or as a browser global\n * Export code from https://github.com/umdjs/umd/blob/master/returnExports.js\n */\n;\n(function(root, factory) {\n  if (typeof define === 'function' && define.amd) {\n    // AMD. Register as an anonymous module.\n    define(factory)\n  } else if (typeof exports === 'object') {\n    /**\n     * Node. Does not work with strict CommonJS, but\n     * only CommonJS-like environments that support module.exports,\n     * like Node.\n     */\n    module.exports = factory()\n  } else {\n    // Browser globals (root is window)\n    factory()(root.lunr);\n  }\n}(this, function() {\n  /**\n   * Just return a value to define the module export.\n   * This example returns an object, but the module\n   * can return a function as the exported value.\n   */\n  return function(lunr) {\n    /* throw error if lunr is not yet included */\n    if ('undefined' === typeof lunr) {\n      throw new Error('Lunr is not present. Please include / require Lunr before this script.');\n    }\n\n    /* throw error if lunr stemmer support is not yet included */\n    if ('undefined' === typeof lunr.stemmerSupport) {\n      throw new Error('Lunr stemmer support is not present. Please include / require Lunr stemmer support before this script.');\n    }\n\n    /* register specific locale function */\n    lunr.da = function() {\n      this.pipeline.reset();\n      this.pipeline.add(\n        lunr.da.trimmer,\n        lunr.da.stopWordFilter,\n        lunr.da.stemmer\n      );\n    };\n\n    /* lunr trimmer function */\n    lunr.da.wordCharacters = \"A-Za-z\\xAA\\xBA\\xC0-\\xD6\\xD8-\\xF6\\xF8-\\u02B8\\u02E0-\\u02E4\\u1D00-\\u1D25\\u1D2C-\\u1D5C\\u1D62-\\u1D65\\u1D6B-\\u1D77\\u1D79-\\u1DBE\\u1E00-\\u1EFF\\u2071\\u207F\\u2090-\\u209C\\u212A\\u212B\\u2132\\u214E\\u2160-\\u2188\\u2C60-\\u2C7F\\uA722-\\uA787\\uA78B-\\uA7AD\\uA7B0-\\uA7B7\\uA7F7-\\uA7FF\\uAB30-\\uAB5A\\uAB5C-\\uAB64\\uFB00-\\uFB06\\uFF21-\\uFF3A\\uFF41-\\uFF5A\";\n    lunr.da.trimmer = lunr.trimmerSupport.generateTrimmer(lunr.da.wordCharacters);\n\n    lunr.Pipeline.registerFunction(lunr.da.trimmer, 'trimmer-da');\n\n    /* lunr stemmer function */\n    lunr.da.stemmer = (function() {\n      /* create the wrapped stemmer object */\n      var Among = lunr.stemmerSupport.Among,\n        SnowballProgram = lunr.stemmerSupport.SnowballProgram,\n        st = new function DanishStemmer() {\n          var a_0 = [new Among(\"hed\", -1, 1), new Among(\"ethed\", 0, 1),\n              new Among(\"ered\", -1, 1), new Among(\"e\", -1, 1),\n              new Among(\"erede\", 3, 1), new Among(\"ende\", 3, 1),\n              new Among(\"erende\", 5, 1), new Among(\"ene\", 3, 1),\n              new Among(\"erne\", 3, 1), new Among(\"ere\", 3, 1),\n              new Among(\"en\", -1, 1), new Among(\"heden\", 10, 1),\n              new Among(\"eren\", 10, 1), new Among(\"er\", -1, 1),\n              new Among(\"heder\", 13, 1), new Among(\"erer\", 13, 1),\n              new Among(\"s\", -1, 2), new Among(\"heds\", 16, 1),\n              new Among(\"es\", 16, 1), new Among(\"endes\", 18, 1),\n              new Among(\"erendes\", 19, 1), new Among(\"enes\", 18, 1),\n              new Among(\"ernes\", 18, 1), new Among(\"eres\", 18, 1),\n              new Among(\"ens\", 16, 1), new Among(\"hedens\", 24, 1),\n              new Among(\"erens\", 24, 1), new Among(\"ers\", 16, 1),\n              new Among(\"ets\", 16, 1), new Among(\"erets\", 28, 1),\n              new Among(\"et\", -1, 1), new Among(\"eret\", 30, 1)\n            ],\n            a_1 = [\n              new Among(\"gd\", -1, -1), new Among(\"dt\", -1, -1),\n              new Among(\"gt\", -1, -1), new Among(\"kt\", -1, -1)\n            ],\n            a_2 = [\n              new Among(\"ig\", -1, 1), new Among(\"lig\", 0, 1),\n              new Among(\"elig\", 1, 1), new Among(\"els\", -1, 1),\n              new Among(\"l\\u00F8st\", -1, 2)\n            ],\n            g_v = [17, 65, 16, 1, 0, 0, 0, 0,\n              0, 0, 0, 0, 0, 0, 0, 0, 48, 0, 128\n            ],\n            g_s_ending = [239, 254, 42, 3,\n              0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 16\n            ],\n            I_x, I_p1, S_ch, sbp = new SnowballProgram();\n          this.setCurrent = function(word) {\n            sbp.setCurrent(word);\n          };\n          this.getCurrent = function() {\n            return sbp.getCurrent();\n          };\n\n          function r_mark_regions() {\n            var v_1, c = sbp.cursor + 3;\n            I_p1 = sbp.limit;\n            if (0 <= c && c <= sbp.limit) {\n              I_x = c;\n              while (true) {\n                v_1 = sbp.cursor;\n                if (sbp.in_grouping(g_v, 97, 248)) {\n                  sbp.cursor = v_1;\n                  break;\n                }\n                sbp.cursor = v_1;\n                if (v_1 >= sbp.limit)\n                  return;\n                sbp.cursor++;\n              }\n              while (!sbp.out_grouping(g_v, 97, 248)) {\n                if (sbp.cursor >= sbp.limit)\n                  return;\n                sbp.cursor++;\n              }\n              I_p1 = sbp.cursor;\n              if (I_p1 < I_x)\n                I_p1 = I_x;\n            }\n          }\n\n          function r_main_suffix() {\n            var among_var, v_1;\n            if (sbp.cursor >= I_p1) {\n              v_1 = sbp.limit_backward;\n              sbp.limit_backward = I_p1;\n              sbp.ket = sbp.cursor;\n              among_var = sbp.find_among_b(a_0, 32);\n              sbp.limit_backward = v_1;\n              if (among_var) {\n                sbp.bra = sbp.cursor;\n                switch (among_var) {\n                  case 1:\n                    sbp.slice_del();\n                    break;\n                  case 2:\n                    if (sbp.in_grouping_b(g_s_ending, 97, 229))\n                      sbp.slice_del();\n                    break;\n                }\n              }\n            }\n          }\n\n          function r_consonant_pair() {\n            var v_1 = sbp.limit - sbp.cursor,\n              v_2;\n            if (sbp.cursor >= I_p1) {\n              v_2 = sbp.limit_backward;\n              sbp.limit_backward = I_p1;\n              sbp.ket = sbp.cursor;\n              if (sbp.find_among_b(a_1, 4)) {\n                sbp.bra = sbp.cursor;\n                sbp.limit_backward = v_2;\n                sbp.cursor = sbp.limit - v_1;\n                if (sbp.cursor > sbp.limit_backward) {\n                  sbp.cursor--;\n                  sbp.bra = sbp.cursor;\n                  sbp.slice_del();\n                }\n              } else\n                sbp.limit_backward = v_2;\n            }\n          }\n\n          function r_other_suffix() {\n            var among_var, v_1 = sbp.limit - sbp.cursor,\n              v_2, v_3;\n            sbp.ket = sbp.cursor;\n            if (sbp.eq_s_b(2, \"st\")) {\n              sbp.bra = sbp.cursor;\n              if (sbp.eq_s_b(2, \"ig\"))\n                sbp.slice_del();\n            }\n            sbp.cursor = sbp.limit - v_1;\n            if (sbp.cursor >= I_p1) {\n              v_2 = sbp.limit_backward;\n              sbp.limit_backward = I_p1;\n              sbp.ket = sbp.cursor;\n              among_var = sbp.find_among_b(a_2, 5);\n              sbp.limit_backward = v_2;\n              if (among_var) {\n                sbp.bra = sbp.cursor;\n                switch (among_var) {\n                  case 1:\n                    sbp.slice_del();\n                    v_3 = sbp.limit - sbp.cursor;\n                    r_consonant_pair();\n                    sbp.cursor = sbp.limit - v_3;\n                    break;\n                  case 2:\n                    sbp.slice_from(\"l\\u00F8s\");\n                    break;\n                }\n              }\n            }\n          }\n\n          function r_undouble() {\n            var v_1;\n            if (sbp.cursor >= I_p1) {\n              v_1 = sbp.limit_backward;\n              sbp.limit_backward = I_p1;\n              sbp.ket = sbp.cursor;\n              if (sbp.out_grouping_b(g_v, 97, 248)) {\n                sbp.bra = sbp.cursor;\n                S_ch = sbp.slice_to(S_ch);\n                sbp.limit_backward = v_1;\n                if (sbp.eq_v_b(S_ch))\n                  sbp.slice_del();\n              } else\n                sbp.limit_backward = v_1;\n            }\n          }\n          this.stem = function() {\n            var v_1 = sbp.cursor;\n            r_mark_regions();\n            sbp.limit_backward = v_1;\n            sbp.cursor = sbp.limit;\n            r_main_suffix();\n            sbp.cursor = sbp.limit;\n            r_consonant_pair();\n            sbp.cursor = sbp.limit;\n            r_other_suffix();\n            sbp.cursor = sbp.limit;\n            r_undouble();\n            return true;\n          }\n        };\n\n      /* and return a function that stems a word for the current locale */\n      return function(word) {\n        st.setCurrent(word);\n        st.stem();\n        return st.getCurrent();\n      }\n    })();\n\n    lunr.Pipeline.registerFunction(lunr.da.stemmer, 'stemmer-da');\n\n    /* stop word filter function */\n    lunr.da.stopWordFilter = function(token) {\n      if (lunr.da.stopWordFilter.stopWords.indexOf(token) === -1) {\n        return token;\n      }\n    };\n\n    lunr.da.stopWordFilter.stopWords = new lunr.SortedSet();\n    lunr.da.stopWordFilter.stopWords.length = 95;\n\n    // The space at the beginning is crucial: It marks the empty string\n    // as a stop word. lunr.js crashes during search when documents\n    // processed by the pipeline still contain the empty string.\n    lunr.da.stopWordFilter.stopWords.elements = ' ad af alle alt anden at blev blive bliver da de dem den denne der deres det dette dig din disse dog du efter eller en end er et for fra ham han hans har havde have hende hendes her hos hun hvad hvis hvor i ikke ind jeg jer jo kunne man mange med meget men mig min mine mit mod ned noget nogle nu når og også om op os over på selv sig sin sine sit skal skulle som sådan thi til ud under var vi vil ville vor være været'.split(' ');\n\n    lunr.Pipeline.registerFunction(lunr.da.stopWordFilter, 'stopWordFilter-da');\n  };\n}))\n"
  },
  {
    "path": "static/js/lunr/lunr.de.js",
    "content": "/*!\n * Lunr languages, `German` language\n * https://github.com/MihaiValentin/lunr-languages\n *\n * Copyright 2014, Mihai Valentin\n * http://www.mozilla.org/MPL/\n */\n/*!\n * based on\n * Snowball JavaScript Library v0.3\n * http://code.google.com/p/urim/\n * http://snowball.tartarus.org/\n *\n * Copyright 2010, Oleg Mazko\n * http://www.mozilla.org/MPL/\n */\n\n/**\n * export the module via AMD, CommonJS or as a browser global\n * Export code from https://github.com/umdjs/umd/blob/master/returnExports.js\n */\n;\n(function(root, factory) {\n  if (typeof define === 'function' && define.amd) {\n    // AMD. Register as an anonymous module.\n    define(factory)\n  } else if (typeof exports === 'object') {\n    /**\n     * Node. Does not work with strict CommonJS, but\n     * only CommonJS-like environments that support module.exports,\n     * like Node.\n     */\n    module.exports = factory()\n  } else {\n    // Browser globals (root is window)\n    factory()(root.lunr);\n  }\n}(this, function() {\n  /**\n   * Just return a value to define the module export.\n   * This example returns an object, but the module\n   * can return a function as the exported value.\n   */\n  return function(lunr) {\n    /* throw error if lunr is not yet included */\n    if ('undefined' === typeof lunr) {\n      throw new Error('Lunr is not present. Please include / require Lunr before this script.');\n    }\n\n    /* throw error if lunr stemmer support is not yet included */\n    if ('undefined' === typeof lunr.stemmerSupport) {\n      throw new Error('Lunr stemmer support is not present. Please include / require Lunr stemmer support before this script.');\n    }\n\n    /* register specific locale function */\n    lunr.de = function() {\n      this.pipeline.reset();\n      this.pipeline.add(\n        lunr.de.trimmer,\n        lunr.de.stopWordFilter,\n        lunr.de.stemmer\n      );\n    };\n\n    /* lunr trimmer function */\n    lunr.de.wordCharacters = \"A-Za-z\\xAA\\xBA\\xC0-\\xD6\\xD8-\\xF6\\xF8-\\u02B8\\u02E0-\\u02E4\\u1D00-\\u1D25\\u1D2C-\\u1D5C\\u1D62-\\u1D65\\u1D6B-\\u1D77\\u1D79-\\u1DBE\\u1E00-\\u1EFF\\u2071\\u207F\\u2090-\\u209C\\u212A\\u212B\\u2132\\u214E\\u2160-\\u2188\\u2C60-\\u2C7F\\uA722-\\uA787\\uA78B-\\uA7AD\\uA7B0-\\uA7B7\\uA7F7-\\uA7FF\\uAB30-\\uAB5A\\uAB5C-\\uAB64\\uFB00-\\uFB06\\uFF21-\\uFF3A\\uFF41-\\uFF5A\";\n    lunr.de.trimmer = lunr.trimmerSupport.generateTrimmer(lunr.de.wordCharacters);\n\n    lunr.Pipeline.registerFunction(lunr.de.trimmer, 'trimmer-de');\n\n    /* lunr stemmer function */\n    lunr.de.stemmer = (function() {\n      /* create the wrapped stemmer object */\n      var Among = lunr.stemmerSupport.Among,\n        SnowballProgram = lunr.stemmerSupport.SnowballProgram,\n        st = new function GermanStemmer() {\n          var a_0 = [new Among(\"\", -1, 6), new Among(\"U\", 0, 2),\n              new Among(\"Y\", 0, 1), new Among(\"\\u00E4\", 0, 3),\n              new Among(\"\\u00F6\", 0, 4), new Among(\"\\u00FC\", 0, 5)\n            ],\n            a_1 = [\n              new Among(\"e\", -1, 2), new Among(\"em\", -1, 1),\n              new Among(\"en\", -1, 2), new Among(\"ern\", -1, 1),\n              new Among(\"er\", -1, 1), new Among(\"s\", -1, 3),\n              new Among(\"es\", 5, 2)\n            ],\n            a_2 = [new Among(\"en\", -1, 1),\n              new Among(\"er\", -1, 1), new Among(\"st\", -1, 2),\n              new Among(\"est\", 2, 1)\n            ],\n            a_3 = [new Among(\"ig\", -1, 1),\n              new Among(\"lich\", -1, 1)\n            ],\n            a_4 = [new Among(\"end\", -1, 1),\n              new Among(\"ig\", -1, 2), new Among(\"ung\", -1, 1),\n              new Among(\"lich\", -1, 3), new Among(\"isch\", -1, 2),\n              new Among(\"ik\", -1, 2), new Among(\"heit\", -1, 3),\n              new Among(\"keit\", -1, 4)\n            ],\n            g_v = [17, 65, 16, 1, 0, 0, 0, 0, 0, 0,\n              0, 0, 0, 0, 0, 0, 8, 0, 32, 8\n            ],\n            g_s_ending = [117, 30, 5],\n            g_st_ending = [\n              117, 30, 4\n            ],\n            I_x, I_p2, I_p1, sbp = new SnowballProgram();\n          this.setCurrent = function(word) {\n            sbp.setCurrent(word);\n          };\n          this.getCurrent = function() {\n            return sbp.getCurrent();\n          };\n\n          function habr1(c1, c2, v_1) {\n            if (sbp.eq_s(1, c1)) {\n              sbp.ket = sbp.cursor;\n              if (sbp.in_grouping(g_v, 97, 252)) {\n                sbp.slice_from(c2);\n                sbp.cursor = v_1;\n                return true;\n              }\n            }\n            return false;\n          }\n\n          function r_prelude() {\n            var v_1 = sbp.cursor,\n              v_2, v_3, v_4, v_5;\n            while (true) {\n              v_2 = sbp.cursor;\n              sbp.bra = v_2;\n              if (sbp.eq_s(1, \"\\u00DF\")) {\n                sbp.ket = sbp.cursor;\n                sbp.slice_from(\"ss\");\n              } else {\n                if (v_2 >= sbp.limit)\n                  break;\n                sbp.cursor = v_2 + 1;\n              }\n            }\n            sbp.cursor = v_1;\n            while (true) {\n              v_3 = sbp.cursor;\n              while (true) {\n                v_4 = sbp.cursor;\n                if (sbp.in_grouping(g_v, 97, 252)) {\n                  v_5 = sbp.cursor;\n                  sbp.bra = v_5;\n                  if (habr1(\"u\", \"U\", v_4))\n                    break;\n                  sbp.cursor = v_5;\n                  if (habr1(\"y\", \"Y\", v_4))\n                    break;\n                }\n                if (v_4 >= sbp.limit) {\n                  sbp.cursor = v_3;\n                  return;\n                }\n                sbp.cursor = v_4 + 1;\n              }\n            }\n          }\n\n          function habr2() {\n            while (!sbp.in_grouping(g_v, 97, 252)) {\n              if (sbp.cursor >= sbp.limit)\n                return true;\n              sbp.cursor++;\n            }\n            while (!sbp.out_grouping(g_v, 97, 252)) {\n              if (sbp.cursor >= sbp.limit)\n                return true;\n              sbp.cursor++;\n            }\n            return false;\n          }\n\n          function r_mark_regions() {\n            I_p1 = sbp.limit;\n            I_p2 = I_p1;\n            var c = sbp.cursor + 3;\n            if (0 <= c && c <= sbp.limit) {\n              I_x = c;\n              if (!habr2()) {\n                I_p1 = sbp.cursor;\n                if (I_p1 < I_x)\n                  I_p1 = I_x;\n                if (!habr2())\n                  I_p2 = sbp.cursor;\n              }\n            }\n          }\n\n          function r_postlude() {\n            var among_var, v_1;\n            while (true) {\n              v_1 = sbp.cursor;\n              sbp.bra = v_1;\n              among_var = sbp.find_among(a_0, 6);\n              if (!among_var)\n                return;\n              sbp.ket = sbp.cursor;\n              switch (among_var) {\n                case 1:\n                  sbp.slice_from(\"y\");\n                  break;\n                case 2:\n                case 5:\n                  sbp.slice_from(\"u\");\n                  break;\n                case 3:\n                  sbp.slice_from(\"a\");\n                  break;\n                case 4:\n                  sbp.slice_from(\"o\");\n                  break;\n                case 6:\n                  if (sbp.cursor >= sbp.limit)\n                    return;\n                  sbp.cursor++;\n                  break;\n              }\n            }\n          }\n\n          function r_R1() {\n            return I_p1 <= sbp.cursor;\n          }\n\n          function r_R2() {\n            return I_p2 <= sbp.cursor;\n          }\n\n          function r_standard_suffix() {\n            var among_var, v_1 = sbp.limit - sbp.cursor,\n              v_2, v_3, v_4;\n            sbp.ket = sbp.cursor;\n            among_var = sbp.find_among_b(a_1, 7);\n            if (among_var) {\n              sbp.bra = sbp.cursor;\n              if (r_R1()) {\n                switch (among_var) {\n                  case 1:\n                    sbp.slice_del();\n                    break;\n                  case 2:\n                    sbp.slice_del();\n                    sbp.ket = sbp.cursor;\n                    if (sbp.eq_s_b(1, \"s\")) {\n                      sbp.bra = sbp.cursor;\n                      if (sbp.eq_s_b(3, \"nis\"))\n                        sbp.slice_del();\n                    }\n                    break;\n                  case 3:\n                    if (sbp.in_grouping_b(g_s_ending, 98, 116))\n                      sbp.slice_del();\n                    break;\n                }\n              }\n            }\n            sbp.cursor = sbp.limit - v_1;\n            sbp.ket = sbp.cursor;\n            among_var = sbp.find_among_b(a_2, 4);\n            if (among_var) {\n              sbp.bra = sbp.cursor;\n              if (r_R1()) {\n                switch (among_var) {\n                  case 1:\n                    sbp.slice_del();\n                    break;\n                  case 2:\n                    if (sbp.in_grouping_b(g_st_ending, 98, 116)) {\n                      var c = sbp.cursor - 3;\n                      if (sbp.limit_backward <= c && c <= sbp.limit) {\n                        sbp.cursor = c;\n                        sbp.slice_del();\n                      }\n                    }\n                    break;\n                }\n              }\n            }\n            sbp.cursor = sbp.limit - v_1;\n            sbp.ket = sbp.cursor;\n            among_var = sbp.find_among_b(a_4, 8);\n            if (among_var) {\n              sbp.bra = sbp.cursor;\n              if (r_R2()) {\n                switch (among_var) {\n                  case 1:\n                    sbp.slice_del();\n                    sbp.ket = sbp.cursor;\n                    if (sbp.eq_s_b(2, \"ig\")) {\n                      sbp.bra = sbp.cursor;\n                      v_2 = sbp.limit - sbp.cursor;\n                      if (!sbp.eq_s_b(1, \"e\")) {\n                        sbp.cursor = sbp.limit - v_2;\n                        if (r_R2())\n                          sbp.slice_del();\n                      }\n                    }\n                    break;\n                  case 2:\n                    v_3 = sbp.limit - sbp.cursor;\n                    if (!sbp.eq_s_b(1, \"e\")) {\n                      sbp.cursor = sbp.limit - v_3;\n                      sbp.slice_del();\n                    }\n                    break;\n                  case 3:\n                    sbp.slice_del();\n                    sbp.ket = sbp.cursor;\n                    v_4 = sbp.limit - sbp.cursor;\n                    if (!sbp.eq_s_b(2, \"er\")) {\n                      sbp.cursor = sbp.limit - v_4;\n                      if (!sbp.eq_s_b(2, \"en\"))\n                        break;\n                    }\n                    sbp.bra = sbp.cursor;\n                    if (r_R1())\n                      sbp.slice_del();\n                    break;\n                  case 4:\n                    sbp.slice_del();\n                    sbp.ket = sbp.cursor;\n                    among_var = sbp.find_among_b(a_3, 2);\n                    if (among_var) {\n                      sbp.bra = sbp.cursor;\n                      if (r_R2() && among_var == 1)\n                        sbp.slice_del();\n                    }\n                    break;\n                }\n              }\n            }\n          }\n          this.stem = function() {\n            var v_1 = sbp.cursor;\n            r_prelude();\n            sbp.cursor = v_1;\n            r_mark_regions();\n            sbp.limit_backward = v_1;\n            sbp.cursor = sbp.limit;\n            r_standard_suffix();\n            sbp.cursor = sbp.limit_backward;\n            r_postlude();\n            return true;\n          }\n        };\n\n      /* and return a function that stems a word for the current locale */\n      return function(word) {\n        st.setCurrent(word);\n        st.stem();\n        return st.getCurrent();\n      }\n    })();\n\n    lunr.Pipeline.registerFunction(lunr.de.stemmer, 'stemmer-de');\n\n    /* stop word filter function */\n    lunr.de.stopWordFilter = function(token) {\n      if (lunr.de.stopWordFilter.stopWords.indexOf(token) === -1) {\n        return token;\n      }\n    };\n\n    lunr.de.stopWordFilter.stopWords = new lunr.SortedSet();\n    lunr.de.stopWordFilter.stopWords.length = 232;\n\n    // The space at the beginning is crucial: It marks the empty string\n    // as a stop word. lunr.js crashes during search when documents\n    // processed by the pipeline still contain the empty string.\n    lunr.de.stopWordFilter.stopWords.elements = ' aber alle allem allen aller alles als also am an ander andere anderem anderen anderer anderes anderm andern anderr anders auch auf aus bei bin bis bist da damit dann das dasselbe dazu daß dein deine deinem deinen deiner deines dem demselben den denn denselben der derer derselbe derselben des desselben dessen dich die dies diese dieselbe dieselben diesem diesen dieser dieses dir doch dort du durch ein eine einem einen einer eines einig einige einigem einigen einiger einiges einmal er es etwas euch euer eure eurem euren eurer eures für gegen gewesen hab habe haben hat hatte hatten hier hin hinter ich ihm ihn ihnen ihr ihre ihrem ihren ihrer ihres im in indem ins ist jede jedem jeden jeder jedes jene jenem jenen jener jenes jetzt kann kein keine keinem keinen keiner keines können könnte machen man manche manchem manchen mancher manches mein meine meinem meinen meiner meines mich mir mit muss musste nach nicht nichts noch nun nur ob oder ohne sehr sein seine seinem seinen seiner seines selbst sich sie sind so solche solchem solchen solcher solches soll sollte sondern sonst um und uns unse unsem unsen unser unses unter viel vom von vor war waren warst was weg weil weiter welche welchem welchen welcher welches wenn werde werden wie wieder will wir wird wirst wo wollen wollte während würde würden zu zum zur zwar zwischen über'.split(' ');\n\n    lunr.Pipeline.registerFunction(lunr.de.stopWordFilter, 'stopWordFilter-de');\n  };\n}))\n"
  },
  {
    "path": "static/js/lunr/lunr.du.js",
    "content": "/*!\n * Lunr languages, `Dutch` language\n * https://github.com/MihaiValentin/lunr-languages\n *\n * Copyright 2014, Mihai Valentin\n * http://www.mozilla.org/MPL/\n */\n/*!\n * based on\n * Snowball JavaScript Library v0.3\n * http://code.google.com/p/urim/\n * http://snowball.tartarus.org/\n *\n * Copyright 2010, Oleg Mazko\n * http://www.mozilla.org/MPL/\n */\n\n/**\n * export the module via AMD, CommonJS or as a browser global\n * Export code from https://github.com/umdjs/umd/blob/master/returnExports.js\n */\n;\n(function(root, factory) {\n  if (typeof define === 'function' && define.amd) {\n    // AMD. Register as an anonymous module.\n    define(factory)\n  } else if (typeof exports === 'object') {\n    /**\n     * Node. Does not work with strict CommonJS, but\n     * only CommonJS-like environments that support module.exports,\n     * like Node.\n     */\n    module.exports = factory()\n  } else {\n    // Browser globals (root is window)\n    factory()(root.lunr);\n  }\n}(this, function() {\n  /**\n   * Just return a value to define the module export.\n   * This example returns an object, but the module\n   * can return a function as the exported value.\n   */\n  return function(lunr) {\n    /* throw error if lunr is not yet included */\n    if ('undefined' === typeof lunr) {\n      throw new Error('Lunr is not present. Please include / require Lunr before this script.');\n    }\n\n    /* throw error if lunr stemmer support is not yet included */\n    if ('undefined' === typeof lunr.stemmerSupport) {\n      throw new Error('Lunr stemmer support is not present. Please include / require Lunr stemmer support before this script.');\n    }\n\n    /* register specific locale function */\n    lunr.du = function() {\n      this.pipeline.reset();\n      this.pipeline.add(\n        lunr.du.trimmer,\n        lunr.du.stopWordFilter,\n        lunr.du.stemmer\n      );\n    };\n\n    /* lunr trimmer function */\n    lunr.du.wordCharacters = \"A-Za-z\\xAA\\xBA\\xC0-\\xD6\\xD8-\\xF6\\xF8-\\u02B8\\u02E0-\\u02E4\\u1D00-\\u1D25\\u1D2C-\\u1D5C\\u1D62-\\u1D65\\u1D6B-\\u1D77\\u1D79-\\u1DBE\\u1E00-\\u1EFF\\u2071\\u207F\\u2090-\\u209C\\u212A\\u212B\\u2132\\u214E\\u2160-\\u2188\\u2C60-\\u2C7F\\uA722-\\uA787\\uA78B-\\uA7AD\\uA7B0-\\uA7B7\\uA7F7-\\uA7FF\\uAB30-\\uAB5A\\uAB5C-\\uAB64\\uFB00-\\uFB06\\uFF21-\\uFF3A\\uFF41-\\uFF5A\";\n    lunr.du.trimmer = lunr.trimmerSupport.generateTrimmer(lunr.du.wordCharacters);\n\n    lunr.Pipeline.registerFunction(lunr.du.trimmer, 'trimmer-du');\n\n    /* lunr stemmer function */\n    lunr.du.stemmer = (function() {\n      /* create the wrapped stemmer object */\n      var Among = lunr.stemmerSupport.Among,\n        SnowballProgram = lunr.stemmerSupport.SnowballProgram,\n        st = new function DutchStemmer() {\n          var a_0 = [new Among(\"\", -1, 6), new Among(\"\\u00E1\", 0, 1),\n              new Among(\"\\u00E4\", 0, 1), new Among(\"\\u00E9\", 0, 2),\n              new Among(\"\\u00EB\", 0, 2), new Among(\"\\u00ED\", 0, 3),\n              new Among(\"\\u00EF\", 0, 3), new Among(\"\\u00F3\", 0, 4),\n              new Among(\"\\u00F6\", 0, 4), new Among(\"\\u00FA\", 0, 5),\n              new Among(\"\\u00FC\", 0, 5)\n            ],\n            a_1 = [new Among(\"\", -1, 3),\n              new Among(\"I\", 0, 2), new Among(\"Y\", 0, 1)\n            ],\n            a_2 = [\n              new Among(\"dd\", -1, -1), new Among(\"kk\", -1, -1),\n              new Among(\"tt\", -1, -1)\n            ],\n            a_3 = [new Among(\"ene\", -1, 2),\n              new Among(\"se\", -1, 3), new Among(\"en\", -1, 2),\n              new Among(\"heden\", 2, 1), new Among(\"s\", -1, 3)\n            ],\n            a_4 = [\n              new Among(\"end\", -1, 1), new Among(\"ig\", -1, 2),\n              new Among(\"ing\", -1, 1), new Among(\"lijk\", -1, 3),\n              new Among(\"baar\", -1, 4), new Among(\"bar\", -1, 5)\n            ],\n            a_5 = [\n              new Among(\"aa\", -1, -1), new Among(\"ee\", -1, -1),\n              new Among(\"oo\", -1, -1), new Among(\"uu\", -1, -1)\n            ],\n            g_v = [17, 65,\n              16, 1, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 128\n            ],\n            g_v_I = [1, 0, 0,\n              17, 65, 16, 1, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 128\n            ],\n            g_v_j = [\n              17, 67, 16, 1, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 128\n            ],\n            I_p2, I_p1, B_e_found, sbp = new SnowballProgram();\n          this.setCurrent = function(word) {\n            sbp.setCurrent(word);\n          };\n          this.getCurrent = function() {\n            return sbp.getCurrent();\n          };\n\n          function r_prelude() {\n            var among_var, v_1 = sbp.cursor,\n              v_2, v_3;\n            while (true) {\n              sbp.bra = sbp.cursor;\n              among_var = sbp.find_among(a_0, 11);\n              if (among_var) {\n                sbp.ket = sbp.cursor;\n                switch (among_var) {\n                  case 1:\n                    sbp.slice_from(\"a\");\n                    continue;\n                  case 2:\n                    sbp.slice_from(\"e\");\n                    continue;\n                  case 3:\n                    sbp.slice_from(\"i\");\n                    continue;\n                  case 4:\n                    sbp.slice_from(\"o\");\n                    continue;\n                  case 5:\n                    sbp.slice_from(\"u\");\n                    continue;\n                  case 6:\n                    if (sbp.cursor >= sbp.limit)\n                      break;\n                    sbp.cursor++;\n                    continue;\n                }\n              }\n              break;\n            }\n            sbp.cursor = v_1;\n            sbp.bra = v_1;\n            if (sbp.eq_s(1, \"y\")) {\n              sbp.ket = sbp.cursor;\n              sbp.slice_from(\"Y\");\n            } else\n              sbp.cursor = v_1;\n            while (true) {\n              v_2 = sbp.cursor;\n              if (sbp.in_grouping(g_v, 97, 232)) {\n                v_3 = sbp.cursor;\n                sbp.bra = v_3;\n                if (sbp.eq_s(1, \"i\")) {\n                  sbp.ket = sbp.cursor;\n                  if (sbp.in_grouping(g_v, 97, 232)) {\n                    sbp.slice_from(\"I\");\n                    sbp.cursor = v_2;\n                  }\n                } else {\n                  sbp.cursor = v_3;\n                  if (sbp.eq_s(1, \"y\")) {\n                    sbp.ket = sbp.cursor;\n                    sbp.slice_from(\"Y\");\n                    sbp.cursor = v_2;\n                  } else if (habr1(v_2))\n                    break;\n                }\n              } else if (habr1(v_2))\n                break;\n            }\n          }\n\n          function habr1(v_1) {\n            sbp.cursor = v_1;\n            if (v_1 >= sbp.limit)\n              return true;\n            sbp.cursor++;\n            return false;\n          }\n\n          function r_mark_regions() {\n            I_p1 = sbp.limit;\n            I_p2 = I_p1;\n            if (!habr2()) {\n              I_p1 = sbp.cursor;\n              if (I_p1 < 3)\n                I_p1 = 3;\n              if (!habr2())\n                I_p2 = sbp.cursor;\n            }\n          }\n\n          function habr2() {\n            while (!sbp.in_grouping(g_v, 97, 232)) {\n              if (sbp.cursor >= sbp.limit)\n                return true;\n              sbp.cursor++;\n            }\n            while (!sbp.out_grouping(g_v, 97, 232)) {\n              if (sbp.cursor >= sbp.limit)\n                return true;\n              sbp.cursor++;\n            }\n            return false;\n          }\n\n          function r_postlude() {\n            var among_var;\n            while (true) {\n              sbp.bra = sbp.cursor;\n              among_var = sbp.find_among(a_1, 3);\n              if (among_var) {\n                sbp.ket = sbp.cursor;\n                switch (among_var) {\n                  case 1:\n                    sbp.slice_from(\"y\");\n                    break;\n                  case 2:\n                    sbp.slice_from(\"i\");\n                    break;\n                  case 3:\n                    if (sbp.cursor >= sbp.limit)\n                      return;\n                    sbp.cursor++;\n                    break;\n                }\n              }\n            }\n          }\n\n          function r_R1() {\n            return I_p1 <= sbp.cursor;\n          }\n\n          function r_R2() {\n            return I_p2 <= sbp.cursor;\n          }\n\n          function r_undouble() {\n            var v_1 = sbp.limit - sbp.cursor;\n            if (sbp.find_among_b(a_2, 3)) {\n              sbp.cursor = sbp.limit - v_1;\n              sbp.ket = sbp.cursor;\n              if (sbp.cursor > sbp.limit_backward) {\n                sbp.cursor--;\n                sbp.bra = sbp.cursor;\n                sbp.slice_del();\n              }\n            }\n          }\n\n          function r_e_ending() {\n            var v_1;\n            B_e_found = false;\n            sbp.ket = sbp.cursor;\n            if (sbp.eq_s_b(1, \"e\")) {\n              sbp.bra = sbp.cursor;\n              if (r_R1()) {\n                v_1 = sbp.limit - sbp.cursor;\n                if (sbp.out_grouping_b(g_v, 97, 232)) {\n                  sbp.cursor = sbp.limit - v_1;\n                  sbp.slice_del();\n                  B_e_found = true;\n                  r_undouble();\n                }\n              }\n            }\n          }\n\n          function r_en_ending() {\n            var v_1;\n            if (r_R1()) {\n              v_1 = sbp.limit - sbp.cursor;\n              if (sbp.out_grouping_b(g_v, 97, 232)) {\n                sbp.cursor = sbp.limit - v_1;\n                if (!sbp.eq_s_b(3, \"gem\")) {\n                  sbp.cursor = sbp.limit - v_1;\n                  sbp.slice_del();\n                  r_undouble();\n                }\n              }\n            }\n          }\n\n          function r_standard_suffix() {\n            var among_var, v_1 = sbp.limit - sbp.cursor,\n              v_2, v_3, v_4, v_5, v_6;\n            sbp.ket = sbp.cursor;\n            among_var = sbp.find_among_b(a_3, 5);\n            if (among_var) {\n              sbp.bra = sbp.cursor;\n              switch (among_var) {\n                case 1:\n                  if (r_R1())\n                    sbp.slice_from(\"heid\");\n                  break;\n                case 2:\n                  r_en_ending();\n                  break;\n                case 3:\n                  if (r_R1() && sbp.out_grouping_b(g_v_j, 97, 232))\n                    sbp.slice_del();\n                  break;\n              }\n            }\n            sbp.cursor = sbp.limit - v_1;\n            r_e_ending();\n            sbp.cursor = sbp.limit - v_1;\n            sbp.ket = sbp.cursor;\n            if (sbp.eq_s_b(4, \"heid\")) {\n              sbp.bra = sbp.cursor;\n              if (r_R2()) {\n                v_2 = sbp.limit - sbp.cursor;\n                if (!sbp.eq_s_b(1, \"c\")) {\n                  sbp.cursor = sbp.limit - v_2;\n                  sbp.slice_del();\n                  sbp.ket = sbp.cursor;\n                  if (sbp.eq_s_b(2, \"en\")) {\n                    sbp.bra = sbp.cursor;\n                    r_en_ending();\n                  }\n                }\n              }\n            }\n            sbp.cursor = sbp.limit - v_1;\n            sbp.ket = sbp.cursor;\n            among_var = sbp.find_among_b(a_4, 6);\n            if (among_var) {\n              sbp.bra = sbp.cursor;\n              switch (among_var) {\n                case 1:\n                  if (r_R2()) {\n                    sbp.slice_del();\n                    v_3 = sbp.limit - sbp.cursor;\n                    sbp.ket = sbp.cursor;\n                    if (sbp.eq_s_b(2, \"ig\")) {\n                      sbp.bra = sbp.cursor;\n                      if (r_R2()) {\n                        v_4 = sbp.limit - sbp.cursor;\n                        if (!sbp.eq_s_b(1, \"e\")) {\n                          sbp.cursor = sbp.limit - v_4;\n                          sbp.slice_del();\n                          break;\n                        }\n                      }\n                    }\n                    sbp.cursor = sbp.limit - v_3;\n                    r_undouble();\n                  }\n                  break;\n                case 2:\n                  if (r_R2()) {\n                    v_5 = sbp.limit - sbp.cursor;\n                    if (!sbp.eq_s_b(1, \"e\")) {\n                      sbp.cursor = sbp.limit - v_5;\n                      sbp.slice_del();\n                    }\n                  }\n                  break;\n                case 3:\n                  if (r_R2()) {\n                    sbp.slice_del();\n                    r_e_ending();\n                  }\n                  break;\n                case 4:\n                  if (r_R2())\n                    sbp.slice_del();\n                  break;\n                case 5:\n                  if (r_R2() && B_e_found)\n                    sbp.slice_del();\n                  break;\n              }\n            }\n            sbp.cursor = sbp.limit - v_1;\n            if (sbp.out_grouping_b(g_v_I, 73, 232)) {\n              v_6 = sbp.limit - sbp.cursor;\n              if (sbp.find_among_b(a_5, 4) && sbp.out_grouping_b(g_v, 97, 232)) {\n                sbp.cursor = sbp.limit - v_6;\n                sbp.ket = sbp.cursor;\n                if (sbp.cursor > sbp.limit_backward) {\n                  sbp.cursor--;\n                  sbp.bra = sbp.cursor;\n                  sbp.slice_del();\n                }\n              }\n            }\n          }\n          this.stem = function() {\n            var v_1 = sbp.cursor;\n            r_prelude();\n            sbp.cursor = v_1;\n            r_mark_regions();\n            sbp.limit_backward = v_1;\n            sbp.cursor = sbp.limit;\n            r_standard_suffix();\n            sbp.cursor = sbp.limit_backward;\n            r_postlude();\n            return true;\n          }\n        };\n\n      /* and return a function that stems a word for the current locale */\n      return function(word) {\n        st.setCurrent(word);\n        st.stem();\n        return st.getCurrent();\n      }\n    })();\n\n    lunr.Pipeline.registerFunction(lunr.du.stemmer, 'stemmer-du');\n\n    /* stop word filter function */\n    lunr.du.stopWordFilter = function(token) {\n      if (lunr.du.stopWordFilter.stopWords.indexOf(token) === -1) {\n        return token;\n      }\n    };\n\n    lunr.du.stopWordFilter.stopWords = new lunr.SortedSet();\n    lunr.du.stopWordFilter.stopWords.length = 103;\n\n    // The space at the beginning is crucial: It marks the empty string\n    // as a stop word. lunr.js crashes during search when documents\n    // processed by the pipeline still contain the empty string.\n    lunr.du.stopWordFilter.stopWords.elements = '  aan al alles als altijd andere ben bij daar dan dat de der deze die dit doch doen door dus een eens en er ge geen geweest haar had heb hebben heeft hem het hier hij hoe hun iemand iets ik in is ja je kan kon kunnen maar me meer men met mij mijn moet na naar niet niets nog nu of om omdat onder ons ook op over reeds te tegen toch toen tot u uit uw van veel voor want waren was wat werd wezen wie wil worden wordt zal ze zelf zich zij zijn zo zonder zou'.split(' ');\n\n    lunr.Pipeline.registerFunction(lunr.du.stopWordFilter, 'stopWordFilter-du');\n  };\n}))\n"
  },
  {
    "path": "static/js/lunr/lunr.es.js",
    "content": "/*!\n * Lunr languages, `Spanish` language\n * https://github.com/MihaiValentin/lunr-languages\n *\n * Copyright 2014, Mihai Valentin\n * http://www.mozilla.org/MPL/\n */\n/*!\n * based on\n * Snowball JavaScript Library v0.3\n * http://code.google.com/p/urim/\n * http://snowball.tartarus.org/\n *\n * Copyright 2010, Oleg Mazko\n * http://www.mozilla.org/MPL/\n */\n\n/**\n * export the module via AMD, CommonJS or as a browser global\n * Export code from https://github.com/umdjs/umd/blob/master/returnExports.js\n */\n;\n(function(root, factory) {\n  if (typeof define === 'function' && define.amd) {\n    // AMD. Register as an anonymous module.\n    define(factory)\n  } else if (typeof exports === 'object') {\n    /**\n     * Node. Does not work with strict CommonJS, but\n     * only CommonJS-like environments that support module.exports,\n     * like Node.\n     */\n    module.exports = factory()\n  } else {\n    // Browser globals (root is window)\n    factory()(root.lunr);\n  }\n}(this, function() {\n  /**\n   * Just return a value to define the module export.\n   * This example returns an object, but the module\n   * can return a function as the exported value.\n   */\n  return function(lunr) {\n    /* throw error if lunr is not yet included */\n    if ('undefined' === typeof lunr) {\n      throw new Error('Lunr is not present. Please include / require Lunr before this script.');\n    }\n\n    /* throw error if lunr stemmer support is not yet included */\n    if ('undefined' === typeof lunr.stemmerSupport) {\n      throw new Error('Lunr stemmer support is not present. Please include / require Lunr stemmer support before this script.');\n    }\n\n    /* register specific locale function */\n    lunr.es = function() {\n      this.pipeline.reset();\n      this.pipeline.add(\n        lunr.es.trimmer,\n        lunr.es.stopWordFilter,\n        lunr.es.stemmer\n      );\n    };\n\n    /* lunr trimmer function */\n    lunr.es.wordCharacters = \"A-Za-z\\xAA\\xBA\\xC0-\\xD6\\xD8-\\xF6\\xF8-\\u02B8\\u02E0-\\u02E4\\u1D00-\\u1D25\\u1D2C-\\u1D5C\\u1D62-\\u1D65\\u1D6B-\\u1D77\\u1D79-\\u1DBE\\u1E00-\\u1EFF\\u2071\\u207F\\u2090-\\u209C\\u212A\\u212B\\u2132\\u214E\\u2160-\\u2188\\u2C60-\\u2C7F\\uA722-\\uA787\\uA78B-\\uA7AD\\uA7B0-\\uA7B7\\uA7F7-\\uA7FF\\uAB30-\\uAB5A\\uAB5C-\\uAB64\\uFB00-\\uFB06\\uFF21-\\uFF3A\\uFF41-\\uFF5A\";\n    lunr.es.trimmer = lunr.trimmerSupport.generateTrimmer(lunr.es.wordCharacters);\n\n    lunr.Pipeline.registerFunction(lunr.es.trimmer, 'trimmer-es');\n\n    /* lunr stemmer function */\n    lunr.es.stemmer = (function() {\n      /* create the wrapped stemmer object */\n      var Among = lunr.stemmerSupport.Among,\n        SnowballProgram = lunr.stemmerSupport.SnowballProgram,\n        st = new function SpanishStemmer() {\n          var a_0 = [new Among(\"\", -1, 6), new Among(\"\\u00E1\", 0, 1),\n              new Among(\"\\u00E9\", 0, 2), new Among(\"\\u00ED\", 0, 3),\n              new Among(\"\\u00F3\", 0, 4), new Among(\"\\u00FA\", 0, 5)\n            ],\n            a_1 = [\n              new Among(\"la\", -1, -1), new Among(\"sela\", 0, -1),\n              new Among(\"le\", -1, -1), new Among(\"me\", -1, -1),\n              new Among(\"se\", -1, -1), new Among(\"lo\", -1, -1),\n              new Among(\"selo\", 5, -1), new Among(\"las\", -1, -1),\n              new Among(\"selas\", 7, -1), new Among(\"les\", -1, -1),\n              new Among(\"los\", -1, -1), new Among(\"selos\", 10, -1),\n              new Among(\"nos\", -1, -1)\n            ],\n            a_2 = [new Among(\"ando\", -1, 6),\n              new Among(\"iendo\", -1, 6), new Among(\"yendo\", -1, 7),\n              new Among(\"\\u00E1ndo\", -1, 2), new Among(\"i\\u00E9ndo\", -1, 1),\n              new Among(\"ar\", -1, 6), new Among(\"er\", -1, 6),\n              new Among(\"ir\", -1, 6), new Among(\"\\u00E1r\", -1, 3),\n              new Among(\"\\u00E9r\", -1, 4), new Among(\"\\u00EDr\", -1, 5)\n            ],\n            a_3 = [\n              new Among(\"ic\", -1, -1), new Among(\"ad\", -1, -1),\n              new Among(\"os\", -1, -1), new Among(\"iv\", -1, 1)\n            ],\n            a_4 = [\n              new Among(\"able\", -1, 1), new Among(\"ible\", -1, 1),\n              new Among(\"ante\", -1, 1)\n            ],\n            a_5 = [new Among(\"ic\", -1, 1),\n              new Among(\"abil\", -1, 1), new Among(\"iv\", -1, 1)\n            ],\n            a_6 = [\n              new Among(\"ica\", -1, 1), new Among(\"ancia\", -1, 2),\n              new Among(\"encia\", -1, 5), new Among(\"adora\", -1, 2),\n              new Among(\"osa\", -1, 1), new Among(\"ista\", -1, 1),\n              new Among(\"iva\", -1, 9), new Among(\"anza\", -1, 1),\n              new Among(\"log\\u00EDa\", -1, 3), new Among(\"idad\", -1, 8),\n              new Among(\"able\", -1, 1), new Among(\"ible\", -1, 1),\n              new Among(\"ante\", -1, 2), new Among(\"mente\", -1, 7),\n              new Among(\"amente\", 13, 6), new Among(\"aci\\u00F3n\", -1, 2),\n              new Among(\"uci\\u00F3n\", -1, 4), new Among(\"ico\", -1, 1),\n              new Among(\"ismo\", -1, 1), new Among(\"oso\", -1, 1),\n              new Among(\"amiento\", -1, 1), new Among(\"imiento\", -1, 1),\n              new Among(\"ivo\", -1, 9), new Among(\"ador\", -1, 2),\n              new Among(\"icas\", -1, 1), new Among(\"ancias\", -1, 2),\n              new Among(\"encias\", -1, 5), new Among(\"adoras\", -1, 2),\n              new Among(\"osas\", -1, 1), new Among(\"istas\", -1, 1),\n              new Among(\"ivas\", -1, 9), new Among(\"anzas\", -1, 1),\n              new Among(\"log\\u00EDas\", -1, 3), new Among(\"idades\", -1, 8),\n              new Among(\"ables\", -1, 1), new Among(\"ibles\", -1, 1),\n              new Among(\"aciones\", -1, 2), new Among(\"uciones\", -1, 4),\n              new Among(\"adores\", -1, 2), new Among(\"antes\", -1, 2),\n              new Among(\"icos\", -1, 1), new Among(\"ismos\", -1, 1),\n              new Among(\"osos\", -1, 1), new Among(\"amientos\", -1, 1),\n              new Among(\"imientos\", -1, 1), new Among(\"ivos\", -1, 9)\n            ],\n            a_7 = [\n              new Among(\"ya\", -1, 1), new Among(\"ye\", -1, 1),\n              new Among(\"yan\", -1, 1), new Among(\"yen\", -1, 1),\n              new Among(\"yeron\", -1, 1), new Among(\"yendo\", -1, 1),\n              new Among(\"yo\", -1, 1), new Among(\"yas\", -1, 1),\n              new Among(\"yes\", -1, 1), new Among(\"yais\", -1, 1),\n              new Among(\"yamos\", -1, 1), new Among(\"y\\u00F3\", -1, 1)\n            ],\n            a_8 = [\n              new Among(\"aba\", -1, 2), new Among(\"ada\", -1, 2),\n              new Among(\"ida\", -1, 2), new Among(\"ara\", -1, 2),\n              new Among(\"iera\", -1, 2), new Among(\"\\u00EDa\", -1, 2),\n              new Among(\"ar\\u00EDa\", 5, 2), new Among(\"er\\u00EDa\", 5, 2),\n              new Among(\"ir\\u00EDa\", 5, 2), new Among(\"ad\", -1, 2),\n              new Among(\"ed\", -1, 2), new Among(\"id\", -1, 2),\n              new Among(\"ase\", -1, 2), new Among(\"iese\", -1, 2),\n              new Among(\"aste\", -1, 2), new Among(\"iste\", -1, 2),\n              new Among(\"an\", -1, 2), new Among(\"aban\", 16, 2),\n              new Among(\"aran\", 16, 2), new Among(\"ieran\", 16, 2),\n              new Among(\"\\u00EDan\", 16, 2), new Among(\"ar\\u00EDan\", 20, 2),\n              new Among(\"er\\u00EDan\", 20, 2), new Among(\"ir\\u00EDan\", 20, 2),\n              new Among(\"en\", -1, 1), new Among(\"asen\", 24, 2),\n              new Among(\"iesen\", 24, 2), new Among(\"aron\", -1, 2),\n              new Among(\"ieron\", -1, 2), new Among(\"ar\\u00E1n\", -1, 2),\n              new Among(\"er\\u00E1n\", -1, 2), new Among(\"ir\\u00E1n\", -1, 2),\n              new Among(\"ado\", -1, 2), new Among(\"ido\", -1, 2),\n              new Among(\"ando\", -1, 2), new Among(\"iendo\", -1, 2),\n              new Among(\"ar\", -1, 2), new Among(\"er\", -1, 2),\n              new Among(\"ir\", -1, 2), new Among(\"as\", -1, 2),\n              new Among(\"abas\", 39, 2), new Among(\"adas\", 39, 2),\n              new Among(\"idas\", 39, 2), new Among(\"aras\", 39, 2),\n              new Among(\"ieras\", 39, 2), new Among(\"\\u00EDas\", 39, 2),\n              new Among(\"ar\\u00EDas\", 45, 2), new Among(\"er\\u00EDas\", 45, 2),\n              new Among(\"ir\\u00EDas\", 45, 2), new Among(\"es\", -1, 1),\n              new Among(\"ases\", 49, 2), new Among(\"ieses\", 49, 2),\n              new Among(\"abais\", -1, 2), new Among(\"arais\", -1, 2),\n              new Among(\"ierais\", -1, 2), new Among(\"\\u00EDais\", -1, 2),\n              new Among(\"ar\\u00EDais\", 55, 2), new Among(\"er\\u00EDais\", 55, 2),\n              new Among(\"ir\\u00EDais\", 55, 2), new Among(\"aseis\", -1, 2),\n              new Among(\"ieseis\", -1, 2), new Among(\"asteis\", -1, 2),\n              new Among(\"isteis\", -1, 2), new Among(\"\\u00E1is\", -1, 2),\n              new Among(\"\\u00E9is\", -1, 1), new Among(\"ar\\u00E9is\", 64, 2),\n              new Among(\"er\\u00E9is\", 64, 2), new Among(\"ir\\u00E9is\", 64, 2),\n              new Among(\"ados\", -1, 2), new Among(\"idos\", -1, 2),\n              new Among(\"amos\", -1, 2), new Among(\"\\u00E1bamos\", 70, 2),\n              new Among(\"\\u00E1ramos\", 70, 2), new Among(\"i\\u00E9ramos\", 70, 2),\n              new Among(\"\\u00EDamos\", 70, 2), new Among(\"ar\\u00EDamos\", 74, 2),\n              new Among(\"er\\u00EDamos\", 74, 2), new Among(\"ir\\u00EDamos\", 74, 2),\n              new Among(\"emos\", -1, 1), new Among(\"aremos\", 78, 2),\n              new Among(\"eremos\", 78, 2), new Among(\"iremos\", 78, 2),\n              new Among(\"\\u00E1semos\", 78, 2), new Among(\"i\\u00E9semos\", 78, 2),\n              new Among(\"imos\", -1, 2), new Among(\"ar\\u00E1s\", -1, 2),\n              new Among(\"er\\u00E1s\", -1, 2), new Among(\"ir\\u00E1s\", -1, 2),\n              new Among(\"\\u00EDs\", -1, 2), new Among(\"ar\\u00E1\", -1, 2),\n              new Among(\"er\\u00E1\", -1, 2), new Among(\"ir\\u00E1\", -1, 2),\n              new Among(\"ar\\u00E9\", -1, 2), new Among(\"er\\u00E9\", -1, 2),\n              new Among(\"ir\\u00E9\", -1, 2), new Among(\"i\\u00F3\", -1, 2)\n            ],\n            a_9 = [\n              new Among(\"a\", -1, 1), new Among(\"e\", -1, 2),\n              new Among(\"o\", -1, 1), new Among(\"os\", -1, 1),\n              new Among(\"\\u00E1\", -1, 1), new Among(\"\\u00E9\", -1, 2),\n              new Among(\"\\u00ED\", -1, 1), new Among(\"\\u00F3\", -1, 1)\n            ],\n            g_v = [17,\n              65, 16, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 1, 17, 4, 10\n            ],\n            I_p2, I_p1, I_pV, sbp = new SnowballProgram();\n          this.setCurrent = function(word) {\n            sbp.setCurrent(word);\n          };\n          this.getCurrent = function() {\n            return sbp.getCurrent();\n          };\n\n          function habr1() {\n            if (sbp.out_grouping(g_v, 97, 252)) {\n              while (!sbp.in_grouping(g_v, 97, 252)) {\n                if (sbp.cursor >= sbp.limit)\n                  return true;\n                sbp.cursor++;\n              }\n              return false;\n            }\n            return true;\n          }\n\n          function habr2() {\n            if (sbp.in_grouping(g_v, 97, 252)) {\n              var v_1 = sbp.cursor;\n              if (habr1()) {\n                sbp.cursor = v_1;\n                if (!sbp.in_grouping(g_v, 97, 252))\n                  return true;\n                while (!sbp.out_grouping(g_v, 97, 252)) {\n                  if (sbp.cursor >= sbp.limit)\n                    return true;\n                  sbp.cursor++;\n                }\n              }\n              return false;\n            }\n            return true;\n          }\n\n          function habr3() {\n            var v_1 = sbp.cursor,\n              v_2;\n            if (habr2()) {\n              sbp.cursor = v_1;\n              if (!sbp.out_grouping(g_v, 97, 252))\n                return;\n              v_2 = sbp.cursor;\n              if (habr1()) {\n                sbp.cursor = v_2;\n                if (!sbp.in_grouping(g_v, 97, 252) || sbp.cursor >= sbp.limit)\n                  return;\n                sbp.cursor++;\n              }\n            }\n            I_pV = sbp.cursor;\n          }\n\n          function habr4() {\n            while (!sbp.in_grouping(g_v, 97, 252)) {\n              if (sbp.cursor >= sbp.limit)\n                return false;\n              sbp.cursor++;\n            }\n            while (!sbp.out_grouping(g_v, 97, 252)) {\n              if (sbp.cursor >= sbp.limit)\n                return false;\n              sbp.cursor++;\n            }\n            return true;\n          }\n\n          function r_mark_regions() {\n            var v_1 = sbp.cursor;\n            I_pV = sbp.limit;\n            I_p1 = I_pV;\n            I_p2 = I_pV;\n            habr3();\n            sbp.cursor = v_1;\n            if (habr4()) {\n              I_p1 = sbp.cursor;\n              if (habr4())\n                I_p2 = sbp.cursor;\n            }\n          }\n\n          function r_postlude() {\n            var among_var;\n            while (true) {\n              sbp.bra = sbp.cursor;\n              among_var = sbp.find_among(a_0, 6);\n              if (among_var) {\n                sbp.ket = sbp.cursor;\n                switch (among_var) {\n                  case 1:\n                    sbp.slice_from(\"a\");\n                    continue;\n                  case 2:\n                    sbp.slice_from(\"e\");\n                    continue;\n                  case 3:\n                    sbp.slice_from(\"i\");\n                    continue;\n                  case 4:\n                    sbp.slice_from(\"o\");\n                    continue;\n                  case 5:\n                    sbp.slice_from(\"u\");\n                    continue;\n                  case 6:\n                    if (sbp.cursor >= sbp.limit)\n                      break;\n                    sbp.cursor++;\n                    continue;\n                }\n              }\n              break;\n            }\n          }\n\n          function r_RV() {\n            return I_pV <= sbp.cursor;\n          }\n\n          function r_R1() {\n            return I_p1 <= sbp.cursor;\n          }\n\n          function r_R2() {\n            return I_p2 <= sbp.cursor;\n          }\n\n          function r_attached_pronoun() {\n            var among_var;\n            sbp.ket = sbp.cursor;\n            if (sbp.find_among_b(a_1, 13)) {\n              sbp.bra = sbp.cursor;\n              among_var = sbp.find_among_b(a_2, 11);\n              if (among_var && r_RV())\n                switch (among_var) {\n                  case 1:\n                    sbp.bra = sbp.cursor;\n                    sbp.slice_from(\"iendo\");\n                    break;\n                  case 2:\n                    sbp.bra = sbp.cursor;\n                    sbp.slice_from(\"ando\");\n                    break;\n                  case 3:\n                    sbp.bra = sbp.cursor;\n                    sbp.slice_from(\"ar\");\n                    break;\n                  case 4:\n                    sbp.bra = sbp.cursor;\n                    sbp.slice_from(\"er\");\n                    break;\n                  case 5:\n                    sbp.bra = sbp.cursor;\n                    sbp.slice_from(\"ir\");\n                    break;\n                  case 6:\n                    sbp.slice_del();\n                    break;\n                  case 7:\n                    if (sbp.eq_s_b(1, \"u\"))\n                      sbp.slice_del();\n                    break;\n                }\n            }\n          }\n\n          function habr5(a, n) {\n            if (!r_R2())\n              return true;\n            sbp.slice_del();\n            sbp.ket = sbp.cursor;\n            var among_var = sbp.find_among_b(a, n);\n            if (among_var) {\n              sbp.bra = sbp.cursor;\n              if (among_var == 1 && r_R2())\n                sbp.slice_del();\n            }\n            return false;\n          }\n\n          function habr6(c1) {\n            if (!r_R2())\n              return true;\n            sbp.slice_del();\n            sbp.ket = sbp.cursor;\n            if (sbp.eq_s_b(2, c1)) {\n              sbp.bra = sbp.cursor;\n              if (r_R2())\n                sbp.slice_del();\n            }\n            return false;\n          }\n\n          function r_standard_suffix() {\n            var among_var;\n            sbp.ket = sbp.cursor;\n            among_var = sbp.find_among_b(a_6, 46);\n            if (among_var) {\n              sbp.bra = sbp.cursor;\n              switch (among_var) {\n                case 1:\n                  if (!r_R2())\n                    return false;\n                  sbp.slice_del();\n                  break;\n                case 2:\n                  if (habr6(\"ic\"))\n                    return false;\n                  break;\n                case 3:\n                  if (!r_R2())\n                    return false;\n                  sbp.slice_from(\"log\");\n                  break;\n                case 4:\n                  if (!r_R2())\n                    return false;\n                  sbp.slice_from(\"u\");\n                  break;\n                case 5:\n                  if (!r_R2())\n                    return false;\n                  sbp.slice_from(\"ente\");\n                  break;\n                case 6:\n                  if (!r_R1())\n                    return false;\n                  sbp.slice_del();\n                  sbp.ket = sbp.cursor;\n                  among_var = sbp.find_among_b(a_3, 4);\n                  if (among_var) {\n                    sbp.bra = sbp.cursor;\n                    if (r_R2()) {\n                      sbp.slice_del();\n                      if (among_var == 1) {\n                        sbp.ket = sbp.cursor;\n                        if (sbp.eq_s_b(2, \"at\")) {\n                          sbp.bra = sbp.cursor;\n                          if (r_R2())\n                            sbp.slice_del();\n                        }\n                      }\n                    }\n                  }\n                  break;\n                case 7:\n                  if (habr5(a_4, 3))\n                    return false;\n                  break;\n                case 8:\n                  if (habr5(a_5, 3))\n                    return false;\n                  break;\n                case 9:\n                  if (habr6(\"at\"))\n                    return false;\n                  break;\n              }\n              return true;\n            }\n            return false;\n          }\n\n          function r_y_verb_suffix() {\n            var among_var, v_1;\n            if (sbp.cursor >= I_pV) {\n              v_1 = sbp.limit_backward;\n              sbp.limit_backward = I_pV;\n              sbp.ket = sbp.cursor;\n              among_var = sbp.find_among_b(a_7, 12);\n              sbp.limit_backward = v_1;\n              if (among_var) {\n                sbp.bra = sbp.cursor;\n                if (among_var == 1) {\n                  if (!sbp.eq_s_b(1, \"u\"))\n                    return false;\n                  sbp.slice_del();\n                }\n                return true;\n              }\n            }\n            return false;\n          }\n\n          function r_verb_suffix() {\n            var among_var, v_1, v_2, v_3;\n            if (sbp.cursor >= I_pV) {\n              v_1 = sbp.limit_backward;\n              sbp.limit_backward = I_pV;\n              sbp.ket = sbp.cursor;\n              among_var = sbp.find_among_b(a_8, 96);\n              sbp.limit_backward = v_1;\n              if (among_var) {\n                sbp.bra = sbp.cursor;\n                switch (among_var) {\n                  case 1:\n                    v_2 = sbp.limit - sbp.cursor;\n                    if (sbp.eq_s_b(1, \"u\")) {\n                      v_3 = sbp.limit - sbp.cursor;\n                      if (sbp.eq_s_b(1, \"g\"))\n                        sbp.cursor = sbp.limit - v_3;\n                      else\n                        sbp.cursor = sbp.limit - v_2;\n                    } else\n                      sbp.cursor = sbp.limit - v_2;\n                    sbp.bra = sbp.cursor;\n                  case 2:\n                    sbp.slice_del();\n                    break;\n                }\n              }\n            }\n          }\n\n          function r_residual_suffix() {\n            var among_var, v_1;\n            sbp.ket = sbp.cursor;\n            among_var = sbp.find_among_b(a_9, 8);\n            if (among_var) {\n              sbp.bra = sbp.cursor;\n              switch (among_var) {\n                case 1:\n                  if (r_RV())\n                    sbp.slice_del();\n                  break;\n                case 2:\n                  if (r_RV()) {\n                    sbp.slice_del();\n                    sbp.ket = sbp.cursor;\n                    if (sbp.eq_s_b(1, \"u\")) {\n                      sbp.bra = sbp.cursor;\n                      v_1 = sbp.limit - sbp.cursor;\n                      if (sbp.eq_s_b(1, \"g\")) {\n                        sbp.cursor = sbp.limit - v_1;\n                        if (r_RV())\n                          sbp.slice_del();\n                      }\n                    }\n                  }\n                  break;\n              }\n            }\n          }\n          this.stem = function() {\n            var v_1 = sbp.cursor;\n            r_mark_regions();\n            sbp.limit_backward = v_1;\n            sbp.cursor = sbp.limit;\n            r_attached_pronoun();\n            sbp.cursor = sbp.limit;\n            if (!r_standard_suffix()) {\n              sbp.cursor = sbp.limit;\n              if (!r_y_verb_suffix()) {\n                sbp.cursor = sbp.limit;\n                r_verb_suffix();\n              }\n            }\n            sbp.cursor = sbp.limit;\n            r_residual_suffix();\n            sbp.cursor = sbp.limit_backward;\n            r_postlude();\n            return true;\n          }\n        };\n\n      /* and return a function that stems a word for the current locale */\n      return function(word) {\n        st.setCurrent(word);\n        st.stem();\n        return st.getCurrent();\n      }\n    })();\n\n    lunr.Pipeline.registerFunction(lunr.es.stemmer, 'stemmer-es');\n\n    /* stop word filter function */\n    lunr.es.stopWordFilter = function(token) {\n      if (lunr.es.stopWordFilter.stopWords.indexOf(token) === -1) {\n        return token;\n      }\n    };\n\n    lunr.es.stopWordFilter.stopWords = new lunr.SortedSet();\n    lunr.es.stopWordFilter.stopWords.length = 309;\n\n    // The space at the beginning is crucial: It marks the empty string\n    // as a stop word. lunr.js crashes during search when documents\n    // processed by the pipeline still contain the empty string.\n    lunr.es.stopWordFilter.stopWords.elements = ' a al algo algunas algunos ante antes como con contra cual cuando de del desde donde durante e el ella ellas ellos en entre era erais eran eras eres es esa esas ese eso esos esta estaba estabais estaban estabas estad estada estadas estado estados estamos estando estar estaremos estará estarán estarás estaré estaréis estaría estaríais estaríamos estarían estarías estas este estemos esto estos estoy estuve estuviera estuvierais estuvieran estuvieras estuvieron estuviese estuvieseis estuviesen estuvieses estuvimos estuviste estuvisteis estuviéramos estuviésemos estuvo está estábamos estáis están estás esté estéis estén estés fue fuera fuerais fueran fueras fueron fuese fueseis fuesen fueses fui fuimos fuiste fuisteis fuéramos fuésemos ha habida habidas habido habidos habiendo habremos habrá habrán habrás habré habréis habría habríais habríamos habrían habrías habéis había habíais habíamos habían habías han has hasta hay haya hayamos hayan hayas hayáis he hemos hube hubiera hubierais hubieran hubieras hubieron hubiese hubieseis hubiesen hubieses hubimos hubiste hubisteis hubiéramos hubiésemos hubo la las le les lo los me mi mis mucho muchos muy más mí mía mías mío míos nada ni no nos nosotras nosotros nuestra nuestras nuestro nuestros o os otra otras otro otros para pero poco por porque que quien quienes qué se sea seamos sean seas seremos será serán serás seré seréis sería seríais seríamos serían serías seáis sido siendo sin sobre sois somos son soy su sus suya suyas suyo suyos sí también tanto te tendremos tendrá tendrán tendrás tendré tendréis tendría tendríais tendríamos tendrían tendrías tened tenemos tenga tengamos tengan tengas tengo tengáis tenida tenidas tenido tenidos teniendo tenéis tenía teníais teníamos tenían tenías ti tiene tienen tienes todo todos tu tus tuve tuviera tuvierais tuvieran tuvieras tuvieron tuviese tuvieseis tuviesen tuvieses tuvimos tuviste tuvisteis tuviéramos tuviésemos tuvo tuya tuyas tuyo tuyos tú un una uno unos vosotras vosotros vuestra vuestras vuestro vuestros y ya yo él éramos'.split(' ');\n\n    lunr.Pipeline.registerFunction(lunr.es.stopWordFilter, 'stopWordFilter-es');\n  };\n}))\n"
  },
  {
    "path": "static/js/lunr/lunr.fi.js",
    "content": "/*!\n * Lunr languages, `Finnish` language\n * https://github.com/MihaiValentin/lunr-languages\n *\n * Copyright 2014, Mihai Valentin\n * http://www.mozilla.org/MPL/\n */\n/*!\n * based on\n * Snowball JavaScript Library v0.3\n * http://code.google.com/p/urim/\n * http://snowball.tartarus.org/\n *\n * Copyright 2010, Oleg Mazko\n * http://www.mozilla.org/MPL/\n */\n\n/**\n * export the module via AMD, CommonJS or as a browser global\n * Export code from https://github.com/umdjs/umd/blob/master/returnExports.js\n */\n;\n(function(root, factory) {\n  if (typeof define === 'function' && define.amd) {\n    // AMD. Register as an anonymous module.\n    define(factory)\n  } else if (typeof exports === 'object') {\n    /**\n     * Node. Does not work with strict CommonJS, but\n     * only CommonJS-like environments that support module.exports,\n     * like Node.\n     */\n    module.exports = factory()\n  } else {\n    // Browser globals (root is window)\n    factory()(root.lunr);\n  }\n}(this, function() {\n  /**\n   * Just return a value to define the module export.\n   * This example returns an object, but the module\n   * can return a function as the exported value.\n   */\n  return function(lunr) {\n    /* throw error if lunr is not yet included */\n    if ('undefined' === typeof lunr) {\n      throw new Error('Lunr is not present. Please include / require Lunr before this script.');\n    }\n\n    /* throw error if lunr stemmer support is not yet included */\n    if ('undefined' === typeof lunr.stemmerSupport) {\n      throw new Error('Lunr stemmer support is not present. Please include / require Lunr stemmer support before this script.');\n    }\n\n    /* register specific locale function */\n    lunr.fi = function() {\n      this.pipeline.reset();\n      this.pipeline.add(\n        lunr.fi.trimmer,\n        lunr.fi.stopWordFilter,\n        lunr.fi.stemmer\n      );\n    };\n\n    /* lunr trimmer function */\n    lunr.fi.wordCharacters = \"A-Za-z\\xAA\\xBA\\xC0-\\xD6\\xD8-\\xF6\\xF8-\\u02B8\\u02E0-\\u02E4\\u1D00-\\u1D25\\u1D2C-\\u1D5C\\u1D62-\\u1D65\\u1D6B-\\u1D77\\u1D79-\\u1DBE\\u1E00-\\u1EFF\\u2071\\u207F\\u2090-\\u209C\\u212A\\u212B\\u2132\\u214E\\u2160-\\u2188\\u2C60-\\u2C7F\\uA722-\\uA787\\uA78B-\\uA7AD\\uA7B0-\\uA7B7\\uA7F7-\\uA7FF\\uAB30-\\uAB5A\\uAB5C-\\uAB64\\uFB00-\\uFB06\\uFF21-\\uFF3A\\uFF41-\\uFF5A\";\n    lunr.fi.trimmer = lunr.trimmerSupport.generateTrimmer(lunr.fi.wordCharacters);\n\n    lunr.Pipeline.registerFunction(lunr.fi.trimmer, 'trimmer-fi');\n\n    /* lunr stemmer function */\n    lunr.fi.stemmer = (function() {\n      /* create the wrapped stemmer object */\n      var Among = lunr.stemmerSupport.Among,\n        SnowballProgram = lunr.stemmerSupport.SnowballProgram,\n        st = new function FinnishStemmer() {\n          var a_0 = [new Among(\"pa\", -1, 1), new Among(\"sti\", -1, 2),\n              new Among(\"kaan\", -1, 1), new Among(\"han\", -1, 1),\n              new Among(\"kin\", -1, 1), new Among(\"h\\u00E4n\", -1, 1),\n              new Among(\"k\\u00E4\\u00E4n\", -1, 1), new Among(\"ko\", -1, 1),\n              new Among(\"p\\u00E4\", -1, 1), new Among(\"k\\u00F6\", -1, 1)\n            ],\n            a_1 = [\n              new Among(\"lla\", -1, -1), new Among(\"na\", -1, -1),\n              new Among(\"ssa\", -1, -1), new Among(\"ta\", -1, -1),\n              new Among(\"lta\", 3, -1), new Among(\"sta\", 3, -1)\n            ],\n            a_2 = [\n              new Among(\"ll\\u00E4\", -1, -1), new Among(\"n\\u00E4\", -1, -1),\n              new Among(\"ss\\u00E4\", -1, -1), new Among(\"t\\u00E4\", -1, -1),\n              new Among(\"lt\\u00E4\", 3, -1), new Among(\"st\\u00E4\", 3, -1)\n            ],\n            a_3 = [\n              new Among(\"lle\", -1, -1), new Among(\"ine\", -1, -1)\n            ],\n            a_4 = [\n              new Among(\"nsa\", -1, 3), new Among(\"mme\", -1, 3),\n              new Among(\"nne\", -1, 3), new Among(\"ni\", -1, 2),\n              new Among(\"si\", -1, 1), new Among(\"an\", -1, 4),\n              new Among(\"en\", -1, 6), new Among(\"\\u00E4n\", -1, 5),\n              new Among(\"ns\\u00E4\", -1, 3)\n            ],\n            a_5 = [new Among(\"aa\", -1, -1),\n              new Among(\"ee\", -1, -1), new Among(\"ii\", -1, -1),\n              new Among(\"oo\", -1, -1), new Among(\"uu\", -1, -1),\n              new Among(\"\\u00E4\\u00E4\", -1, -1),\n              new Among(\"\\u00F6\\u00F6\", -1, -1)\n            ],\n            a_6 = [new Among(\"a\", -1, 8),\n              new Among(\"lla\", 0, -1), new Among(\"na\", 0, -1),\n              new Among(\"ssa\", 0, -1), new Among(\"ta\", 0, -1),\n              new Among(\"lta\", 4, -1), new Among(\"sta\", 4, -1),\n              new Among(\"tta\", 4, 9), new Among(\"lle\", -1, -1),\n              new Among(\"ine\", -1, -1), new Among(\"ksi\", -1, -1),\n              new Among(\"n\", -1, 7), new Among(\"han\", 11, 1),\n              new Among(\"den\", 11, -1, r_VI), new Among(\"seen\", 11, -1, r_LONG),\n              new Among(\"hen\", 11, 2), new Among(\"tten\", 11, -1, r_VI),\n              new Among(\"hin\", 11, 3), new Among(\"siin\", 11, -1, r_VI),\n              new Among(\"hon\", 11, 4), new Among(\"h\\u00E4n\", 11, 5),\n              new Among(\"h\\u00F6n\", 11, 6), new Among(\"\\u00E4\", -1, 8),\n              new Among(\"ll\\u00E4\", 22, -1), new Among(\"n\\u00E4\", 22, -1),\n              new Among(\"ss\\u00E4\", 22, -1), new Among(\"t\\u00E4\", 22, -1),\n              new Among(\"lt\\u00E4\", 26, -1), new Among(\"st\\u00E4\", 26, -1),\n              new Among(\"tt\\u00E4\", 26, 9)\n            ],\n            a_7 = [new Among(\"eja\", -1, -1),\n              new Among(\"mma\", -1, 1), new Among(\"imma\", 1, -1),\n              new Among(\"mpa\", -1, 1), new Among(\"impa\", 3, -1),\n              new Among(\"mmi\", -1, 1), new Among(\"immi\", 5, -1),\n              new Among(\"mpi\", -1, 1), new Among(\"impi\", 7, -1),\n              new Among(\"ej\\u00E4\", -1, -1), new Among(\"mm\\u00E4\", -1, 1),\n              new Among(\"imm\\u00E4\", 10, -1), new Among(\"mp\\u00E4\", -1, 1),\n              new Among(\"imp\\u00E4\", 12, -1)\n            ],\n            a_8 = [new Among(\"i\", -1, -1),\n              new Among(\"j\", -1, -1)\n            ],\n            a_9 = [new Among(\"mma\", -1, 1),\n              new Among(\"imma\", 0, -1)\n            ],\n            g_AEI = [17, 1, 0, 0, 0, 0, 0, 0, 0, 0,\n              0, 0, 0, 0, 0, 0, 8\n            ],\n            g_V1 = [17, 65, 16, 1, 0, 0, 0, 0, 0, 0, 0,\n              0, 0, 0, 0, 0, 8, 0, 32\n            ],\n            g_V2 = [17, 65, 16, 0, 0, 0, 0, 0, 0, 0,\n              0, 0, 0, 0, 0, 0, 8, 0, 32\n            ],\n            g_particle_end = [17, 97, 24, 1, 0, 0,\n              0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 8, 0, 32\n            ],\n            B_ending_removed, S_x, I_p2, I_p1, sbp = new SnowballProgram();\n          this.setCurrent = function(word) {\n            sbp.setCurrent(word);\n          };\n          this.getCurrent = function() {\n            return sbp.getCurrent();\n          };\n\n          function r_mark_regions() {\n            I_p1 = sbp.limit;\n            I_p2 = I_p1;\n            if (!habr1()) {\n              I_p1 = sbp.cursor;\n              if (!habr1())\n                I_p2 = sbp.cursor;\n            }\n          }\n\n          function habr1() {\n            var v_1;\n            while (true) {\n              v_1 = sbp.cursor;\n              if (sbp.in_grouping(g_V1, 97, 246))\n                break;\n              sbp.cursor = v_1;\n              if (v_1 >= sbp.limit)\n                return true;\n              sbp.cursor++;\n            }\n            sbp.cursor = v_1;\n            while (!sbp.out_grouping(g_V1, 97, 246)) {\n              if (sbp.cursor >= sbp.limit)\n                return true;\n              sbp.cursor++;\n            }\n            return false;\n          }\n\n          function r_R2() {\n            return I_p2 <= sbp.cursor;\n          }\n\n          function r_particle_etc() {\n            var among_var, v_1;\n            if (sbp.cursor >= I_p1) {\n              v_1 = sbp.limit_backward;\n              sbp.limit_backward = I_p1;\n              sbp.ket = sbp.cursor;\n              among_var = sbp.find_among_b(a_0, 10);\n              if (among_var) {\n                sbp.bra = sbp.cursor;\n                sbp.limit_backward = v_1;\n                switch (among_var) {\n                  case 1:\n                    if (!sbp.in_grouping_b(g_particle_end, 97, 246))\n                      return;\n                    break;\n                  case 2:\n                    if (!r_R2())\n                      return;\n                    break;\n                }\n                sbp.slice_del();\n              } else\n                sbp.limit_backward = v_1;\n            }\n          }\n\n          function r_possessive() {\n            var among_var, v_1, v_2;\n            if (sbp.cursor >= I_p1) {\n              v_1 = sbp.limit_backward;\n              sbp.limit_backward = I_p1;\n              sbp.ket = sbp.cursor;\n              among_var = sbp.find_among_b(a_4, 9);\n              if (among_var) {\n                sbp.bra = sbp.cursor;\n                sbp.limit_backward = v_1;\n                switch (among_var) {\n                  case 1:\n                    v_2 = sbp.limit - sbp.cursor;\n                    if (!sbp.eq_s_b(1, \"k\")) {\n                      sbp.cursor = sbp.limit - v_2;\n                      sbp.slice_del();\n                    }\n                    break;\n                  case 2:\n                    sbp.slice_del();\n                    sbp.ket = sbp.cursor;\n                    if (sbp.eq_s_b(3, \"kse\")) {\n                      sbp.bra = sbp.cursor;\n                      sbp.slice_from(\"ksi\");\n                    }\n                    break;\n                  case 3:\n                    sbp.slice_del();\n                    break;\n                  case 4:\n                    if (sbp.find_among_b(a_1, 6))\n                      sbp.slice_del();\n                    break;\n                  case 5:\n                    if (sbp.find_among_b(a_2, 6))\n                      sbp.slice_del();\n                    break;\n                  case 6:\n                    if (sbp.find_among_b(a_3, 2))\n                      sbp.slice_del();\n                    break;\n                }\n              } else\n                sbp.limit_backward = v_1;\n            }\n          }\n\n          function r_LONG() {\n            return sbp.find_among_b(a_5, 7);\n          }\n\n          function r_VI() {\n            return sbp.eq_s_b(1, \"i\") && sbp.in_grouping_b(g_V2, 97, 246);\n          }\n\n          function r_case_ending() {\n            var among_var, v_1, v_2;\n            if (sbp.cursor >= I_p1) {\n              v_1 = sbp.limit_backward;\n              sbp.limit_backward = I_p1;\n              sbp.ket = sbp.cursor;\n              among_var = sbp.find_among_b(a_6, 30);\n              if (among_var) {\n                sbp.bra = sbp.cursor;\n                sbp.limit_backward = v_1;\n                switch (among_var) {\n                  case 1:\n                    if (!sbp.eq_s_b(1, \"a\"))\n                      return;\n                    break;\n                  case 2:\n                  case 9:\n                    if (!sbp.eq_s_b(1, \"e\"))\n                      return;\n                    break;\n                  case 3:\n                    if (!sbp.eq_s_b(1, \"i\"))\n                      return;\n                    break;\n                  case 4:\n                    if (!sbp.eq_s_b(1, \"o\"))\n                      return;\n                    break;\n                  case 5:\n                    if (!sbp.eq_s_b(1, \"\\u00E4\"))\n                      return;\n                    break;\n                  case 6:\n                    if (!sbp.eq_s_b(1, \"\\u00F6\"))\n                      return;\n                    break;\n                  case 7:\n                    v_2 = sbp.limit - sbp.cursor;\n                    if (!r_LONG()) {\n                      sbp.cursor = sbp.limit - v_2;\n                      if (!sbp.eq_s_b(2, \"ie\")) {\n                        sbp.cursor = sbp.limit - v_2;\n                        break;\n                      }\n                    }\n                    sbp.cursor = sbp.limit - v_2;\n                    if (sbp.cursor <= sbp.limit_backward) {\n                      sbp.cursor = sbp.limit - v_2;\n                      break;\n                    }\n                    sbp.cursor--;\n                    sbp.bra = sbp.cursor;\n                    break;\n                  case 8:\n                    if (!sbp.in_grouping_b(g_V1, 97, 246) || !sbp.out_grouping_b(g_V1, 97, 246))\n                      return;\n                    break;\n                }\n                sbp.slice_del();\n                B_ending_removed = true;\n              } else\n                sbp.limit_backward = v_1;\n            }\n          }\n\n          function r_other_endings() {\n            var among_var, v_1, v_2;\n            if (sbp.cursor >= I_p2) {\n              v_1 = sbp.limit_backward;\n              sbp.limit_backward = I_p2;\n              sbp.ket = sbp.cursor;\n              among_var = sbp.find_among_b(a_7, 14);\n              if (among_var) {\n                sbp.bra = sbp.cursor;\n                sbp.limit_backward = v_1;\n                if (among_var == 1) {\n                  v_2 = sbp.limit - sbp.cursor;\n                  if (sbp.eq_s_b(2, \"po\"))\n                    return;\n                  sbp.cursor = sbp.limit - v_2;\n                }\n                sbp.slice_del();\n              } else\n                sbp.limit_backward = v_1;\n            }\n          }\n\n          function r_i_plural() {\n            var v_1;\n            if (sbp.cursor >= I_p1) {\n              v_1 = sbp.limit_backward;\n              sbp.limit_backward = I_p1;\n              sbp.ket = sbp.cursor;\n              if (sbp.find_among_b(a_8, 2)) {\n                sbp.bra = sbp.cursor;\n                sbp.limit_backward = v_1;\n                sbp.slice_del();\n              } else\n                sbp.limit_backward = v_1;\n            }\n          }\n\n          function r_t_plural() {\n            var among_var, v_1, v_2, v_3, v_4, v_5;\n            if (sbp.cursor >= I_p1) {\n              v_1 = sbp.limit_backward;\n              sbp.limit_backward = I_p1;\n              sbp.ket = sbp.cursor;\n              if (sbp.eq_s_b(1, \"t\")) {\n                sbp.bra = sbp.cursor;\n                v_2 = sbp.limit - sbp.cursor;\n                if (sbp.in_grouping_b(g_V1, 97, 246)) {\n                  sbp.cursor = sbp.limit - v_2;\n                  sbp.slice_del();\n                  sbp.limit_backward = v_1;\n                  v_3 = sbp.limit - sbp.cursor;\n                  if (sbp.cursor >= I_p2) {\n                    sbp.cursor = I_p2;\n                    v_4 = sbp.limit_backward;\n                    sbp.limit_backward = sbp.cursor;\n                    sbp.cursor = sbp.limit - v_3;\n                    sbp.ket = sbp.cursor;\n                    among_var = sbp.find_among_b(a_9, 2);\n                    if (among_var) {\n                      sbp.bra = sbp.cursor;\n                      sbp.limit_backward = v_4;\n                      if (among_var == 1) {\n                        v_5 = sbp.limit - sbp.cursor;\n                        if (sbp.eq_s_b(2, \"po\"))\n                          return;\n                        sbp.cursor = sbp.limit - v_5;\n                      }\n                      sbp.slice_del();\n                      return;\n                    }\n                  }\n                }\n              }\n              sbp.limit_backward = v_1;\n            }\n          }\n\n          function r_tidy() {\n            var v_1, v_2, v_3, v_4;\n            if (sbp.cursor >= I_p1) {\n              v_1 = sbp.limit_backward;\n              sbp.limit_backward = I_p1;\n              v_2 = sbp.limit - sbp.cursor;\n              if (r_LONG()) {\n                sbp.cursor = sbp.limit - v_2;\n                sbp.ket = sbp.cursor;\n                if (sbp.cursor > sbp.limit_backward) {\n                  sbp.cursor--;\n                  sbp.bra = sbp.cursor;\n                  sbp.slice_del();\n                }\n              }\n              sbp.cursor = sbp.limit - v_2;\n              sbp.ket = sbp.cursor;\n              if (sbp.in_grouping_b(g_AEI, 97, 228)) {\n                sbp.bra = sbp.cursor;\n                if (sbp.out_grouping_b(g_V1, 97, 246))\n                  sbp.slice_del();\n              }\n              sbp.cursor = sbp.limit - v_2;\n              sbp.ket = sbp.cursor;\n              if (sbp.eq_s_b(1, \"j\")) {\n                sbp.bra = sbp.cursor;\n                v_3 = sbp.limit - sbp.cursor;\n                if (!sbp.eq_s_b(1, \"o\")) {\n                  sbp.cursor = sbp.limit - v_3;\n                  if (sbp.eq_s_b(1, \"u\"))\n                    sbp.slice_del();\n                } else\n                  sbp.slice_del();\n              }\n              sbp.cursor = sbp.limit - v_2;\n              sbp.ket = sbp.cursor;\n              if (sbp.eq_s_b(1, \"o\")) {\n                sbp.bra = sbp.cursor;\n                if (sbp.eq_s_b(1, \"j\"))\n                  sbp.slice_del();\n              }\n              sbp.cursor = sbp.limit - v_2;\n              sbp.limit_backward = v_1;\n              while (true) {\n                v_4 = sbp.limit - sbp.cursor;\n                if (sbp.out_grouping_b(g_V1, 97, 246)) {\n                  sbp.cursor = sbp.limit - v_4;\n                  break;\n                }\n                sbp.cursor = sbp.limit - v_4;\n                if (sbp.cursor <= sbp.limit_backward)\n                  return;\n                sbp.cursor--;\n              }\n              sbp.ket = sbp.cursor;\n              if (sbp.cursor > sbp.limit_backward) {\n                sbp.cursor--;\n                sbp.bra = sbp.cursor;\n                S_x = sbp.slice_to();\n                if (sbp.eq_v_b(S_x))\n                  sbp.slice_del();\n              }\n            }\n          }\n          this.stem = function() {\n            var v_1 = sbp.cursor;\n            r_mark_regions();\n            B_ending_removed = false;\n            sbp.limit_backward = v_1;\n            sbp.cursor = sbp.limit;\n            r_particle_etc();\n            sbp.cursor = sbp.limit;\n            r_possessive();\n            sbp.cursor = sbp.limit;\n            r_case_ending();\n            sbp.cursor = sbp.limit;\n            r_other_endings();\n            sbp.cursor = sbp.limit;\n            if (B_ending_removed) {\n              r_i_plural();\n              sbp.cursor = sbp.limit;\n            } else {\n              sbp.cursor = sbp.limit;\n              r_t_plural();\n              sbp.cursor = sbp.limit;\n            }\n            r_tidy();\n            return true;\n          }\n        };\n\n      /* and return a function that stems a word for the current locale */\n      return function(word) {\n        st.setCurrent(word);\n        st.stem();\n        return st.getCurrent();\n      }\n    })();\n\n    lunr.Pipeline.registerFunction(lunr.fi.stemmer, 'stemmer-fi');\n\n    /* stop word filter function */\n    lunr.fi.stopWordFilter = function(token) {\n      if (lunr.fi.stopWordFilter.stopWords.indexOf(token) === -1) {\n        return token;\n      }\n    };\n\n    lunr.fi.stopWordFilter.stopWords = new lunr.SortedSet();\n    lunr.fi.stopWordFilter.stopWords.length = 236;\n\n    // The space at the beginning is crucial: It marks the empty string\n    // as a stop word. lunr.js crashes during search when documents\n    // processed by the pipeline still contain the empty string.\n    lunr.fi.stopWordFilter.stopWords.elements = ' ei eivät emme en et ette että he heidän heidät heihin heille heillä heiltä heissä heistä heitä hän häneen hänelle hänellä häneltä hänen hänessä hänestä hänet häntä itse ja johon joiden joihin joiksi joilla joille joilta joina joissa joista joita joka joksi jolla jolle jolta jona jonka jos jossa josta jota jotka kanssa keiden keihin keiksi keille keillä keiltä keinä keissä keistä keitä keneen keneksi kenelle kenellä keneltä kenen kenenä kenessä kenestä kenet ketkä ketkä ketä koska kuin kuka kun me meidän meidät meihin meille meillä meiltä meissä meistä meitä mihin miksi mikä mille millä miltä minkä minkä minua minulla minulle minulta minun minussa minusta minut minuun minä minä missä mistä mitkä mitä mukaan mutta ne niiden niihin niiksi niille niillä niiltä niin niin niinä niissä niistä niitä noiden noihin noiksi noilla noille noilta noin noina noissa noista noita nuo nyt näiden näihin näiksi näille näillä näiltä näinä näissä näistä näitä nämä ole olemme olen olet olette oli olimme olin olisi olisimme olisin olisit olisitte olisivat olit olitte olivat olla olleet ollut on ovat poikki se sekä sen siihen siinä siitä siksi sille sillä sillä siltä sinua sinulla sinulle sinulta sinun sinussa sinusta sinut sinuun sinä sinä sitä tai te teidän teidät teihin teille teillä teiltä teissä teistä teitä tuo tuohon tuoksi tuolla tuolle tuolta tuon tuona tuossa tuosta tuota tähän täksi tälle tällä tältä tämä tämän tänä tässä tästä tätä vaan vai vaikka yli'.split(' ');\n\n    lunr.Pipeline.registerFunction(lunr.fi.stopWordFilter, 'stopWordFilter-fi');\n  };\n}))\n"
  },
  {
    "path": "static/js/lunr/lunr.fr.js",
    "content": "/*!\n * Lunr languages, `French` language\n * https://github.com/MihaiValentin/lunr-languages\n *\n * Copyright 2014, Mihai Valentin\n * http://www.mozilla.org/MPL/\n */\n/*!\n * based on\n * Snowball JavaScript Library v0.3\n * http://code.google.com/p/urim/\n * http://snowball.tartarus.org/\n *\n * Copyright 2010, Oleg Mazko\n * http://www.mozilla.org/MPL/\n */\n\n/**\n * export the module via AMD, CommonJS or as a browser global\n * Export code from https://github.com/umdjs/umd/blob/master/returnExports.js\n */\n;\n(function(root, factory) {\n  if (typeof define === 'function' && define.amd) {\n    // AMD. Register as an anonymous module.\n    define(factory)\n  } else if (typeof exports === 'object') {\n    /**\n     * Node. Does not work with strict CommonJS, but\n     * only CommonJS-like environments that support module.exports,\n     * like Node.\n     */\n    module.exports = factory()\n  } else {\n    // Browser globals (root is window)\n    factory()(root.lunr);\n  }\n}(this, function() {\n  /**\n   * Just return a value to define the module export.\n   * This example returns an object, but the module\n   * can return a function as the exported value.\n   */\n  return function(lunr) {\n    /* throw error if lunr is not yet included */\n    if ('undefined' === typeof lunr) {\n      throw new Error('Lunr is not present. Please include / require Lunr before this script.');\n    }\n\n    /* throw error if lunr stemmer support is not yet included */\n    if ('undefined' === typeof lunr.stemmerSupport) {\n      throw new Error('Lunr stemmer support is not present. Please include / require Lunr stemmer support before this script.');\n    }\n\n    /* register specific locale function */\n    lunr.fr = function() {\n      this.pipeline.reset();\n      this.pipeline.add(\n        lunr.fr.trimmer,\n        lunr.fr.stopWordFilter,\n        lunr.fr.stemmer\n      );\n    };\n\n    /* lunr trimmer function */\n    lunr.fr.wordCharacters = \"A-Za-z\\xAA\\xBA\\xC0-\\xD6\\xD8-\\xF6\\xF8-\\u02B8\\u02E0-\\u02E4\\u1D00-\\u1D25\\u1D2C-\\u1D5C\\u1D62-\\u1D65\\u1D6B-\\u1D77\\u1D79-\\u1DBE\\u1E00-\\u1EFF\\u2071\\u207F\\u2090-\\u209C\\u212A\\u212B\\u2132\\u214E\\u2160-\\u2188\\u2C60-\\u2C7F\\uA722-\\uA787\\uA78B-\\uA7AD\\uA7B0-\\uA7B7\\uA7F7-\\uA7FF\\uAB30-\\uAB5A\\uAB5C-\\uAB64\\uFB00-\\uFB06\\uFF21-\\uFF3A\\uFF41-\\uFF5A\";\n    lunr.fr.trimmer = lunr.trimmerSupport.generateTrimmer(lunr.fr.wordCharacters);\n\n    lunr.Pipeline.registerFunction(lunr.fr.trimmer, 'trimmer-fr');\n\n    /* lunr stemmer function */\n    lunr.fr.stemmer = (function() {\n      /* create the wrapped stemmer object */\n      var Among = lunr.stemmerSupport.Among,\n        SnowballProgram = lunr.stemmerSupport.SnowballProgram,\n        st = new function FrenchStemmer() {\n          var a_0 = [new Among(\"col\", -1, -1), new Among(\"par\", -1, -1),\n              new Among(\"tap\", -1, -1)\n            ],\n            a_1 = [new Among(\"\", -1, 4),\n              new Among(\"I\", 0, 1), new Among(\"U\", 0, 2), new Among(\"Y\", 0, 3)\n            ],\n            a_2 = [\n              new Among(\"iqU\", -1, 3), new Among(\"abl\", -1, 3),\n              new Among(\"I\\u00E8r\", -1, 4), new Among(\"i\\u00E8r\", -1, 4),\n              new Among(\"eus\", -1, 2), new Among(\"iv\", -1, 1)\n            ],\n            a_3 = [\n              new Among(\"ic\", -1, 2), new Among(\"abil\", -1, 1),\n              new Among(\"iv\", -1, 3)\n            ],\n            a_4 = [new Among(\"iqUe\", -1, 1),\n              new Among(\"atrice\", -1, 2), new Among(\"ance\", -1, 1),\n              new Among(\"ence\", -1, 5), new Among(\"logie\", -1, 3),\n              new Among(\"able\", -1, 1), new Among(\"isme\", -1, 1),\n              new Among(\"euse\", -1, 11), new Among(\"iste\", -1, 1),\n              new Among(\"ive\", -1, 8), new Among(\"if\", -1, 8),\n              new Among(\"usion\", -1, 4), new Among(\"ation\", -1, 2),\n              new Among(\"ution\", -1, 4), new Among(\"ateur\", -1, 2),\n              new Among(\"iqUes\", -1, 1), new Among(\"atrices\", -1, 2),\n              new Among(\"ances\", -1, 1), new Among(\"ences\", -1, 5),\n              new Among(\"logies\", -1, 3), new Among(\"ables\", -1, 1),\n              new Among(\"ismes\", -1, 1), new Among(\"euses\", -1, 11),\n              new Among(\"istes\", -1, 1), new Among(\"ives\", -1, 8),\n              new Among(\"ifs\", -1, 8), new Among(\"usions\", -1, 4),\n              new Among(\"ations\", -1, 2), new Among(\"utions\", -1, 4),\n              new Among(\"ateurs\", -1, 2), new Among(\"ments\", -1, 15),\n              new Among(\"ements\", 30, 6), new Among(\"issements\", 31, 12),\n              new Among(\"it\\u00E9s\", -1, 7), new Among(\"ment\", -1, 15),\n              new Among(\"ement\", 34, 6), new Among(\"issement\", 35, 12),\n              new Among(\"amment\", 34, 13), new Among(\"emment\", 34, 14),\n              new Among(\"aux\", -1, 10), new Among(\"eaux\", 39, 9),\n              new Among(\"eux\", -1, 1), new Among(\"it\\u00E9\", -1, 7)\n            ],\n            a_5 = [\n              new Among(\"ira\", -1, 1), new Among(\"ie\", -1, 1),\n              new Among(\"isse\", -1, 1), new Among(\"issante\", -1, 1),\n              new Among(\"i\", -1, 1), new Among(\"irai\", 4, 1),\n              new Among(\"ir\", -1, 1), new Among(\"iras\", -1, 1),\n              new Among(\"ies\", -1, 1), new Among(\"\\u00EEmes\", -1, 1),\n              new Among(\"isses\", -1, 1), new Among(\"issantes\", -1, 1),\n              new Among(\"\\u00EEtes\", -1, 1), new Among(\"is\", -1, 1),\n              new Among(\"irais\", 13, 1), new Among(\"issais\", 13, 1),\n              new Among(\"irions\", -1, 1), new Among(\"issions\", -1, 1),\n              new Among(\"irons\", -1, 1), new Among(\"issons\", -1, 1),\n              new Among(\"issants\", -1, 1), new Among(\"it\", -1, 1),\n              new Among(\"irait\", 21, 1), new Among(\"issait\", 21, 1),\n              new Among(\"issant\", -1, 1), new Among(\"iraIent\", -1, 1),\n              new Among(\"issaIent\", -1, 1), new Among(\"irent\", -1, 1),\n              new Among(\"issent\", -1, 1), new Among(\"iront\", -1, 1),\n              new Among(\"\\u00EEt\", -1, 1), new Among(\"iriez\", -1, 1),\n              new Among(\"issiez\", -1, 1), new Among(\"irez\", -1, 1),\n              new Among(\"issez\", -1, 1)\n            ],\n            a_6 = [new Among(\"a\", -1, 3),\n              new Among(\"era\", 0, 2), new Among(\"asse\", -1, 3),\n              new Among(\"ante\", -1, 3), new Among(\"\\u00E9e\", -1, 2),\n              new Among(\"ai\", -1, 3), new Among(\"erai\", 5, 2),\n              new Among(\"er\", -1, 2), new Among(\"as\", -1, 3),\n              new Among(\"eras\", 8, 2), new Among(\"\\u00E2mes\", -1, 3),\n              new Among(\"asses\", -1, 3), new Among(\"antes\", -1, 3),\n              new Among(\"\\u00E2tes\", -1, 3), new Among(\"\\u00E9es\", -1, 2),\n              new Among(\"ais\", -1, 3), new Among(\"erais\", 15, 2),\n              new Among(\"ions\", -1, 1), new Among(\"erions\", 17, 2),\n              new Among(\"assions\", 17, 3), new Among(\"erons\", -1, 2),\n              new Among(\"ants\", -1, 3), new Among(\"\\u00E9s\", -1, 2),\n              new Among(\"ait\", -1, 3), new Among(\"erait\", 23, 2),\n              new Among(\"ant\", -1, 3), new Among(\"aIent\", -1, 3),\n              new Among(\"eraIent\", 26, 2), new Among(\"\\u00E8rent\", -1, 2),\n              new Among(\"assent\", -1, 3), new Among(\"eront\", -1, 2),\n              new Among(\"\\u00E2t\", -1, 3), new Among(\"ez\", -1, 2),\n              new Among(\"iez\", 32, 2), new Among(\"eriez\", 33, 2),\n              new Among(\"assiez\", 33, 3), new Among(\"erez\", 32, 2),\n              new Among(\"\\u00E9\", -1, 2)\n            ],\n            a_7 = [new Among(\"e\", -1, 3),\n              new Among(\"I\\u00E8re\", 0, 2), new Among(\"i\\u00E8re\", 0, 2),\n              new Among(\"ion\", -1, 1), new Among(\"Ier\", -1, 2),\n              new Among(\"ier\", -1, 2), new Among(\"\\u00EB\", -1, 4)\n            ],\n            a_8 = [\n              new Among(\"ell\", -1, -1), new Among(\"eill\", -1, -1),\n              new Among(\"enn\", -1, -1), new Among(\"onn\", -1, -1),\n              new Among(\"ett\", -1, -1)\n            ],\n            g_v = [17, 65, 16, 1, 0, 0, 0, 0, 0, 0,\n              0, 0, 0, 0, 0, 128, 130, 103, 8, 5\n            ],\n            g_keep_with_s = [1, 65, 20, 0,\n              0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 128\n            ],\n            I_p2, I_p1, I_pV, sbp = new SnowballProgram();\n          this.setCurrent = function(word) {\n            sbp.setCurrent(word);\n          };\n          this.getCurrent = function() {\n            return sbp.getCurrent();\n          };\n\n          function habr1(c1, c2, v_1) {\n            if (sbp.eq_s(1, c1)) {\n              sbp.ket = sbp.cursor;\n              if (sbp.in_grouping(g_v, 97, 251)) {\n                sbp.slice_from(c2);\n                sbp.cursor = v_1;\n                return true;\n              }\n            }\n            return false;\n          }\n\n          function habr2(c1, c2, v_1) {\n            if (sbp.eq_s(1, c1)) {\n              sbp.ket = sbp.cursor;\n              sbp.slice_from(c2);\n              sbp.cursor = v_1;\n              return true;\n            }\n            return false;\n          }\n\n          function r_prelude() {\n            var v_1, v_2;\n            while (true) {\n              v_1 = sbp.cursor;\n              if (sbp.in_grouping(g_v, 97, 251)) {\n                sbp.bra = sbp.cursor;\n                v_2 = sbp.cursor;\n                if (habr1(\"u\", \"U\", v_1))\n                  continue;\n                sbp.cursor = v_2;\n                if (habr1(\"i\", \"I\", v_1))\n                  continue;\n                sbp.cursor = v_2;\n                if (habr2(\"y\", \"Y\", v_1))\n                  continue;\n              }\n              sbp.cursor = v_1;\n              sbp.bra = v_1;\n              if (!habr1(\"y\", \"Y\", v_1)) {\n                sbp.cursor = v_1;\n                if (sbp.eq_s(1, \"q\")) {\n                  sbp.bra = sbp.cursor;\n                  if (habr2(\"u\", \"U\", v_1))\n                    continue;\n                }\n                sbp.cursor = v_1;\n                if (v_1 >= sbp.limit)\n                  return;\n                sbp.cursor++;\n              }\n            }\n          }\n\n          function habr3() {\n            while (!sbp.in_grouping(g_v, 97, 251)) {\n              if (sbp.cursor >= sbp.limit)\n                return true;\n              sbp.cursor++;\n            }\n            while (!sbp.out_grouping(g_v, 97, 251)) {\n              if (sbp.cursor >= sbp.limit)\n                return true;\n              sbp.cursor++;\n            }\n            return false;\n          }\n\n          function r_mark_regions() {\n            var v_1 = sbp.cursor;\n            I_pV = sbp.limit;\n            I_p1 = I_pV;\n            I_p2 = I_pV;\n            if (sbp.in_grouping(g_v, 97, 251) && sbp.in_grouping(g_v, 97, 251) && sbp.cursor < sbp.limit)\n              sbp.cursor++;\n            else {\n              sbp.cursor = v_1;\n              if (!sbp.find_among(a_0, 3)) {\n                sbp.cursor = v_1;\n                do {\n                  if (sbp.cursor >= sbp.limit) {\n                    sbp.cursor = I_pV;\n                    break;\n                  }\n                  sbp.cursor++;\n                } while (!sbp.in_grouping(g_v, 97, 251));\n              }\n            }\n            I_pV = sbp.cursor;\n            sbp.cursor = v_1;\n            if (!habr3()) {\n              I_p1 = sbp.cursor;\n              if (!habr3())\n                I_p2 = sbp.cursor;\n            }\n          }\n\n          function r_postlude() {\n            var among_var, v_1;\n            while (true) {\n              v_1 = sbp.cursor;\n              sbp.bra = v_1;\n              among_var = sbp.find_among(a_1, 4);\n              if (!among_var)\n                break;\n              sbp.ket = sbp.cursor;\n              switch (among_var) {\n                case 1:\n                  sbp.slice_from(\"i\");\n                  break;\n                case 2:\n                  sbp.slice_from(\"u\");\n                  break;\n                case 3:\n                  sbp.slice_from(\"y\");\n                  break;\n                case 4:\n                  if (sbp.cursor >= sbp.limit)\n                    return;\n                  sbp.cursor++;\n                  break;\n              }\n            }\n          }\n\n          function r_RV() {\n            return I_pV <= sbp.cursor;\n          }\n\n          function r_R1() {\n            return I_p1 <= sbp.cursor;\n          }\n\n          function r_R2() {\n            return I_p2 <= sbp.cursor;\n          }\n\n          function r_standard_suffix() {\n            var among_var, v_1;\n            sbp.ket = sbp.cursor;\n            among_var = sbp.find_among_b(a_4, 43);\n            if (among_var) {\n              sbp.bra = sbp.cursor;\n              switch (among_var) {\n                case 1:\n                  if (!r_R2())\n                    return false;\n                  sbp.slice_del();\n                  break;\n                case 2:\n                  if (!r_R2())\n                    return false;\n                  sbp.slice_del();\n                  sbp.ket = sbp.cursor;\n                  if (sbp.eq_s_b(2, \"ic\")) {\n                    sbp.bra = sbp.cursor;\n                    if (!r_R2())\n                      sbp.slice_from(\"iqU\");\n                    else\n                      sbp.slice_del();\n                  }\n                  break;\n                case 3:\n                  if (!r_R2())\n                    return false;\n                  sbp.slice_from(\"log\");\n                  break;\n                case 4:\n                  if (!r_R2())\n                    return false;\n                  sbp.slice_from(\"u\");\n                  break;\n                case 5:\n                  if (!r_R2())\n                    return false;\n                  sbp.slice_from(\"ent\");\n                  break;\n                case 6:\n                  if (!r_RV())\n                    return false;\n                  sbp.slice_del();\n                  sbp.ket = sbp.cursor;\n                  among_var = sbp.find_among_b(a_2, 6);\n                  if (among_var) {\n                    sbp.bra = sbp.cursor;\n                    switch (among_var) {\n                      case 1:\n                        if (r_R2()) {\n                          sbp.slice_del();\n                          sbp.ket = sbp.cursor;\n                          if (sbp.eq_s_b(2, \"at\")) {\n                            sbp.bra = sbp.cursor;\n                            if (r_R2())\n                              sbp.slice_del();\n                          }\n                        }\n                        break;\n                      case 2:\n                        if (r_R2())\n                          sbp.slice_del();\n                        else if (r_R1())\n                          sbp.slice_from(\"eux\");\n                        break;\n                      case 3:\n                        if (r_R2())\n                          sbp.slice_del();\n                        break;\n                      case 4:\n                        if (r_RV())\n                          sbp.slice_from(\"i\");\n                        break;\n                    }\n                  }\n                  break;\n                case 7:\n                  if (!r_R2())\n                    return false;\n                  sbp.slice_del();\n                  sbp.ket = sbp.cursor;\n                  among_var = sbp.find_among_b(a_3, 3);\n                  if (among_var) {\n                    sbp.bra = sbp.cursor;\n                    switch (among_var) {\n                      case 1:\n                        if (r_R2())\n                          sbp.slice_del();\n                        else\n                          sbp.slice_from(\"abl\");\n                        break;\n                      case 2:\n                        if (r_R2())\n                          sbp.slice_del();\n                        else\n                          sbp.slice_from(\"iqU\");\n                        break;\n                      case 3:\n                        if (r_R2())\n                          sbp.slice_del();\n                        break;\n                    }\n                  }\n                  break;\n                case 8:\n                  if (!r_R2())\n                    return false;\n                  sbp.slice_del();\n                  sbp.ket = sbp.cursor;\n                  if (sbp.eq_s_b(2, \"at\")) {\n                    sbp.bra = sbp.cursor;\n                    if (r_R2()) {\n                      sbp.slice_del();\n                      sbp.ket = sbp.cursor;\n                      if (sbp.eq_s_b(2, \"ic\")) {\n                        sbp.bra = sbp.cursor;\n                        if (r_R2())\n                          sbp.slice_del();\n                        else\n                          sbp.slice_from(\"iqU\");\n                        break;\n                      }\n                    }\n                  }\n                  break;\n                case 9:\n                  sbp.slice_from(\"eau\");\n                  break;\n                case 10:\n                  if (!r_R1())\n                    return false;\n                  sbp.slice_from(\"al\");\n                  break;\n                case 11:\n                  if (r_R2())\n                    sbp.slice_del();\n                  else if (!r_R1())\n                    return false;\n                  else\n                    sbp.slice_from(\"eux\");\n                  break;\n                case 12:\n                  if (!r_R1() || !sbp.out_grouping_b(g_v, 97, 251))\n                    return false;\n                  sbp.slice_del();\n                  break;\n                case 13:\n                  if (r_RV())\n                    sbp.slice_from(\"ant\");\n                  return false;\n                case 14:\n                  if (r_RV())\n                    sbp.slice_from(\"ent\");\n                  return false;\n                case 15:\n                  v_1 = sbp.limit - sbp.cursor;\n                  if (sbp.in_grouping_b(g_v, 97, 251) && r_RV()) {\n                    sbp.cursor = sbp.limit - v_1;\n                    sbp.slice_del();\n                  }\n                  return false;\n              }\n              return true;\n            }\n            return false;\n          }\n\n          function r_i_verb_suffix() {\n            var among_var, v_1;\n            if (sbp.cursor < I_pV)\n              return false;\n            v_1 = sbp.limit_backward;\n            sbp.limit_backward = I_pV;\n            sbp.ket = sbp.cursor;\n            among_var = sbp.find_among_b(a_5, 35);\n            if (!among_var) {\n              sbp.limit_backward = v_1;\n              return false;\n            }\n            sbp.bra = sbp.cursor;\n            if (among_var == 1) {\n              if (!sbp.out_grouping_b(g_v, 97, 251)) {\n                sbp.limit_backward = v_1;\n                return false;\n              }\n              sbp.slice_del();\n            }\n            sbp.limit_backward = v_1;\n            return true;\n          }\n\n          function r_verb_suffix() {\n            var among_var, v_2, v_3;\n            if (sbp.cursor < I_pV)\n              return false;\n            v_2 = sbp.limit_backward;\n            sbp.limit_backward = I_pV;\n            sbp.ket = sbp.cursor;\n            among_var = sbp.find_among_b(a_6, 38);\n            if (!among_var) {\n              sbp.limit_backward = v_2;\n              return false;\n            }\n            sbp.bra = sbp.cursor;\n            switch (among_var) {\n              case 1:\n                if (!r_R2()) {\n                  sbp.limit_backward = v_2;\n                  return false;\n                }\n                sbp.slice_del();\n                break;\n              case 2:\n                sbp.slice_del();\n                break;\n              case 3:\n                sbp.slice_del();\n                v_3 = sbp.limit - sbp.cursor;\n                sbp.ket = sbp.cursor;\n                if (sbp.eq_s_b(1, \"e\")) {\n                  sbp.bra = sbp.cursor;\n                  sbp.slice_del();\n                } else\n                  sbp.cursor = sbp.limit - v_3;\n                break;\n            }\n            sbp.limit_backward = v_2;\n            return true;\n          }\n\n          function r_residual_suffix() {\n            var among_var, v_1 = sbp.limit - sbp.cursor,\n              v_2, v_4, v_5;\n            sbp.ket = sbp.cursor;\n            if (sbp.eq_s_b(1, \"s\")) {\n              sbp.bra = sbp.cursor;\n              v_2 = sbp.limit - sbp.cursor;\n              if (sbp.out_grouping_b(g_keep_with_s, 97, 232)) {\n                sbp.cursor = sbp.limit - v_2;\n                sbp.slice_del();\n              } else\n                sbp.cursor = sbp.limit - v_1;\n            } else\n              sbp.cursor = sbp.limit - v_1;\n            if (sbp.cursor >= I_pV) {\n              v_4 = sbp.limit_backward;\n              sbp.limit_backward = I_pV;\n              sbp.ket = sbp.cursor;\n              among_var = sbp.find_among_b(a_7, 7);\n              if (among_var) {\n                sbp.bra = sbp.cursor;\n                switch (among_var) {\n                  case 1:\n                    if (r_R2()) {\n                      v_5 = sbp.limit - sbp.cursor;\n                      if (!sbp.eq_s_b(1, \"s\")) {\n                        sbp.cursor = sbp.limit - v_5;\n                        if (!sbp.eq_s_b(1, \"t\"))\n                          break;\n                      }\n                      sbp.slice_del();\n                    }\n                    break;\n                  case 2:\n                    sbp.slice_from(\"i\");\n                    break;\n                  case 3:\n                    sbp.slice_del();\n                    break;\n                  case 4:\n                    if (sbp.eq_s_b(2, \"gu\"))\n                      sbp.slice_del();\n                    break;\n                }\n              }\n              sbp.limit_backward = v_4;\n            }\n          }\n\n          function r_un_double() {\n            var v_1 = sbp.limit - sbp.cursor;\n            if (sbp.find_among_b(a_8, 5)) {\n              sbp.cursor = sbp.limit - v_1;\n              sbp.ket = sbp.cursor;\n              if (sbp.cursor > sbp.limit_backward) {\n                sbp.cursor--;\n                sbp.bra = sbp.cursor;\n                sbp.slice_del();\n              }\n            }\n          }\n\n          function r_un_accent() {\n            var v_1, v_2 = 1;\n            while (sbp.out_grouping_b(g_v, 97, 251))\n              v_2--;\n            if (v_2 <= 0) {\n              sbp.ket = sbp.cursor;\n              v_1 = sbp.limit - sbp.cursor;\n              if (!sbp.eq_s_b(1, \"\\u00E9\")) {\n                sbp.cursor = sbp.limit - v_1;\n                if (!sbp.eq_s_b(1, \"\\u00E8\"))\n                  return;\n              }\n              sbp.bra = sbp.cursor;\n              sbp.slice_from(\"e\");\n            }\n          }\n\n          function habr5() {\n            if (!r_standard_suffix()) {\n              sbp.cursor = sbp.limit;\n              if (!r_i_verb_suffix()) {\n                sbp.cursor = sbp.limit;\n                if (!r_verb_suffix()) {\n                  sbp.cursor = sbp.limit;\n                  r_residual_suffix();\n                  return;\n                }\n              }\n            }\n            sbp.cursor = sbp.limit;\n            sbp.ket = sbp.cursor;\n            if (sbp.eq_s_b(1, \"Y\")) {\n              sbp.bra = sbp.cursor;\n              sbp.slice_from(\"i\");\n            } else {\n              sbp.cursor = sbp.limit;\n              if (sbp.eq_s_b(1, \"\\u00E7\")) {\n                sbp.bra = sbp.cursor;\n                sbp.slice_from(\"c\");\n              }\n            }\n          }\n          this.stem = function() {\n            var v_1 = sbp.cursor;\n            r_prelude();\n            sbp.cursor = v_1;\n            r_mark_regions();\n            sbp.limit_backward = v_1;\n            sbp.cursor = sbp.limit;\n            habr5();\n            sbp.cursor = sbp.limit;\n            r_un_double();\n            sbp.cursor = sbp.limit;\n            r_un_accent();\n            sbp.cursor = sbp.limit_backward;\n            r_postlude();\n            return true;\n          }\n        };\n\n      /* and return a function that stems a word for the current locale */\n      return function(word) {\n        st.setCurrent(word);\n        st.stem();\n        return st.getCurrent();\n      }\n    })();\n\n    lunr.Pipeline.registerFunction(lunr.fr.stemmer, 'stemmer-fr');\n\n    /* stop word filter function */\n    lunr.fr.stopWordFilter = function(token) {\n      if (lunr.fr.stopWordFilter.stopWords.indexOf(token) === -1) {\n        return token;\n      }\n    };\n\n    lunr.fr.stopWordFilter.stopWords = new lunr.SortedSet();\n    lunr.fr.stopWordFilter.stopWords.length = 164;\n\n    // The space at the beginning is crucial: It marks the empty string\n    // as a stop word. lunr.js crashes during search when documents\n    // processed by the pipeline still contain the empty string.\n    lunr.fr.stopWordFilter.stopWords.elements = ' ai aie aient aies ait as au aura aurai auraient aurais aurait auras aurez auriez aurions aurons auront aux avaient avais avait avec avez aviez avions avons ayant ayez ayons c ce ceci celà ces cet cette d dans de des du elle en es est et eu eue eues eurent eus eusse eussent eusses eussiez eussions eut eux eûmes eût eûtes furent fus fusse fussent fusses fussiez fussions fut fûmes fût fûtes ici il ils j je l la le les leur leurs lui m ma mais me mes moi mon même n ne nos notre nous on ont ou par pas pour qu que quel quelle quelles quels qui s sa sans se sera serai seraient serais serait seras serez seriez serions serons seront ses soi soient sois soit sommes son sont soyez soyons suis sur t ta te tes toi ton tu un une vos votre vous y à étaient étais était étant étiez étions été étée étées étés êtes'.split(' ');\n\n    lunr.Pipeline.registerFunction(lunr.fr.stopWordFilter, 'stopWordFilter-fr');\n  };\n}))\n"
  },
  {
    "path": "static/js/lunr/lunr.hu.js",
    "content": "/*!\n * Lunr languages, `Hungarian` language\n * https://github.com/MihaiValentin/lunr-languages\n *\n * Copyright 2014, Mihai Valentin\n * http://www.mozilla.org/MPL/\n */\n/*!\n * based on\n * Snowball JavaScript Library v0.3\n * http://code.google.com/p/urim/\n * http://snowball.tartarus.org/\n *\n * Copyright 2010, Oleg Mazko\n * http://www.mozilla.org/MPL/\n */\n\n/**\n * export the module via AMD, CommonJS or as a browser global\n * Export code from https://github.com/umdjs/umd/blob/master/returnExports.js\n */\n;\n(function(root, factory) {\n  if (typeof define === 'function' && define.amd) {\n    // AMD. Register as an anonymous module.\n    define(factory)\n  } else if (typeof exports === 'object') {\n    /**\n     * Node. Does not work with strict CommonJS, but\n     * only CommonJS-like environments that support module.exports,\n     * like Node.\n     */\n    module.exports = factory()\n  } else {\n    // Browser globals (root is window)\n    factory()(root.lunr);\n  }\n}(this, function() {\n  /**\n   * Just return a value to define the module export.\n   * This example returns an object, but the module\n   * can return a function as the exported value.\n   */\n  return function(lunr) {\n    /* throw error if lunr is not yet included */\n    if ('undefined' === typeof lunr) {\n      throw new Error('Lunr is not present. Please include / require Lunr before this script.');\n    }\n\n    /* throw error if lunr stemmer support is not yet included */\n    if ('undefined' === typeof lunr.stemmerSupport) {\n      throw new Error('Lunr stemmer support is not present. Please include / require Lunr stemmer support before this script.');\n    }\n\n    /* register specific locale function */\n    lunr.hu = function() {\n      this.pipeline.reset();\n      this.pipeline.add(\n        lunr.hu.trimmer,\n        lunr.hu.stopWordFilter,\n        lunr.hu.stemmer\n      );\n    };\n\n    /* lunr trimmer function */\n    lunr.hu.wordCharacters = \"A-Za-z\\xAA\\xBA\\xC0-\\xD6\\xD8-\\xF6\\xF8-\\u02B8\\u02E0-\\u02E4\\u1D00-\\u1D25\\u1D2C-\\u1D5C\\u1D62-\\u1D65\\u1D6B-\\u1D77\\u1D79-\\u1DBE\\u1E00-\\u1EFF\\u2071\\u207F\\u2090-\\u209C\\u212A\\u212B\\u2132\\u214E\\u2160-\\u2188\\u2C60-\\u2C7F\\uA722-\\uA787\\uA78B-\\uA7AD\\uA7B0-\\uA7B7\\uA7F7-\\uA7FF\\uAB30-\\uAB5A\\uAB5C-\\uAB64\\uFB00-\\uFB06\\uFF21-\\uFF3A\\uFF41-\\uFF5A\";\n    lunr.hu.trimmer = lunr.trimmerSupport.generateTrimmer(lunr.hu.wordCharacters);\n\n    lunr.Pipeline.registerFunction(lunr.hu.trimmer, 'trimmer-hu');\n\n    /* lunr stemmer function */\n    lunr.hu.stemmer = (function() {\n      /* create the wrapped stemmer object */\n      var Among = lunr.stemmerSupport.Among,\n        SnowballProgram = lunr.stemmerSupport.SnowballProgram,\n        st = new function HungarianStemmer() {\n          var a_0 = [new Among(\"cs\", -1, -1), new Among(\"dzs\", -1, -1),\n              new Among(\"gy\", -1, -1), new Among(\"ly\", -1, -1),\n              new Among(\"ny\", -1, -1), new Among(\"sz\", -1, -1),\n              new Among(\"ty\", -1, -1), new Among(\"zs\", -1, -1)\n            ],\n            a_1 = [\n              new Among(\"\\u00E1\", -1, 1), new Among(\"\\u00E9\", -1, 2)\n            ],\n            a_2 = [\n              new Among(\"bb\", -1, -1), new Among(\"cc\", -1, -1),\n              new Among(\"dd\", -1, -1), new Among(\"ff\", -1, -1),\n              new Among(\"gg\", -1, -1), new Among(\"jj\", -1, -1),\n              new Among(\"kk\", -1, -1), new Among(\"ll\", -1, -1),\n              new Among(\"mm\", -1, -1), new Among(\"nn\", -1, -1),\n              new Among(\"pp\", -1, -1), new Among(\"rr\", -1, -1),\n              new Among(\"ccs\", -1, -1), new Among(\"ss\", -1, -1),\n              new Among(\"zzs\", -1, -1), new Among(\"tt\", -1, -1),\n              new Among(\"vv\", -1, -1), new Among(\"ggy\", -1, -1),\n              new Among(\"lly\", -1, -1), new Among(\"nny\", -1, -1),\n              new Among(\"tty\", -1, -1), new Among(\"ssz\", -1, -1),\n              new Among(\"zz\", -1, -1)\n            ],\n            a_3 = [new Among(\"al\", -1, 1),\n              new Among(\"el\", -1, 2)\n            ],\n            a_4 = [new Among(\"ba\", -1, -1),\n              new Among(\"ra\", -1, -1), new Among(\"be\", -1, -1),\n              new Among(\"re\", -1, -1), new Among(\"ig\", -1, -1),\n              new Among(\"nak\", -1, -1), new Among(\"nek\", -1, -1),\n              new Among(\"val\", -1, -1), new Among(\"vel\", -1, -1),\n              new Among(\"ul\", -1, -1), new Among(\"n\\u00E1l\", -1, -1),\n              new Among(\"n\\u00E9l\", -1, -1), new Among(\"b\\u00F3l\", -1, -1),\n              new Among(\"r\\u00F3l\", -1, -1), new Among(\"t\\u00F3l\", -1, -1),\n              new Among(\"b\\u00F5l\", -1, -1), new Among(\"r\\u00F5l\", -1, -1),\n              new Among(\"t\\u00F5l\", -1, -1), new Among(\"\\u00FCl\", -1, -1),\n              new Among(\"n\", -1, -1), new Among(\"an\", 19, -1),\n              new Among(\"ban\", 20, -1), new Among(\"en\", 19, -1),\n              new Among(\"ben\", 22, -1), new Among(\"k\\u00E9ppen\", 22, -1),\n              new Among(\"on\", 19, -1), new Among(\"\\u00F6n\", 19, -1),\n              new Among(\"k\\u00E9pp\", -1, -1), new Among(\"kor\", -1, -1),\n              new Among(\"t\", -1, -1), new Among(\"at\", 29, -1),\n              new Among(\"et\", 29, -1), new Among(\"k\\u00E9nt\", 29, -1),\n              new Among(\"ank\\u00E9nt\", 32, -1), new Among(\"enk\\u00E9nt\", 32, -1),\n              new Among(\"onk\\u00E9nt\", 32, -1), new Among(\"ot\", 29, -1),\n              new Among(\"\\u00E9rt\", 29, -1), new Among(\"\\u00F6t\", 29, -1),\n              new Among(\"hez\", -1, -1), new Among(\"hoz\", -1, -1),\n              new Among(\"h\\u00F6z\", -1, -1), new Among(\"v\\u00E1\", -1, -1),\n              new Among(\"v\\u00E9\", -1, -1)\n            ],\n            a_5 = [new Among(\"\\u00E1n\", -1, 2),\n              new Among(\"\\u00E9n\", -1, 1), new Among(\"\\u00E1nk\\u00E9nt\", -1, 3)\n            ],\n            a_6 = [\n              new Among(\"stul\", -1, 2), new Among(\"astul\", 0, 1),\n              new Among(\"\\u00E1stul\", 0, 3), new Among(\"st\\u00FCl\", -1, 2),\n              new Among(\"est\\u00FCl\", 3, 1), new Among(\"\\u00E9st\\u00FCl\", 3, 4)\n            ],\n            a_7 = [\n              new Among(\"\\u00E1\", -1, 1), new Among(\"\\u00E9\", -1, 2)\n            ],\n            a_8 = [\n              new Among(\"k\", -1, 7), new Among(\"ak\", 0, 4),\n              new Among(\"ek\", 0, 6), new Among(\"ok\", 0, 5),\n              new Among(\"\\u00E1k\", 0, 1), new Among(\"\\u00E9k\", 0, 2),\n              new Among(\"\\u00F6k\", 0, 3)\n            ],\n            a_9 = [new Among(\"\\u00E9i\", -1, 7),\n              new Among(\"\\u00E1\\u00E9i\", 0, 6), new Among(\"\\u00E9\\u00E9i\", 0, 5),\n              new Among(\"\\u00E9\", -1, 9), new Among(\"k\\u00E9\", 3, 4),\n              new Among(\"ak\\u00E9\", 4, 1), new Among(\"ek\\u00E9\", 4, 1),\n              new Among(\"ok\\u00E9\", 4, 1), new Among(\"\\u00E1k\\u00E9\", 4, 3),\n              new Among(\"\\u00E9k\\u00E9\", 4, 2), new Among(\"\\u00F6k\\u00E9\", 4, 1),\n              new Among(\"\\u00E9\\u00E9\", 3, 8)\n            ],\n            a_10 = [new Among(\"a\", -1, 18),\n              new Among(\"ja\", 0, 17), new Among(\"d\", -1, 16),\n              new Among(\"ad\", 2, 13), new Among(\"ed\", 2, 13),\n              new Among(\"od\", 2, 13), new Among(\"\\u00E1d\", 2, 14),\n              new Among(\"\\u00E9d\", 2, 15), new Among(\"\\u00F6d\", 2, 13),\n              new Among(\"e\", -1, 18), new Among(\"je\", 9, 17),\n              new Among(\"nk\", -1, 4), new Among(\"unk\", 11, 1),\n              new Among(\"\\u00E1nk\", 11, 2), new Among(\"\\u00E9nk\", 11, 3),\n              new Among(\"\\u00FCnk\", 11, 1), new Among(\"uk\", -1, 8),\n              new Among(\"juk\", 16, 7), new Among(\"\\u00E1juk\", 17, 5),\n              new Among(\"\\u00FCk\", -1, 8), new Among(\"j\\u00FCk\", 19, 7),\n              new Among(\"\\u00E9j\\u00FCk\", 20, 6), new Among(\"m\", -1, 12),\n              new Among(\"am\", 22, 9), new Among(\"em\", 22, 9),\n              new Among(\"om\", 22, 9), new Among(\"\\u00E1m\", 22, 10),\n              new Among(\"\\u00E9m\", 22, 11), new Among(\"o\", -1, 18),\n              new Among(\"\\u00E1\", -1, 19), new Among(\"\\u00E9\", -1, 20)\n            ],\n            a_11 = [\n              new Among(\"id\", -1, 10), new Among(\"aid\", 0, 9),\n              new Among(\"jaid\", 1, 6), new Among(\"eid\", 0, 9),\n              new Among(\"jeid\", 3, 6), new Among(\"\\u00E1id\", 0, 7),\n              new Among(\"\\u00E9id\", 0, 8), new Among(\"i\", -1, 15),\n              new Among(\"ai\", 7, 14), new Among(\"jai\", 8, 11),\n              new Among(\"ei\", 7, 14), new Among(\"jei\", 10, 11),\n              new Among(\"\\u00E1i\", 7, 12), new Among(\"\\u00E9i\", 7, 13),\n              new Among(\"itek\", -1, 24), new Among(\"eitek\", 14, 21),\n              new Among(\"jeitek\", 15, 20), new Among(\"\\u00E9itek\", 14, 23),\n              new Among(\"ik\", -1, 29), new Among(\"aik\", 18, 26),\n              new Among(\"jaik\", 19, 25), new Among(\"eik\", 18, 26),\n              new Among(\"jeik\", 21, 25), new Among(\"\\u00E1ik\", 18, 27),\n              new Among(\"\\u00E9ik\", 18, 28), new Among(\"ink\", -1, 20),\n              new Among(\"aink\", 25, 17), new Among(\"jaink\", 26, 16),\n              new Among(\"eink\", 25, 17), new Among(\"jeink\", 28, 16),\n              new Among(\"\\u00E1ink\", 25, 18), new Among(\"\\u00E9ink\", 25, 19),\n              new Among(\"aitok\", -1, 21), new Among(\"jaitok\", 32, 20),\n              new Among(\"\\u00E1itok\", -1, 22), new Among(\"im\", -1, 5),\n              new Among(\"aim\", 35, 4), new Among(\"jaim\", 36, 1),\n              new Among(\"eim\", 35, 4), new Among(\"jeim\", 38, 1),\n              new Among(\"\\u00E1im\", 35, 2), new Among(\"\\u00E9im\", 35, 3)\n            ],\n            g_v = [\n              17, 65, 16, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 1, 17, 52, 14\n            ],\n            I_p1, sbp = new SnowballProgram();\n          this.setCurrent = function(word) {\n            sbp.setCurrent(word);\n          };\n          this.getCurrent = function() {\n            return sbp.getCurrent();\n          };\n\n          function r_mark_regions() {\n            var v_1 = sbp.cursor,\n              v_2;\n            I_p1 = sbp.limit;\n            if (sbp.in_grouping(g_v, 97, 252)) {\n              while (true) {\n                v_2 = sbp.cursor;\n                if (sbp.out_grouping(g_v, 97, 252)) {\n                  sbp.cursor = v_2;\n                  if (!sbp.find_among(a_0, 8)) {\n                    sbp.cursor = v_2;\n                    if (v_2 < sbp.limit)\n                      sbp.cursor++;\n                  }\n                  I_p1 = sbp.cursor;\n                  return;\n                }\n                sbp.cursor = v_2;\n                if (v_2 >= sbp.limit) {\n                  I_p1 = v_2;\n                  return;\n                }\n                sbp.cursor++;\n              }\n            }\n            sbp.cursor = v_1;\n            if (sbp.out_grouping(g_v, 97, 252)) {\n              while (!sbp.in_grouping(g_v, 97, 252)) {\n                if (sbp.cursor >= sbp.limit)\n                  return;\n                sbp.cursor++;\n              }\n              I_p1 = sbp.cursor;\n            }\n          }\n\n          function r_R1() {\n            return I_p1 <= sbp.cursor;\n          }\n\n          function r_v_ending() {\n            var among_var;\n            sbp.ket = sbp.cursor;\n            among_var = sbp.find_among_b(a_1, 2);\n            if (among_var) {\n              sbp.bra = sbp.cursor;\n              if (r_R1()) {\n                switch (among_var) {\n                  case 1:\n                    sbp.slice_from(\"a\");\n                    break;\n                  case 2:\n                    sbp.slice_from(\"e\");\n                    break;\n                }\n              }\n            }\n          }\n\n          function r_double() {\n            var v_1 = sbp.limit - sbp.cursor;\n            if (!sbp.find_among_b(a_2, 23))\n              return false;\n            sbp.cursor = sbp.limit - v_1;\n            return true;\n          }\n\n          function r_undouble() {\n            if (sbp.cursor > sbp.limit_backward) {\n              sbp.cursor--;\n              sbp.ket = sbp.cursor;\n              var c = sbp.cursor - 1;\n              if (sbp.limit_backward <= c && c <= sbp.limit) {\n                sbp.cursor = c;\n                sbp.bra = c;\n                sbp.slice_del();\n              }\n            }\n          }\n\n          function r_instrum() {\n            var among_var;\n            sbp.ket = sbp.cursor;\n            among_var = sbp.find_among_b(a_3, 2);\n            if (among_var) {\n              sbp.bra = sbp.cursor;\n              if (r_R1()) {\n                if (among_var == 1 || among_var == 2)\n                  if (!r_double())\n                    return;\n                sbp.slice_del();\n                r_undouble();\n              }\n            }\n          }\n\n          function r_case() {\n            sbp.ket = sbp.cursor;\n            if (sbp.find_among_b(a_4, 44)) {\n              sbp.bra = sbp.cursor;\n              if (r_R1()) {\n                sbp.slice_del();\n                r_v_ending();\n              }\n            }\n          }\n\n          function r_case_special() {\n            var among_var;\n            sbp.ket = sbp.cursor;\n            among_var = sbp.find_among_b(a_5, 3);\n            if (among_var) {\n              sbp.bra = sbp.cursor;\n              if (r_R1()) {\n                switch (among_var) {\n                  case 1:\n                    sbp.slice_from(\"e\");\n                    break;\n                  case 2:\n                  case 3:\n                    sbp.slice_from(\"a\");\n                    break;\n                }\n              }\n            }\n          }\n\n          function r_case_other() {\n            var among_var;\n            sbp.ket = sbp.cursor;\n            among_var = sbp.find_among_b(a_6, 6);\n            if (among_var) {\n              sbp.bra = sbp.cursor;\n              if (r_R1()) {\n                switch (among_var) {\n                  case 1:\n                  case 2:\n                    sbp.slice_del();\n                    break;\n                  case 3:\n                    sbp.slice_from(\"a\");\n                    break;\n                  case 4:\n                    sbp.slice_from(\"e\");\n                    break;\n                }\n              }\n            }\n          }\n\n          function r_factive() {\n            var among_var;\n            sbp.ket = sbp.cursor;\n            among_var = sbp.find_among_b(a_7, 2);\n            if (among_var) {\n              sbp.bra = sbp.cursor;\n              if (r_R1()) {\n                if (among_var == 1 || among_var == 2)\n                  if (!r_double())\n                    return;\n                sbp.slice_del();\n                r_undouble()\n              }\n            }\n          }\n\n          function r_plural() {\n            var among_var;\n            sbp.ket = sbp.cursor;\n            among_var = sbp.find_among_b(a_8, 7);\n            if (among_var) {\n              sbp.bra = sbp.cursor;\n              if (r_R1()) {\n                switch (among_var) {\n                  case 1:\n                    sbp.slice_from(\"a\");\n                    break;\n                  case 2:\n                    sbp.slice_from(\"e\");\n                    break;\n                  case 3:\n                  case 4:\n                  case 5:\n                  case 6:\n                  case 7:\n                    sbp.slice_del();\n                    break;\n                }\n              }\n            }\n          }\n\n          function r_owned() {\n            var among_var;\n            sbp.ket = sbp.cursor;\n            among_var = sbp.find_among_b(a_9, 12);\n            if (among_var) {\n              sbp.bra = sbp.cursor;\n              if (r_R1()) {\n                switch (among_var) {\n                  case 1:\n                  case 4:\n                  case 7:\n                  case 9:\n                    sbp.slice_del();\n                    break;\n                  case 2:\n                  case 5:\n                  case 8:\n                    sbp.slice_from(\"e\");\n                    break;\n                  case 3:\n                  case 6:\n                    sbp.slice_from(\"a\");\n                    break;\n                }\n              }\n            }\n          }\n\n          function r_sing_owner() {\n            var among_var;\n            sbp.ket = sbp.cursor;\n            among_var = sbp.find_among_b(a_10, 31);\n            if (among_var) {\n              sbp.bra = sbp.cursor;\n              if (r_R1()) {\n                switch (among_var) {\n                  case 1:\n                  case 4:\n                  case 7:\n                  case 8:\n                  case 9:\n                  case 12:\n                  case 13:\n                  case 16:\n                  case 17:\n                  case 18:\n                    sbp.slice_del();\n                    break;\n                  case 2:\n                  case 5:\n                  case 10:\n                  case 14:\n                  case 19:\n                    sbp.slice_from(\"a\");\n                    break;\n                  case 3:\n                  case 6:\n                  case 11:\n                  case 15:\n                  case 20:\n                    sbp.slice_from(\"e\");\n                    break;\n                }\n              }\n            }\n          }\n\n          function r_plur_owner() {\n            var among_var;\n            sbp.ket = sbp.cursor;\n            among_var = sbp.find_among_b(a_11, 42);\n            if (among_var) {\n              sbp.bra = sbp.cursor;\n              if (r_R1()) {\n                switch (among_var) {\n                  case 1:\n                  case 4:\n                  case 5:\n                  case 6:\n                  case 9:\n                  case 10:\n                  case 11:\n                  case 14:\n                  case 15:\n                  case 16:\n                  case 17:\n                  case 20:\n                  case 21:\n                  case 24:\n                  case 25:\n                  case 26:\n                  case 29:\n                    sbp.slice_del();\n                    break;\n                  case 2:\n                  case 7:\n                  case 12:\n                  case 18:\n                  case 22:\n                  case 27:\n                    sbp.slice_from(\"a\");\n                    break;\n                  case 3:\n                  case 8:\n                  case 13:\n                  case 19:\n                  case 23:\n                  case 28:\n                    sbp.slice_from(\"e\");\n                    break;\n                }\n              }\n            }\n          }\n          this.stem = function() {\n            var v_1 = sbp.cursor;\n            r_mark_regions();\n            sbp.limit_backward = v_1;\n            sbp.cursor = sbp.limit;\n            r_instrum();\n            sbp.cursor = sbp.limit;\n            r_case();\n            sbp.cursor = sbp.limit;\n            r_case_special();\n            sbp.cursor = sbp.limit;\n            r_case_other();\n            sbp.cursor = sbp.limit;\n            r_factive();\n            sbp.cursor = sbp.limit;\n            r_owned();\n            sbp.cursor = sbp.limit;\n            r_sing_owner();\n            sbp.cursor = sbp.limit;\n            r_plur_owner();\n            sbp.cursor = sbp.limit;\n            r_plural();\n            return true;\n          }\n        };\n\n      /* and return a function that stems a word for the current locale */\n      return function(word) {\n        st.setCurrent(word);\n        st.stem();\n        return st.getCurrent();\n      }\n    })();\n\n    lunr.Pipeline.registerFunction(lunr.hu.stemmer, 'stemmer-hu');\n\n    /* stop word filter function */\n    lunr.hu.stopWordFilter = function(token) {\n      if (lunr.hu.stopWordFilter.stopWords.indexOf(token) === -1) {\n        return token;\n      }\n    };\n\n    lunr.hu.stopWordFilter.stopWords = new lunr.SortedSet();\n    lunr.hu.stopWordFilter.stopWords.length = 200;\n\n    // The space at the beginning is crucial: It marks the empty string\n    // as a stop word. lunr.js crashes during search when documents\n    // processed by the pipeline still contain the empty string.\n    lunr.hu.stopWordFilter.stopWords.elements = ' a abban ahhoz ahogy ahol aki akik akkor alatt amely amelyek amelyekben amelyeket amelyet amelynek ami amikor amit amolyan amíg annak arra arról az azok azon azonban azt aztán azután azzal azért be belül benne bár cikk cikkek cikkeket csak de e ebben eddig egy egyes egyetlen egyik egyre egyéb egész ehhez ekkor el ellen elsõ elég elõ elõször elõtt emilyen ennek erre ez ezek ezen ezt ezzel ezért fel felé hanem hiszen hogy hogyan igen ill ill. illetve ilyen ilyenkor ismét ison itt jobban jó jól kell kellett keressünk keresztül ki kívül között közül legalább legyen lehet lehetett lenne lenni lesz lett maga magát majd majd meg mellett mely melyek mert mi mikor milyen minden mindenki mindent mindig mint mintha mit mivel miért most már más másik még míg nagy nagyobb nagyon ne nekem neki nem nincs néha néhány nélkül olyan ott pedig persze rá s saját sem semmi sok sokat sokkal szemben szerint szinte számára talán tehát teljes tovább továbbá több ugyanis utolsó után utána vagy vagyis vagyok valaki valami valamint való van vannak vele vissza viszont volna volt voltak voltam voltunk által általában át én éppen és így õ õk õket össze úgy új újabb újra'.split(' ');\n\n    lunr.Pipeline.registerFunction(lunr.hu.stopWordFilter, 'stopWordFilter-hu');\n  };\n}))\n"
  },
  {
    "path": "static/js/lunr/lunr.it.js",
    "content": "/*!\n * Lunr languages, `Italian` language\n * https://github.com/MihaiValentin/lunr-languages\n *\n * Copyright 2014, Mihai Valentin\n * http://www.mozilla.org/MPL/\n */\n/*!\n * based on\n * Snowball JavaScript Library v0.3\n * http://code.google.com/p/urim/\n * http://snowball.tartarus.org/\n *\n * Copyright 2010, Oleg Mazko\n * http://www.mozilla.org/MPL/\n */\n\n/**\n * export the module via AMD, CommonJS or as a browser global\n * Export code from https://github.com/umdjs/umd/blob/master/returnExports.js\n */\n;\n(function(root, factory) {\n  if (typeof define === 'function' && define.amd) {\n    // AMD. Register as an anonymous module.\n    define(factory)\n  } else if (typeof exports === 'object') {\n    /**\n     * Node. Does not work with strict CommonJS, but\n     * only CommonJS-like environments that support module.exports,\n     * like Node.\n     */\n    module.exports = factory()\n  } else {\n    // Browser globals (root is window)\n    factory()(root.lunr);\n  }\n}(this, function() {\n  /**\n   * Just return a value to define the module export.\n   * This example returns an object, but the module\n   * can return a function as the exported value.\n   */\n  return function(lunr) {\n    /* throw error if lunr is not yet included */\n    if ('undefined' === typeof lunr) {\n      throw new Error('Lunr is not present. Please include / require Lunr before this script.');\n    }\n\n    /* throw error if lunr stemmer support is not yet included */\n    if ('undefined' === typeof lunr.stemmerSupport) {\n      throw new Error('Lunr stemmer support is not present. Please include / require Lunr stemmer support before this script.');\n    }\n\n    /* register specific locale function */\n    lunr.it = function() {\n      this.pipeline.reset();\n      this.pipeline.add(\n        lunr.it.trimmer,\n        lunr.it.stopWordFilter,\n        lunr.it.stemmer\n      );\n    };\n\n    /* lunr trimmer function */\n    lunr.it.wordCharacters = \"A-Za-z\\xAA\\xBA\\xC0-\\xD6\\xD8-\\xF6\\xF8-\\u02B8\\u02E0-\\u02E4\\u1D00-\\u1D25\\u1D2C-\\u1D5C\\u1D62-\\u1D65\\u1D6B-\\u1D77\\u1D79-\\u1DBE\\u1E00-\\u1EFF\\u2071\\u207F\\u2090-\\u209C\\u212A\\u212B\\u2132\\u214E\\u2160-\\u2188\\u2C60-\\u2C7F\\uA722-\\uA787\\uA78B-\\uA7AD\\uA7B0-\\uA7B7\\uA7F7-\\uA7FF\\uAB30-\\uAB5A\\uAB5C-\\uAB64\\uFB00-\\uFB06\\uFF21-\\uFF3A\\uFF41-\\uFF5A\";\n    lunr.it.trimmer = lunr.trimmerSupport.generateTrimmer(lunr.it.wordCharacters);\n\n    lunr.Pipeline.registerFunction(lunr.it.trimmer, 'trimmer-it');\n\n    /* lunr stemmer function */\n    lunr.it.stemmer = (function() {\n      /* create the wrapped stemmer object */\n      var Among = lunr.stemmerSupport.Among,\n        SnowballProgram = lunr.stemmerSupport.SnowballProgram,\n        st = new function ItalianStemmer() {\n          var a_0 = [new Among(\"\", -1, 7), new Among(\"qu\", 0, 6),\n              new Among(\"\\u00E1\", 0, 1), new Among(\"\\u00E9\", 0, 2),\n              new Among(\"\\u00ED\", 0, 3), new Among(\"\\u00F3\", 0, 4),\n              new Among(\"\\u00FA\", 0, 5)\n            ],\n            a_1 = [new Among(\"\", -1, 3),\n              new Among(\"I\", 0, 1), new Among(\"U\", 0, 2)\n            ],\n            a_2 = [\n              new Among(\"la\", -1, -1), new Among(\"cela\", 0, -1),\n              new Among(\"gliela\", 0, -1), new Among(\"mela\", 0, -1),\n              new Among(\"tela\", 0, -1), new Among(\"vela\", 0, -1),\n              new Among(\"le\", -1, -1), new Among(\"cele\", 6, -1),\n              new Among(\"gliele\", 6, -1), new Among(\"mele\", 6, -1),\n              new Among(\"tele\", 6, -1), new Among(\"vele\", 6, -1),\n              new Among(\"ne\", -1, -1), new Among(\"cene\", 12, -1),\n              new Among(\"gliene\", 12, -1), new Among(\"mene\", 12, -1),\n              new Among(\"sene\", 12, -1), new Among(\"tene\", 12, -1),\n              new Among(\"vene\", 12, -1), new Among(\"ci\", -1, -1),\n              new Among(\"li\", -1, -1), new Among(\"celi\", 20, -1),\n              new Among(\"glieli\", 20, -1), new Among(\"meli\", 20, -1),\n              new Among(\"teli\", 20, -1), new Among(\"veli\", 20, -1),\n              new Among(\"gli\", 20, -1), new Among(\"mi\", -1, -1),\n              new Among(\"si\", -1, -1), new Among(\"ti\", -1, -1),\n              new Among(\"vi\", -1, -1), new Among(\"lo\", -1, -1),\n              new Among(\"celo\", 31, -1), new Among(\"glielo\", 31, -1),\n              new Among(\"melo\", 31, -1), new Among(\"telo\", 31, -1),\n              new Among(\"velo\", 31, -1)\n            ],\n            a_3 = [new Among(\"ando\", -1, 1),\n              new Among(\"endo\", -1, 1), new Among(\"ar\", -1, 2),\n              new Among(\"er\", -1, 2), new Among(\"ir\", -1, 2)\n            ],\n            a_4 = [\n              new Among(\"ic\", -1, -1), new Among(\"abil\", -1, -1),\n              new Among(\"os\", -1, -1), new Among(\"iv\", -1, 1)\n            ],\n            a_5 = [\n              new Among(\"ic\", -1, 1), new Among(\"abil\", -1, 1),\n              new Among(\"iv\", -1, 1)\n            ],\n            a_6 = [new Among(\"ica\", -1, 1),\n              new Among(\"logia\", -1, 3), new Among(\"osa\", -1, 1),\n              new Among(\"ista\", -1, 1), new Among(\"iva\", -1, 9),\n              new Among(\"anza\", -1, 1), new Among(\"enza\", -1, 5),\n              new Among(\"ice\", -1, 1), new Among(\"atrice\", 7, 1),\n              new Among(\"iche\", -1, 1), new Among(\"logie\", -1, 3),\n              new Among(\"abile\", -1, 1), new Among(\"ibile\", -1, 1),\n              new Among(\"usione\", -1, 4), new Among(\"azione\", -1, 2),\n              new Among(\"uzione\", -1, 4), new Among(\"atore\", -1, 2),\n              new Among(\"ose\", -1, 1), new Among(\"ante\", -1, 1),\n              new Among(\"mente\", -1, 1), new Among(\"amente\", 19, 7),\n              new Among(\"iste\", -1, 1), new Among(\"ive\", -1, 9),\n              new Among(\"anze\", -1, 1), new Among(\"enze\", -1, 5),\n              new Among(\"ici\", -1, 1), new Among(\"atrici\", 25, 1),\n              new Among(\"ichi\", -1, 1), new Among(\"abili\", -1, 1),\n              new Among(\"ibili\", -1, 1), new Among(\"ismi\", -1, 1),\n              new Among(\"usioni\", -1, 4), new Among(\"azioni\", -1, 2),\n              new Among(\"uzioni\", -1, 4), new Among(\"atori\", -1, 2),\n              new Among(\"osi\", -1, 1), new Among(\"anti\", -1, 1),\n              new Among(\"amenti\", -1, 6), new Among(\"imenti\", -1, 6),\n              new Among(\"isti\", -1, 1), new Among(\"ivi\", -1, 9),\n              new Among(\"ico\", -1, 1), new Among(\"ismo\", -1, 1),\n              new Among(\"oso\", -1, 1), new Among(\"amento\", -1, 6),\n              new Among(\"imento\", -1, 6), new Among(\"ivo\", -1, 9),\n              new Among(\"it\\u00E0\", -1, 8), new Among(\"ist\\u00E0\", -1, 1),\n              new Among(\"ist\\u00E8\", -1, 1), new Among(\"ist\\u00EC\", -1, 1)\n            ],\n            a_7 = [\n              new Among(\"isca\", -1, 1), new Among(\"enda\", -1, 1),\n              new Among(\"ata\", -1, 1), new Among(\"ita\", -1, 1),\n              new Among(\"uta\", -1, 1), new Among(\"ava\", -1, 1),\n              new Among(\"eva\", -1, 1), new Among(\"iva\", -1, 1),\n              new Among(\"erebbe\", -1, 1), new Among(\"irebbe\", -1, 1),\n              new Among(\"isce\", -1, 1), new Among(\"ende\", -1, 1),\n              new Among(\"are\", -1, 1), new Among(\"ere\", -1, 1),\n              new Among(\"ire\", -1, 1), new Among(\"asse\", -1, 1),\n              new Among(\"ate\", -1, 1), new Among(\"avate\", 16, 1),\n              new Among(\"evate\", 16, 1), new Among(\"ivate\", 16, 1),\n              new Among(\"ete\", -1, 1), new Among(\"erete\", 20, 1),\n              new Among(\"irete\", 20, 1), new Among(\"ite\", -1, 1),\n              new Among(\"ereste\", -1, 1), new Among(\"ireste\", -1, 1),\n              new Among(\"ute\", -1, 1), new Among(\"erai\", -1, 1),\n              new Among(\"irai\", -1, 1), new Among(\"isci\", -1, 1),\n              new Among(\"endi\", -1, 1), new Among(\"erei\", -1, 1),\n              new Among(\"irei\", -1, 1), new Among(\"assi\", -1, 1),\n              new Among(\"ati\", -1, 1), new Among(\"iti\", -1, 1),\n              new Among(\"eresti\", -1, 1), new Among(\"iresti\", -1, 1),\n              new Among(\"uti\", -1, 1), new Among(\"avi\", -1, 1),\n              new Among(\"evi\", -1, 1), new Among(\"ivi\", -1, 1),\n              new Among(\"isco\", -1, 1), new Among(\"ando\", -1, 1),\n              new Among(\"endo\", -1, 1), new Among(\"Yamo\", -1, 1),\n              new Among(\"iamo\", -1, 1), new Among(\"avamo\", -1, 1),\n              new Among(\"evamo\", -1, 1), new Among(\"ivamo\", -1, 1),\n              new Among(\"eremo\", -1, 1), new Among(\"iremo\", -1, 1),\n              new Among(\"assimo\", -1, 1), new Among(\"ammo\", -1, 1),\n              new Among(\"emmo\", -1, 1), new Among(\"eremmo\", 54, 1),\n              new Among(\"iremmo\", 54, 1), new Among(\"immo\", -1, 1),\n              new Among(\"ano\", -1, 1), new Among(\"iscano\", 58, 1),\n              new Among(\"avano\", 58, 1), new Among(\"evano\", 58, 1),\n              new Among(\"ivano\", 58, 1), new Among(\"eranno\", -1, 1),\n              new Among(\"iranno\", -1, 1), new Among(\"ono\", -1, 1),\n              new Among(\"iscono\", 65, 1), new Among(\"arono\", 65, 1),\n              new Among(\"erono\", 65, 1), new Among(\"irono\", 65, 1),\n              new Among(\"erebbero\", -1, 1), new Among(\"irebbero\", -1, 1),\n              new Among(\"assero\", -1, 1), new Among(\"essero\", -1, 1),\n              new Among(\"issero\", -1, 1), new Among(\"ato\", -1, 1),\n              new Among(\"ito\", -1, 1), new Among(\"uto\", -1, 1),\n              new Among(\"avo\", -1, 1), new Among(\"evo\", -1, 1),\n              new Among(\"ivo\", -1, 1), new Among(\"ar\", -1, 1),\n              new Among(\"ir\", -1, 1), new Among(\"er\\u00E0\", -1, 1),\n              new Among(\"ir\\u00E0\", -1, 1), new Among(\"er\\u00F2\", -1, 1),\n              new Among(\"ir\\u00F2\", -1, 1)\n            ],\n            g_v = [17, 65, 16, 0, 0, 0, 0, 0, 0,\n              0, 0, 0, 0, 0, 0, 128, 128, 8, 2, 1\n            ],\n            g_AEIO = [17, 65, 0, 0, 0, 0,\n              0, 0, 0, 0, 0, 0, 0, 0, 0, 128, 128, 8, 2\n            ],\n            g_CG = [17],\n            I_p2, I_p1, I_pV, sbp = new SnowballProgram();\n          this.setCurrent = function(word) {\n            sbp.setCurrent(word);\n          };\n          this.getCurrent = function() {\n            return sbp.getCurrent();\n          };\n\n          function habr1(c1, c2, v_1) {\n            if (sbp.eq_s(1, c1)) {\n              sbp.ket = sbp.cursor;\n              if (sbp.in_grouping(g_v, 97, 249)) {\n                sbp.slice_from(c2);\n                sbp.cursor = v_1;\n                return true;\n              }\n            }\n            return false;\n          }\n\n          function r_prelude() {\n            var among_var, v_1 = sbp.cursor,\n              v_2, v_3, v_4;\n            while (true) {\n              sbp.bra = sbp.cursor;\n              among_var = sbp.find_among(a_0, 7);\n              if (among_var) {\n                sbp.ket = sbp.cursor;\n                switch (among_var) {\n                  case 1:\n                    sbp.slice_from(\"\\u00E0\");\n                    continue;\n                  case 2:\n                    sbp.slice_from(\"\\u00E8\");\n                    continue;\n                  case 3:\n                    sbp.slice_from(\"\\u00EC\");\n                    continue;\n                  case 4:\n                    sbp.slice_from(\"\\u00F2\");\n                    continue;\n                  case 5:\n                    sbp.slice_from(\"\\u00F9\");\n                    continue;\n                  case 6:\n                    sbp.slice_from(\"qU\");\n                    continue;\n                  case 7:\n                    if (sbp.cursor >= sbp.limit)\n                      break;\n                    sbp.cursor++;\n                    continue;\n                }\n              }\n              break;\n            }\n            sbp.cursor = v_1;\n            while (true) {\n              v_2 = sbp.cursor;\n              while (true) {\n                v_3 = sbp.cursor;\n                if (sbp.in_grouping(g_v, 97, 249)) {\n                  sbp.bra = sbp.cursor;\n                  v_4 = sbp.cursor;\n                  if (habr1(\"u\", \"U\", v_3))\n                    break;\n                  sbp.cursor = v_4;\n                  if (habr1(\"i\", \"I\", v_3))\n                    break;\n                }\n                sbp.cursor = v_3;\n                if (sbp.cursor >= sbp.limit) {\n                  sbp.cursor = v_2;\n                  return;\n                }\n                sbp.cursor++;\n              }\n            }\n          }\n\n          function habr2(v_1) {\n            sbp.cursor = v_1;\n            if (!sbp.in_grouping(g_v, 97, 249))\n              return false;\n            while (!sbp.out_grouping(g_v, 97, 249)) {\n              if (sbp.cursor >= sbp.limit)\n                return false;\n              sbp.cursor++;\n            }\n            return true;\n          }\n\n          function habr3() {\n            if (sbp.in_grouping(g_v, 97, 249)) {\n              var v_1 = sbp.cursor;\n              if (sbp.out_grouping(g_v, 97, 249)) {\n                while (!sbp.in_grouping(g_v, 97, 249)) {\n                  if (sbp.cursor >= sbp.limit)\n                    return habr2(v_1);\n                  sbp.cursor++;\n                }\n                return true;\n              }\n              return habr2(v_1);\n            }\n            return false;\n          }\n\n          function habr4() {\n            var v_1 = sbp.cursor,\n              v_2;\n            if (!habr3()) {\n              sbp.cursor = v_1;\n              if (!sbp.out_grouping(g_v, 97, 249))\n                return;\n              v_2 = sbp.cursor;\n              if (sbp.out_grouping(g_v, 97, 249)) {\n                while (!sbp.in_grouping(g_v, 97, 249)) {\n                  if (sbp.cursor >= sbp.limit) {\n                    sbp.cursor = v_2;\n                    if (sbp.in_grouping(g_v, 97, 249) && sbp.cursor < sbp.limit)\n                      sbp.cursor++;\n                    return;\n                  }\n                  sbp.cursor++;\n                }\n                I_pV = sbp.cursor;\n                return;\n              }\n              sbp.cursor = v_2;\n              if (!sbp.in_grouping(g_v, 97, 249) || sbp.cursor >= sbp.limit)\n                return;\n              sbp.cursor++;\n            }\n            I_pV = sbp.cursor;\n          }\n\n          function habr5() {\n            while (!sbp.in_grouping(g_v, 97, 249)) {\n              if (sbp.cursor >= sbp.limit)\n                return false;\n              sbp.cursor++;\n            }\n            while (!sbp.out_grouping(g_v, 97, 249)) {\n              if (sbp.cursor >= sbp.limit)\n                return false;\n              sbp.cursor++;\n            }\n            return true;\n          }\n\n          function r_mark_regions() {\n            var v_1 = sbp.cursor;\n            I_pV = sbp.limit;\n            I_p1 = I_pV;\n            I_p2 = I_pV;\n            habr4();\n            sbp.cursor = v_1;\n            if (habr5()) {\n              I_p1 = sbp.cursor;\n              if (habr5())\n                I_p2 = sbp.cursor;\n            }\n          }\n\n          function r_postlude() {\n            var among_var;\n            while (true) {\n              sbp.bra = sbp.cursor;\n              among_var = sbp.find_among(a_1, 3);\n              if (!among_var)\n                break;\n              sbp.ket = sbp.cursor;\n              switch (among_var) {\n                case 1:\n                  sbp.slice_from(\"i\");\n                  break;\n                case 2:\n                  sbp.slice_from(\"u\");\n                  break;\n                case 3:\n                  if (sbp.cursor >= sbp.limit)\n                    return;\n                  sbp.cursor++;\n                  break;\n              }\n            }\n          }\n\n          function r_RV() {\n            return I_pV <= sbp.cursor;\n          }\n\n          function r_R1() {\n            return I_p1 <= sbp.cursor;\n          }\n\n          function r_R2() {\n            return I_p2 <= sbp.cursor;\n          }\n\n          function r_attached_pronoun() {\n            var among_var;\n            sbp.ket = sbp.cursor;\n            if (sbp.find_among_b(a_2, 37)) {\n              sbp.bra = sbp.cursor;\n              among_var = sbp.find_among_b(a_3, 5);\n              if (among_var && r_RV()) {\n                switch (among_var) {\n                  case 1:\n                    sbp.slice_del();\n                    break;\n                  case 2:\n                    sbp.slice_from(\"e\");\n                    break;\n                }\n              }\n            }\n          }\n\n          function r_standard_suffix() {\n            var among_var;\n            sbp.ket = sbp.cursor;\n            among_var = sbp.find_among_b(a_6, 51);\n            if (!among_var)\n              return false;\n            sbp.bra = sbp.cursor;\n            switch (among_var) {\n              case 1:\n                if (!r_R2())\n                  return false;\n                sbp.slice_del();\n                break;\n              case 2:\n                if (!r_R2())\n                  return false;\n                sbp.slice_del();\n                sbp.ket = sbp.cursor;\n                if (sbp.eq_s_b(2, \"ic\")) {\n                  sbp.bra = sbp.cursor;\n                  if (r_R2())\n                    sbp.slice_del();\n                }\n                break;\n              case 3:\n                if (!r_R2())\n                  return false;\n                sbp.slice_from(\"log\");\n                break;\n              case 4:\n                if (!r_R2())\n                  return false;\n                sbp.slice_from(\"u\");\n                break;\n              case 5:\n                if (!r_R2())\n                  return false;\n                sbp.slice_from(\"ente\");\n                break;\n              case 6:\n                if (!r_RV())\n                  return false;\n                sbp.slice_del();\n                break;\n              case 7:\n                if (!r_R1())\n                  return false;\n                sbp.slice_del();\n                sbp.ket = sbp.cursor;\n                among_var = sbp.find_among_b(a_4, 4);\n                if (among_var) {\n                  sbp.bra = sbp.cursor;\n                  if (r_R2()) {\n                    sbp.slice_del();\n                    if (among_var == 1) {\n                      sbp.ket = sbp.cursor;\n                      if (sbp.eq_s_b(2, \"at\")) {\n                        sbp.bra = sbp.cursor;\n                        if (r_R2())\n                          sbp.slice_del();\n                      }\n                    }\n                  }\n                }\n                break;\n              case 8:\n                if (!r_R2())\n                  return false;\n                sbp.slice_del();\n                sbp.ket = sbp.cursor;\n                among_var = sbp.find_among_b(a_5, 3);\n                if (among_var) {\n                  sbp.bra = sbp.cursor;\n                  if (among_var == 1)\n                    if (r_R2())\n                      sbp.slice_del();\n                }\n                break;\n              case 9:\n                if (!r_R2())\n                  return false;\n                sbp.slice_del();\n                sbp.ket = sbp.cursor;\n                if (sbp.eq_s_b(2, \"at\")) {\n                  sbp.bra = sbp.cursor;\n                  if (r_R2()) {\n                    sbp.slice_del();\n                    sbp.ket = sbp.cursor;\n                    if (sbp.eq_s_b(2, \"ic\")) {\n                      sbp.bra = sbp.cursor;\n                      if (r_R2())\n                        sbp.slice_del();\n                    }\n                  }\n                }\n                break;\n            }\n            return true;\n          }\n\n          function r_verb_suffix() {\n            var among_var, v_1;\n            if (sbp.cursor >= I_pV) {\n              v_1 = sbp.limit_backward;\n              sbp.limit_backward = I_pV;\n              sbp.ket = sbp.cursor;\n              among_var = sbp.find_among_b(a_7, 87);\n              if (among_var) {\n                sbp.bra = sbp.cursor;\n                if (among_var == 1)\n                  sbp.slice_del();\n              }\n              sbp.limit_backward = v_1;\n            }\n          }\n\n          function habr6() {\n            var v_1 = sbp.limit - sbp.cursor;\n            sbp.ket = sbp.cursor;\n            if (sbp.in_grouping_b(g_AEIO, 97, 242)) {\n              sbp.bra = sbp.cursor;\n              if (r_RV()) {\n                sbp.slice_del();\n                sbp.ket = sbp.cursor;\n                if (sbp.eq_s_b(1, \"i\")) {\n                  sbp.bra = sbp.cursor;\n                  if (r_RV()) {\n                    sbp.slice_del();\n                    return;\n                  }\n                }\n              }\n            }\n            sbp.cursor = sbp.limit - v_1;\n          }\n\n          function r_vowel_suffix() {\n            habr6();\n            sbp.ket = sbp.cursor;\n            if (sbp.eq_s_b(1, \"h\")) {\n              sbp.bra = sbp.cursor;\n              if (sbp.in_grouping_b(g_CG, 99, 103))\n                if (r_RV())\n                  sbp.slice_del();\n            }\n          }\n          this.stem = function() {\n            var v_1 = sbp.cursor;\n            r_prelude();\n            sbp.cursor = v_1;\n            r_mark_regions();\n            sbp.limit_backward = v_1;\n            sbp.cursor = sbp.limit;\n            r_attached_pronoun();\n            sbp.cursor = sbp.limit;\n            if (!r_standard_suffix()) {\n              sbp.cursor = sbp.limit;\n              r_verb_suffix();\n            }\n            sbp.cursor = sbp.limit;\n            r_vowel_suffix();\n            sbp.cursor = sbp.limit_backward;\n            r_postlude();\n            return true;\n          }\n        };\n\n      /* and return a function that stems a word for the current locale */\n      return function(word) {\n        st.setCurrent(word);\n        st.stem();\n        return st.getCurrent();\n      }\n    })();\n\n    lunr.Pipeline.registerFunction(lunr.it.stemmer, 'stemmer-it');\n\n    /* stop word filter function */\n    lunr.it.stopWordFilter = function(token) {\n      if (lunr.it.stopWordFilter.stopWords.indexOf(token) === -1) {\n        return token;\n      }\n    };\n\n    lunr.it.stopWordFilter.stopWords = new lunr.SortedSet();\n    lunr.it.stopWordFilter.stopWords.length = 280;\n\n    // The space at the beginning is crucial: It marks the empty string\n    // as a stop word. lunr.js crashes during search when documents\n    // processed by the pipeline still contain the empty string.\n    lunr.it.stopWordFilter.stopWords.elements = ' a abbia abbiamo abbiano abbiate ad agl agli ai al all alla alle allo anche avemmo avendo avesse avessero avessi avessimo aveste avesti avete aveva avevamo avevano avevate avevi avevo avrai avranno avrebbe avrebbero avrei avremmo avremo avreste avresti avrete avrà avrò avuta avute avuti avuto c che chi ci coi col come con contro cui da dagl dagli dai dal dall dalla dalle dallo degl degli dei del dell della delle dello di dov dove e ebbe ebbero ebbi ed era erano eravamo eravate eri ero essendo faccia facciamo facciano facciate faccio facemmo facendo facesse facessero facessi facessimo faceste facesti faceva facevamo facevano facevate facevi facevo fai fanno farai faranno farebbe farebbero farei faremmo faremo fareste faresti farete farà farò fece fecero feci fosse fossero fossi fossimo foste fosti fu fui fummo furono gli ha hai hanno ho i il in io l la le lei li lo loro lui ma mi mia mie miei mio ne negl negli nei nel nell nella nelle nello noi non nostra nostre nostri nostro o per perché più quale quanta quante quanti quanto quella quelle quelli quello questa queste questi questo sarai saranno sarebbe sarebbero sarei saremmo saremo sareste saresti sarete sarà sarò se sei si sia siamo siano siate siete sono sta stai stando stanno starai staranno starebbe starebbero starei staremmo staremo stareste staresti starete starà starò stava stavamo stavano stavate stavi stavo stemmo stesse stessero stessi stessimo steste stesti stette stettero stetti stia stiamo stiano stiate sto su sua sue sugl sugli sui sul sull sulla sulle sullo suo suoi ti tra tu tua tue tuo tuoi tutti tutto un una uno vi voi vostra vostre vostri vostro è'.split(' ');\n\n    lunr.Pipeline.registerFunction(lunr.it.stopWordFilter, 'stopWordFilter-it');\n  };\n}))\n"
  },
  {
    "path": "static/js/lunr/lunr.ja.js",
    "content": "/*!\n * Lunr languages, `Japanese` language\n * https://github.com/MihaiValentin/lunr-languages\n *\n * Copyright 2014, Chad Liu\n * http://www.mozilla.org/MPL/\n */\n/*!\n * based on\n * Snowball JavaScript Library v0.3\n * http://code.google.com/p/urim/\n * http://snowball.tartarus.org/\n *\n * Copyright 2010, Oleg Mazko\n * http://www.mozilla.org/MPL/\n */\n\n/**\n * export the module via AMD, CommonJS or as a browser global\n * Export code from https://github.com/umdjs/umd/blob/master/returnExports.js\n */\n ;\n (function(root, factory) {\n   if (typeof define === 'function' && define.amd) {\n     // AMD. Register as an anonymous module.\n     define(factory)\n   } else if (typeof exports === 'object') {\n     /**\n      * Node. Does not work with strict CommonJS, but\n      * only CommonJS-like environments that support module.exports,\n      * like Node.\n      */\n     module.exports = factory()\n   } else {\n     // Browser globals (root is window)\n     factory()(root.lunr);\n   }\n }(this, function() {\n   /**\n    * Just return a value to define the module export.\n    * This example returns an object, but the module\n    * can return a function as the exported value.\n    */\n   return function(lunr) {\n     /* throw error if lunr is not yet included */\n     if ('undefined' === typeof lunr) {\n       throw new Error('Lunr is not present. Please include / require Lunr before this script.');\n     }\n\n     /* throw error if lunr stemmer support is not yet included */\n     if ('undefined' === typeof lunr.stemmerSupport) {\n       throw new Error('Lunr stemmer support is not present. Please include / require Lunr stemmer support before this script.');\n     }\n\n     /* register specific locale function */\n     lunr.ja = function() {\n       this.pipeline.reset();\n       this.pipeline.add(\n         lunr.ja.trimmer,\n         lunr.ja.stopWordFilter,\n         lunr.ja.stemmer\n       );\n       // change the tokenizer for japanese one\n       lunr.tokenizer = lunr.ja.tokenizer;\n     };\n     var segmenter = new TinySegmenter();  // インスタンス生成\n\n     lunr.ja.tokenizer = function (obj) {\n         if (!arguments.length || obj == null || obj == undefined) return []\n         if (Array.isArray(obj)) return obj.map(function (t) { return t.toLowerCase() })\n\n         var str = obj.toString().replace(/^\\s+/, '')\n\n         for (var i = str.length - 1; i >= 0; i--) {\n             if (/\\S/.test(str.charAt(i))) {\n                 str = str.substring(0, i + 1)\n                 break\n             }\n         }\n\n\n         var segs = segmenter.segment(str);  // 単語の配列が返る\n         return segs.filter(function (token) {\n             return !!token\n           })\n           .map(function (token) {\n             return token\n           })\n     }\n\n     /* lunr stemmer function */\n     lunr.ja.stemmer = (function() {\n\n       /* TODO japanese stemmer  */\n       return function(word) {\n         return word;\n       }\n     })();\n\n     lunr.Pipeline.registerFunction(lunr.ja.stemmer, 'stemmer-ja');\n\n     /* lunr trimmer function */\n     lunr.ja.wordCharacters = \"一二三四五六七八九十百千万億兆一-龠々〆ヵヶぁ-んァ-ヴーｱ-ﾝﾞa-zA-Zａ-ｚＡ-Ｚ0-9０-９\";\n     lunr.ja.trimmer = lunr.trimmerSupport.generateTrimmer(lunr.ja.wordCharacters);\n     lunr.Pipeline.registerFunction(lunr.ja.trimmer, 'trimmer-ja');\n\n     /* stop word filter function */\n     lunr.ja.stopWordFilter = function(token) {\n       if (lunr.ja.stopWordFilter.stopWords.indexOf(token) === -1) {\n         return token;\n       }\n     };\n\n     lunr.ja.stopWordFilter.stopWords = new lunr.SortedSet();\n     lunr.ja.stopWordFilter.stopWords.length = 45;\n\n     // The space at the beginning is crucial: It marks the empty string\n     // as a stop word. lunr.js crashes during search when documents\n     // processed by the pipeline still contain the empty string.\n     // stopword for japanese is from http://www.ranks.nl/stopwords/japanese\n     lunr.ja.stopWordFilter.stopWords.elements = ' これ それ あれ この その あの ここ そこ あそこ こちら どこ だれ なに なん 何 私 貴方 貴方方 我々 私達 あの人 あのかた 彼女 彼 です あります おります います は が の に を で え から まで より も どの と し それで しかし'.split(' ');\n     lunr.Pipeline.registerFunction(lunr.ja.stopWordFilter, 'stopWordFilter-ja');\n\n     // alias ja => jp for backward-compatibility.\n     // jp is the country code, while ja is the language code\n     // a new lunr.ja.js has been created, but in order to\n     // keep the backward compatibility, we'll leave the lunr.jp.js\n     // here for a while, and just make it use the new lunr.ja.js\n     lunr.jp = lunr.ja;\n     lunr.Pipeline.registerFunction(lunr.jp.stemmer, 'stemmer-jp');\n     lunr.Pipeline.registerFunction(lunr.jp.trimmer, 'trimmer-jp');\n     lunr.Pipeline.registerFunction(lunr.jp.stopWordFilter, 'stopWordFilter-jp');\n   };\n }))\n"
  },
  {
    "path": "static/js/lunr/lunr.no.js",
    "content": "/*!\n * Lunr languages, `Norwegian` language\n * https://github.com/MihaiValentin/lunr-languages\n *\n * Copyright 2014, Mihai Valentin\n * http://www.mozilla.org/MPL/\n */\n/*!\n * based on\n * Snowball JavaScript Library v0.3\n * http://code.google.com/p/urim/\n * http://snowball.tartarus.org/\n *\n * Copyright 2010, Oleg Mazko\n * http://www.mozilla.org/MPL/\n */\n\n/**\n * export the module via AMD, CommonJS or as a browser global\n * Export code from https://github.com/umdjs/umd/blob/master/returnExports.js\n */\n;\n(function(root, factory) {\n  if (typeof define === 'function' && define.amd) {\n    // AMD. Register as an anonymous module.\n    define(factory)\n  } else if (typeof exports === 'object') {\n    /**\n     * Node. Does not work with strict CommonJS, but\n     * only CommonJS-like environments that support module.exports,\n     * like Node.\n     */\n    module.exports = factory()\n  } else {\n    // Browser globals (root is window)\n    factory()(root.lunr);\n  }\n}(this, function() {\n  /**\n   * Just return a value to define the module export.\n   * This example returns an object, but the module\n   * can return a function as the exported value.\n   */\n  return function(lunr) {\n    /* throw error if lunr is not yet included */\n    if ('undefined' === typeof lunr) {\n      throw new Error('Lunr is not present. Please include / require Lunr before this script.');\n    }\n\n    /* throw error if lunr stemmer support is not yet included */\n    if ('undefined' === typeof lunr.stemmerSupport) {\n      throw new Error('Lunr stemmer support is not present. Please include / require Lunr stemmer support before this script.');\n    }\n\n    /* register specific locale function */\n    lunr.no = function() {\n      this.pipeline.reset();\n      this.pipeline.add(\n        lunr.no.trimmer,\n        lunr.no.stopWordFilter,\n        lunr.no.stemmer\n      );\n    };\n\n    /* lunr trimmer function */\n    lunr.no.wordCharacters = \"A-Za-z\\xAA\\xBA\\xC0-\\xD6\\xD8-\\xF6\\xF8-\\u02B8\\u02E0-\\u02E4\\u1D00-\\u1D25\\u1D2C-\\u1D5C\\u1D62-\\u1D65\\u1D6B-\\u1D77\\u1D79-\\u1DBE\\u1E00-\\u1EFF\\u2071\\u207F\\u2090-\\u209C\\u212A\\u212B\\u2132\\u214E\\u2160-\\u2188\\u2C60-\\u2C7F\\uA722-\\uA787\\uA78B-\\uA7AD\\uA7B0-\\uA7B7\\uA7F7-\\uA7FF\\uAB30-\\uAB5A\\uAB5C-\\uAB64\\uFB00-\\uFB06\\uFF21-\\uFF3A\\uFF41-\\uFF5A\";\n    lunr.no.trimmer = lunr.trimmerSupport.generateTrimmer(lunr.no.wordCharacters);\n\n    lunr.Pipeline.registerFunction(lunr.no.trimmer, 'trimmer-no');\n\n    /* lunr stemmer function */\n    lunr.no.stemmer = (function() {\n      /* create the wrapped stemmer object */\n      var Among = lunr.stemmerSupport.Among,\n        SnowballProgram = lunr.stemmerSupport.SnowballProgram,\n        st = new function NorwegianStemmer() {\n          var a_0 = [new Among(\"a\", -1, 1), new Among(\"e\", -1, 1),\n              new Among(\"ede\", 1, 1), new Among(\"ande\", 1, 1),\n              new Among(\"ende\", 1, 1), new Among(\"ane\", 1, 1),\n              new Among(\"ene\", 1, 1), new Among(\"hetene\", 6, 1),\n              new Among(\"erte\", 1, 3), new Among(\"en\", -1, 1),\n              new Among(\"heten\", 9, 1), new Among(\"ar\", -1, 1),\n              new Among(\"er\", -1, 1), new Among(\"heter\", 12, 1),\n              new Among(\"s\", -1, 2), new Among(\"as\", 14, 1),\n              new Among(\"es\", 14, 1), new Among(\"edes\", 16, 1),\n              new Among(\"endes\", 16, 1), new Among(\"enes\", 16, 1),\n              new Among(\"hetenes\", 19, 1), new Among(\"ens\", 14, 1),\n              new Among(\"hetens\", 21, 1), new Among(\"ers\", 14, 1),\n              new Among(\"ets\", 14, 1), new Among(\"et\", -1, 1),\n              new Among(\"het\", 25, 1), new Among(\"ert\", -1, 3),\n              new Among(\"ast\", -1, 1)\n            ],\n            a_1 = [new Among(\"dt\", -1, -1),\n              new Among(\"vt\", -1, -1)\n            ],\n            a_2 = [new Among(\"leg\", -1, 1),\n              new Among(\"eleg\", 0, 1), new Among(\"ig\", -1, 1),\n              new Among(\"eig\", 2, 1), new Among(\"lig\", 2, 1),\n              new Among(\"elig\", 4, 1), new Among(\"els\", -1, 1),\n              new Among(\"lov\", -1, 1), new Among(\"elov\", 7, 1),\n              new Among(\"slov\", 7, 1), new Among(\"hetslov\", 9, 1)\n            ],\n            g_v = [17,\n              65, 16, 1, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 48, 0, 128\n            ],\n            g_s_ending = [\n              119, 125, 149, 1\n            ],\n            I_x, I_p1, sbp = new SnowballProgram();\n          this.setCurrent = function(word) {\n            sbp.setCurrent(word);\n          };\n          this.getCurrent = function() {\n            return sbp.getCurrent();\n          };\n\n          function r_mark_regions() {\n            var v_1, c = sbp.cursor + 3;\n            I_p1 = sbp.limit;\n            if (0 <= c || c <= sbp.limit) {\n              I_x = c;\n              while (true) {\n                v_1 = sbp.cursor;\n                if (sbp.in_grouping(g_v, 97, 248)) {\n                  sbp.cursor = v_1;\n                  break;\n                }\n                if (v_1 >= sbp.limit)\n                  return;\n                sbp.cursor = v_1 + 1;\n              }\n              while (!sbp.out_grouping(g_v, 97, 248)) {\n                if (sbp.cursor >= sbp.limit)\n                  return;\n                sbp.cursor++;\n              }\n              I_p1 = sbp.cursor;\n              if (I_p1 < I_x)\n                I_p1 = I_x;\n            }\n          }\n\n          function r_main_suffix() {\n            var among_var, v_1, v_2;\n            if (sbp.cursor >= I_p1) {\n              v_1 = sbp.limit_backward;\n              sbp.limit_backward = I_p1;\n              sbp.ket = sbp.cursor;\n              among_var = sbp.find_among_b(a_0, 29);\n              sbp.limit_backward = v_1;\n              if (among_var) {\n                sbp.bra = sbp.cursor;\n                switch (among_var) {\n                  case 1:\n                    sbp.slice_del();\n                    break;\n                  case 2:\n                    v_2 = sbp.limit - sbp.cursor;\n                    if (sbp.in_grouping_b(g_s_ending, 98, 122))\n                      sbp.slice_del();\n                    else {\n                      sbp.cursor = sbp.limit - v_2;\n                      if (sbp.eq_s_b(1, \"k\") && sbp.out_grouping_b(g_v, 97, 248))\n                        sbp.slice_del();\n                    }\n                    break;\n                  case 3:\n                    sbp.slice_from(\"er\");\n                    break;\n                }\n              }\n            }\n          }\n\n          function r_consonant_pair() {\n            var v_1 = sbp.limit - sbp.cursor,\n              v_2;\n            if (sbp.cursor >= I_p1) {\n              v_2 = sbp.limit_backward;\n              sbp.limit_backward = I_p1;\n              sbp.ket = sbp.cursor;\n              if (sbp.find_among_b(a_1, 2)) {\n                sbp.bra = sbp.cursor;\n                sbp.limit_backward = v_2;\n                sbp.cursor = sbp.limit - v_1;\n                if (sbp.cursor > sbp.limit_backward) {\n                  sbp.cursor--;\n                  sbp.bra = sbp.cursor;\n                  sbp.slice_del();\n                }\n              } else\n                sbp.limit_backward = v_2;\n            }\n          }\n\n          function r_other_suffix() {\n            var among_var, v_1;\n            if (sbp.cursor >= I_p1) {\n              v_1 = sbp.limit_backward;\n              sbp.limit_backward = I_p1;\n              sbp.ket = sbp.cursor;\n              among_var = sbp.find_among_b(a_2, 11);\n              if (among_var) {\n                sbp.bra = sbp.cursor;\n                sbp.limit_backward = v_1;\n                if (among_var == 1)\n                  sbp.slice_del();\n              } else\n                sbp.limit_backward = v_1;\n            }\n          }\n          this.stem = function() {\n            var v_1 = sbp.cursor;\n            r_mark_regions();\n            sbp.limit_backward = v_1;\n            sbp.cursor = sbp.limit;\n            r_main_suffix();\n            sbp.cursor = sbp.limit;\n            r_consonant_pair();\n            sbp.cursor = sbp.limit;\n            r_other_suffix();\n            return true;\n          }\n        };\n\n      /* and return a function that stems a word for the current locale */\n      return function(word) {\n        st.setCurrent(word);\n        st.stem();\n        return st.getCurrent();\n      }\n    })();\n\n    lunr.Pipeline.registerFunction(lunr.no.stemmer, 'stemmer-no');\n\n    /* stop word filter function */\n    lunr.no.stopWordFilter = function(token) {\n      if (lunr.no.stopWordFilter.stopWords.indexOf(token) === -1) {\n        return token;\n      }\n    };\n\n    lunr.no.stopWordFilter.stopWords = new lunr.SortedSet();\n    lunr.no.stopWordFilter.stopWords.length = 177;\n\n    // The space at the beginning is crucial: It marks the empty string\n    // as a stop word. lunr.js crashes during search when documents\n    // processed by the pipeline still contain the empty string.\n    lunr.no.stopWordFilter.stopWords.elements = ' alle at av bare begge ble blei bli blir blitt både båe da de deg dei deim deira deires dem den denne der dere deres det dette di din disse ditt du dykk dykkar då eg ein eit eitt eller elles en enn er et ett etter for fordi fra før ha hadde han hans har hennar henne hennes her hjå ho hoe honom hoss hossen hun hva hvem hver hvilke hvilken hvis hvor hvordan hvorfor i ikke ikkje ikkje ingen ingi inkje inn inni ja jeg kan kom korleis korso kun kunne kva kvar kvarhelst kven kvi kvifor man mange me med medan meg meget mellom men mi min mine mitt mot mykje ned no noe noen noka noko nokon nokor nokre nå når og også om opp oss over på samme seg selv si si sia sidan siden sin sine sitt sjøl skal skulle slik so som som somme somt så sånn til um upp ut uten var vart varte ved vere verte vi vil ville vore vors vort vår være være vært å'.split(' ');\n\n    lunr.Pipeline.registerFunction(lunr.no.stopWordFilter, 'stopWordFilter-no');\n  };\n}))\n"
  },
  {
    "path": "static/js/lunr/lunr.pt.js",
    "content": "/*!\n * Lunr languages, `Portuguese` language\n * https://github.com/MihaiValentin/lunr-languages\n *\n * Copyright 2014, Mihai Valentin\n * http://www.mozilla.org/MPL/\n */\n/*!\n * based on\n * Snowball JavaScript Library v0.3\n * http://code.google.com/p/urim/\n * http://snowball.tartarus.org/\n *\n * Copyright 2010, Oleg Mazko\n * http://www.mozilla.org/MPL/\n */\n\n/**\n * export the module via AMD, CommonJS or as a browser global\n * Export code from https://github.com/umdjs/umd/blob/master/returnExports.js\n */\n;\n(function(root, factory) {\n  if (typeof define === 'function' && define.amd) {\n    // AMD. Register as an anonymous module.\n    define(factory)\n  } else if (typeof exports === 'object') {\n    /**\n     * Node. Does not work with strict CommonJS, but\n     * only CommonJS-like environments that support module.exports,\n     * like Node.\n     */\n    module.exports = factory()\n  } else {\n    // Browser globals (root is window)\n    factory()(root.lunr);\n  }\n}(this, function() {\n  /**\n   * Just return a value to define the module export.\n   * This example returns an object, but the module\n   * can return a function as the exported value.\n   */\n  return function(lunr) {\n    /* throw error if lunr is not yet included */\n    if ('undefined' === typeof lunr) {\n      throw new Error('Lunr is not present. Please include / require Lunr before this script.');\n    }\n\n    /* throw error if lunr stemmer support is not yet included */\n    if ('undefined' === typeof lunr.stemmerSupport) {\n      throw new Error('Lunr stemmer support is not present. Please include / require Lunr stemmer support before this script.');\n    }\n\n    /* register specific locale function */\n    lunr.pt = function() {\n      this.pipeline.reset();\n      this.pipeline.add(\n        lunr.pt.trimmer,\n        lunr.pt.stopWordFilter,\n        lunr.pt.stemmer\n      );\n    };\n\n    /* lunr trimmer function */\n    lunr.pt.wordCharacters = \"A-Za-z\\xAA\\xBA\\xC0-\\xD6\\xD8-\\xF6\\xF8-\\u02B8\\u02E0-\\u02E4\\u1D00-\\u1D25\\u1D2C-\\u1D5C\\u1D62-\\u1D65\\u1D6B-\\u1D77\\u1D79-\\u1DBE\\u1E00-\\u1EFF\\u2071\\u207F\\u2090-\\u209C\\u212A\\u212B\\u2132\\u214E\\u2160-\\u2188\\u2C60-\\u2C7F\\uA722-\\uA787\\uA78B-\\uA7AD\\uA7B0-\\uA7B7\\uA7F7-\\uA7FF\\uAB30-\\uAB5A\\uAB5C-\\uAB64\\uFB00-\\uFB06\\uFF21-\\uFF3A\\uFF41-\\uFF5A\";\n    lunr.pt.trimmer = lunr.trimmerSupport.generateTrimmer(lunr.pt.wordCharacters);\n\n    lunr.Pipeline.registerFunction(lunr.pt.trimmer, 'trimmer-pt');\n\n    /* lunr stemmer function */\n    lunr.pt.stemmer = (function() {\n      /* create the wrapped stemmer object */\n      var Among = lunr.stemmerSupport.Among,\n        SnowballProgram = lunr.stemmerSupport.SnowballProgram,\n        st = new function PortugueseStemmer() {\n          var a_0 = [new Among(\"\", -1, 3), new Among(\"\\u00E3\", 0, 1),\n              new Among(\"\\u00F5\", 0, 2)\n            ],\n            a_1 = [new Among(\"\", -1, 3),\n              new Among(\"a~\", 0, 1), new Among(\"o~\", 0, 2)\n            ],\n            a_2 = [\n              new Among(\"ic\", -1, -1), new Among(\"ad\", -1, -1),\n              new Among(\"os\", -1, -1), new Among(\"iv\", -1, 1)\n            ],\n            a_3 = [\n              new Among(\"ante\", -1, 1), new Among(\"avel\", -1, 1),\n              new Among(\"\\u00EDvel\", -1, 1)\n            ],\n            a_4 = [new Among(\"ic\", -1, 1),\n              new Among(\"abil\", -1, 1), new Among(\"iv\", -1, 1)\n            ],\n            a_5 = [\n              new Among(\"ica\", -1, 1), new Among(\"\\u00E2ncia\", -1, 1),\n              new Among(\"\\u00EAncia\", -1, 4), new Among(\"ira\", -1, 9),\n              new Among(\"adora\", -1, 1), new Among(\"osa\", -1, 1),\n              new Among(\"ista\", -1, 1), new Among(\"iva\", -1, 8),\n              new Among(\"eza\", -1, 1), new Among(\"log\\u00EDa\", -1, 2),\n              new Among(\"idade\", -1, 7), new Among(\"ante\", -1, 1),\n              new Among(\"mente\", -1, 6), new Among(\"amente\", 12, 5),\n              new Among(\"\\u00E1vel\", -1, 1), new Among(\"\\u00EDvel\", -1, 1),\n              new Among(\"uci\\u00F3n\", -1, 3), new Among(\"ico\", -1, 1),\n              new Among(\"ismo\", -1, 1), new Among(\"oso\", -1, 1),\n              new Among(\"amento\", -1, 1), new Among(\"imento\", -1, 1),\n              new Among(\"ivo\", -1, 8), new Among(\"a\\u00E7a~o\", -1, 1),\n              new Among(\"ador\", -1, 1), new Among(\"icas\", -1, 1),\n              new Among(\"\\u00EAncias\", -1, 4), new Among(\"iras\", -1, 9),\n              new Among(\"adoras\", -1, 1), new Among(\"osas\", -1, 1),\n              new Among(\"istas\", -1, 1), new Among(\"ivas\", -1, 8),\n              new Among(\"ezas\", -1, 1), new Among(\"log\\u00EDas\", -1, 2),\n              new Among(\"idades\", -1, 7), new Among(\"uciones\", -1, 3),\n              new Among(\"adores\", -1, 1), new Among(\"antes\", -1, 1),\n              new Among(\"a\\u00E7o~es\", -1, 1), new Among(\"icos\", -1, 1),\n              new Among(\"ismos\", -1, 1), new Among(\"osos\", -1, 1),\n              new Among(\"amentos\", -1, 1), new Among(\"imentos\", -1, 1),\n              new Among(\"ivos\", -1, 8)\n            ],\n            a_6 = [new Among(\"ada\", -1, 1),\n              new Among(\"ida\", -1, 1), new Among(\"ia\", -1, 1),\n              new Among(\"aria\", 2, 1), new Among(\"eria\", 2, 1),\n              new Among(\"iria\", 2, 1), new Among(\"ara\", -1, 1),\n              new Among(\"era\", -1, 1), new Among(\"ira\", -1, 1),\n              new Among(\"ava\", -1, 1), new Among(\"asse\", -1, 1),\n              new Among(\"esse\", -1, 1), new Among(\"isse\", -1, 1),\n              new Among(\"aste\", -1, 1), new Among(\"este\", -1, 1),\n              new Among(\"iste\", -1, 1), new Among(\"ei\", -1, 1),\n              new Among(\"arei\", 16, 1), new Among(\"erei\", 16, 1),\n              new Among(\"irei\", 16, 1), new Among(\"am\", -1, 1),\n              new Among(\"iam\", 20, 1), new Among(\"ariam\", 21, 1),\n              new Among(\"eriam\", 21, 1), new Among(\"iriam\", 21, 1),\n              new Among(\"aram\", 20, 1), new Among(\"eram\", 20, 1),\n              new Among(\"iram\", 20, 1), new Among(\"avam\", 20, 1),\n              new Among(\"em\", -1, 1), new Among(\"arem\", 29, 1),\n              new Among(\"erem\", 29, 1), new Among(\"irem\", 29, 1),\n              new Among(\"assem\", 29, 1), new Among(\"essem\", 29, 1),\n              new Among(\"issem\", 29, 1), new Among(\"ado\", -1, 1),\n              new Among(\"ido\", -1, 1), new Among(\"ando\", -1, 1),\n              new Among(\"endo\", -1, 1), new Among(\"indo\", -1, 1),\n              new Among(\"ara~o\", -1, 1), new Among(\"era~o\", -1, 1),\n              new Among(\"ira~o\", -1, 1), new Among(\"ar\", -1, 1),\n              new Among(\"er\", -1, 1), new Among(\"ir\", -1, 1),\n              new Among(\"as\", -1, 1), new Among(\"adas\", 47, 1),\n              new Among(\"idas\", 47, 1), new Among(\"ias\", 47, 1),\n              new Among(\"arias\", 50, 1), new Among(\"erias\", 50, 1),\n              new Among(\"irias\", 50, 1), new Among(\"aras\", 47, 1),\n              new Among(\"eras\", 47, 1), new Among(\"iras\", 47, 1),\n              new Among(\"avas\", 47, 1), new Among(\"es\", -1, 1),\n              new Among(\"ardes\", 58, 1), new Among(\"erdes\", 58, 1),\n              new Among(\"irdes\", 58, 1), new Among(\"ares\", 58, 1),\n              new Among(\"eres\", 58, 1), new Among(\"ires\", 58, 1),\n              new Among(\"asses\", 58, 1), new Among(\"esses\", 58, 1),\n              new Among(\"isses\", 58, 1), new Among(\"astes\", 58, 1),\n              new Among(\"estes\", 58, 1), new Among(\"istes\", 58, 1),\n              new Among(\"is\", -1, 1), new Among(\"ais\", 71, 1),\n              new Among(\"eis\", 71, 1), new Among(\"areis\", 73, 1),\n              new Among(\"ereis\", 73, 1), new Among(\"ireis\", 73, 1),\n              new Among(\"\\u00E1reis\", 73, 1), new Among(\"\\u00E9reis\", 73, 1),\n              new Among(\"\\u00EDreis\", 73, 1), new Among(\"\\u00E1sseis\", 73, 1),\n              new Among(\"\\u00E9sseis\", 73, 1), new Among(\"\\u00EDsseis\", 73, 1),\n              new Among(\"\\u00E1veis\", 73, 1), new Among(\"\\u00EDeis\", 73, 1),\n              new Among(\"ar\\u00EDeis\", 84, 1), new Among(\"er\\u00EDeis\", 84, 1),\n              new Among(\"ir\\u00EDeis\", 84, 1), new Among(\"ados\", -1, 1),\n              new Among(\"idos\", -1, 1), new Among(\"amos\", -1, 1),\n              new Among(\"\\u00E1ramos\", 90, 1), new Among(\"\\u00E9ramos\", 90, 1),\n              new Among(\"\\u00EDramos\", 90, 1), new Among(\"\\u00E1vamos\", 90, 1),\n              new Among(\"\\u00EDamos\", 90, 1), new Among(\"ar\\u00EDamos\", 95, 1),\n              new Among(\"er\\u00EDamos\", 95, 1), new Among(\"ir\\u00EDamos\", 95, 1),\n              new Among(\"emos\", -1, 1), new Among(\"aremos\", 99, 1),\n              new Among(\"eremos\", 99, 1), new Among(\"iremos\", 99, 1),\n              new Among(\"\\u00E1ssemos\", 99, 1), new Among(\"\\u00EAssemos\", 99, 1),\n              new Among(\"\\u00EDssemos\", 99, 1), new Among(\"imos\", -1, 1),\n              new Among(\"armos\", -1, 1), new Among(\"ermos\", -1, 1),\n              new Among(\"irmos\", -1, 1), new Among(\"\\u00E1mos\", -1, 1),\n              new Among(\"ar\\u00E1s\", -1, 1), new Among(\"er\\u00E1s\", -1, 1),\n              new Among(\"ir\\u00E1s\", -1, 1), new Among(\"eu\", -1, 1),\n              new Among(\"iu\", -1, 1), new Among(\"ou\", -1, 1),\n              new Among(\"ar\\u00E1\", -1, 1), new Among(\"er\\u00E1\", -1, 1),\n              new Among(\"ir\\u00E1\", -1, 1)\n            ],\n            a_7 = [new Among(\"a\", -1, 1),\n              new Among(\"i\", -1, 1), new Among(\"o\", -1, 1),\n              new Among(\"os\", -1, 1), new Among(\"\\u00E1\", -1, 1),\n              new Among(\"\\u00ED\", -1, 1), new Among(\"\\u00F3\", -1, 1)\n            ],\n            a_8 = [\n              new Among(\"e\", -1, 1), new Among(\"\\u00E7\", -1, 2),\n              new Among(\"\\u00E9\", -1, 1), new Among(\"\\u00EA\", -1, 1)\n            ],\n            g_v = [17,\n              65, 16, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 3, 19, 12, 2\n            ],\n            I_p2, I_p1, I_pV, sbp = new SnowballProgram();\n          this.setCurrent = function(word) {\n            sbp.setCurrent(word);\n          };\n          this.getCurrent = function() {\n            return sbp.getCurrent();\n          };\n\n          function r_prelude() {\n            var among_var;\n            while (true) {\n              sbp.bra = sbp.cursor;\n              among_var = sbp.find_among(a_0, 3);\n              if (among_var) {\n                sbp.ket = sbp.cursor;\n                switch (among_var) {\n                  case 1:\n                    sbp.slice_from(\"a~\");\n                    continue;\n                  case 2:\n                    sbp.slice_from(\"o~\");\n                    continue;\n                  case 3:\n                    if (sbp.cursor >= sbp.limit)\n                      break;\n                    sbp.cursor++;\n                    continue;\n                }\n              }\n              break;\n            }\n          }\n\n          function habr2() {\n            if (sbp.out_grouping(g_v, 97, 250)) {\n              while (!sbp.in_grouping(g_v, 97, 250)) {\n                if (sbp.cursor >= sbp.limit)\n                  return true;\n                sbp.cursor++;\n              }\n              return false;\n            }\n            return true;\n          }\n\n          function habr3() {\n            if (sbp.in_grouping(g_v, 97, 250)) {\n              while (!sbp.out_grouping(g_v, 97, 250)) {\n                if (sbp.cursor >= sbp.limit)\n                  return false;\n                sbp.cursor++;\n              }\n            }\n            I_pV = sbp.cursor;\n            return true;\n          }\n\n          function habr4() {\n            var v_1 = sbp.cursor,\n              v_2, v_3;\n            if (sbp.in_grouping(g_v, 97, 250)) {\n              v_2 = sbp.cursor;\n              if (habr2()) {\n                sbp.cursor = v_2;\n                if (habr3())\n                  return;\n              } else\n                I_pV = sbp.cursor;\n            }\n            sbp.cursor = v_1;\n            if (sbp.out_grouping(g_v, 97, 250)) {\n              v_3 = sbp.cursor;\n              if (habr2()) {\n                sbp.cursor = v_3;\n                if (!sbp.in_grouping(g_v, 97, 250) || sbp.cursor >= sbp.limit)\n                  return;\n                sbp.cursor++;\n              }\n              I_pV = sbp.cursor;\n            }\n          }\n\n          function habr5() {\n            while (!sbp.in_grouping(g_v, 97, 250)) {\n              if (sbp.cursor >= sbp.limit)\n                return false;\n              sbp.cursor++;\n            }\n            while (!sbp.out_grouping(g_v, 97, 250)) {\n              if (sbp.cursor >= sbp.limit)\n                return false;\n              sbp.cursor++;\n            }\n            return true;\n          }\n\n          function r_mark_regions() {\n            var v_1 = sbp.cursor;\n            I_pV = sbp.limit;\n            I_p1 = I_pV;\n            I_p2 = I_pV;\n            habr4();\n            sbp.cursor = v_1;\n            if (habr5()) {\n              I_p1 = sbp.cursor;\n              if (habr5())\n                I_p2 = sbp.cursor;\n            }\n          }\n\n          function r_postlude() {\n            var among_var;\n            while (true) {\n              sbp.bra = sbp.cursor;\n              among_var = sbp.find_among(a_1, 3);\n              if (among_var) {\n                sbp.ket = sbp.cursor;\n                switch (among_var) {\n                  case 1:\n                    sbp.slice_from(\"\\u00E3\");\n                    continue;\n                  case 2:\n                    sbp.slice_from(\"\\u00F5\");\n                    continue;\n                  case 3:\n                    if (sbp.cursor >= sbp.limit)\n                      break;\n                    sbp.cursor++;\n                    continue;\n                }\n              }\n              break;\n            }\n          }\n\n          function r_RV() {\n            return I_pV <= sbp.cursor;\n          }\n\n          function r_R1() {\n            return I_p1 <= sbp.cursor;\n          }\n\n          function r_R2() {\n            return I_p2 <= sbp.cursor;\n          }\n\n          function r_standard_suffix() {\n            var among_var;\n            sbp.ket = sbp.cursor;\n            among_var = sbp.find_among_b(a_5, 45);\n            if (!among_var)\n              return false;\n            sbp.bra = sbp.cursor;\n            switch (among_var) {\n              case 1:\n                if (!r_R2())\n                  return false;\n                sbp.slice_del();\n                break;\n              case 2:\n                if (!r_R2())\n                  return false;\n                sbp.slice_from(\"log\");\n                break;\n              case 3:\n                if (!r_R2())\n                  return false;\n                sbp.slice_from(\"u\");\n                break;\n              case 4:\n                if (!r_R2())\n                  return false;\n                sbp.slice_from(\"ente\");\n                break;\n              case 5:\n                if (!r_R1())\n                  return false;\n                sbp.slice_del();\n                sbp.ket = sbp.cursor;\n                among_var = sbp.find_among_b(a_2, 4);\n                if (among_var) {\n                  sbp.bra = sbp.cursor;\n                  if (r_R2()) {\n                    sbp.slice_del();\n                    if (among_var == 1) {\n                      sbp.ket = sbp.cursor;\n                      if (sbp.eq_s_b(2, \"at\")) {\n                        sbp.bra = sbp.cursor;\n                        if (r_R2())\n                          sbp.slice_del();\n                      }\n                    }\n                  }\n                }\n                break;\n              case 6:\n                if (!r_R2())\n                  return false;\n                sbp.slice_del();\n                sbp.ket = sbp.cursor;\n                among_var = sbp.find_among_b(a_3, 3);\n                if (among_var) {\n                  sbp.bra = sbp.cursor;\n                  if (among_var == 1)\n                    if (r_R2())\n                      sbp.slice_del();\n                }\n                break;\n              case 7:\n                if (!r_R2())\n                  return false;\n                sbp.slice_del();\n                sbp.ket = sbp.cursor;\n                among_var = sbp.find_among_b(a_4, 3);\n                if (among_var) {\n                  sbp.bra = sbp.cursor;\n                  if (among_var == 1)\n                    if (r_R2())\n                      sbp.slice_del();\n                }\n                break;\n              case 8:\n                if (!r_R2())\n                  return false;\n                sbp.slice_del();\n                sbp.ket = sbp.cursor;\n                if (sbp.eq_s_b(2, \"at\")) {\n                  sbp.bra = sbp.cursor;\n                  if (r_R2())\n                    sbp.slice_del();\n                }\n                break;\n              case 9:\n                if (!r_RV() || !sbp.eq_s_b(1, \"e\"))\n                  return false;\n                sbp.slice_from(\"ir\");\n                break;\n            }\n            return true;\n          }\n\n          function r_verb_suffix() {\n            var among_var, v_1;\n            if (sbp.cursor >= I_pV) {\n              v_1 = sbp.limit_backward;\n              sbp.limit_backward = I_pV;\n              sbp.ket = sbp.cursor;\n              among_var = sbp.find_among_b(a_6, 120);\n              if (among_var) {\n                sbp.bra = sbp.cursor;\n                if (among_var == 1)\n                  sbp.slice_del();\n                sbp.limit_backward = v_1;\n                return true;\n              }\n              sbp.limit_backward = v_1;\n            }\n            return false;\n          }\n\n          function r_residual_suffix() {\n            var among_var;\n            sbp.ket = sbp.cursor;\n            among_var = sbp.find_among_b(a_7, 7);\n            if (among_var) {\n              sbp.bra = sbp.cursor;\n              if (among_var == 1)\n                if (r_RV())\n                  sbp.slice_del();\n            }\n          }\n\n          function habr6(c1, c2) {\n            if (sbp.eq_s_b(1, c1)) {\n              sbp.bra = sbp.cursor;\n              var v_1 = sbp.limit - sbp.cursor;\n              if (sbp.eq_s_b(1, c2)) {\n                sbp.cursor = sbp.limit - v_1;\n                if (r_RV())\n                  sbp.slice_del();\n                return false;\n              }\n            }\n            return true;\n          }\n\n          function r_residual_form() {\n            var among_var, v_1, v_2, v_3;\n            sbp.ket = sbp.cursor;\n            among_var = sbp.find_among_b(a_8, 4);\n            if (among_var) {\n              sbp.bra = sbp.cursor;\n              switch (among_var) {\n                case 1:\n                  if (r_RV()) {\n                    sbp.slice_del();\n                    sbp.ket = sbp.cursor;\n                    v_1 = sbp.limit - sbp.cursor;\n                    if (habr6(\"u\", \"g\"))\n                      habr6(\"i\", \"c\")\n                  }\n                  break;\n                case 2:\n                  sbp.slice_from(\"c\");\n                  break;\n              }\n            }\n          }\n\n          function habr1() {\n            if (!r_standard_suffix()) {\n              sbp.cursor = sbp.limit;\n              if (!r_verb_suffix()) {\n                sbp.cursor = sbp.limit;\n                r_residual_suffix();\n                return;\n              }\n            }\n            sbp.cursor = sbp.limit;\n            sbp.ket = sbp.cursor;\n            if (sbp.eq_s_b(1, \"i\")) {\n              sbp.bra = sbp.cursor;\n              if (sbp.eq_s_b(1, \"c\")) {\n                sbp.cursor = sbp.limit;\n                if (r_RV())\n                  sbp.slice_del();\n              }\n            }\n          }\n          this.stem = function() {\n            var v_1 = sbp.cursor;\n            r_prelude();\n            sbp.cursor = v_1;\n            r_mark_regions();\n            sbp.limit_backward = v_1;\n            sbp.cursor = sbp.limit;\n            habr1();\n            sbp.cursor = sbp.limit;\n            r_residual_form();\n            sbp.cursor = sbp.limit_backward;\n            r_postlude();\n            return true;\n          }\n        };\n\n      /* and return a function that stems a word for the current locale */\n      return function(word) {\n        st.setCurrent(word);\n        st.stem();\n        return st.getCurrent();\n      }\n    })();\n\n    lunr.Pipeline.registerFunction(lunr.pt.stemmer, 'stemmer-pt');\n\n    /* stop word filter function */\n    lunr.pt.stopWordFilter = function(token) {\n      if (lunr.pt.stopWordFilter.stopWords.indexOf(token) === -1) {\n        return token;\n      }\n    };\n\n    lunr.pt.stopWordFilter.stopWords = new lunr.SortedSet();\n    lunr.pt.stopWordFilter.stopWords.length = 204;\n\n    // The space at the beginning is crucial: It marks the empty string\n    // as a stop word. lunr.js crashes during search when documents\n    // processed by the pipeline still contain the empty string.\n    lunr.pt.stopWordFilter.stopWords.elements = ' a ao aos aquela aquelas aquele aqueles aquilo as até com como da das de dela delas dele deles depois do dos e ela elas ele eles em entre era eram essa essas esse esses esta estamos estas estava estavam este esteja estejam estejamos estes esteve estive estivemos estiver estivera estiveram estiverem estivermos estivesse estivessem estivéramos estivéssemos estou está estávamos estão eu foi fomos for fora foram forem formos fosse fossem fui fôramos fôssemos haja hajam hajamos havemos hei houve houvemos houver houvera houveram houverei houverem houveremos houveria houveriam houvermos houverá houverão houveríamos houvesse houvessem houvéramos houvéssemos há hão isso isto já lhe lhes mais mas me mesmo meu meus minha minhas muito na nas nem no nos nossa nossas nosso nossos num numa não nós o os ou para pela pelas pelo pelos por qual quando que quem se seja sejam sejamos sem serei seremos seria seriam será serão seríamos seu seus somos sou sua suas são só também te tem temos tenha tenham tenhamos tenho terei teremos teria teriam terá terão teríamos teu teus teve tinha tinham tive tivemos tiver tivera tiveram tiverem tivermos tivesse tivessem tivéramos tivéssemos tu tua tuas tém tínhamos um uma você vocês vos à às éramos'.split(' ');\n\n    lunr.Pipeline.registerFunction(lunr.pt.stopWordFilter, 'stopWordFilter-pt');\n  };\n}))\n"
  },
  {
    "path": "static/js/lunr/lunr.ro.js",
    "content": "/*!\n * Lunr languages, `Romanian` language\n * https://github.com/MihaiValentin/lunr-languages\n *\n * Copyright 2014, Mihai Valentin\n * http://www.mozilla.org/MPL/\n */\n/*!\n * based on\n * Snowball JavaScript Library v0.3\n * http://code.google.com/p/urim/\n * http://snowball.tartarus.org/\n *\n * Copyright 2010, Oleg Mazko\n * http://www.mozilla.org/MPL/\n */\n\n/**\n * export the module via AMD, CommonJS or as a browser global\n * Export code from https://github.com/umdjs/umd/blob/master/returnExports.js\n */\n;\n(function(root, factory) {\n  if (typeof define === 'function' && define.amd) {\n    // AMD. Register as an anonymous module.\n    define(factory)\n  } else if (typeof exports === 'object') {\n    /**\n     * Node. Does not work with strict CommonJS, but\n     * only CommonJS-like environments that support module.exports,\n     * like Node.\n     */\n    module.exports = factory()\n  } else {\n    // Browser globals (root is window)\n    factory()(root.lunr);\n  }\n}(this, function() {\n  /**\n   * Just return a value to define the module export.\n   * This example returns an object, but the module\n   * can return a function as the exported value.\n   */\n  return function(lunr) {\n    /* throw error if lunr is not yet included */\n    if ('undefined' === typeof lunr) {\n      throw new Error('Lunr is not present. Please include / require Lunr before this script.');\n    }\n\n    /* throw error if lunr stemmer support is not yet included */\n    if ('undefined' === typeof lunr.stemmerSupport) {\n      throw new Error('Lunr stemmer support is not present. Please include / require Lunr stemmer support before this script.');\n    }\n\n    /* register specific locale function */\n    lunr.ro = function() {\n      this.pipeline.reset();\n      this.pipeline.add(\n        lunr.ro.trimmer,\n        lunr.ro.stopWordFilter,\n        lunr.ro.stemmer\n      );\n    };\n\n    /* lunr trimmer function */\n    lunr.ro.wordCharacters = \"A-Za-z\\xAA\\xBA\\xC0-\\xD6\\xD8-\\xF6\\xF8-\\u02B8\\u02E0-\\u02E4\\u1D00-\\u1D25\\u1D2C-\\u1D5C\\u1D62-\\u1D65\\u1D6B-\\u1D77\\u1D79-\\u1DBE\\u1E00-\\u1EFF\\u2071\\u207F\\u2090-\\u209C\\u212A\\u212B\\u2132\\u214E\\u2160-\\u2188\\u2C60-\\u2C7F\\uA722-\\uA787\\uA78B-\\uA7AD\\uA7B0-\\uA7B7\\uA7F7-\\uA7FF\\uAB30-\\uAB5A\\uAB5C-\\uAB64\\uFB00-\\uFB06\\uFF21-\\uFF3A\\uFF41-\\uFF5A\";\n    lunr.ro.trimmer = lunr.trimmerSupport.generateTrimmer(lunr.ro.wordCharacters);\n\n    lunr.Pipeline.registerFunction(lunr.ro.trimmer, 'trimmer-ro');\n\n    /* lunr stemmer function */\n    lunr.ro.stemmer = (function() {\n      /* create the wrapped stemmer object */\n      var Among = lunr.stemmerSupport.Among,\n        SnowballProgram = lunr.stemmerSupport.SnowballProgram,\n        st = new function RomanianStemmer() {\n          var a_0 = [new Among(\"\", -1, 3), new Among(\"I\", 0, 1), new Among(\"U\", 0, 2)],\n            a_1 = [\n              new Among(\"ea\", -1, 3), new Among(\"a\\u0163ia\", -1, 7),\n              new Among(\"aua\", -1, 2), new Among(\"iua\", -1, 4),\n              new Among(\"a\\u0163ie\", -1, 7), new Among(\"ele\", -1, 3),\n              new Among(\"ile\", -1, 5), new Among(\"iile\", 6, 4),\n              new Among(\"iei\", -1, 4), new Among(\"atei\", -1, 6),\n              new Among(\"ii\", -1, 4), new Among(\"ului\", -1, 1),\n              new Among(\"ul\", -1, 1), new Among(\"elor\", -1, 3),\n              new Among(\"ilor\", -1, 4), new Among(\"iilor\", 14, 4)\n            ],\n            a_2 = [\n              new Among(\"icala\", -1, 4), new Among(\"iciva\", -1, 4),\n              new Among(\"ativa\", -1, 5), new Among(\"itiva\", -1, 6),\n              new Among(\"icale\", -1, 4), new Among(\"a\\u0163iune\", -1, 5),\n              new Among(\"i\\u0163iune\", -1, 6), new Among(\"atoare\", -1, 5),\n              new Among(\"itoare\", -1, 6), new Among(\"\\u0103toare\", -1, 5),\n              new Among(\"icitate\", -1, 4), new Among(\"abilitate\", -1, 1),\n              new Among(\"ibilitate\", -1, 2), new Among(\"ivitate\", -1, 3),\n              new Among(\"icive\", -1, 4), new Among(\"ative\", -1, 5),\n              new Among(\"itive\", -1, 6), new Among(\"icali\", -1, 4),\n              new Among(\"atori\", -1, 5), new Among(\"icatori\", 18, 4),\n              new Among(\"itori\", -1, 6), new Among(\"\\u0103tori\", -1, 5),\n              new Among(\"icitati\", -1, 4), new Among(\"abilitati\", -1, 1),\n              new Among(\"ivitati\", -1, 3), new Among(\"icivi\", -1, 4),\n              new Among(\"ativi\", -1, 5), new Among(\"itivi\", -1, 6),\n              new Among(\"icit\\u0103i\", -1, 4), new Among(\"abilit\\u0103i\", -1, 1),\n              new Among(\"ivit\\u0103i\", -1, 3),\n              new Among(\"icit\\u0103\\u0163i\", -1, 4),\n              new Among(\"abilit\\u0103\\u0163i\", -1, 1),\n              new Among(\"ivit\\u0103\\u0163i\", -1, 3), new Among(\"ical\", -1, 4),\n              new Among(\"ator\", -1, 5), new Among(\"icator\", 35, 4),\n              new Among(\"itor\", -1, 6), new Among(\"\\u0103tor\", -1, 5),\n              new Among(\"iciv\", -1, 4), new Among(\"ativ\", -1, 5),\n              new Among(\"itiv\", -1, 6), new Among(\"ical\\u0103\", -1, 4),\n              new Among(\"iciv\\u0103\", -1, 4), new Among(\"ativ\\u0103\", -1, 5),\n              new Among(\"itiv\\u0103\", -1, 6)\n            ],\n            a_3 = [new Among(\"ica\", -1, 1),\n              new Among(\"abila\", -1, 1), new Among(\"ibila\", -1, 1),\n              new Among(\"oasa\", -1, 1), new Among(\"ata\", -1, 1),\n              new Among(\"ita\", -1, 1), new Among(\"anta\", -1, 1),\n              new Among(\"ista\", -1, 3), new Among(\"uta\", -1, 1),\n              new Among(\"iva\", -1, 1), new Among(\"ic\", -1, 1),\n              new Among(\"ice\", -1, 1), new Among(\"abile\", -1, 1),\n              new Among(\"ibile\", -1, 1), new Among(\"isme\", -1, 3),\n              new Among(\"iune\", -1, 2), new Among(\"oase\", -1, 1),\n              new Among(\"ate\", -1, 1), new Among(\"itate\", 17, 1),\n              new Among(\"ite\", -1, 1), new Among(\"ante\", -1, 1),\n              new Among(\"iste\", -1, 3), new Among(\"ute\", -1, 1),\n              new Among(\"ive\", -1, 1), new Among(\"ici\", -1, 1),\n              new Among(\"abili\", -1, 1), new Among(\"ibili\", -1, 1),\n              new Among(\"iuni\", -1, 2), new Among(\"atori\", -1, 1),\n              new Among(\"osi\", -1, 1), new Among(\"ati\", -1, 1),\n              new Among(\"itati\", 30, 1), new Among(\"iti\", -1, 1),\n              new Among(\"anti\", -1, 1), new Among(\"isti\", -1, 3),\n              new Among(\"uti\", -1, 1), new Among(\"i\\u015Fti\", -1, 3),\n              new Among(\"ivi\", -1, 1), new Among(\"it\\u0103i\", -1, 1),\n              new Among(\"o\\u015Fi\", -1, 1), new Among(\"it\\u0103\\u0163i\", -1, 1),\n              new Among(\"abil\", -1, 1), new Among(\"ibil\", -1, 1),\n              new Among(\"ism\", -1, 3), new Among(\"ator\", -1, 1),\n              new Among(\"os\", -1, 1), new Among(\"at\", -1, 1),\n              new Among(\"it\", -1, 1), new Among(\"ant\", -1, 1),\n              new Among(\"ist\", -1, 3), new Among(\"ut\", -1, 1),\n              new Among(\"iv\", -1, 1), new Among(\"ic\\u0103\", -1, 1),\n              new Among(\"abil\\u0103\", -1, 1), new Among(\"ibil\\u0103\", -1, 1),\n              new Among(\"oas\\u0103\", -1, 1), new Among(\"at\\u0103\", -1, 1),\n              new Among(\"it\\u0103\", -1, 1), new Among(\"ant\\u0103\", -1, 1),\n              new Among(\"ist\\u0103\", -1, 3), new Among(\"ut\\u0103\", -1, 1),\n              new Among(\"iv\\u0103\", -1, 1)\n            ],\n            a_4 = [new Among(\"ea\", -1, 1),\n              new Among(\"ia\", -1, 1), new Among(\"esc\", -1, 1),\n              new Among(\"\\u0103sc\", -1, 1), new Among(\"ind\", -1, 1),\n              new Among(\"\\u00E2nd\", -1, 1), new Among(\"are\", -1, 1),\n              new Among(\"ere\", -1, 1), new Among(\"ire\", -1, 1),\n              new Among(\"\\u00E2re\", -1, 1), new Among(\"se\", -1, 2),\n              new Among(\"ase\", 10, 1), new Among(\"sese\", 10, 2),\n              new Among(\"ise\", 10, 1), new Among(\"use\", 10, 1),\n              new Among(\"\\u00E2se\", 10, 1), new Among(\"e\\u015Fte\", -1, 1),\n              new Among(\"\\u0103\\u015Fte\", -1, 1), new Among(\"eze\", -1, 1),\n              new Among(\"ai\", -1, 1), new Among(\"eai\", 19, 1),\n              new Among(\"iai\", 19, 1), new Among(\"sei\", -1, 2),\n              new Among(\"e\\u015Fti\", -1, 1), new Among(\"\\u0103\\u015Fti\", -1, 1),\n              new Among(\"ui\", -1, 1), new Among(\"ezi\", -1, 1),\n              new Among(\"\\u00E2i\", -1, 1), new Among(\"a\\u015Fi\", -1, 1),\n              new Among(\"se\\u015Fi\", -1, 2), new Among(\"ase\\u015Fi\", 29, 1),\n              new Among(\"sese\\u015Fi\", 29, 2), new Among(\"ise\\u015Fi\", 29, 1),\n              new Among(\"use\\u015Fi\", 29, 1),\n              new Among(\"\\u00E2se\\u015Fi\", 29, 1), new Among(\"i\\u015Fi\", -1, 1),\n              new Among(\"u\\u015Fi\", -1, 1), new Among(\"\\u00E2\\u015Fi\", -1, 1),\n              new Among(\"a\\u0163i\", -1, 2), new Among(\"ea\\u0163i\", 38, 1),\n              new Among(\"ia\\u0163i\", 38, 1), new Among(\"e\\u0163i\", -1, 2),\n              new Among(\"i\\u0163i\", -1, 2), new Among(\"\\u00E2\\u0163i\", -1, 2),\n              new Among(\"ar\\u0103\\u0163i\", -1, 1),\n              new Among(\"ser\\u0103\\u0163i\", -1, 2),\n              new Among(\"aser\\u0103\\u0163i\", 45, 1),\n              new Among(\"seser\\u0103\\u0163i\", 45, 2),\n              new Among(\"iser\\u0103\\u0163i\", 45, 1),\n              new Among(\"user\\u0103\\u0163i\", 45, 1),\n              new Among(\"\\u00E2ser\\u0103\\u0163i\", 45, 1),\n              new Among(\"ir\\u0103\\u0163i\", -1, 1),\n              new Among(\"ur\\u0103\\u0163i\", -1, 1),\n              new Among(\"\\u00E2r\\u0103\\u0163i\", -1, 1), new Among(\"am\", -1, 1),\n              new Among(\"eam\", 54, 1), new Among(\"iam\", 54, 1),\n              new Among(\"em\", -1, 2), new Among(\"asem\", 57, 1),\n              new Among(\"sesem\", 57, 2), new Among(\"isem\", 57, 1),\n              new Among(\"usem\", 57, 1), new Among(\"\\u00E2sem\", 57, 1),\n              new Among(\"im\", -1, 2), new Among(\"\\u00E2m\", -1, 2),\n              new Among(\"\\u0103m\", -1, 2), new Among(\"ar\\u0103m\", 65, 1),\n              new Among(\"ser\\u0103m\", 65, 2), new Among(\"aser\\u0103m\", 67, 1),\n              new Among(\"seser\\u0103m\", 67, 2), new Among(\"iser\\u0103m\", 67, 1),\n              new Among(\"user\\u0103m\", 67, 1),\n              new Among(\"\\u00E2ser\\u0103m\", 67, 1),\n              new Among(\"ir\\u0103m\", 65, 1), new Among(\"ur\\u0103m\", 65, 1),\n              new Among(\"\\u00E2r\\u0103m\", 65, 1), new Among(\"au\", -1, 1),\n              new Among(\"eau\", 76, 1), new Among(\"iau\", 76, 1),\n              new Among(\"indu\", -1, 1), new Among(\"\\u00E2ndu\", -1, 1),\n              new Among(\"ez\", -1, 1), new Among(\"easc\\u0103\", -1, 1),\n              new Among(\"ar\\u0103\", -1, 1), new Among(\"ser\\u0103\", -1, 2),\n              new Among(\"aser\\u0103\", 84, 1), new Among(\"seser\\u0103\", 84, 2),\n              new Among(\"iser\\u0103\", 84, 1), new Among(\"user\\u0103\", 84, 1),\n              new Among(\"\\u00E2ser\\u0103\", 84, 1), new Among(\"ir\\u0103\", -1, 1),\n              new Among(\"ur\\u0103\", -1, 1), new Among(\"\\u00E2r\\u0103\", -1, 1),\n              new Among(\"eaz\\u0103\", -1, 1)\n            ],\n            a_5 = [new Among(\"a\", -1, 1),\n              new Among(\"e\", -1, 1), new Among(\"ie\", 1, 1),\n              new Among(\"i\", -1, 1), new Among(\"\\u0103\", -1, 1)\n            ],\n            g_v = [17, 65,\n              16, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 2, 32, 0, 0, 4\n            ],\n            B_standard_suffix_removed, I_p2, I_p1, I_pV, sbp = new SnowballProgram();\n          this.setCurrent = function(word) {\n            sbp.setCurrent(word);\n          };\n          this.getCurrent = function() {\n            return sbp.getCurrent();\n          };\n\n          function habr1(c1, c2) {\n            if (sbp.eq_s(1, c1)) {\n              sbp.ket = sbp.cursor;\n              if (sbp.in_grouping(g_v, 97, 259))\n                sbp.slice_from(c2);\n            }\n          }\n\n          function r_prelude() {\n            var v_1, v_2;\n            while (true) {\n              v_1 = sbp.cursor;\n              if (sbp.in_grouping(g_v, 97, 259)) {\n                v_2 = sbp.cursor;\n                sbp.bra = v_2;\n                habr1(\"u\", \"U\");\n                sbp.cursor = v_2;\n                habr1(\"i\", \"I\");\n              }\n              sbp.cursor = v_1;\n              if (sbp.cursor >= sbp.limit) {\n                break;\n              }\n              sbp.cursor++;\n            }\n          }\n\n          function habr2() {\n            if (sbp.out_grouping(g_v, 97, 259)) {\n              while (!sbp.in_grouping(g_v, 97, 259)) {\n                if (sbp.cursor >= sbp.limit)\n                  return true;\n                sbp.cursor++;\n              }\n              return false;\n            }\n            return true;\n          }\n\n          function habr3() {\n            if (sbp.in_grouping(g_v, 97, 259)) {\n              while (!sbp.out_grouping(g_v, 97, 259)) {\n                if (sbp.cursor >= sbp.limit)\n                  return true;\n                sbp.cursor++;\n              }\n            }\n            return false;\n          }\n\n          function habr4() {\n            var v_1 = sbp.cursor,\n              v_2, v_3;\n            if (sbp.in_grouping(g_v, 97, 259)) {\n              v_2 = sbp.cursor;\n              if (habr2()) {\n                sbp.cursor = v_2;\n                if (!habr3()) {\n                  I_pV = sbp.cursor;\n                  return;\n                }\n              } else {\n                I_pV = sbp.cursor;\n                return;\n              }\n            }\n            sbp.cursor = v_1;\n            if (sbp.out_grouping(g_v, 97, 259)) {\n              v_3 = sbp.cursor;\n              if (habr2()) {\n                sbp.cursor = v_3;\n                if (sbp.in_grouping(g_v, 97, 259) && sbp.cursor < sbp.limit)\n                  sbp.cursor++;\n              }\n              I_pV = sbp.cursor;\n            }\n          }\n\n          function habr5() {\n            while (!sbp.in_grouping(g_v, 97, 259)) {\n              if (sbp.cursor >= sbp.limit)\n                return false;\n              sbp.cursor++;\n            }\n            while (!sbp.out_grouping(g_v, 97, 259)) {\n              if (sbp.cursor >= sbp.limit)\n                return false;\n              sbp.cursor++;\n            }\n            return true;\n          }\n\n          function r_mark_regions() {\n            var v_1 = sbp.cursor;\n            I_pV = sbp.limit;\n            I_p1 = I_pV;\n            I_p2 = I_pV;\n            habr4();\n            sbp.cursor = v_1;\n            if (habr5()) {\n              I_p1 = sbp.cursor;\n              if (habr5())\n                I_p2 = sbp.cursor;\n            }\n          }\n\n          function r_postlude() {\n            var among_var;\n            while (true) {\n              sbp.bra = sbp.cursor;\n              among_var = sbp.find_among(a_0, 3);\n              if (among_var) {\n                sbp.ket = sbp.cursor;\n                switch (among_var) {\n                  case 1:\n                    sbp.slice_from(\"i\");\n                    continue;\n                  case 2:\n                    sbp.slice_from(\"u\");\n                    continue;\n                  case 3:\n                    if (sbp.cursor >= sbp.limit)\n                      break;\n                    sbp.cursor++;\n                    continue;\n                }\n              }\n              break;\n            }\n          }\n\n          function r_RV() {\n            return I_pV <= sbp.cursor;\n          }\n\n          function r_R1() {\n            return I_p1 <= sbp.cursor;\n          }\n\n          function r_R2() {\n            return I_p2 <= sbp.cursor;\n          }\n\n          function r_step_0() {\n            var among_var, v_1;\n            sbp.ket = sbp.cursor;\n            among_var = sbp.find_among_b(a_1, 16);\n            if (among_var) {\n              sbp.bra = sbp.cursor;\n              if (r_R1()) {\n                switch (among_var) {\n                  case 1:\n                    sbp.slice_del();\n                    break;\n                  case 2:\n                    sbp.slice_from(\"a\");\n                    break;\n                  case 3:\n                    sbp.slice_from(\"e\");\n                    break;\n                  case 4:\n                    sbp.slice_from(\"i\");\n                    break;\n                  case 5:\n                    v_1 = sbp.limit - sbp.cursor;\n                    if (!sbp.eq_s_b(2, \"ab\")) {\n                      sbp.cursor = sbp.limit - v_1;\n                      sbp.slice_from(\"i\");\n                    }\n                    break;\n                  case 6:\n                    sbp.slice_from(\"at\");\n                    break;\n                  case 7:\n                    sbp.slice_from(\"a\\u0163i\");\n                    break;\n                }\n              }\n            }\n          }\n\n          function r_combo_suffix() {\n            var among_var, v_1 = sbp.limit - sbp.cursor;\n            sbp.ket = sbp.cursor;\n            among_var = sbp.find_among_b(a_2, 46);\n            if (among_var) {\n              sbp.bra = sbp.cursor;\n              if (r_R1()) {\n                switch (among_var) {\n                  case 1:\n                    sbp.slice_from(\"abil\");\n                    break;\n                  case 2:\n                    sbp.slice_from(\"ibil\");\n                    break;\n                  case 3:\n                    sbp.slice_from(\"iv\");\n                    break;\n                  case 4:\n                    sbp.slice_from(\"ic\");\n                    break;\n                  case 5:\n                    sbp.slice_from(\"at\");\n                    break;\n                  case 6:\n                    sbp.slice_from(\"it\");\n                    break;\n                }\n                B_standard_suffix_removed = true;\n                sbp.cursor = sbp.limit - v_1;\n                return true;\n              }\n            }\n            return false;\n          }\n\n          function r_standard_suffix() {\n            var among_var, v_1;\n            B_standard_suffix_removed = false;\n            while (true) {\n              v_1 = sbp.limit - sbp.cursor;\n              if (!r_combo_suffix()) {\n                sbp.cursor = sbp.limit - v_1;\n                break;\n              }\n            }\n            sbp.ket = sbp.cursor;\n            among_var = sbp.find_among_b(a_3, 62);\n            if (among_var) {\n              sbp.bra = sbp.cursor;\n              if (r_R2()) {\n                switch (among_var) {\n                  case 1:\n                    sbp.slice_del();\n                    break;\n                  case 2:\n                    if (sbp.eq_s_b(1, \"\\u0163\")) {\n                      sbp.bra = sbp.cursor;\n                      sbp.slice_from(\"t\");\n                    }\n                    break;\n                  case 3:\n                    sbp.slice_from(\"ist\");\n                    break;\n                }\n                B_standard_suffix_removed = true;\n              }\n            }\n          }\n\n          function r_verb_suffix() {\n            var among_var, v_1, v_2;\n            if (sbp.cursor >= I_pV) {\n              v_1 = sbp.limit_backward;\n              sbp.limit_backward = I_pV;\n              sbp.ket = sbp.cursor;\n              among_var = sbp.find_among_b(a_4, 94);\n              if (among_var) {\n                sbp.bra = sbp.cursor;\n                switch (among_var) {\n                  case 1:\n                    v_2 = sbp.limit - sbp.cursor;\n                    if (!sbp.out_grouping_b(g_v, 97, 259)) {\n                      sbp.cursor = sbp.limit - v_2;\n                      if (!sbp.eq_s_b(1, \"u\"))\n                        break;\n                    }\n                  case 2:\n                    sbp.slice_del();\n                    break;\n                }\n              }\n              sbp.limit_backward = v_1;\n            }\n          }\n\n          function r_vowel_suffix() {\n            var among_var;\n            sbp.ket = sbp.cursor;\n            among_var = sbp.find_among_b(a_5, 5);\n            if (among_var) {\n              sbp.bra = sbp.cursor;\n              if (r_RV() && among_var == 1)\n                sbp.slice_del();\n            }\n          }\n          this.stem = function() {\n            var v_1 = sbp.cursor;\n            r_prelude();\n            sbp.cursor = v_1;\n            r_mark_regions();\n            sbp.limit_backward = v_1;\n            sbp.cursor = sbp.limit;\n            r_step_0();\n            sbp.cursor = sbp.limit;\n            r_standard_suffix();\n            sbp.cursor = sbp.limit;\n            if (!B_standard_suffix_removed) {\n              sbp.cursor = sbp.limit;\n              r_verb_suffix();\n              sbp.cursor = sbp.limit;\n            }\n            r_vowel_suffix();\n            sbp.cursor = sbp.limit_backward;\n            r_postlude();\n            return true;\n          }\n        };\n\n      /* and return a function that stems a word for the current locale */\n      return function(word) {\n        st.setCurrent(word);\n        st.stem();\n        return st.getCurrent();\n      }\n    })();\n\n    lunr.Pipeline.registerFunction(lunr.ro.stemmer, 'stemmer-ro');\n\n    /* stop word filter function */\n    lunr.ro.stopWordFilter = function(token) {\n      if (lunr.ro.stopWordFilter.stopWords.indexOf(token) === -1) {\n        return token;\n      }\n    };\n\n    lunr.ro.stopWordFilter.stopWords = new lunr.SortedSet();\n    lunr.ro.stopWordFilter.stopWords.length = 282;\n\n    // The space at the beginning is crucial: It marks the empty string\n    // as a stop word. lunr.js crashes during search when documents\n    // processed by the pipeline still contain the empty string.\n    lunr.ro.stopWordFilter.stopWords.elements = ' acea aceasta această aceea acei aceia acel acela acele acelea acest acesta aceste acestea aceşti aceştia acolo acord acum ai aia aibă aici al ale alea altceva altcineva am ar are asemenea asta astea astăzi asupra au avea avem aveţi azi aş aşadar aţi bine bucur bună ca care caut ce cel ceva chiar cinci cine cineva contra cu cum cumva curând curînd când cât câte câtva câţi cînd cît cîte cîtva cîţi că căci cărei căror cărui către da dacă dar datorită dată dau de deci deja deoarece departe deşi din dinaintea dintr- dintre doi doilea două drept după dă ea ei el ele eram este eu eşti face fata fi fie fiecare fii fim fiu fiţi frumos fără graţie halbă iar ieri la le li lor lui lângă lîngă mai mea mei mele mereu meu mi mie mine mult multă mulţi mulţumesc mâine mîine mă ne nevoie nici nicăieri nimeni nimeri nimic nişte noastre noastră noi noroc nostru nouă noştri nu opt ori oricare orice oricine oricum oricând oricât oricînd oricît oriunde patra patru patrulea pe pentru peste pic poate pot prea prima primul prin puţin puţina puţină până pînă rog sa sale sau se spate spre sub sunt suntem sunteţi sută sînt sîntem sînteţi să săi său ta tale te timp tine toate toată tot totuşi toţi trei treia treilea tu tăi tău un una unde undeva unei uneia unele uneori unii unor unora unu unui unuia unul vi voastre voastră voi vostru vouă voştri vreme vreo vreun vă zece zero zi zice îi îl îmi împotriva în  înainte înaintea încotro încât încît între întrucât întrucît îţi ăla ălea ăsta ăstea ăştia şapte şase şi ştiu ţi ţie'.split(' ');\n\n    lunr.Pipeline.registerFunction(lunr.ro.stopWordFilter, 'stopWordFilter-ro');\n  };\n}))\n"
  },
  {
    "path": "static/js/lunr/lunr.ru.js",
    "content": "/*!\n * Lunr languages, `Russian` language\n * https://github.com/MihaiValentin/lunr-languages\n *\n * Copyright 2014, Mihai Valentin\n * http://www.mozilla.org/MPL/\n */\n/*!\n * based on\n * Snowball JavaScript Library v0.3\n * http://code.google.com/p/urim/\n * http://snowball.tartarus.org/\n *\n * Copyright 2010, Oleg Mazko\n * http://www.mozilla.org/MPL/\n */\n\n/**\n * export the module via AMD, CommonJS or as a browser global\n * Export code from https://github.com/umdjs/umd/blob/master/returnExports.js\n */\n;\n(function(root, factory) {\n  if (typeof define === 'function' && define.amd) {\n    // AMD. Register as an anonymous module.\n    define(factory)\n  } else if (typeof exports === 'object') {\n    /**\n     * Node. Does not work with strict CommonJS, but\n     * only CommonJS-like environments that support module.exports,\n     * like Node.\n     */\n    module.exports = factory()\n  } else {\n    // Browser globals (root is window)\n    factory()(root.lunr);\n  }\n}(this, function() {\n  /**\n   * Just return a value to define the module export.\n   * This example returns an object, but the module\n   * can return a function as the exported value.\n   */\n  return function(lunr) {\n    /* throw error if lunr is not yet included */\n    if ('undefined' === typeof lunr) {\n      throw new Error('Lunr is not present. Please include / require Lunr before this script.');\n    }\n\n    /* throw error if lunr stemmer support is not yet included */\n    if ('undefined' === typeof lunr.stemmerSupport) {\n      throw new Error('Lunr stemmer support is not present. Please include / require Lunr stemmer support before this script.');\n    }\n\n    /* register specific locale function */\n    lunr.ru = function() {\n      this.pipeline.reset();\n      this.pipeline.add(\n        lunr.ru.trimmer,\n        lunr.ru.stopWordFilter,\n        lunr.ru.stemmer\n      );\n    };\n\n    /* lunr trimmer function */\n    lunr.ru.wordCharacters = \"\\u0400-\\u0484\\u0487-\\u052F\\u1D2B\\u1D78\\u2DE0-\\u2DFF\\uA640-\\uA69F\\uFE2E\\uFE2F\";\n    lunr.ru.trimmer = lunr.trimmerSupport.generateTrimmer(lunr.ru.wordCharacters);\n\n    lunr.Pipeline.registerFunction(lunr.ru.trimmer, 'trimmer-ru');\n\n    /* lunr stemmer function */\n    lunr.ru.stemmer = (function() {\n      /* create the wrapped stemmer object */\n      var Among = lunr.stemmerSupport.Among,\n        SnowballProgram = lunr.stemmerSupport.SnowballProgram,\n        st = new function RussianStemmer() {\n          var a_0 = [new Among(\"\\u0432\", -1, 1), new Among(\"\\u0438\\u0432\", 0, 2),\n              new Among(\"\\u044B\\u0432\", 0, 2),\n              new Among(\"\\u0432\\u0448\\u0438\", -1, 1),\n              new Among(\"\\u0438\\u0432\\u0448\\u0438\", 3, 2),\n              new Among(\"\\u044B\\u0432\\u0448\\u0438\", 3, 2),\n              new Among(\"\\u0432\\u0448\\u0438\\u0441\\u044C\", -1, 1),\n              new Among(\"\\u0438\\u0432\\u0448\\u0438\\u0441\\u044C\", 6, 2),\n              new Among(\"\\u044B\\u0432\\u0448\\u0438\\u0441\\u044C\", 6, 2)\n            ],\n            a_1 = [\n              new Among(\"\\u0435\\u0435\", -1, 1), new Among(\"\\u0438\\u0435\", -1, 1),\n              new Among(\"\\u043E\\u0435\", -1, 1), new Among(\"\\u044B\\u0435\", -1, 1),\n              new Among(\"\\u0438\\u043C\\u0438\", -1, 1),\n              new Among(\"\\u044B\\u043C\\u0438\", -1, 1),\n              new Among(\"\\u0435\\u0439\", -1, 1), new Among(\"\\u0438\\u0439\", -1, 1),\n              new Among(\"\\u043E\\u0439\", -1, 1), new Among(\"\\u044B\\u0439\", -1, 1),\n              new Among(\"\\u0435\\u043C\", -1, 1), new Among(\"\\u0438\\u043C\", -1, 1),\n              new Among(\"\\u043E\\u043C\", -1, 1), new Among(\"\\u044B\\u043C\", -1, 1),\n              new Among(\"\\u0435\\u0433\\u043E\", -1, 1),\n              new Among(\"\\u043E\\u0433\\u043E\", -1, 1),\n              new Among(\"\\u0435\\u043C\\u0443\", -1, 1),\n              new Among(\"\\u043E\\u043C\\u0443\", -1, 1),\n              new Among(\"\\u0438\\u0445\", -1, 1), new Among(\"\\u044B\\u0445\", -1, 1),\n              new Among(\"\\u0435\\u044E\", -1, 1), new Among(\"\\u043E\\u044E\", -1, 1),\n              new Among(\"\\u0443\\u044E\", -1, 1), new Among(\"\\u044E\\u044E\", -1, 1),\n              new Among(\"\\u0430\\u044F\", -1, 1), new Among(\"\\u044F\\u044F\", -1, 1)\n            ],\n            a_2 = [\n              new Among(\"\\u0435\\u043C\", -1, 1), new Among(\"\\u043D\\u043D\", -1, 1),\n              new Among(\"\\u0432\\u0448\", -1, 1),\n              new Among(\"\\u0438\\u0432\\u0448\", 2, 2),\n              new Among(\"\\u044B\\u0432\\u0448\", 2, 2), new Among(\"\\u0449\", -1, 1),\n              new Among(\"\\u044E\\u0449\", 5, 1),\n              new Among(\"\\u0443\\u044E\\u0449\", 6, 2)\n            ],\n            a_3 = [\n              new Among(\"\\u0441\\u044C\", -1, 1), new Among(\"\\u0441\\u044F\", -1, 1)\n            ],\n            a_4 = [\n              new Among(\"\\u043B\\u0430\", -1, 1),\n              new Among(\"\\u0438\\u043B\\u0430\", 0, 2),\n              new Among(\"\\u044B\\u043B\\u0430\", 0, 2),\n              new Among(\"\\u043D\\u0430\", -1, 1),\n              new Among(\"\\u0435\\u043D\\u0430\", 3, 2),\n              new Among(\"\\u0435\\u0442\\u0435\", -1, 1),\n              new Among(\"\\u0438\\u0442\\u0435\", -1, 2),\n              new Among(\"\\u0439\\u0442\\u0435\", -1, 1),\n              new Among(\"\\u0435\\u0439\\u0442\\u0435\", 7, 2),\n              new Among(\"\\u0443\\u0439\\u0442\\u0435\", 7, 2),\n              new Among(\"\\u043B\\u0438\", -1, 1),\n              new Among(\"\\u0438\\u043B\\u0438\", 10, 2),\n              new Among(\"\\u044B\\u043B\\u0438\", 10, 2), new Among(\"\\u0439\", -1, 1),\n              new Among(\"\\u0435\\u0439\", 13, 2), new Among(\"\\u0443\\u0439\", 13, 2),\n              new Among(\"\\u043B\", -1, 1), new Among(\"\\u0438\\u043B\", 16, 2),\n              new Among(\"\\u044B\\u043B\", 16, 2), new Among(\"\\u0435\\u043C\", -1, 1),\n              new Among(\"\\u0438\\u043C\", -1, 2), new Among(\"\\u044B\\u043C\", -1, 2),\n              new Among(\"\\u043D\", -1, 1), new Among(\"\\u0435\\u043D\", 22, 2),\n              new Among(\"\\u043B\\u043E\", -1, 1),\n              new Among(\"\\u0438\\u043B\\u043E\", 24, 2),\n              new Among(\"\\u044B\\u043B\\u043E\", 24, 2),\n              new Among(\"\\u043D\\u043E\", -1, 1),\n              new Among(\"\\u0435\\u043D\\u043E\", 27, 2),\n              new Among(\"\\u043D\\u043D\\u043E\", 27, 1),\n              new Among(\"\\u0435\\u0442\", -1, 1),\n              new Among(\"\\u0443\\u0435\\u0442\", 30, 2),\n              new Among(\"\\u0438\\u0442\", -1, 2), new Among(\"\\u044B\\u0442\", -1, 2),\n              new Among(\"\\u044E\\u0442\", -1, 1),\n              new Among(\"\\u0443\\u044E\\u0442\", 34, 2),\n              new Among(\"\\u044F\\u0442\", -1, 2), new Among(\"\\u043D\\u044B\", -1, 1),\n              new Among(\"\\u0435\\u043D\\u044B\", 37, 2),\n              new Among(\"\\u0442\\u044C\", -1, 1),\n              new Among(\"\\u0438\\u0442\\u044C\", 39, 2),\n              new Among(\"\\u044B\\u0442\\u044C\", 39, 2),\n              new Among(\"\\u0435\\u0448\\u044C\", -1, 1),\n              new Among(\"\\u0438\\u0448\\u044C\", -1, 2), new Among(\"\\u044E\", -1, 2),\n              new Among(\"\\u0443\\u044E\", 44, 2)\n            ],\n            a_5 = [\n              new Among(\"\\u0430\", -1, 1), new Among(\"\\u0435\\u0432\", -1, 1),\n              new Among(\"\\u043E\\u0432\", -1, 1), new Among(\"\\u0435\", -1, 1),\n              new Among(\"\\u0438\\u0435\", 3, 1), new Among(\"\\u044C\\u0435\", 3, 1),\n              new Among(\"\\u0438\", -1, 1), new Among(\"\\u0435\\u0438\", 6, 1),\n              new Among(\"\\u0438\\u0438\", 6, 1),\n              new Among(\"\\u0430\\u043C\\u0438\", 6, 1),\n              new Among(\"\\u044F\\u043C\\u0438\", 6, 1),\n              new Among(\"\\u0438\\u044F\\u043C\\u0438\", 10, 1),\n              new Among(\"\\u0439\", -1, 1), new Among(\"\\u0435\\u0439\", 12, 1),\n              new Among(\"\\u0438\\u0435\\u0439\", 13, 1),\n              new Among(\"\\u0438\\u0439\", 12, 1), new Among(\"\\u043E\\u0439\", 12, 1),\n              new Among(\"\\u0430\\u043C\", -1, 1), new Among(\"\\u0435\\u043C\", -1, 1),\n              new Among(\"\\u0438\\u0435\\u043C\", 18, 1),\n              new Among(\"\\u043E\\u043C\", -1, 1), new Among(\"\\u044F\\u043C\", -1, 1),\n              new Among(\"\\u0438\\u044F\\u043C\", 21, 1), new Among(\"\\u043E\", -1, 1),\n              new Among(\"\\u0443\", -1, 1), new Among(\"\\u0430\\u0445\", -1, 1),\n              new Among(\"\\u044F\\u0445\", -1, 1),\n              new Among(\"\\u0438\\u044F\\u0445\", 26, 1), new Among(\"\\u044B\", -1, 1),\n              new Among(\"\\u044C\", -1, 1), new Among(\"\\u044E\", -1, 1),\n              new Among(\"\\u0438\\u044E\", 30, 1), new Among(\"\\u044C\\u044E\", 30, 1),\n              new Among(\"\\u044F\", -1, 1), new Among(\"\\u0438\\u044F\", 33, 1),\n              new Among(\"\\u044C\\u044F\", 33, 1)\n            ],\n            a_6 = [\n              new Among(\"\\u043E\\u0441\\u0442\", -1, 1),\n              new Among(\"\\u043E\\u0441\\u0442\\u044C\", -1, 1)\n            ],\n            a_7 = [\n              new Among(\"\\u0435\\u0439\\u0448\\u0435\", -1, 1),\n              new Among(\"\\u043D\", -1, 2), new Among(\"\\u0435\\u0439\\u0448\", -1, 1),\n              new Among(\"\\u044C\", -1, 3)\n            ],\n            g_v = [33, 65, 8, 232],\n            I_p2, I_pV, sbp = new SnowballProgram();\n          this.setCurrent = function(word) {\n            sbp.setCurrent(word);\n          };\n          this.getCurrent = function() {\n            return sbp.getCurrent();\n          };\n\n          function habr3() {\n            while (!sbp.in_grouping(g_v, 1072, 1103)) {\n              if (sbp.cursor >= sbp.limit)\n                return false;\n              sbp.cursor++;\n            }\n            return true;\n          }\n\n          function habr4() {\n            while (!sbp.out_grouping(g_v, 1072, 1103)) {\n              if (sbp.cursor >= sbp.limit)\n                return false;\n              sbp.cursor++;\n            }\n            return true;\n          }\n\n          function r_mark_regions() {\n            I_pV = sbp.limit;\n            I_p2 = I_pV;\n            if (habr3()) {\n              I_pV = sbp.cursor;\n              if (habr4())\n                if (habr3())\n                  if (habr4())\n                    I_p2 = sbp.cursor;\n            }\n          }\n\n          function r_R2() {\n            return I_p2 <= sbp.cursor;\n          }\n\n          function habr2(a, n) {\n            var among_var, v_1;\n            sbp.ket = sbp.cursor;\n            among_var = sbp.find_among_b(a, n);\n            if (among_var) {\n              sbp.bra = sbp.cursor;\n              switch (among_var) {\n                case 1:\n                  v_1 = sbp.limit - sbp.cursor;\n                  if (!sbp.eq_s_b(1, \"\\u0430\")) {\n                    sbp.cursor = sbp.limit - v_1;\n                    if (!sbp.eq_s_b(1, \"\\u044F\"))\n                      return false;\n                  }\n                case 2:\n                  sbp.slice_del();\n                  break;\n              }\n              return true;\n            }\n            return false;\n          }\n\n          function r_perfective_gerund() {\n            return habr2(a_0, 9);\n          }\n\n          function habr1(a, n) {\n            var among_var;\n            sbp.ket = sbp.cursor;\n            among_var = sbp.find_among_b(a, n);\n            if (among_var) {\n              sbp.bra = sbp.cursor;\n              if (among_var == 1)\n                sbp.slice_del();\n              return true;\n            }\n            return false;\n          }\n\n          function r_adjective() {\n            return habr1(a_1, 26);\n          }\n\n          function r_adjectival() {\n            var among_var;\n            if (r_adjective()) {\n              habr2(a_2, 8);\n              return true;\n            }\n            return false;\n          }\n\n          function r_reflexive() {\n            return habr1(a_3, 2);\n          }\n\n          function r_verb() {\n            return habr2(a_4, 46);\n          }\n\n          function r_noun() {\n            habr1(a_5, 36);\n          }\n\n          function r_derivational() {\n            var among_var;\n            sbp.ket = sbp.cursor;\n            among_var = sbp.find_among_b(a_6, 2);\n            if (among_var) {\n              sbp.bra = sbp.cursor;\n              if (r_R2() && among_var == 1)\n                sbp.slice_del();\n            }\n          }\n\n          function r_tidy_up() {\n            var among_var;\n            sbp.ket = sbp.cursor;\n            among_var = sbp.find_among_b(a_7, 4);\n            if (among_var) {\n              sbp.bra = sbp.cursor;\n              switch (among_var) {\n                case 1:\n                  sbp.slice_del();\n                  sbp.ket = sbp.cursor;\n                  if (!sbp.eq_s_b(1, \"\\u043D\"))\n                    break;\n                  sbp.bra = sbp.cursor;\n                case 2:\n                  if (!sbp.eq_s_b(1, \"\\u043D\"))\n                    break;\n                case 3:\n                  sbp.slice_del();\n                  break;\n              }\n            }\n          }\n          this.stem = function() {\n            r_mark_regions();\n            sbp.cursor = sbp.limit;\n            if (sbp.cursor < I_pV)\n              return false;\n            sbp.limit_backward = I_pV;\n            if (!r_perfective_gerund()) {\n              sbp.cursor = sbp.limit;\n              if (!r_reflexive())\n                sbp.cursor = sbp.limit;\n              if (!r_adjectival()) {\n                sbp.cursor = sbp.limit;\n                if (!r_verb()) {\n                  sbp.cursor = sbp.limit;\n                  r_noun();\n                }\n              }\n            }\n            sbp.cursor = sbp.limit;\n            sbp.ket = sbp.cursor;\n            if (sbp.eq_s_b(1, \"\\u0438\")) {\n              sbp.bra = sbp.cursor;\n              sbp.slice_del();\n            } else\n              sbp.cursor = sbp.limit;\n            r_derivational();\n            sbp.cursor = sbp.limit;\n            r_tidy_up();\n            return true;\n          }\n        };\n\n      /* and return a function that stems a word for the current locale */\n      return function(word) {\n        st.setCurrent(word);\n        st.stem();\n        return st.getCurrent();\n      }\n    })();\n\n    lunr.Pipeline.registerFunction(lunr.ru.stemmer, 'stemmer-ru');\n\n    /* stop word filter function */\n    lunr.ru.stopWordFilter = function(token) {\n      if (lunr.ru.stopWordFilter.stopWords.indexOf(token) === -1) {\n        return token;\n      }\n    };\n\n    lunr.ru.stopWordFilter.stopWords = new lunr.SortedSet();\n    lunr.ru.stopWordFilter.stopWords.length = 422;\n\n    // The space at the beginning is crucial: It marks the empty string\n    // as a stop word. lunr.js crashes during search when documents\n    // processed by the pipeline still contain the empty string.\n    lunr.ru.stopWordFilter.stopWords.elements = ' алло без близко более больше будем будет будете будешь будто буду будут будь бы бывает бывь был была были было быть в важная важное важные важный вам вами вас ваш ваша ваше ваши вверх вдали вдруг ведь везде весь вниз внизу во вокруг вон восемнадцатый восемнадцать восемь восьмой вот впрочем времени время все всегда всего всем всеми всему всех всею всю всюду вся всё второй вы г где говорил говорит год года году да давно даже далеко дальше даром два двадцатый двадцать две двенадцатый двенадцать двух девятнадцатый девятнадцать девятый девять действительно дел день десятый десять для до довольно долго должно другая другие других друго другое другой е его ее ей ему если есть еще ещё ею её ж же жизнь за занят занята занято заняты затем зато зачем здесь значит и из или им именно иметь ими имя иногда их к каждая каждое каждые каждый кажется как какая какой кем когда кого ком кому конечно которая которого которой которые который которых кроме кругом кто куда лет ли лишь лучше люди м мало между меля менее меньше меня миллионов мимо мира мне много многочисленная многочисленное многочисленные многочисленный мной мною мог могут мож может можно можхо мои мой мор мочь моя моё мы на наверху над надо назад наиболее наконец нам нами нас начала наш наша наше наши не него недавно недалеко нее ней нельзя нем немного нему непрерывно нередко несколько нет нею неё ни нибудь ниже низко никогда никуда ними них ничего но ну нужно нх о об оба обычно один одиннадцатый одиннадцать однажды однако одного одной около он она они оно опять особенно от отовсюду отсюда очень первый перед по под пожалуйста позже пока пор пора после посреди потом потому почему почти прекрасно при про просто против процентов пятнадцатый пятнадцать пятый пять раз разве рано раньше рядом с сам сама сами самим самими самих само самого самой самом самому саму свое своего своей свои своих свою сеаой себе себя сегодня седьмой сейчас семнадцатый семнадцать семь сих сказал сказала сказать сколько слишком сначала снова со собой собою совсем спасибо стал суть т та так такая также такие такое такой там твой твоя твоё те тебе тебя тем теми теперь тех то тобой тобою тогда того тоже только том тому тот тою третий три тринадцатый тринадцать ту туда тут ты тысяч у уж уже уметь хорошо хотеть хоть хотя хочешь часто чаще чего человек чем чему через четвертый четыре четырнадцатый четырнадцать что чтоб чтобы чуть шестнадцатый шестнадцать шестой шесть эта эти этим этими этих это этого этой этом этому этот эту я ﻿а'.split(' ');\n\n    lunr.Pipeline.registerFunction(lunr.ru.stopWordFilter, 'stopWordFilter-ru');\n  };\n}))\n"
  },
  {
    "path": "static/js/lunr/lunr.sv.js",
    "content": "/*!\n * Lunr languages, `Swedish` language\n * https://github.com/MihaiValentin/lunr-languages\n *\n * Copyright 2014, Mihai Valentin\n * http://www.mozilla.org/MPL/\n */\n/*!\n * based on\n * Snowball JavaScript Library v0.3\n * http://code.google.com/p/urim/\n * http://snowball.tartarus.org/\n *\n * Copyright 2010, Oleg Mazko\n * http://www.mozilla.org/MPL/\n */\n\n/**\n * export the module via AMD, CommonJS or as a browser global\n * Export code from https://github.com/umdjs/umd/blob/master/returnExports.js\n */\n;\n(function(root, factory) {\n  if (typeof define === 'function' && define.amd) {\n    // AMD. Register as an anonymous module.\n    define(factory)\n  } else if (typeof exports === 'object') {\n    /**\n     * Node. Does not work with strict CommonJS, but\n     * only CommonJS-like environments that support module.exports,\n     * like Node.\n     */\n    module.exports = factory()\n  } else {\n    // Browser globals (root is window)\n    factory()(root.lunr);\n  }\n}(this, function() {\n  /**\n   * Just return a value to define the module export.\n   * This example returns an object, but the module\n   * can return a function as the exported value.\n   */\n  return function(lunr) {\n    /* throw error if lunr is not yet included */\n    if ('undefined' === typeof lunr) {\n      throw new Error('Lunr is not present. Please include / require Lunr before this script.');\n    }\n\n    /* throw error if lunr stemmer support is not yet included */\n    if ('undefined' === typeof lunr.stemmerSupport) {\n      throw new Error('Lunr stemmer support is not present. Please include / require Lunr stemmer support before this script.');\n    }\n\n    /* register specific locale function */\n    lunr.sv = function() {\n      this.pipeline.reset();\n      this.pipeline.add(\n        lunr.sv.trimmer,\n        lunr.sv.stopWordFilter,\n        lunr.sv.stemmer\n      );\n    };\n\n    /* lunr trimmer function */\n    lunr.sv.wordCharacters = \"A-Za-z\\xAA\\xBA\\xC0-\\xD6\\xD8-\\xF6\\xF8-\\u02B8\\u02E0-\\u02E4\\u1D00-\\u1D25\\u1D2C-\\u1D5C\\u1D62-\\u1D65\\u1D6B-\\u1D77\\u1D79-\\u1DBE\\u1E00-\\u1EFF\\u2071\\u207F\\u2090-\\u209C\\u212A\\u212B\\u2132\\u214E\\u2160-\\u2188\\u2C60-\\u2C7F\\uA722-\\uA787\\uA78B-\\uA7AD\\uA7B0-\\uA7B7\\uA7F7-\\uA7FF\\uAB30-\\uAB5A\\uAB5C-\\uAB64\\uFB00-\\uFB06\\uFF21-\\uFF3A\\uFF41-\\uFF5A\";\n    lunr.sv.trimmer = lunr.trimmerSupport.generateTrimmer(lunr.sv.wordCharacters);\n\n    lunr.Pipeline.registerFunction(lunr.sv.trimmer, 'trimmer-sv');\n\n    /* lunr stemmer function */\n    lunr.sv.stemmer = (function() {\n      /* create the wrapped stemmer object */\n      var Among = lunr.stemmerSupport.Among,\n        SnowballProgram = lunr.stemmerSupport.SnowballProgram,\n        st = new function SwedishStemmer() {\n          var a_0 = [new Among(\"a\", -1, 1), new Among(\"arna\", 0, 1),\n              new Among(\"erna\", 0, 1), new Among(\"heterna\", 2, 1),\n              new Among(\"orna\", 0, 1), new Among(\"ad\", -1, 1),\n              new Among(\"e\", -1, 1), new Among(\"ade\", 6, 1),\n              new Among(\"ande\", 6, 1), new Among(\"arne\", 6, 1),\n              new Among(\"are\", 6, 1), new Among(\"aste\", 6, 1),\n              new Among(\"en\", -1, 1), new Among(\"anden\", 12, 1),\n              new Among(\"aren\", 12, 1), new Among(\"heten\", 12, 1),\n              new Among(\"ern\", -1, 1), new Among(\"ar\", -1, 1),\n              new Among(\"er\", -1, 1), new Among(\"heter\", 18, 1),\n              new Among(\"or\", -1, 1), new Among(\"s\", -1, 2),\n              new Among(\"as\", 21, 1), new Among(\"arnas\", 22, 1),\n              new Among(\"ernas\", 22, 1), new Among(\"ornas\", 22, 1),\n              new Among(\"es\", 21, 1), new Among(\"ades\", 26, 1),\n              new Among(\"andes\", 26, 1), new Among(\"ens\", 21, 1),\n              new Among(\"arens\", 29, 1), new Among(\"hetens\", 29, 1),\n              new Among(\"erns\", 21, 1), new Among(\"at\", -1, 1),\n              new Among(\"andet\", -1, 1), new Among(\"het\", -1, 1),\n              new Among(\"ast\", -1, 1)\n            ],\n            a_1 = [new Among(\"dd\", -1, -1),\n              new Among(\"gd\", -1, -1), new Among(\"nn\", -1, -1),\n              new Among(\"dt\", -1, -1), new Among(\"gt\", -1, -1),\n              new Among(\"kt\", -1, -1), new Among(\"tt\", -1, -1)\n            ],\n            a_2 = [\n              new Among(\"ig\", -1, 1), new Among(\"lig\", 0, 1),\n              new Among(\"els\", -1, 1), new Among(\"fullt\", -1, 3),\n              new Among(\"l\\u00F6st\", -1, 2)\n            ],\n            g_v = [17, 65, 16, 1, 0, 0, 0, 0,\n              0, 0, 0, 0, 0, 0, 0, 0, 24, 0, 32\n            ],\n            g_s_ending = [119, 127, 149],\n            I_x, I_p1, sbp = new SnowballProgram();\n          this.setCurrent = function(word) {\n            sbp.setCurrent(word);\n          };\n          this.getCurrent = function() {\n            return sbp.getCurrent();\n          };\n\n          function r_mark_regions() {\n            var v_1, c = sbp.cursor + 3;\n            I_p1 = sbp.limit;\n            if (0 <= c || c <= sbp.limit) {\n              I_x = c;\n              while (true) {\n                v_1 = sbp.cursor;\n                if (sbp.in_grouping(g_v, 97, 246)) {\n                  sbp.cursor = v_1;\n                  break;\n                }\n                sbp.cursor = v_1;\n                if (sbp.cursor >= sbp.limit)\n                  return;\n                sbp.cursor++;\n              }\n              while (!sbp.out_grouping(g_v, 97, 246)) {\n                if (sbp.cursor >= sbp.limit)\n                  return;\n                sbp.cursor++;\n              }\n              I_p1 = sbp.cursor;\n              if (I_p1 < I_x)\n                I_p1 = I_x;\n            }\n          }\n\n          function r_main_suffix() {\n            var among_var, v_2 = sbp.limit_backward;\n            if (sbp.cursor >= I_p1) {\n              sbp.limit_backward = I_p1;\n              sbp.cursor = sbp.limit;\n              sbp.ket = sbp.cursor;\n              among_var = sbp.find_among_b(a_0, 37);\n              sbp.limit_backward = v_2;\n              if (among_var) {\n                sbp.bra = sbp.cursor;\n                switch (among_var) {\n                  case 1:\n                    sbp.slice_del();\n                    break;\n                  case 2:\n                    if (sbp.in_grouping_b(g_s_ending, 98, 121))\n                      sbp.slice_del();\n                    break;\n                }\n              }\n            }\n          }\n\n          function r_consonant_pair() {\n            var v_1 = sbp.limit_backward;\n            if (sbp.cursor >= I_p1) {\n              sbp.limit_backward = I_p1;\n              sbp.cursor = sbp.limit;\n              if (sbp.find_among_b(a_1, 7)) {\n                sbp.cursor = sbp.limit;\n                sbp.ket = sbp.cursor;\n                if (sbp.cursor > sbp.limit_backward) {\n                  sbp.bra = --sbp.cursor;\n                  sbp.slice_del();\n                }\n              }\n              sbp.limit_backward = v_1;\n            }\n          }\n\n          function r_other_suffix() {\n            var among_var, v_2;\n            if (sbp.cursor >= I_p1) {\n              v_2 = sbp.limit_backward;\n              sbp.limit_backward = I_p1;\n              sbp.cursor = sbp.limit;\n              sbp.ket = sbp.cursor;\n              among_var = sbp.find_among_b(a_2, 5);\n              if (among_var) {\n                sbp.bra = sbp.cursor;\n                switch (among_var) {\n                  case 1:\n                    sbp.slice_del();\n                    break;\n                  case 2:\n                    sbp.slice_from(\"l\\u00F6s\");\n                    break;\n                  case 3:\n                    sbp.slice_from(\"full\");\n                    break;\n                }\n              }\n              sbp.limit_backward = v_2;\n            }\n          }\n          this.stem = function() {\n            var v_1 = sbp.cursor;\n            r_mark_regions();\n            sbp.limit_backward = v_1;\n            sbp.cursor = sbp.limit;\n            r_main_suffix();\n            sbp.cursor = sbp.limit;\n            r_consonant_pair();\n            sbp.cursor = sbp.limit;\n            r_other_suffix();\n            return true;\n          }\n        };\n\n      /* and return a function that stems a word for the current locale */\n      return function(word) {\n        st.setCurrent(word);\n        st.stem();\n        return st.getCurrent();\n      }\n    })();\n\n    lunr.Pipeline.registerFunction(lunr.sv.stemmer, 'stemmer-sv');\n\n    /* stop word filter function */\n    lunr.sv.stopWordFilter = function(token) {\n      if (lunr.sv.stopWordFilter.stopWords.indexOf(token) === -1) {\n        return token;\n      }\n    };\n\n    lunr.sv.stopWordFilter.stopWords = new lunr.SortedSet();\n    lunr.sv.stopWordFilter.stopWords.length = 115;\n\n    // The space at the beginning is crucial: It marks the empty string\n    // as a stop word. lunr.js crashes during search when documents\n    // processed by the pipeline still contain the empty string.\n    lunr.sv.stopWordFilter.stopWords.elements = ' alla allt att av blev bli blir blivit de dem den denna deras dess dessa det detta dig din dina ditt du där då efter ej eller en er era ert ett från för ha hade han hans har henne hennes hon honom hur här i icke ingen inom inte jag ju kan kunde man med mellan men mig min mina mitt mot mycket ni nu när någon något några och om oss på samma sedan sig sin sina sitta själv skulle som så sådan sådana sådant till under upp ut utan vad var vara varför varit varje vars vart vem vi vid vilka vilkas vilken vilket vår våra vårt än är åt över'.split(' ');\n\n    lunr.Pipeline.registerFunction(lunr.sv.stopWordFilter, 'stopWordFilter-sv');\n  };\n}))\n"
  },
  {
    "path": "static/js/lunr/lunr.tr.js",
    "content": "/*!\n * Lunr languages, `Turkish` language\n * https://github.com/MihaiValentin/lunr-languages\n *\n * Copyright 2014, Mihai Valentin\n * http://www.mozilla.org/MPL/\n */\n/*!\n * based on\n * Snowball JavaScript Library v0.3\n * http://code.google.com/p/urim/\n * http://snowball.tartarus.org/\n *\n * Copyright 2010, Oleg Mazko\n * http://www.mozilla.org/MPL/\n */\n\n/**\n * export the module via AMD, CommonJS or as a browser global\n * Export code from https://github.com/umdjs/umd/blob/master/returnExports.js\n */\n;\n(function(root, factory) {\n  if (typeof define === 'function' && define.amd) {\n    // AMD. Register as an anonymous module.\n    define(factory)\n  } else if (typeof exports === 'object') {\n    /**\n     * Node. Does not work with strict CommonJS, but\n     * only CommonJS-like environments that support module.exports,\n     * like Node.\n     */\n    module.exports = factory()\n  } else {\n    // Browser globals (root is window)\n    factory()(root.lunr);\n  }\n}(this, function() {\n  /**\n   * Just return a value to define the module export.\n   * This example returns an object, but the module\n   * can return a function as the exported value.\n   */\n  return function(lunr) {\n    /* throw error if lunr is not yet included */\n    if ('undefined' === typeof lunr) {\n      throw new Error('Lunr is not present. Please include / require Lunr before this script.');\n    }\n\n    /* throw error if lunr stemmer support is not yet included */\n    if ('undefined' === typeof lunr.stemmerSupport) {\n      throw new Error('Lunr stemmer support is not present. Please include / require Lunr stemmer support before this script.');\n    }\n\n    /* register specific locale function */\n    lunr.tr = function() {\n      this.pipeline.reset();\n      this.pipeline.add(\n        lunr.tr.trimmer,\n        lunr.tr.stopWordFilter,\n        lunr.tr.stemmer\n      );\n    };\n\n    /* lunr trimmer function */\n    lunr.tr.wordCharacters = \"A-Za-z\\xAA\\xBA\\xC0-\\xD6\\xD8-\\xF6\\xF8-\\u02B8\\u02E0-\\u02E4\\u1D00-\\u1D25\\u1D2C-\\u1D5C\\u1D62-\\u1D65\\u1D6B-\\u1D77\\u1D79-\\u1DBE\\u1E00-\\u1EFF\\u2071\\u207F\\u2090-\\u209C\\u212A\\u212B\\u2132\\u214E\\u2160-\\u2188\\u2C60-\\u2C7F\\uA722-\\uA787\\uA78B-\\uA7AD\\uA7B0-\\uA7B7\\uA7F7-\\uA7FF\\uAB30-\\uAB5A\\uAB5C-\\uAB64\\uFB00-\\uFB06\\uFF21-\\uFF3A\\uFF41-\\uFF5A\";\n    lunr.tr.trimmer = lunr.trimmerSupport.generateTrimmer(lunr.tr.wordCharacters);\n\n    lunr.Pipeline.registerFunction(lunr.tr.trimmer, 'trimmer-tr');\n\n    /* lunr stemmer function */\n    lunr.tr.stemmer = (function() {\n      /* create the wrapped stemmer object */\n      var Among = lunr.stemmerSupport.Among,\n        SnowballProgram = lunr.stemmerSupport.SnowballProgram,\n        st = new function TurkishStemmer() {\n          var a_0 = [new Among(\"m\", -1, -1), new Among(\"n\", -1, -1),\n              new Among(\"miz\", -1, -1), new Among(\"niz\", -1, -1),\n              new Among(\"muz\", -1, -1), new Among(\"nuz\", -1, -1),\n              new Among(\"m\\u00FCz\", -1, -1), new Among(\"n\\u00FCz\", -1, -1),\n              new Among(\"m\\u0131z\", -1, -1), new Among(\"n\\u0131z\", -1, -1)\n            ],\n            a_1 = [\n              new Among(\"leri\", -1, -1), new Among(\"lar\\u0131\", -1, -1)\n            ],\n            a_2 = [\n              new Among(\"ni\", -1, -1), new Among(\"nu\", -1, -1),\n              new Among(\"n\\u00FC\", -1, -1), new Among(\"n\\u0131\", -1, -1)\n            ],\n            a_3 = [\n              new Among(\"in\", -1, -1), new Among(\"un\", -1, -1),\n              new Among(\"\\u00FCn\", -1, -1), new Among(\"\\u0131n\", -1, -1)\n            ],\n            a_4 = [\n              new Among(\"a\", -1, -1), new Among(\"e\", -1, -1)\n            ],\n            a_5 = [\n              new Among(\"na\", -1, -1), new Among(\"ne\", -1, -1)\n            ],\n            a_6 = [\n              new Among(\"da\", -1, -1), new Among(\"ta\", -1, -1),\n              new Among(\"de\", -1, -1), new Among(\"te\", -1, -1)\n            ],\n            a_7 = [\n              new Among(\"nda\", -1, -1), new Among(\"nde\", -1, -1)\n            ],\n            a_8 = [\n              new Among(\"dan\", -1, -1), new Among(\"tan\", -1, -1),\n              new Among(\"den\", -1, -1), new Among(\"ten\", -1, -1)\n            ],\n            a_9 = [\n              new Among(\"ndan\", -1, -1), new Among(\"nden\", -1, -1)\n            ],\n            a_10 = [\n              new Among(\"la\", -1, -1), new Among(\"le\", -1, -1)\n            ],\n            a_11 = [\n              new Among(\"ca\", -1, -1), new Among(\"ce\", -1, -1)\n            ],\n            a_12 = [\n              new Among(\"im\", -1, -1), new Among(\"um\", -1, -1),\n              new Among(\"\\u00FCm\", -1, -1), new Among(\"\\u0131m\", -1, -1)\n            ],\n            a_13 = [\n              new Among(\"sin\", -1, -1), new Among(\"sun\", -1, -1),\n              new Among(\"s\\u00FCn\", -1, -1), new Among(\"s\\u0131n\", -1, -1)\n            ],\n            a_14 = [\n              new Among(\"iz\", -1, -1), new Among(\"uz\", -1, -1),\n              new Among(\"\\u00FCz\", -1, -1), new Among(\"\\u0131z\", -1, -1)\n            ],\n            a_15 = [\n              new Among(\"siniz\", -1, -1), new Among(\"sunuz\", -1, -1),\n              new Among(\"s\\u00FCn\\u00FCz\", -1, -1),\n              new Among(\"s\\u0131n\\u0131z\", -1, -1)\n            ],\n            a_16 = [\n              new Among(\"lar\", -1, -1), new Among(\"ler\", -1, -1)\n            ],\n            a_17 = [\n              new Among(\"niz\", -1, -1), new Among(\"nuz\", -1, -1),\n              new Among(\"n\\u00FCz\", -1, -1), new Among(\"n\\u0131z\", -1, -1)\n            ],\n            a_18 = [\n              new Among(\"dir\", -1, -1), new Among(\"tir\", -1, -1),\n              new Among(\"dur\", -1, -1), new Among(\"tur\", -1, -1),\n              new Among(\"d\\u00FCr\", -1, -1), new Among(\"t\\u00FCr\", -1, -1),\n              new Among(\"d\\u0131r\", -1, -1), new Among(\"t\\u0131r\", -1, -1)\n            ],\n            a_19 = [\n              new Among(\"cas\\u0131na\", -1, -1), new Among(\"cesine\", -1, -1)\n            ],\n            a_20 = [\n              new Among(\"di\", -1, -1), new Among(\"ti\", -1, -1),\n              new Among(\"dik\", -1, -1), new Among(\"tik\", -1, -1),\n              new Among(\"duk\", -1, -1), new Among(\"tuk\", -1, -1),\n              new Among(\"d\\u00FCk\", -1, -1), new Among(\"t\\u00FCk\", -1, -1),\n              new Among(\"d\\u0131k\", -1, -1), new Among(\"t\\u0131k\", -1, -1),\n              new Among(\"dim\", -1, -1), new Among(\"tim\", -1, -1),\n              new Among(\"dum\", -1, -1), new Among(\"tum\", -1, -1),\n              new Among(\"d\\u00FCm\", -1, -1), new Among(\"t\\u00FCm\", -1, -1),\n              new Among(\"d\\u0131m\", -1, -1), new Among(\"t\\u0131m\", -1, -1),\n              new Among(\"din\", -1, -1), new Among(\"tin\", -1, -1),\n              new Among(\"dun\", -1, -1), new Among(\"tun\", -1, -1),\n              new Among(\"d\\u00FCn\", -1, -1), new Among(\"t\\u00FCn\", -1, -1),\n              new Among(\"d\\u0131n\", -1, -1), new Among(\"t\\u0131n\", -1, -1),\n              new Among(\"du\", -1, -1), new Among(\"tu\", -1, -1),\n              new Among(\"d\\u00FC\", -1, -1), new Among(\"t\\u00FC\", -1, -1),\n              new Among(\"d\\u0131\", -1, -1), new Among(\"t\\u0131\", -1, -1)\n            ],\n            a_21 = [\n              new Among(\"sa\", -1, -1), new Among(\"se\", -1, -1),\n              new Among(\"sak\", -1, -1), new Among(\"sek\", -1, -1),\n              new Among(\"sam\", -1, -1), new Among(\"sem\", -1, -1),\n              new Among(\"san\", -1, -1), new Among(\"sen\", -1, -1)\n            ],\n            a_22 = [\n              new Among(\"mi\\u015F\", -1, -1), new Among(\"mu\\u015F\", -1, -1),\n              new Among(\"m\\u00FC\\u015F\", -1, -1),\n              new Among(\"m\\u0131\\u015F\", -1, -1)\n            ],\n            a_23 = [new Among(\"b\", -1, 1),\n              new Among(\"c\", -1, 2), new Among(\"d\", -1, 3),\n              new Among(\"\\u011F\", -1, 4)\n            ],\n            g_vowel = [17, 65, 16, 0, 0, 0, 0, 0,\n              0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 32, 8, 0, 0, 0, 0, 0, 0, 1\n            ],\n            g_U = [\n              1, 16, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 8, 0, 0, 0,\n              0, 0, 0, 1\n            ],\n            g_vowel1 = [1, 64, 16, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0,\n              0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 1\n            ],\n            g_vowel2 = [17, 0, 0, 0,\n              0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 130\n            ],\n            g_vowel3 = [1, 0,\n              0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0,\n              0, 0, 1\n            ],\n            g_vowel4 = [17],\n            g_vowel5 = [65],\n            g_vowel6 = [65],\n            B_c_s_n_s, I_strlen, g_habr = [\n              [\"a\", g_vowel1, 97, 305],\n              [\"e\", g_vowel2, 101, 252],\n              [\"\\u0131\", g_vowel3, 97, 305],\n              [\"i\", g_vowel4, 101, 105],\n              [\"o\", g_vowel5, 111, 117],\n              [\"\\u00F6\", g_vowel6, 246, 252],\n              [\"u\", g_vowel5, 111, 117]\n            ],\n            sbp = new SnowballProgram();\n          this.setCurrent = function(word) {\n            sbp.setCurrent(word);\n          };\n          this.getCurrent = function() {\n            return sbp.getCurrent();\n          };\n\n          function habr1(g_v, n1, n2) {\n            while (true) {\n              var v_1 = sbp.limit - sbp.cursor;\n              if (sbp.in_grouping_b(g_v, n1, n2)) {\n                sbp.cursor = sbp.limit - v_1;\n                break;\n              }\n              sbp.cursor = sbp.limit - v_1;\n              if (sbp.cursor <= sbp.limit_backward)\n                return false;\n              sbp.cursor--;\n            }\n            return true;\n          }\n\n          function r_check_vowel_harmony() {\n            var v_1, v_2;\n            v_1 = sbp.limit - sbp.cursor;\n            habr1(g_vowel, 97, 305);\n            for (var i = 0; i < g_habr.length; i++) {\n              v_2 = sbp.limit - sbp.cursor;\n              var habr = g_habr[i];\n              if (sbp.eq_s_b(1, habr[0]) && habr1(habr[1], habr[2], habr[3])) {\n                sbp.cursor = sbp.limit - v_1;\n                return true;\n              }\n              sbp.cursor = sbp.limit - v_2;\n            }\n            sbp.cursor = sbp.limit - v_2;\n            if (!sbp.eq_s_b(1, \"\\u00FC\") || !habr1(g_vowel6, 246, 252))\n              return false;\n            sbp.cursor = sbp.limit - v_1;\n            return true;\n          }\n\n          function habr2(f1, f2) {\n            var v_1 = sbp.limit - sbp.cursor,\n              v_2;\n            if (f1()) {\n              sbp.cursor = sbp.limit - v_1;\n              if (sbp.cursor > sbp.limit_backward) {\n                sbp.cursor--;\n                v_2 = sbp.limit - sbp.cursor;\n                if (f2()) {\n                  sbp.cursor = sbp.limit - v_2;\n                  return true;\n                }\n              }\n            }\n            sbp.cursor = sbp.limit - v_1;\n            if (f1()) {\n              sbp.cursor = sbp.limit - v_1;\n              return false;\n            }\n            sbp.cursor = sbp.limit - v_1;\n            if (sbp.cursor <= sbp.limit_backward)\n              return false;\n            sbp.cursor--;\n            if (!f2())\n              return false;\n            sbp.cursor = sbp.limit - v_1;\n            return true;\n          }\n\n          function habr3(f1) {\n            return habr2(f1, function() {\n              return sbp.in_grouping_b(g_vowel, 97, 305);\n            });\n          }\n\n          function r_mark_suffix_with_optional_n_consonant() {\n            return habr3(function() {\n              return sbp.eq_s_b(1, \"n\");\n            });\n          }\n\n          function r_mark_suffix_with_optional_s_consonant() {\n            return habr3(function() {\n              return sbp.eq_s_b(1, \"s\");\n            });\n          }\n\n          function r_mark_suffix_with_optional_y_consonant() {\n            return habr3(function() {\n              return sbp.eq_s_b(1, \"y\");\n            });\n          }\n\n          function r_mark_suffix_with_optional_U_vowel() {\n            return habr2(function() {\n              return sbp.in_grouping_b(g_U, 105, 305);\n            }, function() {\n              return sbp.out_grouping_b(g_vowel, 97, 305);\n            });\n          }\n\n          function r_mark_possessives() {\n            return sbp.find_among_b(a_0, 10) && r_mark_suffix_with_optional_U_vowel();\n          }\n\n          function r_mark_sU() {\n            return r_check_vowel_harmony() && sbp.in_grouping_b(g_U, 105, 305) && r_mark_suffix_with_optional_s_consonant();\n          }\n\n          function r_mark_lArI() {\n            return sbp.find_among_b(a_1, 2);\n          }\n\n          function r_mark_yU() {\n            return r_check_vowel_harmony() && sbp.in_grouping_b(g_U, 105, 305) && r_mark_suffix_with_optional_y_consonant();\n          }\n\n          function r_mark_nU() {\n            return r_check_vowel_harmony() && sbp.find_among_b(a_2, 4);\n          }\n\n          function r_mark_nUn() {\n            return r_check_vowel_harmony() && sbp.find_among_b(a_3, 4) && r_mark_suffix_with_optional_n_consonant();\n          }\n\n          function r_mark_yA() {\n            return r_check_vowel_harmony() && sbp.find_among_b(a_4, 2) && r_mark_suffix_with_optional_y_consonant();\n          }\n\n          function r_mark_nA() {\n            return r_check_vowel_harmony() && sbp.find_among_b(a_5, 2);\n          }\n\n          function r_mark_DA() {\n            return r_check_vowel_harmony() && sbp.find_among_b(a_6, 4);\n          }\n\n          function r_mark_ndA() {\n            return r_check_vowel_harmony() && sbp.find_among_b(a_7, 2);\n          }\n\n          function r_mark_DAn() {\n            return r_check_vowel_harmony() && sbp.find_among_b(a_8, 4);\n          }\n\n          function r_mark_ndAn() {\n            return r_check_vowel_harmony() && sbp.find_among_b(a_9, 2);\n          }\n\n          function r_mark_ylA() {\n            return r_check_vowel_harmony() && sbp.find_among_b(a_10, 2) && r_mark_suffix_with_optional_y_consonant();\n          }\n\n          function r_mark_ki() {\n            return sbp.eq_s_b(2, \"ki\");\n          }\n\n          function r_mark_ncA() {\n            return r_check_vowel_harmony() && sbp.find_among_b(a_11, 2) && r_mark_suffix_with_optional_n_consonant();\n          }\n\n          function r_mark_yUm() {\n            return r_check_vowel_harmony() && sbp.find_among_b(a_12, 4) && r_mark_suffix_with_optional_y_consonant();\n          }\n\n          function r_mark_sUn() {\n            return r_check_vowel_harmony() && sbp.find_among_b(a_13, 4);\n          }\n\n          function r_mark_yUz() {\n            return r_check_vowel_harmony() && sbp.find_among_b(a_14, 4) && r_mark_suffix_with_optional_y_consonant();\n          }\n\n          function r_mark_sUnUz() {\n            return sbp.find_among_b(a_15, 4);\n          }\n\n          function r_mark_lAr() {\n            return r_check_vowel_harmony() && sbp.find_among_b(a_16, 2);\n          }\n\n          function r_mark_nUz() {\n            return r_check_vowel_harmony() && sbp.find_among_b(a_17, 4);\n          }\n\n          function r_mark_DUr() {\n            return r_check_vowel_harmony() && sbp.find_among_b(a_18, 8);\n          }\n\n          function r_mark_cAsInA() {\n            return sbp.find_among_b(a_19, 2);\n          }\n\n          function r_mark_yDU() {\n            return r_check_vowel_harmony() && sbp.find_among_b(a_20, 32) && r_mark_suffix_with_optional_y_consonant();\n          }\n\n          function r_mark_ysA() {\n            return sbp.find_among_b(a_21, 8) && r_mark_suffix_with_optional_y_consonant();\n          }\n\n          function r_mark_ymUs_() {\n            return r_check_vowel_harmony() && sbp.find_among_b(a_22, 4) && r_mark_suffix_with_optional_y_consonant();\n          }\n\n          function r_mark_yken() {\n            return sbp.eq_s_b(3, \"ken\") && r_mark_suffix_with_optional_y_consonant();\n          }\n\n          function habr4() {\n            var v_1 = sbp.limit - sbp.cursor;\n            if (!r_mark_ymUs_()) {\n              sbp.cursor = sbp.limit - v_1;\n              if (!r_mark_yDU()) {\n                sbp.cursor = sbp.limit - v_1;\n                if (!r_mark_ysA()) {\n                  sbp.cursor = sbp.limit - v_1;\n                  if (!r_mark_yken())\n                    return true;\n                }\n              }\n            }\n            return false;\n          }\n\n          function habr5() {\n            if (r_mark_cAsInA()) {\n              var v_1 = sbp.limit - sbp.cursor;\n              if (!r_mark_sUnUz()) {\n                sbp.cursor = sbp.limit - v_1;\n                if (!r_mark_lAr()) {\n                  sbp.cursor = sbp.limit - v_1;\n                  if (!r_mark_yUm()) {\n                    sbp.cursor = sbp.limit - v_1;\n                    if (!r_mark_sUn()) {\n                      sbp.cursor = sbp.limit - v_1;\n                      if (!r_mark_yUz())\n                        sbp.cursor = sbp.limit - v_1;\n                    }\n                  }\n                }\n              }\n              if (r_mark_ymUs_())\n                return false;\n            }\n            return true;\n          }\n\n          function habr6() {\n            if (r_mark_lAr()) {\n              sbp.bra = sbp.cursor;\n              sbp.slice_del();\n              var v_1 = sbp.limit - sbp.cursor;\n              sbp.ket = sbp.cursor;\n              if (!r_mark_DUr()) {\n                sbp.cursor = sbp.limit - v_1;\n                if (!r_mark_yDU()) {\n                  sbp.cursor = sbp.limit - v_1;\n                  if (!r_mark_ysA()) {\n                    sbp.cursor = sbp.limit - v_1;\n                    if (!r_mark_ymUs_())\n                      sbp.cursor = sbp.limit - v_1;\n                  }\n                }\n              }\n              B_c_s_n_s = false;\n              return false;\n            }\n            return true;\n          }\n\n          function habr7() {\n            if (!r_mark_nUz())\n              return true;\n            var v_1 = sbp.limit - sbp.cursor;\n            if (!r_mark_yDU()) {\n              sbp.cursor = sbp.limit - v_1;\n              if (!r_mark_ysA())\n                return true;\n            }\n            return false;\n          }\n\n          function habr8() {\n            var v_1 = sbp.limit - sbp.cursor,\n              v_2;\n            if (!r_mark_sUnUz()) {\n              sbp.cursor = sbp.limit - v_1;\n              if (!r_mark_yUz()) {\n                sbp.cursor = sbp.limit - v_1;\n                if (!r_mark_sUn()) {\n                  sbp.cursor = sbp.limit - v_1;\n                  if (!r_mark_yUm())\n                    return true;\n                }\n              }\n            }\n            sbp.bra = sbp.cursor;\n            sbp.slice_del();\n            v_2 = sbp.limit - sbp.cursor;\n            sbp.ket = sbp.cursor;\n            if (!r_mark_ymUs_())\n              sbp.cursor = sbp.limit - v_2;\n            return false;\n          }\n\n          function r_stem_nominal_verb_suffixes() {\n            var v_1 = sbp.limit - sbp.cursor,\n              v_2;\n            sbp.ket = sbp.cursor;\n            B_c_s_n_s = true;\n            if (habr4()) {\n              sbp.cursor = sbp.limit - v_1;\n              if (habr5()) {\n                sbp.cursor = sbp.limit - v_1;\n                if (habr6()) {\n                  sbp.cursor = sbp.limit - v_1;\n                  if (habr7()) {\n                    sbp.cursor = sbp.limit - v_1;\n                    if (habr8()) {\n                      sbp.cursor = sbp.limit - v_1;\n                      if (!r_mark_DUr())\n                        return;\n                      sbp.bra = sbp.cursor;\n                      sbp.slice_del();\n                      sbp.ket = sbp.cursor;\n                      v_2 = sbp.limit - sbp.cursor;\n                      if (!r_mark_sUnUz()) {\n                        sbp.cursor = sbp.limit - v_2;\n                        if (!r_mark_lAr()) {\n                          sbp.cursor = sbp.limit - v_2;\n                          if (!r_mark_yUm()) {\n                            sbp.cursor = sbp.limit - v_2;\n                            if (!r_mark_sUn()) {\n                              sbp.cursor = sbp.limit - v_2;\n                              if (!r_mark_yUz())\n                                sbp.cursor = sbp.limit - v_2;\n                            }\n                          }\n                        }\n                      }\n                      if (!r_mark_ymUs_())\n                        sbp.cursor = sbp.limit - v_2;\n                    }\n                  }\n                }\n              }\n            }\n            sbp.bra = sbp.cursor;\n            sbp.slice_del();\n          }\n\n          function r_stem_suffix_chain_before_ki() {\n            var v_1, v_2, v_3, v_4;\n            sbp.ket = sbp.cursor;\n            if (r_mark_ki()) {\n              v_1 = sbp.limit - sbp.cursor;\n              if (r_mark_DA()) {\n                sbp.bra = sbp.cursor;\n                sbp.slice_del();\n                v_2 = sbp.limit - sbp.cursor;\n                sbp.ket = sbp.cursor;\n                if (r_mark_lAr()) {\n                  sbp.bra = sbp.cursor;\n                  sbp.slice_del();\n                  r_stem_suffix_chain_before_ki();\n                } else {\n                  sbp.cursor = sbp.limit - v_2;\n                  if (r_mark_possessives()) {\n                    sbp.bra = sbp.cursor;\n                    sbp.slice_del();\n                    sbp.ket = sbp.cursor;\n                    if (r_mark_lAr()) {\n                      sbp.bra = sbp.cursor;\n                      sbp.slice_del();\n                      r_stem_suffix_chain_before_ki();\n                    }\n                  }\n                }\n                return true;\n              }\n              sbp.cursor = sbp.limit - v_1;\n              if (r_mark_nUn()) {\n                sbp.bra = sbp.cursor;\n                sbp.slice_del();\n                sbp.ket = sbp.cursor;\n                v_3 = sbp.limit - sbp.cursor;\n                if (r_mark_lArI()) {\n                  sbp.bra = sbp.cursor;\n                  sbp.slice_del();\n                } else {\n                  sbp.cursor = sbp.limit - v_3;\n                  sbp.ket = sbp.cursor;\n                  if (!r_mark_possessives()) {\n                    sbp.cursor = sbp.limit - v_3;\n                    if (!r_mark_sU()) {\n                      sbp.cursor = sbp.limit - v_3;\n                      if (!r_stem_suffix_chain_before_ki())\n                        return true;\n                    }\n                  }\n                  sbp.bra = sbp.cursor;\n                  sbp.slice_del();\n                  sbp.ket = sbp.cursor;\n                  if (r_mark_lAr()) {\n                    sbp.bra = sbp.cursor;\n                    sbp.slice_del();\n                    r_stem_suffix_chain_before_ki()\n                  }\n                }\n                return true;\n              }\n              sbp.cursor = sbp.limit - v_1;\n              if (r_mark_ndA()) {\n                v_4 = sbp.limit - sbp.cursor;\n                if (r_mark_lArI()) {\n                  sbp.bra = sbp.cursor;\n                  sbp.slice_del();\n                } else {\n                  sbp.cursor = sbp.limit - v_4;\n                  if (r_mark_sU()) {\n                    sbp.bra = sbp.cursor;\n                    sbp.slice_del();\n                    sbp.ket = sbp.cursor;\n                    if (r_mark_lAr()) {\n                      sbp.bra = sbp.cursor;\n                      sbp.slice_del();\n                      r_stem_suffix_chain_before_ki();\n                    }\n                  } else {\n                    sbp.cursor = sbp.limit - v_4;\n                    if (!r_stem_suffix_chain_before_ki())\n                      return false;\n                  }\n                }\n                return true;\n              }\n            }\n            return false;\n          }\n\n          function habr9(v_1) {\n            sbp.ket = sbp.cursor;\n            if (!r_mark_ndA()) {\n              sbp.cursor = sbp.limit - v_1;\n              if (!r_mark_nA())\n                return false;\n            }\n            var v_2 = sbp.limit - sbp.cursor;\n            if (r_mark_lArI()) {\n              sbp.bra = sbp.cursor;\n              sbp.slice_del();\n            } else {\n              sbp.cursor = sbp.limit - v_2;\n              if (r_mark_sU()) {\n                sbp.bra = sbp.cursor;\n                sbp.slice_del();\n                sbp.ket = sbp.cursor;\n                if (r_mark_lAr()) {\n                  sbp.bra = sbp.cursor;\n                  sbp.slice_del();\n                  r_stem_suffix_chain_before_ki();\n                }\n              } else {\n                sbp.cursor = sbp.limit - v_2;\n                if (!r_stem_suffix_chain_before_ki())\n                  return false;\n              }\n            }\n            return true;\n          }\n\n          function habr10(v_1) {\n            sbp.ket = sbp.cursor;\n            if (!r_mark_ndAn()) {\n              sbp.cursor = sbp.limit - v_1;\n              if (!r_mark_nU())\n                return false;\n            }\n            var v_2 = sbp.limit - sbp.cursor;\n            if (!r_mark_sU()) {\n              sbp.cursor = sbp.limit - v_2;\n              if (!r_mark_lArI())\n                return false;\n            }\n            sbp.bra = sbp.cursor;\n            sbp.slice_del();\n            sbp.ket = sbp.cursor;\n            if (r_mark_lAr()) {\n              sbp.bra = sbp.cursor;\n              sbp.slice_del();\n              r_stem_suffix_chain_before_ki();\n            }\n            return true;\n          }\n\n          function habr11() {\n            var v_1 = sbp.limit - sbp.cursor,\n              v_2;\n            sbp.ket = sbp.cursor;\n            if (!r_mark_nUn()) {\n              sbp.cursor = sbp.limit - v_1;\n              if (!r_mark_ylA())\n                return false;\n            }\n            sbp.bra = sbp.cursor;\n            sbp.slice_del();\n            v_2 = sbp.limit - sbp.cursor;\n            sbp.ket = sbp.cursor;\n            if (r_mark_lAr()) {\n              sbp.bra = sbp.cursor;\n              sbp.slice_del();\n              if (r_stem_suffix_chain_before_ki())\n                return true;\n            }\n            sbp.cursor = sbp.limit - v_2;\n            sbp.ket = sbp.cursor;\n            if (!r_mark_possessives()) {\n              sbp.cursor = sbp.limit - v_2;\n              if (!r_mark_sU()) {\n                sbp.cursor = sbp.limit - v_2;\n                if (!r_stem_suffix_chain_before_ki())\n                  return true;\n              }\n            }\n            sbp.bra = sbp.cursor;\n            sbp.slice_del();\n            sbp.ket = sbp.cursor;\n            if (r_mark_lAr()) {\n              sbp.bra = sbp.cursor;\n              sbp.slice_del();\n              r_stem_suffix_chain_before_ki();\n            }\n            return true;\n          }\n\n          function habr12() {\n            var v_1 = sbp.limit - sbp.cursor,\n              v_2, v_3;\n            sbp.ket = sbp.cursor;\n            if (!r_mark_DA()) {\n              sbp.cursor = sbp.limit - v_1;\n              if (!r_mark_yU()) {\n                sbp.cursor = sbp.limit - v_1;\n                if (!r_mark_yA())\n                  return false;\n              }\n            }\n            sbp.bra = sbp.cursor;\n            sbp.slice_del();\n            sbp.ket = sbp.cursor;\n            v_2 = sbp.limit - sbp.cursor;\n            if (r_mark_possessives()) {\n              sbp.bra = sbp.cursor;\n              sbp.slice_del();\n              v_3 = sbp.limit - sbp.cursor;\n              sbp.ket = sbp.cursor;\n              if (!r_mark_lAr())\n                sbp.cursor = sbp.limit - v_3;\n            } else {\n              sbp.cursor = sbp.limit - v_2;\n              if (!r_mark_lAr())\n                return true;\n            }\n            sbp.bra = sbp.cursor;\n            sbp.slice_del();\n            sbp.ket = sbp.cursor;\n            r_stem_suffix_chain_before_ki();\n            return true;\n          }\n\n          function r_stem_noun_suffixes() {\n            var v_1 = sbp.limit - sbp.cursor,\n              v_2, v_3;\n            sbp.ket = sbp.cursor;\n            if (r_mark_lAr()) {\n              sbp.bra = sbp.cursor;\n              sbp.slice_del();\n              r_stem_suffix_chain_before_ki();\n              return;\n            }\n            sbp.cursor = sbp.limit - v_1;\n            sbp.ket = sbp.cursor;\n            if (r_mark_ncA()) {\n              sbp.bra = sbp.cursor;\n              sbp.slice_del();\n              v_2 = sbp.limit - sbp.cursor;\n              sbp.ket = sbp.cursor;\n              if (r_mark_lArI()) {\n                sbp.bra = sbp.cursor;\n                sbp.slice_del();\n              } else {\n                sbp.cursor = sbp.limit - v_2;\n                sbp.ket = sbp.cursor;\n                if (!r_mark_possessives()) {\n                  sbp.cursor = sbp.limit - v_2;\n                  if (!r_mark_sU()) {\n                    sbp.cursor = sbp.limit - v_2;\n                    sbp.ket = sbp.cursor;\n                    if (!r_mark_lAr())\n                      return;\n                    sbp.bra = sbp.cursor;\n                    sbp.slice_del();\n                    if (!r_stem_suffix_chain_before_ki())\n                      return;\n                  }\n                }\n                sbp.bra = sbp.cursor;\n                sbp.slice_del();\n                sbp.ket = sbp.cursor;\n                if (r_mark_lAr()) {\n                  sbp.bra = sbp.cursor;\n                  sbp.slice_del();\n                  r_stem_suffix_chain_before_ki();\n                }\n              }\n              return;\n            }\n            sbp.cursor = sbp.limit - v_1;\n            if (habr9(v_1))\n              return;\n            sbp.cursor = sbp.limit - v_1;\n            if (habr10(v_1))\n              return;\n            sbp.cursor = sbp.limit - v_1;\n            sbp.ket = sbp.cursor;\n            if (r_mark_DAn()) {\n              sbp.bra = sbp.cursor;\n              sbp.slice_del();\n              sbp.ket = sbp.cursor;\n              v_3 = sbp.limit - sbp.cursor;\n              if (r_mark_possessives()) {\n                sbp.bra = sbp.cursor;\n                sbp.slice_del();\n                sbp.ket = sbp.cursor;\n                if (r_mark_lAr()) {\n                  sbp.bra = sbp.cursor;\n                  sbp.slice_del();\n                  r_stem_suffix_chain_before_ki();\n                }\n              } else {\n                sbp.cursor = sbp.limit - v_3;\n                if (r_mark_lAr()) {\n                  sbp.bra = sbp.cursor;\n                  sbp.slice_del();\n                  r_stem_suffix_chain_before_ki();\n                } else {\n                  sbp.cursor = sbp.limit - v_3;\n                  r_stem_suffix_chain_before_ki();\n                }\n              }\n              return;\n            }\n            sbp.cursor = sbp.limit - v_1;\n            if (habr11())\n              return;\n            sbp.cursor = sbp.limit - v_1;\n            if (r_mark_lArI()) {\n              sbp.bra = sbp.cursor;\n              sbp.slice_del();\n              return;\n            }\n            sbp.cursor = sbp.limit - v_1;\n            if (r_stem_suffix_chain_before_ki())\n              return;\n            sbp.cursor = sbp.limit - v_1;\n            if (habr12())\n              return;\n            sbp.cursor = sbp.limit - v_1;\n            sbp.ket = sbp.cursor;\n            if (!r_mark_possessives()) {\n              sbp.cursor = sbp.limit - v_1;\n              if (!r_mark_sU())\n                return;\n            }\n            sbp.bra = sbp.cursor;\n            sbp.slice_del();\n            sbp.ket = sbp.cursor;\n            if (r_mark_lAr()) {\n              sbp.bra = sbp.cursor;\n              sbp.slice_del();\n              r_stem_suffix_chain_before_ki();\n            }\n          }\n\n          function r_post_process_last_consonants() {\n            var among_var;\n            sbp.ket = sbp.cursor;\n            among_var = sbp.find_among_b(a_23, 4);\n            if (among_var) {\n              sbp.bra = sbp.cursor;\n              switch (among_var) {\n                case 1:\n                  sbp.slice_from(\"p\");\n                  break;\n                case 2:\n                  sbp.slice_from(\"\\u00E7\");\n                  break;\n                case 3:\n                  sbp.slice_from(\"t\");\n                  break;\n                case 4:\n                  sbp.slice_from(\"k\");\n                  break;\n              }\n            }\n          }\n\n          function habr13() {\n            while (true) {\n              var v_1 = sbp.limit - sbp.cursor;\n              if (sbp.in_grouping_b(g_vowel, 97, 305)) {\n                sbp.cursor = sbp.limit - v_1;\n                break;\n              }\n              sbp.cursor = sbp.limit - v_1;\n              if (sbp.cursor <= sbp.limit_backward)\n                return false;\n              sbp.cursor--;\n            }\n            return true;\n          }\n\n          function habr14(v_1, c1, c2) {\n            sbp.cursor = sbp.limit - v_1;\n            if (habr13()) {\n              var v_2 = sbp.limit - sbp.cursor;\n              if (!sbp.eq_s_b(1, c1)) {\n                sbp.cursor = sbp.limit - v_2;\n                if (!sbp.eq_s_b(1, c2))\n                  return true;\n              }\n              sbp.cursor = sbp.limit - v_1;\n              var c = sbp.cursor;\n              sbp.insert(sbp.cursor, sbp.cursor, c2);\n              sbp.cursor = c;\n              return false;\n            }\n            return true;\n          }\n\n          function r_append_U_to_stems_ending_with_d_or_g() {\n            var v_1 = sbp.limit - sbp.cursor;\n            if (!sbp.eq_s_b(1, \"d\")) {\n              sbp.cursor = sbp.limit - v_1;\n              if (!sbp.eq_s_b(1, \"g\"))\n                return;\n            }\n            if (habr14(v_1, \"a\", \"\\u0131\"))\n              if (habr14(v_1, \"e\", \"i\"))\n                if (habr14(v_1, \"o\", \"u\"))\n                  habr14(v_1, \"\\u00F6\", \"\\u00FC\")\n          }\n\n          function r_more_than_one_syllable_word() {\n            var v_1 = sbp.cursor,\n              v_2 = 2,\n              v_3;\n            while (true) {\n              v_3 = sbp.cursor;\n              while (!sbp.in_grouping(g_vowel, 97, 305)) {\n                if (sbp.cursor >= sbp.limit) {\n                  sbp.cursor = v_3;\n                  if (v_2 > 0)\n                    return false;\n                  sbp.cursor = v_1;\n                  return true;\n                }\n                sbp.cursor++;\n              }\n              v_2--;\n            }\n          }\n\n          function habr15(v_1, n1, c1) {\n            while (!sbp.eq_s(n1, c1)) {\n              if (sbp.cursor >= sbp.limit)\n                return true;\n              sbp.cursor++;\n            }\n            I_strlen = n1;\n            if (I_strlen != sbp.limit)\n              return true;\n            sbp.cursor = v_1;\n            return false;\n          }\n\n          function r_is_reserved_word() {\n            var v_1 = sbp.cursor;\n            if (habr15(v_1, 2, \"ad\")) {\n              sbp.cursor = v_1;\n              if (habr15(v_1, 5, \"soyad\"))\n                return false;\n            }\n            return true;\n          }\n\n          function r_postlude() {\n            var v_1 = sbp.cursor;\n            if (r_is_reserved_word())\n              return false;\n            sbp.limit_backward = v_1;\n            sbp.cursor = sbp.limit;\n            r_append_U_to_stems_ending_with_d_or_g();\n            sbp.cursor = sbp.limit;\n            r_post_process_last_consonants();\n            return true;\n          }\n          this.stem = function() {\n            if (r_more_than_one_syllable_word()) {\n              sbp.limit_backward = sbp.cursor;\n              sbp.cursor = sbp.limit;\n              r_stem_nominal_verb_suffixes();\n              sbp.cursor = sbp.limit;\n              if (B_c_s_n_s) {\n                r_stem_noun_suffixes();\n                sbp.cursor = sbp.limit_backward;\n                if (r_postlude())\n                  return true;\n              }\n            }\n            return false;\n          }\n        };\n\n      /* and return a function that stems a word for the current locale */\n      return function(word) {\n        st.setCurrent(word);\n        st.stem();\n        return st.getCurrent();\n      }\n    })();\n\n    lunr.Pipeline.registerFunction(lunr.tr.stemmer, 'stemmer-tr');\n\n    /* stop word filter function */\n    lunr.tr.stopWordFilter = function(token) {\n      if (lunr.tr.stopWordFilter.stopWords.indexOf(token) === -1) {\n        return token;\n      }\n    };\n\n    lunr.tr.stopWordFilter.stopWords = new lunr.SortedSet();\n    lunr.tr.stopWordFilter.stopWords.length = 210;\n\n    // The space at the beginning is crucial: It marks the empty string\n    // as a stop word. lunr.js crashes during search when documents\n    // processed by the pipeline still contain the empty string.\n    lunr.tr.stopWordFilter.stopWords.elements = ' acaba altmış altı ama ancak arada aslında ayrıca bana bazı belki ben benden beni benim beri beş bile bin bir biri birkaç birkez birçok birşey birşeyi biz bizden bize bizi bizim bu buna bunda bundan bunlar bunları bunların bunu bunun burada böyle böylece da daha dahi de defa değil diye diğer doksan dokuz dolayı dolayısıyla dört edecek eden ederek edilecek ediliyor edilmesi ediyor elli en etmesi etti ettiği ettiğini eğer gibi göre halen hangi hatta hem henüz hep hepsi her herhangi herkesin hiç hiçbir iki ile ilgili ise itibaren itibariyle için işte kadar karşın katrilyon kendi kendilerine kendini kendisi kendisine kendisini kez ki kim kimden kime kimi kimse kırk milyar milyon mu mü mı nasıl ne neden nedenle nerde nerede nereye niye niçin o olan olarak oldu olduklarını olduğu olduğunu olmadı olmadığı olmak olması olmayan olmaz olsa olsun olup olur olursa oluyor on ona ondan onlar onlardan onları onların onu onun otuz oysa pek rağmen sadece sanki sekiz seksen sen senden seni senin siz sizden sizi sizin tarafından trilyon tüm var vardı ve veya ya yani yapacak yapmak yaptı yaptıkları yaptığı yaptığını yapılan yapılması yapıyor yedi yerine yetmiş yine yirmi yoksa yüz zaten çok çünkü öyle üzere üç şey şeyden şeyi şeyler şu şuna şunda şundan şunları şunu şöyle'.split(' ');\n\n    lunr.Pipeline.registerFunction(lunr.tr.stopWordFilter, 'stopWordFilter-tr');\n  };\n}))\n"
  },
  {
    "path": "static/js/lunr/lunr.zh.js",
    "content": "/*!\n * Lunr languages, `Chinese` language\n * https://github.com/MihaiValentin/lunr-languages\n *\n * Copyright 2019, Felix Lian (repairearth)\n * http://www.mozilla.org/MPL/\n */\n/*!\n * based on\n * Snowball zhvaScript Library v0.3\n * http://code.google.com/p/urim/\n * http://snowball.tartarus.org/\n *\n * Copyright 2010, Oleg Mazko\n * http://www.mozilla.org/MPL/\n */\n\n/**\n * export the module via AMD, CommonJS or as a browser global\n * Export code from https://github.com/umdjs/umd/blob/master/returnExports.js\n */\n;\n(function(root, factory) {\n  if (typeof define === 'function' && define.amd) {\n    // AMD. Register as an anonymous module.\n    define(factory)\n  } else if (typeof exports === 'object') {\n    /**\n     * Node. Does not work with strict CommonJS, but\n     * only CommonJS-like environments that support module.exports,\n     * like Node.\n     */\n    module.exports = factory(require('nodejieba'))\n  } else {\n    // Browser globals (root is window)\n    factory()(root.lunr);\n  }\n}(this, function(nodejieba) {\n  /**\n   * Just return a value to define the module export.\n   * This example returns an object, but the module\n   * can return a function as the exported value.\n   */\n  return function(lunr, nodejiebaDictJson) {\n    /* throw error if lunr is not yet included */\n    if ('undefined' === typeof lunr) {\n      throw new Error('Lunr is not present. Please include / require Lunr before this script.');\n    }\n\n    /* throw error if lunr stemmer support is not yet included */\n    if ('undefined' === typeof lunr.stemmerSupport) {\n      throw new Error('Lunr stemmer support is not present. Please include / require Lunr stemmer support before this script.');\n    }\n\n    /*\n    Chinese tokenization is trickier, since it does not\n    take into account spaces.\n    Since the tokenization function is represented different\n    internally for each of the Lunr versions, this had to be done\n    in order to try to try to pick the best way of doing this based\n    on the Lunr version\n     */\n    var isLunr2 = lunr.version[0] == \"2\";\n\n    /* register specific locale function */\n    lunr.zh = function() {\n      this.pipeline.reset();\n      this.pipeline.add(\n        lunr.zh.trimmer,\n        lunr.zh.stopWordFilter,\n        lunr.zh.stemmer\n      );\n\n      // change the tokenizer for Chinese one\n      if (isLunr2) { // for lunr version 2.0.0\n        this.tokenizer = lunr.zh.tokenizer;\n      } else {\n        if (lunr.tokenizer) { // for lunr version 0.6.0\n          lunr.tokenizer = lunr.zh.tokenizer;\n        }\n        if (this.tokenizerFn) { // for lunr version 0.7.0 -> 1.0.0\n          this.tokenizerFn = lunr.zh.tokenizer;\n        }\n      }\n    };\n\n    lunr.zh.tokenizer = function(obj) {\n      if (!arguments.length || obj == null || obj == undefined) return []\n      if (Array.isArray(obj)) return obj.map(function(t) {\n        return isLunr2 ? new lunr.Token(t.toLowerCase()) : t.toLowerCase()\n      })\n\n      nodejiebaDictJson && nodejieba.load(nodejiebaDictJson)\n\n      var str = obj.toString().trim().toLowerCase();\n      var tokens = [];\n\n      nodejieba.cut(str, true).forEach(function(seg) {\n        tokens = tokens.concat(seg.split(' '))\n      })\n\n      tokens = tokens.filter(function(token) {\n        return !!token;\n      });\n\n      var fromIndex = 0\n\n      return tokens.map(function(token, index) {\n        if (isLunr2) {\n          var start = str.indexOf(token, fromIndex)\n\n          var tokenMetadata = {}\n          tokenMetadata[\"position\"] = [start, token.length]\n          tokenMetadata[\"index\"] = index\n\n          fromIndex = start\n\n          return new lunr.Token(token, tokenMetadata);\n        } else {\n          return token\n        }\n      });\n    }\n\n    /* lunr trimmer function */\n    lunr.zh.wordCharacters = \"\\\\w\\u4e00-\\u9fa5\";\n    lunr.zh.trimmer = lunr.trimmerSupport.generateTrimmer(lunr.zh.wordCharacters);\n    lunr.Pipeline.registerFunction(lunr.zh.trimmer, 'trimmer-zh');\n\n    /* lunr stemmer function */\n    lunr.zh.stemmer = (function() {\n\n      /* TODO Chinese stemmer  */\n      return function(word) {\n        return word;\n      }\n    })();\n    lunr.Pipeline.registerFunction(lunr.zh.stemmer, 'stemmer-zh');\n\n    /* lunr stop word filter. see https://www.ranks.nl/stopwords/chinese-stopwords */\n    lunr.generateStopWordFilter = function (stopWords) {\n        var words = stopWords.reduce(function (memo, stopWord) {\n            memo[stopWord] = stopWord;\n            return memo;\n        }, {});\n\n        return function (token) {\n            if (token && words[token.toString()] !== token.toString()) return token;\n        };\n    }\n\n    lunr.zh.stopWordFilter = lunr.generateStopWordFilter(\n      '的 一 不 在 人 有 是 为 以 于 上 他 而 后 之 来 及 了 因 下 可 到 由 这 与 也 此 但 并 个 其 已 无 小 我 们 起 最 再 今 去 好 只 又 或 很 亦 某 把 那 你 乃 它 吧 被 比 别 趁 当 从 到 得 打 凡 儿 尔 该 各 给 跟 和 何 还 即 几 既 看 据 距 靠 啦 了 另 么 每 们 嘛 拿 哪 那 您 凭 且 却 让 仍 啥 如 若 使 谁 虽 随 同 所 她 哇 嗡 往 哪 些 向 沿 哟 用 于 咱 则 怎 曾 至 致 着 诸 自'.split(' '));\n    lunr.Pipeline.registerFunction(lunr.zh.stopWordFilter, 'stopWordFilter-zh');\n  };\n}))\n"
  },
  {
    "path": "static/js/lunr/lunrStemmerSupport.js",
    "content": "/*!\n * Snowball JavaScript Library v0.3\n * http://code.google.com/p/urim/\n * http://snowball.tartarus.org/\n *\n * Copyright 2010, Oleg Mazko\n * http://www.mozilla.org/MPL/\n */\n\n/**\n * export the module via AMD, CommonJS or as a browser global\n * Export code from https://github.com/umdjs/umd/blob/master/returnExports.js\n */\n; (function (root, factory) {\n    if (typeof define === 'function' && define.amd) {\n        // AMD. Register as an anonymous module.\n        define(factory)\n    } else if (typeof exports === 'object') {\n        /**\n         * Node. Does not work with strict CommonJS, but\n         * only CommonJS-like environments that support module.exports,\n         * like Node.\n         */\n        module.exports = factory()\n    } else {\n        // Browser globals (root is window)\n        factory()(root.lunr);\n    }\n}(this, function () {\n    /**\n     * Just return a value to define the module export.\n     * This example returns an object, but the module\n     * can return a function as the exported value.\n     */\n    return function (lunr) {\n        /* provides utilities for the included stemmers */\n        lunr.stemmerSupport = {\n            Among: function (s, substring_i, result, method) {\n                this.toCharArray = function (s) {\n                    var sLength = s.length, charArr = new Array(sLength);\n                    for (var i = 0; i < sLength; i++)\n                        charArr[i] = s.charCodeAt(i);\n                    return charArr;\n                };\n\n                if ((!s && s != \"\") || (!substring_i && (substring_i != 0)) || !result)\n                    throw (\"Bad Among initialisation: s:\" + s + \", substring_i: \"\n                        + substring_i + \", result: \" + result);\n                this.s_size = s.length;\n                this.s = this.toCharArray(s);\n                this.substring_i = substring_i;\n                this.result = result;\n                this.method = method;\n            },\n            SnowballProgram: function () {\n                var current;\n                return {\n                    bra: 0,\n                    ket: 0,\n                    limit: 0,\n                    cursor: 0,\n                    limit_backward: 0,\n                    setCurrent: function (word) {\n                        current = word;\n                        this.cursor = 0;\n                        this.limit = word.length;\n                        this.limit_backward = 0;\n                        this.bra = this.cursor;\n                        this.ket = this.limit;\n                    },\n                    getCurrent: function () {\n                        var result = current;\n                        current = null;\n                        return result;\n                    },\n                    in_grouping: function (s, min, max) {\n                        if (this.cursor < this.limit) {\n                            var ch = current.charCodeAt(this.cursor);\n                            if (ch <= max && ch >= min) {\n                                ch -= min;\n                                if (s[ch >> 3] & (0X1 << (ch & 0X7))) {\n                                    this.cursor++;\n                                    return true;\n                                }\n                            }\n                        }\n                        return false;\n                    },\n                    in_grouping_b: function (s, min, max) {\n                        if (this.cursor > this.limit_backward) {\n                            var ch = current.charCodeAt(this.cursor - 1);\n                            if (ch <= max && ch >= min) {\n                                ch -= min;\n                                if (s[ch >> 3] & (0X1 << (ch & 0X7))) {\n                                    this.cursor--;\n                                    return true;\n                                }\n                            }\n                        }\n                        return false;\n                    },\n                    out_grouping: function (s, min, max) {\n                        if (this.cursor < this.limit) {\n                            var ch = current.charCodeAt(this.cursor);\n                            if (ch > max || ch < min) {\n                                this.cursor++;\n                                return true;\n                            }\n                            ch -= min;\n                            if (!(s[ch >> 3] & (0X1 << (ch & 0X7)))) {\n                                this.cursor++;\n                                return true;\n                            }\n                        }\n                        return false;\n                    },\n                    out_grouping_b: function (s, min, max) {\n                        if (this.cursor > this.limit_backward) {\n                            var ch = current.charCodeAt(this.cursor - 1);\n                            if (ch > max || ch < min) {\n                                this.cursor--;\n                                return true;\n                            }\n                            ch -= min;\n                            if (!(s[ch >> 3] & (0X1 << (ch & 0X7)))) {\n                                this.cursor--;\n                                return true;\n                            }\n                        }\n                        return false;\n                    },\n                    eq_s: function (s_size, s) {\n                        if (this.limit - this.cursor < s_size)\n                            return false;\n                        for (var i = 0; i < s_size; i++)\n                            if (current.charCodeAt(this.cursor + i) != s.charCodeAt(i))\n                                return false;\n                        this.cursor += s_size;\n                        return true;\n                    },\n                    eq_s_b: function (s_size, s) {\n                        if (this.cursor - this.limit_backward < s_size)\n                            return false;\n                        for (var i = 0; i < s_size; i++)\n                            if (current.charCodeAt(this.cursor - s_size + i) != s\n                                .charCodeAt(i))\n                                return false;\n                        this.cursor -= s_size;\n                        return true;\n                    },\n                    find_among: function (v, v_size) {\n                        var i = 0, j = v_size, c = this.cursor, l = this.limit, common_i = 0, common_j = 0, first_key_inspected = false;\n                        while (true) {\n                            var k = i + ((j - i) >> 1), diff = 0, common = common_i < common_j\n                                ? common_i\n                                : common_j, w = v[k];\n                            for (var i2 = common; i2 < w.s_size; i2++) {\n                                if (c + common == l) {\n                                    diff = -1;\n                                    break;\n                                }\n                                diff = current.charCodeAt(c + common) - w.s[i2];\n                                if (diff)\n                                    break;\n                                common++;\n                            }\n                            if (diff < 0) {\n                                j = k;\n                                common_j = common;\n                            } else {\n                                i = k;\n                                common_i = common;\n                            }\n                            if (j - i <= 1) {\n                                if (i > 0 || j == i || first_key_inspected)\n                                    break;\n                                first_key_inspected = true;\n                            }\n                        }\n                        while (true) {\n                            var w = v[i];\n                            if (common_i >= w.s_size) {\n                                this.cursor = c + w.s_size;\n                                if (!w.method)\n                                    return w.result;\n                                var res = w.method();\n                                this.cursor = c + w.s_size;\n                                if (res)\n                                    return w.result;\n                            }\n                            i = w.substring_i;\n                            if (i < 0)\n                                return 0;\n                        }\n                    },\n                    find_among_b: function (v, v_size) {\n                        var i = 0, j = v_size, c = this.cursor, lb = this.limit_backward, common_i = 0, common_j = 0, first_key_inspected = false;\n                        while (true) {\n                            var k = i + ((j - i) >> 1), diff = 0, common = common_i < common_j\n                                ? common_i\n                                : common_j, w = v[k];\n                            for (var i2 = w.s_size - 1 - common; i2 >= 0; i2--) {\n                                if (c - common == lb) {\n                                    diff = -1;\n                                    break;\n                                }\n                                diff = current.charCodeAt(c - 1 - common) - w.s[i2];\n                                if (diff)\n                                    break;\n                                common++;\n                            }\n                            if (diff < 0) {\n                                j = k;\n                                common_j = common;\n                            } else {\n                                i = k;\n                                common_i = common;\n                            }\n                            if (j - i <= 1) {\n                                if (i > 0 || j == i || first_key_inspected)\n                                    break;\n                                first_key_inspected = true;\n                            }\n                        }\n                        while (true) {\n                            var w = v[i];\n                            if (common_i >= w.s_size) {\n                                this.cursor = c - w.s_size;\n                                if (!w.method)\n                                    return w.result;\n                                var res = w.method();\n                                this.cursor = c - w.s_size;\n                                if (res)\n                                    return w.result;\n                            }\n                            i = w.substring_i;\n                            if (i < 0)\n                                return 0;\n                        }\n                    },\n                    replace_s: function (c_bra, c_ket, s) {\n                        var adjustment = s.length - (c_ket - c_bra), left = current\n                            .substring(0, c_bra), right = current.substring(c_ket);\n                        current = left + s + right;\n                        this.limit += adjustment;\n                        if (this.cursor >= c_ket)\n                            this.cursor += adjustment;\n                        else if (this.cursor > c_bra)\n                            this.cursor = c_bra;\n                        return adjustment;\n                    },\n                    slice_check: function () {\n                        if (this.bra < 0 || this.bra > this.ket || this.ket > this.limit\n                            || this.limit > current.length)\n                            throw (\"faulty slice operation\");\n                    },\n                    slice_from: function (s) {\n                        this.slice_check();\n                        this.replace_s(this.bra, this.ket, s);\n                    },\n                    slice_del: function () {\n                        this.slice_from(\"\");\n                    },\n                    insert: function (c_bra, c_ket, s) {\n                        var adjustment = this.replace_s(c_bra, c_ket, s);\n                        if (c_bra <= this.bra)\n                            this.bra += adjustment;\n                        if (c_bra <= this.ket)\n                            this.ket += adjustment;\n                    },\n                    slice_to: function () {\n                        this.slice_check();\n                        return current.substring(this.bra, this.ket);\n                    },\n                    eq_v_b: function (s) {\n                        return this.eq_s_b(s.length, s);\n                    }\n                };\n            }\n        };\n\n        lunr.trimmerSupport = {\n            generateTrimmer: function (wordCharacters) {\n                var startRegex = new RegExp(\"^[^\" + wordCharacters + \"]+\")\n                var endRegex = new RegExp(\"[^\" + wordCharacters + \"]+$\")\n\n                return function (token) {\n                    // for lunr version 2\n                    if (typeof token.update === \"function\") {\n                        return token.update(function (s) {\n                            return s\n                                .replace(startRegex, '')\n                                .replace(endRegex, '');\n                        })\n                    } else { // for lunr version 1\n                        return token\n                            .replace(startRegex, '')\n                            .replace(endRegex, '');\n                    }\n                };\n            }\n        }\n    }\n}));\n"
  },
  {
    "path": "static/js/lunr/tinyseg.js",
    "content": "// TinySegmenter 0.1 -- Super compact Japanese tokenizer in Javascript\n// (c) 2008 Taku Kudo <taku@chasen.org>\n// TinySegmenter is freely distributable under the terms of a new BSD licence.\n// For details, see http://chasen.org/~taku/software/TinySegmenter/LICENCE.txt\n\nfunction TinySegmenter() {\n    var patterns = {\n        '[一二三四五六七八九十百千万億兆]': 'M',\n        '[一-龠々〆ヵヶ]': 'H',\n        '[ぁ-ん]': 'I',\n        '[ァ-ヴーｱ-ﾝﾞｰ]': 'K',\n        '[a-zA-Zａ-ｚＡ-Ｚ]': 'A',\n        '[0-9０-９]': 'N',\n    };\n    this.chartype_ = [];\n    for (var i in patterns) {\n        var regexp = new RegExp();\n        regexp.compile(i);\n        this.chartype_.push([regexp, patterns[i]]);\n    }\n\n    this.BIAS__ = -332;\n    this.BC1__ = { HH: 6, II: 2461, KH: 406, OH: -1378 };\n    this.BC2__ = {\n        AA: -3267,\n        AI: 2744,\n        AN: -878,\n        HH: -4070,\n        HM: -1711,\n        HN: 4012,\n        HO: 3761,\n        IA: 1327,\n        IH: -1184,\n        II: -1332,\n        IK: 1721,\n        IO: 5492,\n        KI: 3831,\n        KK: -8741,\n        MH: -3132,\n        MK: 3334,\n        OO: -2920,\n    };\n    this.BC3__ = {\n        HH: 996,\n        HI: 626,\n        HK: -721,\n        HN: -1307,\n        HO: -836,\n        IH: -301,\n        KK: 2762,\n        MK: 1079,\n        MM: 4034,\n        OA: -1652,\n        OH: 266,\n    };\n    this.BP1__ = { BB: 295, OB: 304, OO: -125, UB: 352 };\n    this.BP2__ = { BO: 60, OO: -1762 };\n    this.BQ1__ = {\n        BHH: 1150,\n        BHM: 1521,\n        BII: -1158,\n        BIM: 886,\n        BMH: 1208,\n        BNH: 449,\n        BOH: -91,\n        BOO: -2597,\n        OHI: 451,\n        OIH: -296,\n        OKA: 1851,\n        OKH: -1020,\n        OKK: 904,\n        OOO: 2965,\n    };\n    this.BQ2__ = {\n        BHH: 118,\n        BHI: -1159,\n        BHM: 466,\n        BIH: -919,\n        BKK: -1720,\n        BKO: 864,\n        OHH: -1139,\n        OHM: -181,\n        OIH: 153,\n        UHI: -1146,\n    };\n    this.BQ3__ = {\n        BHH: -792,\n        BHI: 2664,\n        BII: -299,\n        BKI: 419,\n        BMH: 937,\n        BMM: 8335,\n        BNN: 998,\n        BOH: 775,\n        OHH: 2174,\n        OHM: 439,\n        OII: 280,\n        OKH: 1798,\n        OKI: -793,\n        OKO: -2242,\n        OMH: -2402,\n        OOO: 11699,\n    };\n    this.BQ4__ = {\n        BHH: -3895,\n        BIH: 3761,\n        BII: -4654,\n        BIK: 1348,\n        BKK: -1806,\n        BMI: -3385,\n        BOO: -12396,\n        OAH: 926,\n        OHH: 266,\n        OHK: -2036,\n        ONN: -973,\n    };\n    this.BW1__ = {\n        ',と': 660,\n        ',同': 727,\n        B1あ: 1404,\n        B1同: 542,\n        '、と': 660,\n        '、同': 727,\n        '」と': 1682,\n        あっ: 1505,\n        いう: 1743,\n        いっ: -2055,\n        いる: 672,\n        うし: -4817,\n        うん: 665,\n        から: 3472,\n        がら: 600,\n        こう: -790,\n        こと: 2083,\n        こん: -1262,\n        さら: -4143,\n        さん: 4573,\n        した: 2641,\n        して: 1104,\n        すで: -3399,\n        そこ: 1977,\n        それ: -871,\n        たち: 1122,\n        ため: 601,\n        った: 3463,\n        つい: -802,\n        てい: 805,\n        てき: 1249,\n        でき: 1127,\n        です: 3445,\n        では: 844,\n        とい: -4915,\n        とみ: 1922,\n        どこ: 3887,\n        ない: 5713,\n        なっ: 3015,\n        など: 7379,\n        なん: -1113,\n        にし: 2468,\n        には: 1498,\n        にも: 1671,\n        に対: -912,\n        の一: -501,\n        の中: 741,\n        ませ: 2448,\n        まで: 1711,\n        まま: 2600,\n        まる: -2155,\n        やむ: -1947,\n        よっ: -2565,\n        れた: 2369,\n        れで: -913,\n        をし: 1860,\n        を見: 731,\n        亡く: -1886,\n        京都: 2558,\n        取り: -2784,\n        大き: -2604,\n        大阪: 1497,\n        平方: -2314,\n        引き: -1336,\n        日本: -195,\n        本当: -2423,\n        毎日: -2113,\n        目指: -724,\n        Ｂ１あ: 1404,\n        Ｂ１同: 542,\n        '｣と': 1682,\n    };\n    this.BW2__ = {\n        '..': -11822,\n        11: -669,\n        '――': -5730,\n        '−−': -13175,\n        いう: -1609,\n        うか: 2490,\n        かし: -1350,\n        かも: -602,\n        から: -7194,\n        かれ: 4612,\n        がい: 853,\n        がら: -3198,\n        きた: 1941,\n        くな: -1597,\n        こと: -8392,\n        この: -4193,\n        させ: 4533,\n        され: 13168,\n        さん: -3977,\n        しい: -1819,\n        しか: -545,\n        した: 5078,\n        して: 972,\n        しな: 939,\n        その: -3744,\n        たい: -1253,\n        たた: -662,\n        ただ: -3857,\n        たち: -786,\n        たと: 1224,\n        たは: -939,\n        った: 4589,\n        って: 1647,\n        っと: -2094,\n        てい: 6144,\n        てき: 3640,\n        てく: 2551,\n        ては: -3110,\n        ても: -3065,\n        でい: 2666,\n        でき: -1528,\n        でし: -3828,\n        です: -4761,\n        でも: -4203,\n        とい: 1890,\n        とこ: -1746,\n        とと: -2279,\n        との: 720,\n        とみ: 5168,\n        とも: -3941,\n        ない: -2488,\n        なが: -1313,\n        など: -6509,\n        なの: 2614,\n        なん: 3099,\n        にお: -1615,\n        にし: 2748,\n        にな: 2454,\n        によ: -7236,\n        に対: -14943,\n        に従: -4688,\n        に関: -11388,\n        のか: 2093,\n        ので: -7059,\n        のに: -6041,\n        のの: -6125,\n        はい: 1073,\n        はが: -1033,\n        はず: -2532,\n        ばれ: 1813,\n        まし: -1316,\n        まで: -6621,\n        まれ: 5409,\n        めて: -3153,\n        もい: 2230,\n        もの: -10713,\n        らか: -944,\n        らし: -1611,\n        らに: -1897,\n        りし: 651,\n        りま: 1620,\n        れた: 4270,\n        れて: 849,\n        れば: 4114,\n        ろう: 6067,\n        われ: 7901,\n        を通: -11877,\n        んだ: 728,\n        んな: -4115,\n        一人: 602,\n        一方: -1375,\n        一日: 970,\n        一部: -1051,\n        上が: -4479,\n        会社: -1116,\n        出て: 2163,\n        分の: -7758,\n        同党: 970,\n        同日: -913,\n        大阪: -2471,\n        委員: -1250,\n        少な: -1050,\n        年度: -8669,\n        年間: -1626,\n        府県: -2363,\n        手権: -1982,\n        新聞: -4066,\n        日新: -722,\n        日本: -7068,\n        日米: 3372,\n        曜日: -601,\n        朝鮮: -2355,\n        本人: -2697,\n        東京: -1543,\n        然と: -1384,\n        社会: -1276,\n        立て: -990,\n        第に: -1612,\n        米国: -4268,\n        '１１': -669,\n    };\n    this.BW3__ = {\n        あた: -2194,\n        あり: 719,\n        ある: 3846,\n        'い.': -1185,\n        'い。': -1185,\n        いい: 5308,\n        いえ: 2079,\n        いく: 3029,\n        いた: 2056,\n        いっ: 1883,\n        いる: 5600,\n        いわ: 1527,\n        うち: 1117,\n        うと: 4798,\n        えと: 1454,\n        'か.': 2857,\n        'か。': 2857,\n        かけ: -743,\n        かっ: -4098,\n        かに: -669,\n        から: 6520,\n        かり: -2670,\n        'が,': 1816,\n        'が、': 1816,\n        がき: -4855,\n        がけ: -1127,\n        がっ: -913,\n        がら: -4977,\n        がり: -2064,\n        きた: 1645,\n        けど: 1374,\n        こと: 7397,\n        この: 1542,\n        ころ: -2757,\n        さい: -714,\n        さを: 976,\n        'し,': 1557,\n        'し、': 1557,\n        しい: -3714,\n        した: 3562,\n        して: 1449,\n        しな: 2608,\n        しま: 1200,\n        'す.': -1310,\n        'す。': -1310,\n        する: 6521,\n        'ず,': 3426,\n        'ず、': 3426,\n        ずに: 841,\n        そう: 428,\n        'た.': 8875,\n        'た。': 8875,\n        たい: -594,\n        たの: 812,\n        たり: -1183,\n        たる: -853,\n        'だ.': 4098,\n        'だ。': 4098,\n        だっ: 1004,\n        った: -4748,\n        って: 300,\n        てい: 6240,\n        てお: 855,\n        ても: 302,\n        です: 1437,\n        でに: -1482,\n        では: 2295,\n        とう: -1387,\n        とし: 2266,\n        との: 541,\n        とも: -3543,\n        どう: 4664,\n        ない: 1796,\n        なく: -903,\n        など: 2135,\n        'に,': -1021,\n        'に、': -1021,\n        にし: 1771,\n        にな: 1906,\n        には: 2644,\n        'の,': -724,\n        'の、': -724,\n        の子: -1000,\n        'は,': 1337,\n        'は、': 1337,\n        べき: 2181,\n        まし: 1113,\n        ます: 6943,\n        まっ: -1549,\n        まで: 6154,\n        まれ: -793,\n        らし: 1479,\n        られ: 6820,\n        るる: 3818,\n        'れ,': 854,\n        'れ、': 854,\n        れた: 1850,\n        れて: 1375,\n        れば: -3246,\n        れる: 1091,\n        われ: -605,\n        んだ: 606,\n        んで: 798,\n        カ月: 990,\n        会議: 860,\n        入り: 1232,\n        大会: 2217,\n        始め: 1681,\n        市: 965,\n        新聞: -5055,\n        '日,': 974,\n        '日、': 974,\n        社会: 2024,\n        ｶ月: 990,\n    };\n    this.TC1__ = {\n        AAA: 1093,\n        HHH: 1029,\n        HHM: 580,\n        HII: 998,\n        HOH: -390,\n        HOM: -331,\n        IHI: 1169,\n        IOH: -142,\n        IOI: -1015,\n        IOM: 467,\n        MMH: 187,\n        OOI: -1832,\n    };\n    this.TC2__ = {\n        HHO: 2088,\n        HII: -1023,\n        HMM: -1154,\n        IHI: -1965,\n        KKH: 703,\n        OII: -2649,\n    };\n    this.TC3__ = {\n        AAA: -294,\n        HHH: 346,\n        HHI: -341,\n        HII: -1088,\n        HIK: 731,\n        HOH: -1486,\n        IHH: 128,\n        IHI: -3041,\n        IHO: -1935,\n        IIH: -825,\n        IIM: -1035,\n        IOI: -542,\n        KHH: -1216,\n        KKA: 491,\n        KKH: -1217,\n        KOK: -1009,\n        MHH: -2694,\n        MHM: -457,\n        MHO: 123,\n        MMH: -471,\n        NNH: -1689,\n        NNO: 662,\n        OHO: -3393,\n    };\n    this.TC4__ = {\n        HHH: -203,\n        HHI: 1344,\n        HHK: 365,\n        HHM: -122,\n        HHN: 182,\n        HHO: 669,\n        HIH: 804,\n        HII: 679,\n        HOH: 446,\n        IHH: 695,\n        IHO: -2324,\n        IIH: 321,\n        III: 1497,\n        IIO: 656,\n        IOO: 54,\n        KAK: 4845,\n        KKA: 3386,\n        KKK: 3065,\n        MHH: -405,\n        MHI: 201,\n        MMH: -241,\n        MMM: 661,\n        MOM: 841,\n    };\n    this.TQ1__ = {\n        BHHH: -227,\n        BHHI: 316,\n        BHIH: -132,\n        BIHH: 60,\n        BIII: 1595,\n        BNHH: -744,\n        BOHH: 225,\n        BOOO: -908,\n        OAKK: 482,\n        OHHH: 281,\n        OHIH: 249,\n        OIHI: 200,\n        OIIH: -68,\n    };\n    this.TQ2__ = { BIHH: -1401, BIII: -1033, BKAK: -543, BOOO: -5591 };\n    this.TQ3__ = {\n        BHHH: 478,\n        BHHM: -1073,\n        BHIH: 222,\n        BHII: -504,\n        BIIH: -116,\n        BIII: -105,\n        BMHI: -863,\n        BMHM: -464,\n        BOMH: 620,\n        OHHH: 346,\n        OHHI: 1729,\n        OHII: 997,\n        OHMH: 481,\n        OIHH: 623,\n        OIIH: 1344,\n        OKAK: 2792,\n        OKHH: 587,\n        OKKA: 679,\n        OOHH: 110,\n        OOII: -685,\n    };\n    this.TQ4__ = {\n        BHHH: -721,\n        BHHM: -3604,\n        BHII: -966,\n        BIIH: -607,\n        BIII: -2181,\n        OAAA: -2763,\n        OAKK: 180,\n        OHHH: -294,\n        OHHI: 2446,\n        OHHO: 480,\n        OHIH: -1573,\n        OIHH: 1935,\n        OIHI: -493,\n        OIIH: 626,\n        OIII: -4007,\n        OKAK: -8156,\n    };\n    this.TW1__ = { につい: -4681, 東京都: 2026 };\n    this.TW2__ = {\n        ある程: -2049,\n        いった: -1256,\n        ころが: -2434,\n        しょう: 3873,\n        その後: -4430,\n        だって: -1049,\n        ていた: 1833,\n        として: -4657,\n        ともに: -4517,\n        もので: 1882,\n        一気に: -792,\n        初めて: -1512,\n        同時に: -8097,\n        大きな: -1255,\n        対して: -2721,\n        社会党: -3216,\n    };\n    this.TW3__ = {\n        いただ: -1734,\n        してい: 1314,\n        として: -4314,\n        につい: -5483,\n        にとっ: -5989,\n        に当た: -6247,\n        'ので,': -727,\n        'ので、': -727,\n        のもの: -600,\n        れから: -3752,\n        十二月: -2287,\n    };\n    this.TW4__ = {\n        'いう.': 8576,\n        'いう。': 8576,\n        からな: -2348,\n        してい: 2958,\n        'たが,': 1516,\n        'たが、': 1516,\n        ている: 1538,\n        という: 1349,\n        ました: 5543,\n        ません: 1097,\n        ようと: -4258,\n        よると: 5865,\n    };\n    this.UC1__ = { A: 484, K: 93, M: 645, O: -505 };\n    this.UC2__ = { A: 819, H: 1059, I: 409, M: 3987, N: 5775, O: 646 };\n    this.UC3__ = { A: -1370, I: 2311 };\n    this.UC4__ = { A: -2643, H: 1809, I: -1032, K: -3450, M: 3565, N: 3876, O: 6646 };\n    this.UC5__ = { H: 313, I: -1238, K: -799, M: 539, O: -831 };\n    this.UC6__ = { H: -506, I: -253, K: 87, M: 247, O: -387 };\n    this.UP1__ = { O: -214 };\n    this.UP2__ = { B: 69, O: 935 };\n    this.UP3__ = { B: 189 };\n    this.UQ1__ = {\n        BH: 21,\n        BI: -12,\n        BK: -99,\n        BN: 142,\n        BO: -56,\n        OH: -95,\n        OI: 477,\n        OK: 410,\n        OO: -2422,\n    };\n    this.UQ2__ = { BH: 216, BI: 113, OK: 1759 };\n    this.UQ3__ = {\n        BA: -479,\n        BH: 42,\n        BI: 1913,\n        BK: -7198,\n        BM: 3160,\n        BN: 6427,\n        BO: 14761,\n        OI: -827,\n        ON: -3212,\n    };\n    this.UW1__ = {\n        ',': 156,\n        '、': 156,\n        '「': -463,\n        あ: -941,\n        う: -127,\n        が: -553,\n        き: 121,\n        こ: 505,\n        で: -201,\n        と: -547,\n        ど: -123,\n        に: -789,\n        の: -185,\n        は: -847,\n        も: -466,\n        や: -470,\n        よ: 182,\n        ら: -292,\n        り: 208,\n        れ: 169,\n        を: -446,\n        ん: -137,\n        '・': -135,\n        主: -402,\n        京: -268,\n        区: -912,\n        午: 871,\n        国: -460,\n        大: 561,\n        委: 729,\n        市: -411,\n        日: -141,\n        理: 361,\n        生: -408,\n        県: -386,\n        都: -718,\n        '｢': -463,\n        '･': -135,\n    };\n    this.UW2__ = {\n        ',': -829,\n        '、': -829,\n        〇: 892,\n        '「': -645,\n        '」': 3145,\n        あ: -538,\n        い: 505,\n        う: 134,\n        お: -502,\n        か: 1454,\n        が: -856,\n        く: -412,\n        こ: 1141,\n        さ: 878,\n        ざ: 540,\n        し: 1529,\n        す: -675,\n        せ: 300,\n        そ: -1011,\n        た: 188,\n        だ: 1837,\n        つ: -949,\n        て: -291,\n        で: -268,\n        と: -981,\n        ど: 1273,\n        な: 1063,\n        に: -1764,\n        の: 130,\n        は: -409,\n        ひ: -1273,\n        べ: 1261,\n        ま: 600,\n        も: -1263,\n        や: -402,\n        よ: 1639,\n        り: -579,\n        る: -694,\n        れ: 571,\n        を: -2516,\n        ん: 2095,\n        ア: -587,\n        カ: 306,\n        キ: 568,\n        ッ: 831,\n        三: -758,\n        不: -2150,\n        世: -302,\n        中: -968,\n        主: -861,\n        事: 492,\n        人: -123,\n        会: 978,\n        保: 362,\n        入: 548,\n        初: -3025,\n        副: -1566,\n        北: -3414,\n        区: -422,\n        大: -1769,\n        天: -865,\n        太: -483,\n        子: -1519,\n        学: 760,\n        実: 1023,\n        小: -2009,\n        市: -813,\n        年: -1060,\n        強: 1067,\n        手: -1519,\n        揺: -1033,\n        政: 1522,\n        文: -1355,\n        新: -1682,\n        日: -1815,\n        明: -1462,\n        最: -630,\n        朝: -1843,\n        本: -1650,\n        東: -931,\n        果: -665,\n        次: -2378,\n        民: -180,\n        気: -1740,\n        理: 752,\n        発: 529,\n        目: -1584,\n        相: -242,\n        県: -1165,\n        立: -763,\n        第: 810,\n        米: 509,\n        自: -1353,\n        行: 838,\n        西: -744,\n        見: -3874,\n        調: 1010,\n        議: 1198,\n        込: 3041,\n        開: 1758,\n        間: -1257,\n        '｢': -645,\n        '｣': 3145,\n        ｯ: 831,\n        ｱ: -587,\n        ｶ: 306,\n        ｷ: 568,\n    };\n    this.UW3__ = {\n        ',': 4889,\n        1: -800,\n        '−': -1723,\n        '、': 4889,\n        々: -2311,\n        〇: 5827,\n        '」': 2670,\n        '〓': -3573,\n        あ: -2696,\n        い: 1006,\n        う: 2342,\n        え: 1983,\n        お: -4864,\n        か: -1163,\n        が: 3271,\n        く: 1004,\n        け: 388,\n        げ: 401,\n        こ: -3552,\n        ご: -3116,\n        さ: -1058,\n        し: -395,\n        す: 584,\n        せ: 3685,\n        そ: -5228,\n        た: 842,\n        ち: -521,\n        っ: -1444,\n        つ: -1081,\n        て: 6167,\n        で: 2318,\n        と: 1691,\n        ど: -899,\n        な: -2788,\n        に: 2745,\n        の: 4056,\n        は: 4555,\n        ひ: -2171,\n        ふ: -1798,\n        へ: 1199,\n        ほ: -5516,\n        ま: -4384,\n        み: -120,\n        め: 1205,\n        も: 2323,\n        や: -788,\n        よ: -202,\n        ら: 727,\n        り: 649,\n        る: 5905,\n        れ: 2773,\n        わ: -1207,\n        を: 6620,\n        ん: -518,\n        ア: 551,\n        グ: 1319,\n        ス: 874,\n        ッ: -1350,\n        ト: 521,\n        ム: 1109,\n        ル: 1591,\n        ロ: 2201,\n        ン: 278,\n        '・': -3794,\n        一: -1619,\n        下: -1759,\n        世: -2087,\n        両: 3815,\n        中: 653,\n        主: -758,\n        予: -1193,\n        二: 974,\n        人: 2742,\n        今: 792,\n        他: 1889,\n        以: -1368,\n        低: 811,\n        何: 4265,\n        作: -361,\n        保: -2439,\n        元: 4858,\n        党: 3593,\n        全: 1574,\n        公: -3030,\n        六: 755,\n        共: -1880,\n        円: 5807,\n        再: 3095,\n        分: 457,\n        初: 2475,\n        別: 1129,\n        前: 2286,\n        副: 4437,\n        力: 365,\n        動: -949,\n        務: -1872,\n        化: 1327,\n        北: -1038,\n        区: 4646,\n        千: -2309,\n        午: -783,\n        協: -1006,\n        口: 483,\n        右: 1233,\n        各: 3588,\n        合: -241,\n        同: 3906,\n        和: -837,\n        員: 4513,\n        国: 642,\n        型: 1389,\n        場: 1219,\n        外: -241,\n        妻: 2016,\n        学: -1356,\n        安: -423,\n        実: -1008,\n        家: 1078,\n        小: -513,\n        少: -3102,\n        州: 1155,\n        市: 3197,\n        平: -1804,\n        年: 2416,\n        広: -1030,\n        府: 1605,\n        度: 1452,\n        建: -2352,\n        当: -3885,\n        得: 1905,\n        思: -1291,\n        性: 1822,\n        戸: -488,\n        指: -3973,\n        政: -2013,\n        教: -1479,\n        数: 3222,\n        文: -1489,\n        新: 1764,\n        日: 2099,\n        旧: 5792,\n        昨: -661,\n        時: -1248,\n        曜: -951,\n        最: -937,\n        月: 4125,\n        期: 360,\n        李: 3094,\n        村: 364,\n        東: -805,\n        核: 5156,\n        森: 2438,\n        業: 484,\n        氏: 2613,\n        民: -1694,\n        決: -1073,\n        法: 1868,\n        海: -495,\n        無: 979,\n        物: 461,\n        特: -3850,\n        生: -273,\n        用: 914,\n        町: 1215,\n        的: 7313,\n        直: -1835,\n        省: 792,\n        県: 6293,\n        知: -1528,\n        私: 4231,\n        税: 401,\n        立: -960,\n        第: 1201,\n        米: 7767,\n        系: 3066,\n        約: 3663,\n        級: 1384,\n        統: -4229,\n        総: 1163,\n        線: 1255,\n        者: 6457,\n        能: 725,\n        自: -2869,\n        英: 785,\n        見: 1044,\n        調: -562,\n        財: -733,\n        費: 1777,\n        車: 1835,\n        軍: 1375,\n        込: -1504,\n        通: -1136,\n        選: -681,\n        郎: 1026,\n        郡: 4404,\n        部: 1200,\n        金: 2163,\n        長: 421,\n        開: -1432,\n        間: 1302,\n        関: -1282,\n        雨: 2009,\n        電: -1045,\n        非: 2066,\n        駅: 1620,\n        '１': -800,\n        '｣': 2670,\n        '･': -3794,\n        ｯ: -1350,\n        ｱ: 551,\n        ｸﾞ: 1319,\n        ｽ: 874,\n        ﾄ: 521,\n        ﾑ: 1109,\n        ﾙ: 1591,\n        ﾛ: 2201,\n        ﾝ: 278,\n    };\n    this.UW4__ = {\n        ',': 3930,\n        '.': 3508,\n        '―': -4841,\n        '、': 3930,\n        '。': 3508,\n        〇: 4999,\n        '「': 1895,\n        '」': 3798,\n        '〓': -5156,\n        あ: 4752,\n        い: -3435,\n        う: -640,\n        え: -2514,\n        お: 2405,\n        か: 530,\n        が: 6006,\n        き: -4482,\n        ぎ: -3821,\n        く: -3788,\n        け: -4376,\n        げ: -4734,\n        こ: 2255,\n        ご: 1979,\n        さ: 2864,\n        し: -843,\n        じ: -2506,\n        す: -731,\n        ず: 1251,\n        せ: 181,\n        そ: 4091,\n        た: 5034,\n        だ: 5408,\n        ち: -3654,\n        っ: -5882,\n        つ: -1659,\n        て: 3994,\n        で: 7410,\n        と: 4547,\n        な: 5433,\n        に: 6499,\n        ぬ: 1853,\n        ね: 1413,\n        の: 7396,\n        は: 8578,\n        ば: 1940,\n        ひ: 4249,\n        び: -4134,\n        ふ: 1345,\n        へ: 6665,\n        べ: -744,\n        ほ: 1464,\n        ま: 1051,\n        み: -2082,\n        む: -882,\n        め: -5046,\n        も: 4169,\n        ゃ: -2666,\n        や: 2795,\n        ょ: -1544,\n        よ: 3351,\n        ら: -2922,\n        り: -9726,\n        る: -14896,\n        れ: -2613,\n        ろ: -4570,\n        わ: -1783,\n        を: 13150,\n        ん: -2352,\n        カ: 2145,\n        コ: 1789,\n        セ: 1287,\n        ッ: -724,\n        ト: -403,\n        メ: -1635,\n        ラ: -881,\n        リ: -541,\n        ル: -856,\n        ン: -3637,\n        '・': -4371,\n        ー: -11870,\n        一: -2069,\n        中: 2210,\n        予: 782,\n        事: -190,\n        井: -1768,\n        人: 1036,\n        以: 544,\n        会: 950,\n        体: -1286,\n        作: 530,\n        側: 4292,\n        先: 601,\n        党: -2006,\n        共: -1212,\n        内: 584,\n        円: 788,\n        初: 1347,\n        前: 1623,\n        副: 3879,\n        力: -302,\n        動: -740,\n        務: -2715,\n        化: 776,\n        区: 4517,\n        協: 1013,\n        参: 1555,\n        合: -1834,\n        和: -681,\n        員: -910,\n        器: -851,\n        回: 1500,\n        国: -619,\n        園: -1200,\n        地: 866,\n        場: -1410,\n        塁: -2094,\n        士: -1413,\n        多: 1067,\n        大: 571,\n        子: -4802,\n        学: -1397,\n        定: -1057,\n        寺: -809,\n        小: 1910,\n        屋: -1328,\n        山: -1500,\n        島: -2056,\n        川: -2667,\n        市: 2771,\n        年: 374,\n        庁: -4556,\n        後: 456,\n        性: 553,\n        感: 916,\n        所: -1566,\n        支: 856,\n        改: 787,\n        政: 2182,\n        教: 704,\n        文: 522,\n        方: -856,\n        日: 1798,\n        時: 1829,\n        最: 845,\n        月: -9066,\n        木: -485,\n        来: -442,\n        校: -360,\n        業: -1043,\n        氏: 5388,\n        民: -2716,\n        気: -910,\n        沢: -939,\n        済: -543,\n        物: -735,\n        率: 672,\n        球: -1267,\n        生: -1286,\n        産: -1101,\n        田: -2900,\n        町: 1826,\n        的: 2586,\n        目: 922,\n        省: -3485,\n        県: 2997,\n        空: -867,\n        立: -2112,\n        第: 788,\n        米: 2937,\n        系: 786,\n        約: 2171,\n        経: 1146,\n        統: -1169,\n        総: 940,\n        線: -994,\n        署: 749,\n        者: 2145,\n        能: -730,\n        般: -852,\n        行: -792,\n        規: 792,\n        警: -1184,\n        議: -244,\n        谷: -1000,\n        賞: 730,\n        車: -1481,\n        軍: 1158,\n        輪: -1433,\n        込: -3370,\n        近: 929,\n        道: -1291,\n        選: 2596,\n        郎: -4866,\n        都: 1192,\n        野: -1100,\n        銀: -2213,\n        長: 357,\n        間: -2344,\n        院: -2297,\n        際: -2604,\n        電: -878,\n        領: -1659,\n        題: -792,\n        館: -1984,\n        首: 1749,\n        高: 2120,\n        '｢': 1895,\n        '｣': 3798,\n        '･': -4371,\n        ｯ: -724,\n        ｰ: -11870,\n        ｶ: 2145,\n        ｺ: 1789,\n        ｾ: 1287,\n        ﾄ: -403,\n        ﾒ: -1635,\n        ﾗ: -881,\n        ﾘ: -541,\n        ﾙ: -856,\n        ﾝ: -3637,\n    };\n    this.UW5__ = {\n        ',': 465,\n        '.': -299,\n        1: -514,\n        E2: -32768,\n        ']': -2762,\n        '、': 465,\n        '。': -299,\n        '「': 363,\n        あ: 1655,\n        い: 331,\n        う: -503,\n        え: 1199,\n        お: 527,\n        か: 647,\n        が: -421,\n        き: 1624,\n        ぎ: 1971,\n        く: 312,\n        げ: -983,\n        さ: -1537,\n        し: -1371,\n        す: -852,\n        だ: -1186,\n        ち: 1093,\n        っ: 52,\n        つ: 921,\n        て: -18,\n        で: -850,\n        と: -127,\n        ど: 1682,\n        な: -787,\n        に: -1224,\n        の: -635,\n        は: -578,\n        べ: 1001,\n        み: 502,\n        め: 865,\n        ゃ: 3350,\n        ょ: 854,\n        り: -208,\n        る: 429,\n        れ: 504,\n        わ: 419,\n        を: -1264,\n        ん: 327,\n        イ: 241,\n        ル: 451,\n        ン: -343,\n        中: -871,\n        京: 722,\n        会: -1153,\n        党: -654,\n        務: 3519,\n        区: -901,\n        告: 848,\n        員: 2104,\n        大: -1296,\n        学: -548,\n        定: 1785,\n        嵐: -1304,\n        市: -2991,\n        席: 921,\n        年: 1763,\n        思: 872,\n        所: -814,\n        挙: 1618,\n        新: -1682,\n        日: 218,\n        月: -4353,\n        査: 932,\n        格: 1356,\n        機: -1508,\n        氏: -1347,\n        田: 240,\n        町: -3912,\n        的: -3149,\n        相: 1319,\n        省: -1052,\n        県: -4003,\n        研: -997,\n        社: -278,\n        空: -813,\n        統: 1955,\n        者: -2233,\n        表: 663,\n        語: -1073,\n        議: 1219,\n        選: -1018,\n        郎: -368,\n        長: 786,\n        間: 1191,\n        題: 2368,\n        館: -689,\n        '１': -514,\n        Ｅ２: -32768,\n        '｢': 363,\n        ｲ: 241,\n        ﾙ: 451,\n        ﾝ: -343,\n    };\n    this.UW6__ = {\n        ',': 227,\n        '.': 808,\n        1: -270,\n        E1: 306,\n        '、': 227,\n        '。': 808,\n        あ: -307,\n        う: 189,\n        か: 241,\n        が: -73,\n        く: -121,\n        こ: -200,\n        じ: 1782,\n        す: 383,\n        た: -428,\n        っ: 573,\n        て: -1014,\n        で: 101,\n        と: -105,\n        な: -253,\n        に: -149,\n        の: -417,\n        は: -236,\n        も: -206,\n        り: 187,\n        る: -135,\n        を: 195,\n        ル: -673,\n        ン: -496,\n        一: -277,\n        中: 201,\n        件: -800,\n        会: 624,\n        前: 302,\n        区: 1792,\n        員: -1212,\n        委: 798,\n        学: -960,\n        市: 887,\n        広: -695,\n        後: 535,\n        業: -697,\n        相: 753,\n        社: -507,\n        福: 974,\n        空: -822,\n        者: 1811,\n        連: 463,\n        郎: 1082,\n        '１': -270,\n        Ｅ１: 306,\n        ﾙ: -673,\n        ﾝ: -496,\n    };\n\n    return this;\n}\n\nTinySegmenter.prototype.ctype_ = function (str) {\n    for (var i in this.chartype_) {\n        if (str.match(this.chartype_[i][0])) {\n            return this.chartype_[i][1];\n        }\n    }\n    return 'O';\n};\n\nTinySegmenter.prototype.ts_ = function (v) {\n    if (v) {\n        return v;\n    }\n    return 0;\n};\n\nTinySegmenter.prototype.segment = function (input) {\n    if (input == null || input == undefined || input == '') {\n        return [];\n    }\n    var result = [];\n    var seg = ['B3', 'B2', 'B1'];\n    var ctype = ['O', 'O', 'O'];\n    var o = input.split('');\n    for (i = 0; i < o.length; ++i) {\n        seg.push(o[i]);\n        ctype.push(this.ctype_(o[i]));\n    }\n    seg.push('E1');\n    seg.push('E2');\n    seg.push('E3');\n    ctype.push('O');\n    ctype.push('O');\n    ctype.push('O');\n    var word = seg[3];\n    var p1 = 'U';\n    var p2 = 'U';\n    var p3 = 'U';\n    for (var i = 4; i < seg.length - 3; ++i) {\n        var score = this.BIAS__;\n        var w1 = seg[i - 3];\n        var w2 = seg[i - 2];\n        var w3 = seg[i - 1];\n        var w4 = seg[i];\n        var w5 = seg[i + 1];\n        var w6 = seg[i + 2];\n        var c1 = ctype[i - 3];\n        var c2 = ctype[i - 2];\n        var c3 = ctype[i - 1];\n        var c4 = ctype[i];\n        var c5 = ctype[i + 1];\n        var c6 = ctype[i + 2];\n        score += this.ts_(this.UP1__[p1]);\n        score += this.ts_(this.UP2__[p2]);\n        score += this.ts_(this.UP3__[p3]);\n        score += this.ts_(this.BP1__[p1 + p2]);\n        score += this.ts_(this.BP2__[p2 + p3]);\n        score += this.ts_(this.UW1__[w1]);\n        score += this.ts_(this.UW2__[w2]);\n        score += this.ts_(this.UW3__[w3]);\n        score += this.ts_(this.UW4__[w4]);\n        score += this.ts_(this.UW5__[w5]);\n        score += this.ts_(this.UW6__[w6]);\n        score += this.ts_(this.BW1__[w2 + w3]);\n        score += this.ts_(this.BW2__[w3 + w4]);\n        score += this.ts_(this.BW3__[w4 + w5]);\n        score += this.ts_(this.TW1__[w1 + w2 + w3]);\n        score += this.ts_(this.TW2__[w2 + w3 + w4]);\n        score += this.ts_(this.TW3__[w3 + w4 + w5]);\n        score += this.ts_(this.TW4__[w4 + w5 + w6]);\n        score += this.ts_(this.UC1__[c1]);\n        score += this.ts_(this.UC2__[c2]);\n        score += this.ts_(this.UC3__[c3]);\n        score += this.ts_(this.UC4__[c4]);\n        score += this.ts_(this.UC5__[c5]);\n        score += this.ts_(this.UC6__[c6]);\n        score += this.ts_(this.BC1__[c2 + c3]);\n        score += this.ts_(this.BC2__[c3 + c4]);\n        score += this.ts_(this.BC3__[c4 + c5]);\n        score += this.ts_(this.TC1__[c1 + c2 + c3]);\n        score += this.ts_(this.TC2__[c2 + c3 + c4]);\n        score += this.ts_(this.TC3__[c3 + c4 + c5]);\n        score += this.ts_(this.TC4__[c4 + c5 + c6]);\n        //  score += this.ts_(this.TC5__[c4 + c5 + c6]);\n        score += this.ts_(this.UQ1__[p1 + c1]);\n        score += this.ts_(this.UQ2__[p2 + c2]);\n        score += this.ts_(this.UQ3__[p3 + c3]);\n        score += this.ts_(this.BQ1__[p2 + c2 + c3]);\n        score += this.ts_(this.BQ2__[p2 + c3 + c4]);\n        score += this.ts_(this.BQ3__[p3 + c2 + c3]);\n        score += this.ts_(this.BQ4__[p3 + c3 + c4]);\n        score += this.ts_(this.TQ1__[p2 + c1 + c2 + c3]);\n        score += this.ts_(this.TQ2__[p2 + c2 + c3 + c4]);\n        score += this.ts_(this.TQ3__[p3 + c1 + c2 + c3]);\n        score += this.ts_(this.TQ4__[p3 + c2 + c3 + c4]);\n        var p = 'O';\n        if (score > 0) {\n            result.push(word);\n            word = '';\n            p = 'B';\n        }\n        p1 = p2;\n        p2 = p3;\n        p3 = p;\n        word += seg[i];\n    }\n    result.push(word);\n\n    return result;\n};\n"
  },
  {
    "path": "static/js/searchElasticlunr.js",
    "content": "/**\n * elasticlunr - http://weixsong.github.io\n * Lightweight full-text search engine in Javascript for browser search and offline search. - 0.9.5\n *\n * Copyright (C) 2017 Oliver Nightingale\n * Copyright (C) 2017 Wei Song\n * MIT Licensed\n * @license\n */\n(function () {\n    /*!\n     * elasticlunr.js\n     * Copyright (C) 2017 Oliver Nightingale\n     * Copyright (C) 2017 Wei Song\n     */\n\n    /**\n     * Convenience function for instantiating a new elasticlunr index and configuring it\n     * with the default pipeline functions and the passed config function.\n     *\n     * When using this convenience function a new index will be created with the\n     * following functions already in the pipeline:\n     *\n     * 1. elasticlunr.trimmer - trim non-word character\n     * 2. elasticlunr.StopWordFilter - filters out any stop words before they enter the\n     * index\n     * 3. elasticlunr.stemmer - stems the tokens before entering the index.\n     *\n     *\n     * Example:\n     *\n     *     var idx = elasticlunr(function () {\n     *       this.addField('id');\n     *       this.addField('title');\n     *       this.addField('body');\n     *\n     *       //this.setRef('id'); // default ref is 'id'\n     *\n     *       this.pipeline.add(function () {\n     *         // some custom pipeline function\n     *       });\n     *     });\n     *\n     *    idx.addDoc({\n     *      id: 1,\n     *      title: 'Oracle released database 12g',\n     *      body: 'Yestaday, Oracle has released their latest database, named 12g, more robust. this product will increase Oracle profit.'\n     *    });\n     *\n     *    idx.addDoc({\n     *      id: 2,\n     *      title: 'Oracle released annual profit report',\n     *      body: 'Yestaday, Oracle has released their annual profit report of 2015, total profit is 12.5 Billion.'\n     *    });\n     *\n     *    # simple search\n     *    idx.search('oracle database');\n     *\n     *    # search with query-time boosting\n     *    idx.search('oracle database', {fields: {title: {boost: 2}, body: {boost: 1}}});\n     *\n     * @param {Function} config A function that will be called with the new instance\n     * of the elasticlunr.Index as both its context and first parameter. It can be used to\n     * customize the instance of new elasticlunr.Index.\n     * @namespace\n     * @module\n     * @return {elasticlunr.Index}\n     *\n     */\n    const elasticlunr = function (config) {\n        const idx = new elasticlunr.Index();\n\n        idx.pipeline.add(\n            elasticlunr.trimmer,\n            elasticlunr.stopWordFilter,\n            elasticlunr.stemmer\n        );\n\n        if (config) config.call(idx, idx);\n\n        return idx;\n    };\n\n    elasticlunr.version = '0.9.5';\n\n    // only used this to make elasticlunr.js compatible with lunr-languages\n    // this is a trick to define a global alias of elasticlunr\n    lunr = elasticlunr;\n\n    /*!\n     * elasticlunr.utils\n     * Copyright (C) 2017 Oliver Nightingale\n     * Copyright (C) 2017 Wei Song\n     */\n\n    /**\n     * A namespace containing utils for the rest of the elasticlunr library\n     */\n    elasticlunr.utils = {};\n\n    /**\n     * Print a warning message to the console.\n     *\n     * @param {String} message The message to be printed.\n     * @memberOf Utils\n     */\n    elasticlunr.utils.warn = (function (global) {\n        return function (message) {\n            if (global.console && console.warn) {\n                console.warn(message);\n            }\n        };\n    })(this);\n\n    /**\n     * Convert an object to string.\n     *\n     * In the case of `null` and `undefined` the function returns\n     * an empty string, in all other cases the result of calling\n     * `toString` on the passed object is returned.\n     *\n     * @param {object} obj The object to convert to a string.\n     * @return {String} string representation of the passed object.\n     * @memberOf Utils\n     */\n    elasticlunr.utils.toString = function (obj) {\n        if (obj === void 0 || obj === null) {\n            return '';\n        }\n\n        return obj.toString();\n    };\n    /*!\n     * elasticlunr.EventEmitter\n     * Copyright (C) 2017 Oliver Nightingale\n     * Copyright (C) 2017 Wei Song\n     */\n\n    /**\n     * elasticlunr.EventEmitter is an event emitter for elasticlunr.\n     * It manages adding and removing event handlers and triggering events and their handlers.\n     *\n     * Each event could has multiple corresponding functions,\n     * these functions will be called as the sequence that they are added into the event.\n     *\n     * @constructor\n     */\n    elasticlunr.EventEmitter = function () {\n        this.events = {};\n    };\n\n    /**\n     * Binds a handler function to a specific event(s).\n     *\n     * Can bind a single function to many different events in one call.\n     *\n     * @param {String} [eventName] The name(s) of events to bind this function to.\n     * @param {Function} fn The function to call when an event is fired.\n     * @memberOf EventEmitter\n     */\n    elasticlunr.EventEmitter.prototype.addListener = function () {\n        const args = Array.prototype.slice.call(arguments);\n        const fn = args.pop();\n        const names = args;\n\n        if (typeof fn !== 'function')\n            throw new TypeError('last argument must be a function');\n\n        names.forEach(function (name) {\n            if (!this.hasHandler(name)) this.events[name] = [];\n            this.events[name].push(fn);\n        }, this);\n    };\n\n    /**\n     * Removes a handler function from a specific event.\n     *\n     * @param {String} eventName The name of the event to remove this function from.\n     * @param {Function} fn The function to remove from an event.\n     * @memberOf EventEmitter\n     */\n    elasticlunr.EventEmitter.prototype.removeListener = function (name, fn) {\n        if (!this.hasHandler(name)) return;\n\n        const fnIndex = this.events[name].indexOf(fn);\n        if (fnIndex === -1) return;\n\n        this.events[name].splice(fnIndex, 1);\n\n        if (this.events[name].length === 0) delete this.events[name];\n    };\n\n    /**\n     * Call all functions that bounded to the given event.\n     *\n     * Additional data can be passed to the event handler as arguments to `emit`\n     * after the event name.\n     *\n     * @param {String} eventName The name of the event to emit.\n     * @memberOf EventEmitter\n     */\n    elasticlunr.EventEmitter.prototype.emit = function (name) {\n        if (!this.hasHandler(name)) return;\n\n        const args = Array.prototype.slice.call(arguments, 1);\n\n        this.events[name].forEach(function (fn) {\n            fn.apply(undefined, args);\n        }, this);\n    };\n\n    /**\n     * Checks whether a handler has ever been stored against an event.\n     *\n     * @param {String} eventName The name of the event to check.\n     * @private\n     * @memberOf EventEmitter\n     */\n    elasticlunr.EventEmitter.prototype.hasHandler = function (name) {\n        return name in this.events;\n    };\n    /*!\n     * elasticlunr.tokenizer\n     * Copyright (C) 2017 Oliver Nightingale\n     * Copyright (C) 2017 Wei Song\n     */\n\n    /**\n     * A function for splitting a string into tokens.\n     * Currently English is supported as default.\n     * Uses `elasticlunr.tokenizer.seperator` to split strings, you could change\n     * the value of this property to set how you want strings are split into tokens.\n     * IMPORTANT: use elasticlunr.tokenizer.seperator carefully, if you are not familiar with\n     * text process, then you'd better not change it.\n     *\n     * @module\n     * @param {String} str The string that you want to tokenize.\n     * @see elasticlunr.tokenizer.seperator\n     * @return {Array}\n     */\n    elasticlunr.tokenizer = function (str) {\n        if (!arguments.length || str === null || str === undefined) return [];\n        if (Array.isArray(str)) {\n            let arr = str.filter(function (token) {\n                if (token === null || token === undefined) {\n                    return false;\n                }\n\n                return true;\n            });\n\n            arr = arr.map(function (t) {\n                return elasticlunr.utils.toString(t).toLowerCase();\n            });\n\n            let out = [];\n            arr.forEach(function (item) {\n                const tokens = item.split(elasticlunr.tokenizer.seperator);\n                out = out.concat(tokens);\n            }, this);\n\n            return out;\n        }\n\n        return str\n            .toString()\n            .trim()\n            .toLowerCase()\n            .split(elasticlunr.tokenizer.seperator);\n    };\n\n    /**\n     * Default string seperator.\n     */\n    elasticlunr.tokenizer.defaultSeperator = /[\\s-]+/;\n\n    /**\n     * The sperator used to split a string into tokens. Override this property to change the behaviour of\n     * `elasticlunr.tokenizer` behaviour when tokenizing strings. By default this splits on whitespace and hyphens.\n     *\n     * @static\n     * @see elasticlunr.tokenizer\n     */\n    elasticlunr.tokenizer.seperator = elasticlunr.tokenizer.defaultSeperator;\n\n    /**\n     * Set up customized string seperator\n     *\n     * @param {Object} sep The customized seperator that you want to use to tokenize a string.\n     */\n    elasticlunr.tokenizer.setSeperator = function (sep) {\n        if (sep !== null && sep !== undefined && typeof sep === 'object') {\n            elasticlunr.tokenizer.seperator = sep;\n        }\n    };\n\n    /**\n     * Reset string seperator\n     *\n     */\n    elasticlunr.tokenizer.resetSeperator = function () {\n        elasticlunr.tokenizer.seperator = elasticlunr.tokenizer.defaultSeperator;\n    };\n\n    /**\n     * Get string seperator\n     *\n     */\n    elasticlunr.tokenizer.getSeperator = function () {\n        return elasticlunr.tokenizer.seperator;\n    };\n    /*!\n     * elasticlunr.Pipeline\n     * Copyright (C) 2017 Oliver Nightingale\n     * Copyright (C) 2017 Wei Song\n     */\n\n    /**\n     * elasticlunr.Pipelines maintain an ordered list of functions to be applied to\n     * both documents tokens and query tokens.\n     *\n     * An instance of elasticlunr.Index will contain a pipeline\n     * with a trimmer, a stop word filter, an English stemmer. Extra\n     * functions can be added before or after either of these functions or these\n     * default functions can be removed.\n     *\n     * When run the pipeline, it will call each function in turn.\n     *\n     * The output of the functions in the pipeline will be passed to the next function\n     * in the pipeline. To exclude a token from entering the index the function\n     * should return undefined, the rest of the pipeline will not be called with\n     * this token.\n     *\n     * For serialisation of pipelines to work, all functions used in an instance of\n     * a pipeline should be registered with elasticlunr.Pipeline. Registered functions can\n     * then be loaded. If trying to load a serialised pipeline that uses functions\n     * that are not registered an error will be thrown.\n     *\n     * If not planning on serialising the pipeline then registering pipeline functions\n     * is not necessary.\n     *\n     * @constructor\n     */\n    elasticlunr.Pipeline = function () {\n        this._queue = [];\n    };\n\n    elasticlunr.Pipeline.registeredFunctions = {};\n\n    /**\n     * Register a function in the pipeline.\n     *\n     * Functions that are used in the pipeline should be registered if the pipeline\n     * needs to be serialised, or a serialised pipeline needs to be loaded.\n     *\n     * Registering a function does not add it to a pipeline, functions must still be\n     * added to instances of the pipeline for them to be used when running a pipeline.\n     *\n     * @param {Function} fn The function to register.\n     * @param {String} label The label to register this function with\n     * @memberOf Pipeline\n     */\n    elasticlunr.Pipeline.registerFunction = function (fn, label) {\n        if (label in elasticlunr.Pipeline.registeredFunctions) {\n            elasticlunr.utils.warn(\n                'Overwriting existing registered function: ' + label\n            );\n        }\n\n        fn.label = label;\n        elasticlunr.Pipeline.registeredFunctions[label] = fn;\n    };\n\n    /**\n     * Get a registered function in the pipeline.\n     *\n     * @param {String} label The label of registered function.\n     * @return {Function}\n     * @memberOf Pipeline\n     */\n    elasticlunr.Pipeline.getRegisteredFunction = function (label) {\n        if (label in elasticlunr.Pipeline.registeredFunctions !== true) {\n            return null;\n        }\n\n        return elasticlunr.Pipeline.registeredFunctions[label];\n    };\n\n    /**\n     * Warns if the function is not registered as a Pipeline function.\n     *\n     * @param {Function} fn The function to check for.\n     * @private\n     * @memberOf Pipeline\n     */\n    elasticlunr.Pipeline.warnIfFunctionNotRegistered = function (fn) {\n        const isRegistered = fn.label && fn.label in this.registeredFunctions;\n\n        if (!isRegistered) {\n            elasticlunr.utils.warn(\n                'Function is not registered with pipeline. This may cause problems when serialising the index.\\n',\n                fn\n            );\n        }\n    };\n\n    /**\n     * Loads a previously serialised pipeline.\n     *\n     * All functions to be loaded must already be registered with elasticlunr.Pipeline.\n     * If any function from the serialised data has not been registered then an\n     * error will be thrown.\n     *\n     * @param {Object} serialised The serialised pipeline to load.\n     * @return {elasticlunr.Pipeline}\n     * @memberOf Pipeline\n     */\n    elasticlunr.Pipeline.load = function (serialised) {\n        const pipeline = new elasticlunr.Pipeline();\n\n        serialised.forEach(function (fnName) {\n            const fn = elasticlunr.Pipeline.getRegisteredFunction(fnName);\n\n            if (fn) {\n                pipeline.add(fn);\n            } else {\n                throw new Error('Cannot load un-registered function: ' + fnName);\n            }\n        });\n\n        return pipeline;\n    };\n\n    /**\n     * Adds new functions to the end of the pipeline.\n     *\n     * Logs a warning if the function has not been registered.\n     *\n     * @param {Function} functions Any number of functions to add to the pipeline.\n     * @memberOf Pipeline\n     */\n    elasticlunr.Pipeline.prototype.add = function () {\n        const fns = Array.prototype.slice.call(arguments);\n\n        fns.forEach(function (fn) {\n            elasticlunr.Pipeline.warnIfFunctionNotRegistered(fn);\n            this._queue.push(fn);\n        }, this);\n    };\n\n    /**\n     * Adds a single function after a function that already exists in the\n     * pipeline.\n     *\n     * Logs a warning if the function has not been registered.\n     * If existingFn is not found, throw an Exception.\n     *\n     * @param {Function} existingFn A function that already exists in the pipeline.\n     * @param {Function} newFn The new function to add to the pipeline.\n     * @memberOf Pipeline\n     */\n    elasticlunr.Pipeline.prototype.after = function (existingFn, newFn) {\n        elasticlunr.Pipeline.warnIfFunctionNotRegistered(newFn);\n\n        const pos = this._queue.indexOf(existingFn);\n        if (pos === -1) {\n            throw new Error('Cannot find existingFn');\n        }\n\n        this._queue.splice(pos + 1, 0, newFn);\n    };\n\n    /**\n     * Adds a single function before a function that already exists in the\n     * pipeline.\n     *\n     * Logs a warning if the function has not been registered.\n     * If existingFn is not found, throw an Exception.\n     *\n     * @param {Function} existingFn A function that already exists in the pipeline.\n     * @param {Function} newFn The new function to add to the pipeline.\n     * @memberOf Pipeline\n     */\n    elasticlunr.Pipeline.prototype.before = function (existingFn, newFn) {\n        elasticlunr.Pipeline.warnIfFunctionNotRegistered(newFn);\n\n        const pos = this._queue.indexOf(existingFn);\n        if (pos === -1) {\n            throw new Error('Cannot find existingFn');\n        }\n\n        this._queue.splice(pos, 0, newFn);\n    };\n\n    /**\n     * Removes a function from the pipeline.\n     *\n     * @param {Function} fn The function to remove from the pipeline.\n     * @memberOf Pipeline\n     */\n    elasticlunr.Pipeline.prototype.remove = function (fn) {\n        const pos = this._queue.indexOf(fn);\n        if (pos === -1) {\n            return;\n        }\n\n        this._queue.splice(pos, 1);\n    };\n\n    /**\n     * Runs the current list of functions that registered in the pipeline against the\n     * input tokens.\n     *\n     * @param {Array} tokens The tokens to run through the pipeline.\n     * @return {Array}\n     * @memberOf Pipeline\n     */\n    elasticlunr.Pipeline.prototype.run = function (tokens) {\n        const out = [];\n        const tokenLength = tokens.length;\n        const pipelineLength = this._queue.length;\n\n        for (let i = 0; i < tokenLength; i++) {\n            let token = tokens[i];\n\n            for (let j = 0; j < pipelineLength; j++) {\n                token = this._queue[j](token, i, tokens);\n                if (token === void 0 || token === null) break;\n            }\n\n            if (token !== void 0 && token !== null) out.push(token);\n        }\n\n        return out;\n    };\n\n    /**\n     * Resets the pipeline by removing any existing processors.\n     *\n     * @memberOf Pipeline\n     */\n    elasticlunr.Pipeline.prototype.reset = function () {\n        this._queue = [];\n    };\n\n    /**\n     * Get the pipeline if user want to check the pipeline.\n     *\n     * @memberOf Pipeline\n     */\n    elasticlunr.Pipeline.prototype.get = function () {\n        return this._queue;\n    };\n\n    /**\n     * Returns a representation of the pipeline ready for serialisation.\n     * Only serialize pipeline function's name. Not storing function, so when\n     * loading the archived JSON index file, corresponding pipeline function is\n     * added by registered function of elasticlunr.Pipeline.registeredFunctions\n     *\n     * Logs a warning if the function has not been registered.\n     *\n     * @return {Array}\n     * @memberOf Pipeline\n     */\n    elasticlunr.Pipeline.prototype.toJSON = function () {\n        return this._queue.map(function (fn) {\n            elasticlunr.Pipeline.warnIfFunctionNotRegistered(fn);\n            return fn.label;\n        });\n    };\n    /*!\n     * elasticlunr.Index\n     * Copyright (C) 2017 Oliver Nightingale\n     * Copyright (C) 2017 Wei Song\n     */\n\n    /**\n     * elasticlunr.Index is object that manages a search index.  It contains the indexes\n     * and stores all the tokens and document lookups.  It also provides the main\n     * user facing API for the library.\n     *\n     * @constructor\n     */\n    elasticlunr.Index = function () {\n        this._fields = [];\n        this._ref = 'id';\n        this.pipeline = new elasticlunr.Pipeline();\n        this.documentStore = new elasticlunr.DocumentStore();\n        this.index = {};\n        this.eventEmitter = new elasticlunr.EventEmitter();\n        this._idfCache = {};\n\n        this.on(\n            'add',\n            'remove',\n            'update',\n            function () {\n                this._idfCache = {};\n            }.bind(this)\n        );\n    };\n\n    /**\n     * Bind a handler to events being emitted by the index.\n     *\n     * The handler can be bound to many events at the same time.\n     *\n     * @param {String} [eventName] The name(s) of events to bind the function to.\n     * @param {Function} fn The serialised set to load.\n     * @memberOf Index\n     */\n    elasticlunr.Index.prototype.on = function () {\n        const args = Array.prototype.slice.call(arguments);\n        return this.eventEmitter.addListener.apply(this.eventEmitter, args);\n    };\n\n    /**\n     * Removes a handler from an event being emitted by the index.\n     *\n     * @param {String} eventName The name of events to remove the function from.\n     * @param {Function} fn The serialised set to load.\n     * @memberOf Index\n     */\n    elasticlunr.Index.prototype.off = function (name, fn) {\n        return this.eventEmitter.removeListener(name, fn);\n    };\n\n    /**\n     * Loads a previously serialised index.\n     *\n     * Issues a warning if the index being imported was serialised\n     * by a different version of elasticlunr.\n     *\n     * @param {Object} serialisedData The serialised set to load.\n     * @return {elasticlunr.Index}\n     * @memberOf Index\n     */\n    elasticlunr.Index.load = function (serialisedData) {\n        if (serialisedData.version !== elasticlunr.version) {\n            elasticlunr.utils.warn(\n                'version mismatch: current ' +\n                    elasticlunr.version +\n                    ' importing ' +\n                    serialisedData.version\n            );\n        }\n\n        const idx = new this();\n\n        idx._fields = serialisedData.fields;\n        idx._ref = serialisedData.ref;\n        idx.documentStore = elasticlunr.DocumentStore.load(\n            serialisedData.documentStore\n        );\n        idx.pipeline = elasticlunr.Pipeline.load(serialisedData.pipeline);\n        idx.index = {};\n        for (const field in serialisedData.index) {\n            idx.index[field] = elasticlunr.InvertedIndex.load(\n                serialisedData.index[field]\n            );\n        }\n\n        return idx;\n    };\n\n    /**\n     * Adds a field to the list of fields that will be searchable within documents in the index.\n     *\n     * Remember that inner index is build based on field, which means each field has one inverted index.\n     *\n     * Fields should be added before any documents are added to the index, fields\n     * that are added after documents are added to the index will only apply to new\n     * documents added to the index.\n     *\n     * @param {String} fieldName The name of the field within the document that should be indexed\n     * @return {elasticlunr.Index}\n     * @memberOf Index\n     */\n    elasticlunr.Index.prototype.addField = function (fieldName) {\n        this._fields.push(fieldName);\n        this.index[fieldName] = new elasticlunr.InvertedIndex();\n        return this;\n    };\n\n    /**\n     * Sets the property used to uniquely identify documents added to the index,\n     * by default this property is 'id'.\n     *\n     * This should only be changed before adding documents to the index, changing\n     * the ref property without resetting the index can lead to unexpected results.\n     *\n     * @param {String} refName The property to use to uniquely identify the\n     * documents in the index.\n     * @param {Boolean} emitEvent Whether to emit add events, defaults to true\n     * @return {elasticlunr.Index}\n     * @memberOf Index\n     */\n    elasticlunr.Index.prototype.setRef = function (refName) {\n        this._ref = refName;\n        return this;\n    };\n\n    /**\n     *\n     * Set if the JSON format original documents are save into elasticlunr.DocumentStore\n     *\n     * Defaultly save all the original JSON documents.\n     *\n     * @param {Boolean} save Whether to save the original JSON documents.\n     * @return {elasticlunr.Index}\n     * @memberOf Index\n     */\n    elasticlunr.Index.prototype.saveDocument = function (save) {\n        this.documentStore = new elasticlunr.DocumentStore(save);\n        return this;\n    };\n\n    /**\n     * Add a JSON format document to the index.\n     *\n     * This is the way new documents enter the index, this function will run the\n     * fields from the document through the index's pipeline and then add it to\n     * the index, it will then show up in search results.\n     *\n     * An 'add' event is emitted with the document that has been added and the index\n     * the document has been added to. This event can be silenced by passing false\n     * as the second argument to add.\n     *\n     * @param {Object} doc The JSON format document to add to the index.\n     * @param {Boolean} emitEvent Whether or not to emit events, default true.\n     * @memberOf Index\n     */\n    elasticlunr.Index.prototype.addDoc = function (doc, emitEvent) {\n        if (!doc) return;\n        var emitEvent = emitEvent === undefined ? true : emitEvent;\n\n        const docRef = doc[this._ref];\n\n        this.documentStore.addDoc(docRef, doc);\n        this._fields.forEach(function (field) {\n            const fieldTokens = this.pipeline.run(elasticlunr.tokenizer(doc[field]));\n            this.documentStore.addFieldLength(docRef, field, fieldTokens.length);\n\n            const tokenCount = {};\n            fieldTokens.forEach(function (token) {\n                if (token in tokenCount) tokenCount[token] += 1;\n                else tokenCount[token] = 1;\n            }, this);\n\n            for (const token in tokenCount) {\n                let termFrequency = tokenCount[token];\n                termFrequency = Math.sqrt(termFrequency);\n                this.index[field].addToken(token, { ref: docRef, tf: termFrequency });\n            }\n        }, this);\n\n        if (emitEvent) this.eventEmitter.emit('add', doc, this);\n    };\n\n    /**\n     * Removes a document from the index by doc ref.\n     *\n     * To make sure documents no longer show up in search results they can be\n     * removed from the index using this method.\n     *\n     * A 'remove' event is emitted with the document that has been removed and the index\n     * the document has been removed from. This event can be silenced by passing false\n     * as the second argument to remove.\n     *\n     * If user setting DocumentStore not storing the documents, then remove doc by docRef is not allowed.\n     *\n     * @param {String|Integer} docRef The document ref to remove from the index.\n     * @param {Boolean} emitEvent Whether to emit remove events, defaults to true\n     * @memberOf Index\n     */\n    elasticlunr.Index.prototype.removeDocByRef = function (docRef, emitEvent) {\n        if (!docRef) return;\n        if (this.documentStore.isDocStored() === false) {\n            return;\n        }\n\n        if (!this.documentStore.hasDoc(docRef)) return;\n        const doc = this.documentStore.getDoc(docRef);\n        this.removeDoc(doc, false);\n    };\n\n    /**\n     * Removes a document from the index.\n     * This remove operation could work even the original doc is not store in the DocumentStore.\n     *\n     * To make sure documents no longer show up in search results they can be\n     * removed from the index using this method.\n     *\n     * A 'remove' event is emitted with the document that has been removed and the index\n     * the document has been removed from. This event can be silenced by passing false\n     * as the second argument to remove.\n     *\n     *\n     * @param {Object} doc The document ref to remove from the index.\n     * @param {Boolean} emitEvent Whether to emit remove events, defaults to true\n     * @memberOf Index\n     */\n    elasticlunr.Index.prototype.removeDoc = function (doc, emitEvent) {\n        if (!doc) return;\n\n        var emitEvent = emitEvent === undefined ? true : emitEvent;\n\n        const docRef = doc[this._ref];\n        if (!this.documentStore.hasDoc(docRef)) return;\n\n        this.documentStore.removeDoc(docRef);\n\n        this._fields.forEach(function (field) {\n            const fieldTokens = this.pipeline.run(elasticlunr.tokenizer(doc[field]));\n            fieldTokens.forEach(function (token) {\n                this.index[field].removeToken(token, docRef);\n            }, this);\n        }, this);\n\n        if (emitEvent) this.eventEmitter.emit('remove', doc, this);\n    };\n\n    /**\n     * Updates a document in the index.\n     *\n     * When a document contained within the index gets updated, fields changed,\n     * added or removed, to make sure it correctly matched against search queries,\n     * it should be updated in the index.\n     *\n     * This method is just a wrapper around `remove` and `add`\n     *\n     * An 'update' event is emitted with the document that has been updated and the index.\n     * This event can be silenced by passing false as the second argument to update. Only\n     * an update event will be fired, the 'add' and 'remove' events of the underlying calls\n     * are silenced.\n     *\n     * @param {Object} doc The document to update in the index.\n     * @param {Boolean} emitEvent Whether to emit update events, defaults to true\n     * @see Index.prototype.remove\n     * @see Index.prototype.add\n     * @memberOf Index\n     */\n    elasticlunr.Index.prototype.updateDoc = function (doc, emitEvent) {\n        var emitEvent = emitEvent === undefined ? true : emitEvent;\n\n        this.removeDocByRef(doc[this._ref], false);\n        this.addDoc(doc, false);\n\n        if (emitEvent) this.eventEmitter.emit('update', doc, this);\n    };\n\n    /**\n     * Calculates the inverse document frequency for a token within the index of a field.\n     *\n     * @param {String} token The token to calculate the idf of.\n     * @param {String} field The field to compute idf.\n     * @see Index.prototype.idf\n     * @private\n     * @memberOf Index\n     */\n    elasticlunr.Index.prototype.idf = function (term, field) {\n        const cacheKey = '@' + field + '/' + term;\n        if (Object.prototype.hasOwnProperty.call(this._idfCache, cacheKey))\n            return this._idfCache[cacheKey];\n\n        const df = this.index[field].getDocFreq(term);\n        const idf = 1 + Math.log(this.documentStore.length / (df + 1));\n        this._idfCache[cacheKey] = idf;\n\n        return idf;\n    };\n\n    /**\n     * get fields of current index instance\n     *\n     * @return {Array}\n     */\n    elasticlunr.Index.prototype.getFields = function () {\n        return this._fields.slice();\n    };\n\n    /**\n     * Searches the index using the passed query.\n     * Queries should be a string, multiple words are allowed.\n     *\n     * If config is null, will search all fields defaultly, and lead to OR based query.\n     * If config is specified, will search specified with query time boosting.\n     *\n     * All query tokens are passed through the same pipeline that document tokens\n     * are passed through, so any language processing involved will be run on every\n     * query term.\n     *\n     * Each query term is expanded, so that the term 'he' might be expanded to\n     * 'hello' and 'help' if those terms were already included in the index.\n     *\n     * Matching documents are returned as an array of objects, each object contains\n     * the matching document ref, as set for this index, and the similarity score\n     * for this document against the query.\n     *\n     * @param {String} query The query to search the index with.\n     * @param {JSON} userConfig The user query config, JSON format.\n     * @return {Object}\n     * @see Index.prototype.idf\n     * @see Index.prototype.documentVector\n     * @memberOf Index\n     */\n    elasticlunr.Index.prototype.search = function (query, userConfig) {\n        if (!query) return [];\n        if (typeof query === 'string') {\n            query = { any: query };\n        } else {\n            query = JSON.parse(JSON.stringify(query));\n        }\n\n        let configStr = null;\n        if (userConfig != null) {\n            configStr = JSON.stringify(userConfig);\n        }\n\n        const config = new elasticlunr.Configuration(configStr, this.getFields()).get();\n\n        const queryTokens = {};\n        const queryFields = Object.keys(query);\n\n        for (let i = 0; i < queryFields.length; i++) {\n            const key = queryFields[i];\n\n            queryTokens[key] = this.pipeline.run(elasticlunr.tokenizer(query[key]));\n        }\n\n        const queryResults = {};\n\n        for (const field in config) {\n            const tokens = queryTokens[field] || queryTokens.any;\n            if (!tokens) {\n                continue;\n            }\n\n            const fieldSearchResults = this.fieldSearch(tokens, field, config);\n            const fieldBoost = config[field].boost;\n\n            for (var docRef in fieldSearchResults) {\n                fieldSearchResults[docRef] = fieldSearchResults[docRef] * fieldBoost;\n            }\n\n            for (var docRef in fieldSearchResults) {\n                if (docRef in queryResults) {\n                    queryResults[docRef] += fieldSearchResults[docRef];\n                } else {\n                    queryResults[docRef] = fieldSearchResults[docRef];\n                }\n            }\n        }\n\n        const results = [];\n        let result;\n        for (var docRef in queryResults) {\n            result = { ref: docRef, score: queryResults[docRef] };\n            if (this.documentStore.hasDoc(docRef)) {\n                result.doc = this.documentStore.getDoc(docRef);\n            }\n            results.push(result);\n        }\n\n        results.sort(function (a, b) {\n            return b.score - a.score;\n        });\n        return results;\n    };\n\n    /**\n     * search queryTokens in specified field.\n     *\n     * @param {Array} queryTokens The query tokens to query in this field.\n     * @param {String} field Field to query in.\n     * @param {elasticlunr.Configuration} config The user query config, JSON format.\n     * @return {Object}\n     */\n    elasticlunr.Index.prototype.fieldSearch = function (\n        queryTokens,\n        fieldName,\n        config\n    ) {\n        const booleanType = config[fieldName].bool;\n        const expand = config[fieldName].expand;\n        const boost = config[fieldName].boost;\n        let scores = null;\n        const docTokens = {};\n\n        // Do nothing if the boost is 0\n        if (boost === 0) {\n            return;\n        }\n\n        queryTokens.forEach(function (token) {\n            let tokens = [token];\n            if (expand === true) {\n                tokens = this.index[fieldName].expandToken(token);\n            }\n            // Consider every query token in turn. If expanded, each query token\n            // corresponds to a set of tokens, which is all tokens in the\n            // index matching the pattern queryToken* .\n            // For the set of tokens corresponding to a query token, find and score\n            // all matching documents. Store those scores in queryTokenScores,\n            // keyed by docRef.\n            // Then, depending on the value of booleanType, combine the scores\n            // for this query token with previous scores.  If booleanType is OR,\n            // then merge the scores by summing into the accumulated total, adding\n            // new document scores are required (effectively a union operator).\n            // If booleanType is AND, accumulate scores only if the document\n            // has previously been scored by another query token (an intersection\n            // operation0.\n            // Furthermore, since when booleanType is AND, additional\n            // query tokens can't add new documents to the result set, use the\n            // current document set to limit the processing of each new query\n            // token for efficiency (i.e., incremental intersection).\n\n            const queryTokenScores = {};\n            tokens.forEach(function (key) {\n                let docs = this.index[fieldName].getDocs(key);\n                const idf = this.idf(key, fieldName);\n\n                if (scores && booleanType === 'AND') {\n                    // special case, we can rule out documents that have been\n                    // already been filtered out because they weren't scored\n                    // by previous query token passes.\n                    const filteredDocs = {};\n                    for (var docRef in scores) {\n                        if (docRef in docs) {\n                            filteredDocs[docRef] = docs[docRef];\n                        }\n                    }\n                    docs = filteredDocs;\n                }\n                // only record appeared token for retrieved documents for the\n                // original token, not for expaned token.\n                // beause for doing coordNorm for a retrieved document, coordNorm only care how many\n                // query token appear in that document.\n                // so expanded token should not be added into docTokens, if added, this will pollute the\n                // coordNorm\n                if (key === token) {\n                    this.fieldSearchStats(docTokens, key, docs);\n                }\n\n                for (var docRef in docs) {\n                    const tf = this.index[fieldName].getTermFrequency(key, docRef);\n                    const fieldLength = this.documentStore.getFieldLength(\n                        docRef,\n                        fieldName\n                    );\n                    let fieldLengthNorm = 1;\n                    if (fieldLength !== 0) {\n                        fieldLengthNorm = 1 / Math.sqrt(fieldLength);\n                    }\n\n                    let penality = 1;\n                    if (key !== token) {\n                        // currently I'm not sure if this penality is enough,\n                        // need to do verification\n                        penality =\n                            (1 - (key.length - token.length) / key.length) * 0.15;\n                    }\n\n                    const score = tf * idf * fieldLengthNorm * penality;\n\n                    if (docRef in queryTokenScores) {\n                        queryTokenScores[docRef] += score;\n                    } else {\n                        queryTokenScores[docRef] = score;\n                    }\n                }\n            }, this);\n\n            scores = this.mergeScores(scores, queryTokenScores, booleanType);\n        }, this);\n\n        scores = this.coordNorm(scores, docTokens, queryTokens.length);\n        return scores;\n    };\n\n    /**\n     * Merge the scores from one set of tokens into an accumulated score table.\n     * Exact operation depends on the op parameter. If op is 'AND', then only the\n     * intersection of the two score lists is retained. Otherwise, the union of\n     * the two score lists is returned. For internal use only.\n     *\n     * @param {Object} bool accumulated scores. Should be null on first call.\n     * @param {String} scores new scores to merge into accumScores.\n     * @param {Object} op merge operation (should be 'AND' or 'OR').\n     *\n     */\n\n    elasticlunr.Index.prototype.mergeScores = function (accumScores, scores, op) {\n        if (!accumScores) {\n            return scores;\n        }\n        if (op === 'AND') {\n            const intersection = {};\n            for (var docRef in scores) {\n                if (docRef in accumScores) {\n                    intersection[docRef] = accumScores[docRef] + scores[docRef];\n                }\n            }\n            return intersection;\n        } else {\n            for (var docRef in scores) {\n                if (docRef in accumScores) {\n                    accumScores[docRef] += scores[docRef];\n                } else {\n                    accumScores[docRef] = scores[docRef];\n                }\n            }\n            return accumScores;\n        }\n    };\n\n    /**\n     * Record the occuring query token of retrieved doc specified by doc field.\n     * Only for inner user.\n     *\n     * @param {Object} docTokens a data structure stores which token appears in the retrieved doc.\n     * @param {String} token query token\n     * @param {Object} docs the retrieved documents of the query token\n     *\n     */\n    elasticlunr.Index.prototype.fieldSearchStats = function (docTokens, token, docs) {\n        for (const doc in docs) {\n            if (doc in docTokens) {\n                docTokens[doc].push(token);\n            } else {\n                docTokens[doc] = [token];\n            }\n        }\n    };\n\n    /**\n     * coord norm the score of a doc.\n     * if a doc contain more query tokens, then the score will larger than the doc\n     * contains less query tokens.\n     *\n     * only for inner use.\n     *\n     * @param {Object} results first results\n     * @param {Object} docs field search results of a token\n     * @param {Integer} n query token number\n     * @return {Object}\n     */\n    elasticlunr.Index.prototype.coordNorm = function (scores, docTokens, n) {\n        for (const doc in scores) {\n            if (!(doc in docTokens)) continue;\n            const tokens = docTokens[doc].length;\n            scores[doc] = (scores[doc] * tokens) / n;\n        }\n\n        return scores;\n    };\n\n    /**\n     * Returns a representation of the index ready for serialisation.\n     *\n     * @return {Object}\n     * @memberOf Index\n     */\n    elasticlunr.Index.prototype.toJSON = function () {\n        const indexJson = {};\n        this._fields.forEach(function (field) {\n            indexJson[field] = this.index[field].toJSON();\n        }, this);\n\n        return {\n            version: elasticlunr.version,\n            fields: this._fields,\n            ref: this._ref,\n            documentStore: this.documentStore.toJSON(),\n            index: indexJson,\n            pipeline: this.pipeline.toJSON(),\n        };\n    };\n\n    /**\n     * Applies a plugin to the current index.\n     *\n     * A plugin is a function that is called with the index as its context.\n     * Plugins can be used to customise or extend the behaviour the index\n     * in some way. A plugin is just a function, that encapsulated the custom\n     * behaviour that should be applied to the index.\n     *\n     * The plugin function will be called with the index as its argument, additional\n     * arguments can also be passed when calling use. The function will be called\n     * with the index as its context.\n     *\n     * Example:\n     *\n     *     var myPlugin = function (idx, arg1, arg2) {\n     *       // `this` is the index to be extended\n     *       // apply any extensions etc here.\n     *     }\n     *\n     *     var idx = elasticlunr(function () {\n     *       this.use(myPlugin, 'arg1', 'arg2')\n     *     })\n     *\n     * @param {Function} plugin The plugin to apply.\n     * @memberOf Index\n     */\n    elasticlunr.Index.prototype.use = function (plugin) {\n        const args = Array.prototype.slice.call(arguments, 1);\n        args.unshift(this);\n        plugin.apply(this, args);\n    };\n    /*!\n     * elasticlunr.DocumentStore\n     * Copyright (C) 2017 Wei Song\n     */\n\n    /**\n     * elasticlunr.DocumentStore is a simple key-value document store used for storing sets of tokens for\n     * documents stored in index.\n     *\n     * elasticlunr.DocumentStore store original JSON format documents that you could build search snippet by this original JSON document.\n     *\n     * user could choose whether original JSON format document should be store, if no configuration then document will be stored defaultly.\n     * If user care more about the index size, user could select not store JSON documents, then this will has some defects, such as user\n     * could not use JSON document to generate snippets of search results.\n     *\n     * @param {Boolean} save If the original JSON document should be stored.\n     * @constructor\n     * @module\n     */\n    elasticlunr.DocumentStore = function (save) {\n        if (save === null || save === undefined) {\n            this._save = true;\n        } else {\n            this._save = save;\n        }\n\n        this.docs = {};\n        this.docInfo = {};\n        this.length = 0;\n    };\n\n    /**\n     * Loads a previously serialised document store\n     *\n     * @param {Object} serialisedData The serialised document store to load.\n     * @return {elasticlunr.DocumentStore}\n     */\n    elasticlunr.DocumentStore.load = function (serialisedData) {\n        const store = new this();\n\n        store.length = serialisedData.length;\n        store.docs = serialisedData.docs;\n        store.docInfo = serialisedData.docInfo;\n        store._save = serialisedData.save;\n\n        return store;\n    };\n\n    /**\n     * check if current instance store the original doc\n     *\n     * @return {Boolean}\n     */\n    elasticlunr.DocumentStore.prototype.isDocStored = function () {\n        return this._save;\n    };\n\n    /**\n     * Stores the given doc in the document store against the given id.\n     * If docRef already exist, then update doc.\n     *\n     * Document is store by original JSON format, then you could use original document to generate search snippets.\n     *\n     * @param {Integer|String} docRef The key used to store the JSON format doc.\n     * @param {Object} doc The JSON format doc.\n     */\n    elasticlunr.DocumentStore.prototype.addDoc = function (docRef, doc) {\n        if (!this.hasDoc(docRef)) this.length++;\n\n        if (this._save === true) {\n            this.docs[docRef] = clone(doc);\n        } else {\n            this.docs[docRef] = null;\n        }\n    };\n\n    /**\n     * Retrieves the JSON doc from the document store for a given key.\n     *\n     * If docRef not found, return null.\n     * If user set not storing the documents, return null.\n     *\n     * @param {Integer|String} docRef The key to lookup and retrieve from the document store.\n     * @return {Object}\n     * @memberOf DocumentStore\n     */\n    elasticlunr.DocumentStore.prototype.getDoc = function (docRef) {\n        if (this.hasDoc(docRef) === false) return null;\n        return this.docs[docRef];\n    };\n\n    /**\n     * Checks whether the document store contains a key (docRef).\n     *\n     * @param {Integer|String} docRef The id to look up in the document store.\n     * @return {Boolean}\n     * @memberOf DocumentStore\n     */\n    elasticlunr.DocumentStore.prototype.hasDoc = function (docRef) {\n        return docRef in this.docs;\n    };\n\n    /**\n     * Removes the value for a key in the document store.\n     *\n     * @param {Integer|String} docRef The id to remove from the document store.\n     * @memberOf DocumentStore\n     */\n    elasticlunr.DocumentStore.prototype.removeDoc = function (docRef) {\n        if (!this.hasDoc(docRef)) return;\n\n        delete this.docs[docRef];\n        delete this.docInfo[docRef];\n        this.length--;\n    };\n\n    /**\n     * Add field length of a document's field tokens from pipeline results.\n     * The field length of a document is used to do field length normalization even without the original JSON document stored.\n     *\n     * @param {Integer|String} docRef document's id or reference\n     * @param {String} fieldName field name\n     * @param {Integer} length field length\n     */\n    elasticlunr.DocumentStore.prototype.addFieldLength = function (\n        docRef,\n        fieldName,\n        length\n    ) {\n        if (docRef === null || docRef === undefined) return;\n        if (this.hasDoc(docRef) == false) return;\n\n        if (!this.docInfo[docRef]) this.docInfo[docRef] = {};\n        this.docInfo[docRef][fieldName] = length;\n    };\n\n    /**\n     * Update field length of a document's field tokens from pipeline results.\n     * The field length of a document is used to do field length normalization even without the original JSON document stored.\n     *\n     * @param {Integer|String} docRef document's id or reference\n     * @param {String} fieldName field name\n     * @param {Integer} length field length\n     */\n    elasticlunr.DocumentStore.prototype.updateFieldLength = function (\n        docRef,\n        fieldName,\n        length\n    ) {\n        if (docRef === null || docRef === undefined) return;\n        if (this.hasDoc(docRef) == false) return;\n\n        this.addFieldLength(docRef, fieldName, length);\n    };\n\n    /**\n     * get field length of a document by docRef\n     *\n     * @param {Integer|String} docRef document id or reference\n     * @param {String} fieldName field name\n     * @return {Integer} field length\n     */\n    elasticlunr.DocumentStore.prototype.getFieldLength = function (docRef, fieldName) {\n        if (docRef === null || docRef === undefined) return 0;\n\n        if (!(docRef in this.docs)) return 0;\n        if (!(fieldName in this.docInfo[docRef])) return 0;\n        return this.docInfo[docRef][fieldName];\n    };\n\n    /**\n     * Returns a JSON representation of the document store used for serialisation.\n     *\n     * @return {Object} JSON format\n     * @memberOf DocumentStore\n     */\n    elasticlunr.DocumentStore.prototype.toJSON = function () {\n        return {\n            docs: this.docs,\n            docInfo: this.docInfo,\n            length: this.length,\n            save: this._save,\n        };\n    };\n\n    /**\n     * Cloning object\n     *\n     * @param {Object} object in JSON format\n     * @return {Object} copied object\n     */\n    function clone(obj) {\n        if (obj === null || typeof obj !== 'object') return obj;\n\n        const copy = obj.constructor();\n\n        for (const attr in obj) {\n            if (obj.hasOwnProperty(attr)) copy[attr] = obj[attr];\n        }\n\n        return copy;\n    }\n    /*!\n     * elasticlunr.stemmer\n     * Copyright (C) 2017 Oliver Nightingale\n     * Copyright (C) 2017 Wei Song\n     * Includes code from - http://tartarus.org/~martin/PorterStemmer/js.txt\n     */\n\n    /**\n     * elasticlunr.stemmer is an english language stemmer, this is a JavaScript\n     * implementation of the PorterStemmer taken from http://tartarus.org/~martin\n     *\n     * @module\n     * @param {String} str The string to stem\n     * @return {String}\n     * @see elasticlunr.Pipeline\n     */\n    elasticlunr.stemmer = (function () {\n        const step2list = {\n            ational: 'ate',\n            tional: 'tion',\n            enci: 'ence',\n            anci: 'ance',\n            izer: 'ize',\n            bli: 'ble',\n            alli: 'al',\n            entli: 'ent',\n            eli: 'e',\n            ousli: 'ous',\n            ization: 'ize',\n            ation: 'ate',\n            ator: 'ate',\n            alism: 'al',\n            iveness: 'ive',\n            fulness: 'ful',\n            ousness: 'ous',\n            aliti: 'al',\n            iviti: 'ive',\n            biliti: 'ble',\n            logi: 'log',\n        };\n\n        const step3list = {\n            icate: 'ic',\n            ative: '',\n            alize: 'al',\n            iciti: 'ic',\n            ical: 'ic',\n            ful: '',\n            ness: '',\n        };\n\n        const c = '[^aeiou]'; // consonant\n        const v = '[aeiouy]'; // vowel\n        const C = c + '[^aeiouy]*'; // consonant sequence\n        const V = v + '[aeiou]*'; // vowel sequence\n\n        const mgr0 = '^(' + C + ')?' + V + C; // [C]VC... is m>0\n        const meq1 = '^(' + C + ')?' + V + C + '(' + V + ')?$'; // [C]VC[V] is m=1\n        const mgr1 = '^(' + C + ')?' + V + C + V + C; // [C]VCVC... is m>1\n        const s_v = '^(' + C + ')?' + v; // vowel in stem\n\n        const re_mgr0 = new RegExp(mgr0);\n        const re_mgr1 = new RegExp(mgr1);\n        const re_meq1 = new RegExp(meq1);\n        const re_s_v = new RegExp(s_v);\n\n        const re_1a = /^(.+?)(ss|i)es$/;\n        const re2_1a = /^(.+?)([^s])s$/;\n        const re_1b = /^(.+?)eed$/;\n        const re2_1b = /^(.+?)(ed|ing)$/;\n        const re_1b_2 = /.$/;\n        const re2_1b_2 = /(at|bl|iz)$/;\n        const re3_1b_2 = new RegExp('([^aeiouylsz])\\\\1$');\n        const re4_1b_2 = new RegExp('^' + C + v + '[^aeiouwxy]$');\n\n        const re_1c = /^(.+?[^aeiou])y$/;\n        const re_2 =\n            /^(.+?)(ational|tional|enci|anci|izer|bli|alli|entli|eli|ousli|ization|ation|ator|alism|iveness|fulness|ousness|aliti|iviti|biliti|logi)$/;\n\n        const re_3 = /^(.+?)(icate|ative|alize|iciti|ical|ful|ness)$/;\n\n        const re_4 =\n            /^(.+?)(al|ance|ence|er|ic|able|ible|ant|ement|ment|ent|ou|ism|ate|iti|ous|ive|ize)$/;\n        const re2_4 = /^(.+?)(s|t)(ion)$/;\n\n        const re_5 = /^(.+?)e$/;\n        const re_5_1 = /ll$/;\n        const re3_5 = new RegExp('^' + C + v + '[^aeiouwxy]$');\n\n        const porterStemmer = function porterStemmer(w) {\n            let stem, suffix, firstch, re, re2, re3, re4;\n\n            if (w.length < 3) {\n                return w;\n            }\n\n            firstch = w.substr(0, 1);\n            if (firstch == 'y') {\n                w = firstch.toUpperCase() + w.substr(1);\n            }\n\n            // Step 1a\n            re = re_1a;\n            re2 = re2_1a;\n\n            if (re.test(w)) {\n                w = w.replace(re, '$1$2');\n            } else if (re2.test(w)) {\n                w = w.replace(re2, '$1$2');\n            }\n\n            // Step 1b\n            re = re_1b;\n            re2 = re2_1b;\n            if (re.test(w)) {\n                var fp = re.exec(w);\n                re = re_mgr0;\n                if (re.test(fp[1])) {\n                    re = re_1b_2;\n                    w = w.replace(re, '');\n                }\n            } else if (re2.test(w)) {\n                var fp = re2.exec(w);\n                stem = fp[1];\n                re2 = re_s_v;\n                if (re2.test(stem)) {\n                    w = stem;\n                    re2 = re2_1b_2;\n                    re3 = re3_1b_2;\n                    re4 = re4_1b_2;\n                    if (re2.test(w)) {\n                        w = w + 'e';\n                    } else if (re3.test(w)) {\n                        re = re_1b_2;\n                        w = w.replace(re, '');\n                    } else if (re4.test(w)) {\n                        w = w + 'e';\n                    }\n                }\n            }\n\n            // Step 1c - replace suffix y or Y by i if preceded by a non-vowel which is not the first letter of the word (so cry -> cri, by -> by, say -> say)\n            re = re_1c;\n            if (re.test(w)) {\n                var fp = re.exec(w);\n                stem = fp[1];\n                w = stem + 'i';\n            }\n\n            // Step 2\n            re = re_2;\n            if (re.test(w)) {\n                var fp = re.exec(w);\n                stem = fp[1];\n                suffix = fp[2];\n                re = re_mgr0;\n                if (re.test(stem)) {\n                    w = stem + step2list[suffix];\n                }\n            }\n\n            // Step 3\n            re = re_3;\n            if (re.test(w)) {\n                var fp = re.exec(w);\n                stem = fp[1];\n                suffix = fp[2];\n                re = re_mgr0;\n                if (re.test(stem)) {\n                    w = stem + step3list[suffix];\n                }\n            }\n\n            // Step 4\n            re = re_4;\n            re2 = re2_4;\n            if (re.test(w)) {\n                var fp = re.exec(w);\n                stem = fp[1];\n                re = re_mgr1;\n                if (re.test(stem)) {\n                    w = stem;\n                }\n            } else if (re2.test(w)) {\n                var fp = re2.exec(w);\n                stem = fp[1] + fp[2];\n                re2 = re_mgr1;\n                if (re2.test(stem)) {\n                    w = stem;\n                }\n            }\n\n            // Step 5\n            re = re_5;\n            if (re.test(w)) {\n                var fp = re.exec(w);\n                stem = fp[1];\n                re = re_mgr1;\n                re2 = re_meq1;\n                re3 = re3_5;\n                if (re.test(stem) || (re2.test(stem) && !re3.test(stem))) {\n                    w = stem;\n                }\n            }\n\n            re = re_5_1;\n            re2 = re_mgr1;\n            if (re.test(w) && re2.test(w)) {\n                re = re_1b_2;\n                w = w.replace(re, '');\n            }\n\n            // and turn initial Y back to y\n\n            if (firstch == 'y') {\n                w = firstch.toLowerCase() + w.substr(1);\n            }\n\n            return w;\n        };\n\n        return porterStemmer;\n    })();\n\n    elasticlunr.Pipeline.registerFunction(elasticlunr.stemmer, 'stemmer');\n    /*!\n     * elasticlunr.stopWordFilter\n     * Copyright (C) 2017 Oliver Nightingale\n     * Copyright (C) 2017 Wei Song\n     */\n\n    /**\n     * elasticlunr.stopWordFilter is an English language stop words filter, any words\n     * contained in the stop word list will not be passed through the filter.\n     *\n     * This is intended to be used in the Pipeline. If the token does not pass the\n     * filter then undefined will be returned.\n     * Currently this StopwordFilter using dictionary to do O(1) time complexity stop word filtering.\n     *\n     * @module\n     * @param {String} token The token to pass through the filter\n     * @return {String}\n     * @see elasticlunr.Pipeline\n     */\n    elasticlunr.stopWordFilter = function (token) {\n        if (token && elasticlunr.stopWordFilter.stopWords[token] !== true) {\n            return token;\n        }\n    };\n\n    /**\n     * Remove predefined stop words\n     * if user want to use customized stop words, user could use this function to delete\n     * all predefined stopwords.\n     *\n     * @return {null}\n     */\n    elasticlunr.clearStopWords = function () {\n        elasticlunr.stopWordFilter.stopWords = {};\n    };\n\n    /**\n     * Add customized stop words\n     * user could use this function to add customized stop words\n     *\n     * @params {Array} words customized stop words\n     * @return {null}\n     */\n    elasticlunr.addStopWords = function (words) {\n        if (words == null || Array.isArray(words) === false) return;\n\n        words.forEach(function (word) {\n            elasticlunr.stopWordFilter.stopWords[word] = true;\n        }, this);\n    };\n\n    /**\n     * Reset to default stop words\n     * user could use this function to restore default stop words\n     *\n     * @return {null}\n     */\n    elasticlunr.resetStopWords = function () {\n        elasticlunr.stopWordFilter.stopWords = elasticlunr.defaultStopWords;\n    };\n\n    elasticlunr.defaultStopWords = {\n        '': true,\n        a: true,\n        able: true,\n        about: true,\n        across: true,\n        after: true,\n        all: true,\n        almost: true,\n        also: true,\n        am: true,\n        among: true,\n        an: true,\n        and: true,\n        any: true,\n        are: true,\n        as: true,\n        at: true,\n        be: true,\n        because: true,\n        been: true,\n        but: true,\n        by: true,\n        can: true,\n        cannot: true,\n        could: true,\n        dear: true,\n        did: true,\n        do: true,\n        does: true,\n        either: true,\n        else: true,\n        ever: true,\n        every: true,\n        for: true,\n        from: true,\n        get: true,\n        got: true,\n        had: true,\n        has: true,\n        have: true,\n        he: true,\n        her: true,\n        hers: true,\n        him: true,\n        his: true,\n        how: true,\n        however: true,\n        i: true,\n        if: true,\n        in: true,\n        into: true,\n        is: true,\n        it: true,\n        its: true,\n        just: true,\n        least: true,\n        let: true,\n        like: true,\n        likely: true,\n        may: true,\n        me: true,\n        might: true,\n        most: true,\n        must: true,\n        my: true,\n        neither: true,\n        no: true,\n        nor: true,\n        not: true,\n        of: true,\n        off: true,\n        often: true,\n        on: true,\n        only: true,\n        or: true,\n        other: true,\n        our: true,\n        own: true,\n        rather: true,\n        said: true,\n        say: true,\n        says: true,\n        she: true,\n        should: true,\n        since: true,\n        so: true,\n        some: true,\n        than: true,\n        that: true,\n        the: true,\n        their: true,\n        them: true,\n        then: true,\n        there: true,\n        these: true,\n        they: true,\n        this: true,\n        tis: true,\n        to: true,\n        too: true,\n        twas: true,\n        us: true,\n        wants: true,\n        was: true,\n        we: true,\n        were: true,\n        what: true,\n        when: true,\n        where: true,\n        which: true,\n        while: true,\n        who: true,\n        whom: true,\n        why: true,\n        will: true,\n        with: true,\n        would: true,\n        yet: true,\n        you: true,\n        your: true,\n    };\n\n    elasticlunr.stopWordFilter.stopWords = elasticlunr.defaultStopWords;\n\n    elasticlunr.Pipeline.registerFunction(elasticlunr.stopWordFilter, 'stopWordFilter');\n    /*!\n     * elasticlunr.trimmer\n     * Copyright (C) 2017 Oliver Nightingale\n     * Copyright (C) 2017 Wei Song\n     */\n\n    /**\n     * elasticlunr.trimmer is a pipeline function for trimming non word\n     * characters from the begining and end of tokens before they\n     * enter the index.\n     *\n     * This implementation may not work correctly for non latin\n     * characters and should either be removed or adapted for use\n     * with languages with non-latin characters.\n     *\n     * @module\n     * @param {String} token The token to pass through the filter\n     * @return {String}\n     * @see elasticlunr.Pipeline\n     */\n    elasticlunr.trimmer = function (token) {\n        if (token === null || token === undefined) {\n            throw new Error('token should not be undefined');\n        }\n\n        return token.replace(/^\\W+/, '').replace(/\\W+$/, '');\n    };\n\n    elasticlunr.Pipeline.registerFunction(elasticlunr.trimmer, 'trimmer');\n    /*!\n     * elasticlunr.InvertedIndex\n     * Copyright (C) 2017 Wei Song\n     * Includes code from - http://tartarus.org/~martin/PorterStemmer/js.txt\n     */\n\n    /**\n     * elasticlunr.InvertedIndex is used for efficiently storing and\n     * lookup of documents that contain a given token.\n     *\n     * @constructor\n     */\n    elasticlunr.InvertedIndex = function () {\n        this.root = { docs: {}, df: 0 };\n    };\n\n    /**\n     * Loads a previously serialised inverted index.\n     *\n     * @param {Object} serialisedData The serialised inverted index to load.\n     * @return {elasticlunr.InvertedIndex}\n     */\n    elasticlunr.InvertedIndex.load = function (serialisedData) {\n        const idx = new this();\n        idx.root = serialisedData.root;\n\n        return idx;\n    };\n\n    /**\n     * Adds a {token: tokenInfo} pair to the inverted index.\n     * If the token already exist, then update the tokenInfo.\n     *\n     * tokenInfo format: { ref: 1, tf: 2}\n     * tokenInfor should contains the document's ref and the tf(token frequency) of that token in\n     * the document.\n     *\n     * By default this function starts at the root of the current inverted index, however\n     * it can start at any node of the inverted index if required.\n     *\n     * @param {String} token\n     * @param {Object} tokenInfo format: { ref: 1, tf: 2}\n     * @param {Object} root An optional node at which to start looking for the\n     * correct place to enter the doc, by default the root of this elasticlunr.InvertedIndex\n     * is used.\n     * @memberOf InvertedIndex\n     */\n    elasticlunr.InvertedIndex.prototype.addToken = function (token, tokenInfo, root) {\n        var root = root || this.root;\n        let idx = 0;\n\n        while (idx <= token.length - 1) {\n            const key = token[idx];\n\n            if (!(key in root)) root[key] = { docs: {}, df: 0 };\n            idx += 1;\n            root = root[key];\n        }\n\n        const docRef = tokenInfo.ref;\n        if (!root.docs[docRef]) {\n            // if this doc not exist, then add this doc\n            root.docs[docRef] = { tf: tokenInfo.tf };\n            root.df += 1;\n        } else {\n            // if this doc already exist, then update tokenInfo\n            root.docs[docRef] = { tf: tokenInfo.tf };\n        }\n    };\n\n    /**\n     * Checks whether a token is in this elasticlunr.InvertedIndex.\n     *\n     *\n     * @param {String} token The token to be checked\n     * @return {Boolean}\n     * @memberOf InvertedIndex\n     */\n    elasticlunr.InvertedIndex.prototype.hasToken = function (token) {\n        if (!token) return false;\n\n        let node = this.root;\n\n        for (let i = 0; i < token.length; i++) {\n            if (!node[token[i]]) return false;\n            node = node[token[i]];\n        }\n\n        return true;\n    };\n\n    /**\n     * Retrieve a node from the inverted index for a given token.\n     * If token not found in this InvertedIndex, return null.\n     *\n     *\n     * @param {String} token The token to get the node for.\n     * @return {Object}\n     * @see InvertedIndex.prototype.get\n     * @memberOf InvertedIndex\n     */\n    elasticlunr.InvertedIndex.prototype.getNode = function (token) {\n        if (!token) return null;\n\n        let node = this.root;\n\n        for (let i = 0; i < token.length; i++) {\n            if (!node[token[i]]) return null;\n            node = node[token[i]];\n        }\n\n        return node;\n    };\n\n    /**\n     * Retrieve the documents of a given token.\n     * If token not found, return {}.\n     *\n     *\n     * @param {String} token The token to get the documents for.\n     * @return {Object}\n     * @memberOf InvertedIndex\n     */\n    elasticlunr.InvertedIndex.prototype.getDocs = function (token) {\n        const node = this.getNode(token);\n        if (node == null) {\n            return {};\n        }\n\n        return node.docs;\n    };\n\n    /**\n     * Retrieve term frequency of given token in given docRef.\n     * If token or docRef not found, return 0.\n     *\n     *\n     * @param {String} token The token to get the documents for.\n     * @param {String|Integer} docRef\n     * @return {Integer}\n     * @memberOf InvertedIndex\n     */\n    elasticlunr.InvertedIndex.prototype.getTermFrequency = function (token, docRef) {\n        const node = this.getNode(token);\n\n        if (node == null) {\n            return 0;\n        }\n\n        if (!(docRef in node.docs)) {\n            return 0;\n        }\n\n        return node.docs[docRef].tf;\n    };\n\n    /**\n     * Retrieve the document frequency of given token.\n     * If token not found, return 0.\n     *\n     *\n     * @param {String} token The token to get the documents for.\n     * @return {Object}\n     * @memberOf InvertedIndex\n     */\n    elasticlunr.InvertedIndex.prototype.getDocFreq = function (token) {\n        const node = this.getNode(token);\n\n        if (node == null) {\n            return 0;\n        }\n\n        return node.df;\n    };\n\n    /**\n     * Remove the document identified by document's ref from the token in the inverted index.\n     *\n     *\n     * @param {String} token Remove the document from which token.\n     * @param {String} ref The ref of the document to remove from given token.\n     * @memberOf InvertedIndex\n     */\n    elasticlunr.InvertedIndex.prototype.removeToken = function (token, ref) {\n        if (!token) return;\n        const node = this.getNode(token);\n\n        if (node == null) return;\n\n        if (ref in node.docs) {\n            delete node.docs[ref];\n            node.df -= 1;\n        }\n    };\n\n    /**\n     * Find all the possible suffixes of given token using tokens currently in the inverted index.\n     * If token not found, return empty Array.\n     *\n     * @param {String} token The token to expand.\n     * @return {Array}\n     * @memberOf InvertedIndex\n     */\n    elasticlunr.InvertedIndex.prototype.expandToken = function (token, memo, root) {\n        if (token == null || token == '') return [];\n        var memo = memo || [];\n\n        if (root == void 0) {\n            root = this.getNode(token);\n            if (root == null) return memo;\n        }\n\n        if (root.df > 0) memo.push(token);\n\n        for (const key in root) {\n            if (key === 'docs') continue;\n            if (key === 'df') continue;\n            this.expandToken(token + key, memo, root[key]);\n        }\n\n        return memo;\n    };\n\n    /**\n     * Returns a representation of the inverted index ready for serialisation.\n     *\n     * @return {Object}\n     * @memberOf InvertedIndex\n     */\n    elasticlunr.InvertedIndex.prototype.toJSON = function () {\n        return {\n            root: this.root,\n        };\n    };\n\n    /*!\n     * elasticlunr.Configuration\n     * Copyright (C) 2017 Wei Song\n     */\n\n    /**\n     * elasticlunr.Configuration is used to analyze the user search configuration.\n     *\n     * By elasticlunr.Configuration user could set query-time boosting, boolean model in each field.\n     *\n     * Currently configuration supports:\n     * 1. query-time boosting, user could set how to boost each field.\n     * 2. boolean model chosing, user could choose which boolean model to use for each field.\n     * 3. token expandation, user could set token expand to True to improve Recall. Default is False.\n     *\n     * Query time boosting must be configured by field category, \"boolean\" model could be configured\n     * by both field category or globally as the following example. Field configuration for \"boolean\"\n     * will overwrite global configuration.\n     * Token expand could be configured both by field category or golbally. Local field configuration will\n     * overwrite global configuration.\n     *\n     * configuration example:\n     * {\n     *   fields:{\n     *     title: {boost: 2},\n     *     body: {boost: 1}\n     *   },\n     *   bool: \"OR\"\n     * }\n     *\n     * \"bool\" field configuation overwrite global configuation example:\n     * {\n     *   fields:{\n     *     title: {boost: 2, bool: \"AND\"},\n     *     body: {boost: 1}\n     *   },\n     *   bool: \"OR\"\n     * }\n     *\n     * \"expand\" example:\n     * {\n     *   fields:{\n     *     title: {boost: 2, bool: \"AND\"},\n     *     body: {boost: 1}\n     *   },\n     *   bool: \"OR\",\n     *   expand: true\n     * }\n     *\n     * \"expand\" example for field category:\n     * {\n     *   fields:{\n     *     title: {boost: 2, bool: \"AND\", expand: true},\n     *     body: {boost: 1}\n     *   },\n     *   bool: \"OR\"\n     * }\n     *\n     * setting the boost to 0 ignores the field (this will only search the title):\n     * {\n     *   fields:{\n     *     title: {boost: 1},\n     *     body: {boost: 0}\n     *   }\n     * }\n     *\n     * then, user could search with configuration to do query-time boosting.\n     * idx.search('oracle database', {fields: {title: {boost: 2}, body: {boost: 1}}});\n     *\n     *\n     * @constructor\n     *\n     * @param {String} config user configuration\n     * @param {Array} fields fields of index instance\n     * @module\n     */\n    elasticlunr.Configuration = function (config, fields) {\n        var config = config || '';\n\n        if (fields == undefined || fields == null) {\n            throw new Error('fields should not be null');\n        }\n\n        this.config = {};\n\n        let userConfig;\n        try {\n            userConfig = JSON.parse(config);\n            this.buildUserConfig(userConfig, fields);\n        } catch (error) {\n            elasticlunr.utils.warn(\n                'user configuration parse failed, will use default configuration'\n            );\n            this.buildDefaultConfig(fields);\n        }\n    };\n\n    /**\n     * Build default search configuration.\n     *\n     * @param {Array} fields fields of index instance\n     */\n    elasticlunr.Configuration.prototype.buildDefaultConfig = function (fields) {\n        this.reset();\n        fields.forEach(function (field) {\n            this.config[field] = {\n                boost: 1,\n                bool: 'OR',\n                expand: false,\n            };\n        }, this);\n    };\n\n    /**\n     * Build user configuration.\n     *\n     * @param {JSON} config User JSON configuratoin\n     * @param {Array} fields fields of index instance\n     */\n    elasticlunr.Configuration.prototype.buildUserConfig = function (config, fields) {\n        let global_bool = 'OR';\n        let global_expand = false;\n\n        this.reset();\n        if ('bool' in config) {\n            global_bool = config.bool || global_bool;\n        }\n\n        if ('expand' in config) {\n            global_expand = config.expand || global_expand;\n        }\n\n        if ('fields' in config) {\n            for (const field in config.fields) {\n                if (fields.indexOf(field) > -1) {\n                    const field_config = config.fields[field];\n                    let field_expand = global_expand;\n                    if (field_config.expand != undefined) {\n                        field_expand = field_config.expand;\n                    }\n\n                    this.config[field] = {\n                        boost:\n                            field_config.boost || field_config.boost === 0\n                                ? field_config.boost\n                                : 1,\n                        bool: field_config.bool || global_bool,\n                        expand: field_expand,\n                    };\n                } else {\n                    elasticlunr.utils.warn(\n                        'field name in user configuration not found in index instance fields'\n                    );\n                }\n            }\n        } else {\n            this.addAllFields2UserConfig(global_bool, global_expand, fields);\n        }\n    };\n\n    /**\n     * Add all fields to user search configuration.\n     *\n     * @param {String} bool Boolean model\n     * @param {String} expand Expand model\n     * @param {Array} fields fields of index instance\n     */\n    elasticlunr.Configuration.prototype.addAllFields2UserConfig = function (\n        bool,\n        expand,\n        fields\n    ) {\n        fields.forEach(function (field) {\n            this.config[field] = {\n                boost: 1,\n                bool,\n                expand,\n            };\n        }, this);\n    };\n\n    /**\n     * get current user configuration\n     */\n    elasticlunr.Configuration.prototype.get = function () {\n        return this.config;\n    };\n\n    /**\n     * reset user search configuration.\n     */\n    elasticlunr.Configuration.prototype.reset = function () {\n        this.config = {};\n    };\n    /**\n     * sorted_set.js is added only to make elasticlunr.js compatible with lunr-languages.\n     * if elasticlunr.js support different languages by default, this will make elasticlunr.js\n     * much bigger that not good for browser usage.\n     *\n     */\n\n    /*!\n     * lunr.SortedSet\n     * Copyright (C) 2017 Oliver Nightingale\n     */\n\n    /**\n     * lunr.SortedSets are used to maintain an array of uniq values in a sorted\n     * order.\n     *\n     * @constructor\n     */\n    lunr.SortedSet = function () {\n        this.length = 0;\n        this.elements = [];\n    };\n\n    /**\n     * Loads a previously serialised sorted set.\n     *\n     * @param {Array} serialisedData The serialised set to load.\n     * @returns {lunr.SortedSet}\n     * @memberOf SortedSet\n     */\n    lunr.SortedSet.load = function (serialisedData) {\n        const set = new this();\n\n        set.elements = serialisedData;\n        set.length = serialisedData.length;\n\n        return set;\n    };\n\n    /**\n     * Inserts new items into the set in the correct position to maintain the\n     * order.\n     *\n     * @param {Object} The objects to add to this set.\n     * @memberOf SortedSet\n     */\n    lunr.SortedSet.prototype.add = function () {\n        let i, element;\n\n        for (i = 0; i < arguments.length; i++) {\n            element = arguments[i];\n            if (~this.indexOf(element)) continue;\n            this.elements.splice(this.locationFor(element), 0, element);\n        }\n\n        this.length = this.elements.length;\n    };\n\n    /**\n     * Converts this sorted set into an array.\n     *\n     * @returns {Array}\n     * @memberOf SortedSet\n     */\n    lunr.SortedSet.prototype.toArray = function () {\n        return this.elements.slice();\n    };\n\n    /**\n     * Creates a new array with the results of calling a provided function on every\n     * element in this sorted set.\n     *\n     * Delegates to Array.prototype.map and has the same signature.\n     *\n     * @param {Function} fn The function that is called on each element of the\n     * set.\n     * @param {Object} ctx An optional object that can be used as the context\n     * for the function fn.\n     * @returns {Array}\n     * @memberOf SortedSet\n     */\n    lunr.SortedSet.prototype.map = function (fn, ctx) {\n        return this.elements.map(fn, ctx);\n    };\n\n    /**\n     * Executes a provided function once per sorted set element.\n     *\n     * Delegates to Array.prototype.forEach and has the same signature.\n     *\n     * @param {Function} fn The function that is called on each element of the\n     * set.\n     * @param {Object} ctx An optional object that can be used as the context\n     * @memberOf SortedSet\n     * for the function fn.\n     */\n    lunr.SortedSet.prototype.forEach = function (fn, ctx) {\n        return this.elements.forEach(fn, ctx);\n    };\n\n    /**\n     * Returns the index at which a given element can be found in the\n     * sorted set, or -1 if it is not present.\n     *\n     * @param {Object} elem The object to locate in the sorted set.\n     * @returns {Number}\n     * @memberOf SortedSet\n     */\n    lunr.SortedSet.prototype.indexOf = function (elem) {\n        let start = 0;\n        let end = this.elements.length;\n        let sectionLength = end - start;\n        let pivot = start + Math.floor(sectionLength / 2);\n        let pivotElem = this.elements[pivot];\n\n        while (sectionLength > 1) {\n            if (pivotElem === elem) return pivot;\n\n            if (pivotElem < elem) start = pivot;\n            if (pivotElem > elem) end = pivot;\n\n            sectionLength = end - start;\n            pivot = start + Math.floor(sectionLength / 2);\n            pivotElem = this.elements[pivot];\n        }\n\n        if (pivotElem === elem) return pivot;\n\n        return -1;\n    };\n\n    /**\n     * Returns the position within the sorted set that an element should be\n     * inserted at to maintain the current order of the set.\n     *\n     * This function assumes that the element to search for does not already exist\n     * in the sorted set.\n     *\n     * @param {Object} elem The elem to find the position for in the set\n     * @returns {Number}\n     * @memberOf SortedSet\n     */\n    lunr.SortedSet.prototype.locationFor = function (elem) {\n        let start = 0;\n        let end = this.elements.length;\n        let sectionLength = end - start;\n        let pivot = start + Math.floor(sectionLength / 2);\n        let pivotElem = this.elements[pivot];\n\n        while (sectionLength > 1) {\n            if (pivotElem < elem) start = pivot;\n            if (pivotElem > elem) end = pivot;\n\n            sectionLength = end - start;\n            pivot = start + Math.floor(sectionLength / 2);\n            pivotElem = this.elements[pivot];\n        }\n\n        if (pivotElem > elem) return pivot;\n        if (pivotElem < elem) return pivot + 1;\n    };\n\n    /**\n     * Creates a new lunr.SortedSet that contains the elements in the intersection\n     * of this set and the passed set.\n     *\n     * @param {lunr.SortedSet} otherSet The set to intersect with this set.\n     * @returns {lunr.SortedSet}\n     * @memberOf SortedSet\n     */\n    lunr.SortedSet.prototype.intersect = function (otherSet) {\n        const intersectSet = new lunr.SortedSet();\n        let i = 0;\n        let j = 0;\n        const a_len = this.length;\n        const b_len = otherSet.length;\n        const a = this.elements;\n        const b = otherSet.elements;\n\n        while (true) {\n            if (i > a_len - 1 || j > b_len - 1) break;\n\n            if (a[i] === b[j]) {\n                intersectSet.add(a[i]);\n                i++, j++;\n                continue;\n            }\n\n            if (a[i] < b[j]) {\n                i++;\n                continue;\n            }\n\n            if (a[i] > b[j]) {\n                j++;\n                continue;\n            }\n        }\n\n        return intersectSet;\n    };\n\n    /**\n     * Makes a copy of this set\n     *\n     * @returns {lunr.SortedSet}\n     * @memberOf SortedSet\n     */\n    lunr.SortedSet.prototype.clone = function () {\n        const clone = new lunr.SortedSet();\n\n        clone.elements = this.toArray();\n        clone.length = clone.elements.length;\n\n        return clone;\n    };\n\n    /**\n     * Creates a new lunr.SortedSet that contains the elements in the union\n     * of this set and the passed set.\n     *\n     * @param {lunr.SortedSet} otherSet The set to union with this set.\n     * @returns {lunr.SortedSet}\n     * @memberOf SortedSet\n     */\n    lunr.SortedSet.prototype.union = function (otherSet) {\n        let longSet, shortSet, unionSet;\n\n        if (this.length >= otherSet.length) {\n            (longSet = this), (shortSet = otherSet);\n        } else {\n            (longSet = otherSet), (shortSet = this);\n        }\n\n        unionSet = longSet.clone();\n\n        for (\n            let i = 0, shortSetElements = shortSet.toArray();\n            i < shortSetElements.length;\n            i++\n        ) {\n            unionSet.add(shortSetElements[i]);\n        }\n\n        return unionSet;\n    };\n\n    /**\n     * Returns a representation of the sorted set ready for serialisation.\n     *\n     * @returns {Array}\n     * @memberOf SortedSet\n     */\n    lunr.SortedSet.prototype.toJSON = function () {\n        return this.toArray();\n    };\n    /**\n     * export the module via AMD, CommonJS or as a browser global\n     * Export code from https://github.com/umdjs/umd/blob/master/returnExports.js\n     */\n    (function (root, factory) {\n        if (typeof define === 'function' && define.amd) {\n            // AMD. Register as an anonymous module.\n            define(factory);\n        } else if (typeof exports === 'object') {\n            /**\n             * Node. Does not work with strict CommonJS, but\n             * only CommonJS-like enviroments that support module.exports,\n             * like Node.\n             */\n            module.exports = factory();\n        } else {\n            // Browser globals (root is window)\n            root.elasticlunr = factory();\n        }\n    })(this, function () {\n        /**\n         * Just return a value to define the module export.\n         * This example returns an object, but the module\n         * can return a function as the exported value.\n         */\n        return elasticlunr;\n    });\n})();\n\n// - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - //\n// End of elasticlunr code (http://elasticlunr.com/elasticlunr.js) //\n// - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - //\n\nwindow.onload = function () {\n    if (!document.body.contains(document.getElementById('searchModal'))) {\n        return;\n    }\n\n    const lang = document.documentElement.lang;\n    const searchInput = document.getElementById('searchInput');\n    const searchModal = document.getElementById('searchModal');\n    const searchButton = document.getElementById('search-button');\n    const clearSearchButton = document.getElementById('clear-search');\n    const resultsContainer = document.getElementById('results-container');\n    const results = document.getElementById('results');\n    // Get all spans holding the translated strings, even if they are only used on one language.\n    const zeroResultsSpan = document.getElementById('zero_results');\n    const oneResultsSpan = document.getElementById('one_results');\n    const twoResultsSpan = document.getElementById('two_results');\n    const fewResultsSpan = document.getElementById('few_results');\n    const manyResultsSpan = document.getElementById('many_results');\n\n    // Static mapping of keys to spans.\n    const resultSpans = {\n        zero_results: zeroResultsSpan,\n        one_results: oneResultsSpan,\n        two_results: twoResultsSpan,\n        few_results: fewResultsSpan,\n        many_results: manyResultsSpan,\n    };\n\n    // Replace $SHORTCUT in search icon title with actual OS-specific shortcut.\n    function getShortcut() {\n        const userAgent = window.navigator.userAgent.toLowerCase();\n        if (userAgent.includes('mac')) {\n            return 'Cmd + K';\n        } else {\n            return 'Ctrl + K';\n        }\n    }\n\n    function setAttributes(element, attributeNames) {\n        const shortcut = getShortcut();\n        attributeNames.forEach((attributeName) => {\n            let attributeValue = element.getAttribute(attributeName);\n            if (attributeValue) {\n                attributeValue = attributeValue.replace('$SHORTCUT', shortcut);\n                element.setAttribute(attributeName, attributeValue);\n            }\n        });\n    }\n    setAttributes(searchButton, ['title', 'aria-label']);\n\n    // Make search button keyboard accessible.\n    searchButton.addEventListener('keydown', function (event) {\n        if (event.key === 'Enter' || event.key === ' ') {\n            searchButton.click();\n        }\n    });\n\n    let lastFocusedElement;\n    function openSearchModal() {\n        lastFocusedElement = document.activeElement;\n        loadSearchIndex();\n        searchModal.style.display = 'block';\n        searchInput.focus();\n    }\n\n    function closeModal() {\n        searchModal.style.display = 'none';\n        clearSearch();\n        if (lastFocusedElement && document.body.contains(lastFocusedElement)) {\n            lastFocusedElement.focus();\n        }\n    }\n\n    function toggleModalVisibility() {\n        const isModalOpen = searchModal.style.display === 'block';\n        if (isModalOpen) {\n            closeModal();\n        } else {\n            openSearchModal();\n        }\n    }\n\n    // Function to remove 'selected' class from all divs except the one passed.\n    function clearSelected(exceptDiv = null) {\n        const divs = results.querySelectorAll('#results > div');\n        divs.forEach((div) => {\n            if (div !== exceptDiv) {\n                div.setAttribute('aria-selected', 'false');\n            }\n        });\n    }\n\n    function updateSelection(div) {\n        if (div.getAttribute('aria-selected') !== 'true') {\n            clearSelected(div);\n            div.setAttribute('aria-selected', 'true');\n        }\n        searchInput.setAttribute('aria-activedescendant', div.id);\n    }\n\n    function clearSearch() {\n        searchInput.value = '';\n        results.innerHTML = '';\n        resultsContainer.style.display = 'none';\n        searchInput.removeAttribute('aria-activedescendant');\n        clearSearchButton.style.display = 'none';\n    }\n\n    // Close modal when clicking/tapping outside.\n    function handleModalInteraction(event) {\n        if (event.target === searchModal) {\n            closeModal();\n        }\n    }\n    searchModal.addEventListener('click', handleModalInteraction);\n\n    // Close modal when pressing escape.\n    document.addEventListener('keydown', function (event) {\n        if (event.key === 'Escape') {\n            closeModal();\n        }\n    });\n\n    clearSearchButton.addEventListener('click', function () {\n        clearSearch();\n        searchInput.focus();\n    });\n    clearSearchButton.addEventListener('keydown', function (event) {\n        if (event.key === 'Enter' || event.key === ' ') {\n            clearSearch();\n            searchInput.focus();\n            event.preventDefault();\n        }\n    });\n\n    // The index loads on mouseover/tap.\n    // Clicking/tapping the search button opens the modal.\n    // Index is preloaded on `touchstart` so it's ready by the time the synthetic\n    // click fires; the modal itself is opened from `click` so we don't open it\n    // mid-tap (which causes the follow-up synthetic click to land on the newly\n    // visible overlay and immediately close it again).\n    searchButton.addEventListener('mouseover', loadSearchIndex);\n    searchButton.addEventListener('touchstart', loadSearchIndex, { passive: true });\n    searchButton.addEventListener('click', openSearchModal);\n\n    let searchIndexPromise = null;\n    function loadSearchIndex() {\n        if (!searchIndexPromise) {\n            // Check if the search index is already loaded in the window object\n            if (window.searchIndex) {\n                // If the index is pre-loaded, use it directly.\n                searchIndexPromise = Promise.resolve(\n                    elasticlunr.Index.load(window.searchIndex)\n                );\n            } else {\n                // If the index is not pre-loaded, fetch it from the JSON file.\n                const language = document.documentElement\n                    .getAttribute('lang')\n                    .substring(0, 2);\n                let basePath = document\n                    .querySelector(\"meta[name='base']\")\n                    .getAttribute('content');\n                if (basePath.endsWith('/')) {\n                    basePath = basePath.slice(0, -1);\n                }\n\n                searchIndexPromise = fetch(\n                    basePath + '/search_index.' + language + '.json'\n                )\n                    .then((response) => response.json())\n                    .then((json) => elasticlunr.Index.load(json));\n            }\n        }\n    }\n\n    function getByteByBinary(binaryCode) {\n        // Binary system, starts with `0b` in ES6\n        // Octal number system, starts with `0` in ES5 and starts with `0o` in ES6\n        // Hexadecimal, starts with `0x` in both ES5 and ES6\n        var byteLengthDatas = [0, 1, 2, 3, 4];\n        var len = byteLengthDatas[Math.ceil(binaryCode.length / 8)];\n        return len;\n    }\n\n    function getByteByHex(hexCode) {\n        return getByteByBinary(parseInt(hexCode, 16).toString(2));\n    }\n\n    function substringByByte(str, maxLength) {\n        let result = '';\n        let flag = false;\n        let len = 0;\n        let length = 0;\n        let length2 = 0;\n        for (let i = 0; i < str.length; i++) {\n            const code = str.codePointAt(i).toString(16);\n            if (code.length > 4) {\n                i++;\n                if (i + 1 < str.length) {\n                    flag = str.codePointAt(i + 1).toString(16) === '200d';\n                }\n            }\n            if (flag) {\n                len += getByteByHex(code);\n                if (i == str.length - 1) {\n                    length += len;\n                    if (length <= maxLength) {\n                        result += str.substr(length2, i - length2 + 1);\n                    } else {\n                        break;\n                    }\n                }\n            } else {\n                if (len != 0) {\n                    length += len;\n                    length += getByteByHex(code);\n                    if (length <= maxLength) {\n                        result += str.substr(length2, i - length2 + 1);\n                        length2 = i + 1;\n                    } else {\n                        break;\n                    }\n                    len = 0;\n                    continue;\n                }\n                length += getByteByHex(code);\n                if (length <= maxLength) {\n                    if (code.length <= 4) {\n                        result += str[i];\n                    } else {\n                        result += str[i - 1] + str[i];\n                    }\n                    length2 = i + 1;\n                } else {\n                    break;\n                }\n            }\n        }\n        return result;\n    }\n\n    function generateSnippet(text, searchTerms) {\n        const BASE_SCORE = 2;\n        const FIRST_WORD_SCORE = 8;\n        const HIGHLIGHT_SCORE = 40;\n        const PRE_MATCH_CONTEXT_WORDS = 4;\n        const SNIPPET_LENGTH = 150;\n        const WINDOW_SIZE = 30;\n\n        const stemmedTerms = searchTerms.map(function (term) {\n            return elasticlunr.stemmer(term.toLowerCase());\n        });\n\n        let totalLength = 0;\n        const tokenScores = [];\n        const sentences = text.toLowerCase().split('. ');\n\n        for (const sentence of sentences) {\n            const words = sentence.split(/[\\s\\n]/);\n            let isFirstWord = true;\n\n            for (const word of words) {\n                if (word.length > 0) {\n                    let score = isFirstWord ? FIRST_WORD_SCORE : BASE_SCORE;\n                    for (const stemmedTerm of stemmedTerms) {\n                        if (elasticlunr.stemmer(word).startsWith(stemmedTerm)) {\n                            score = HIGHLIGHT_SCORE;\n                        }\n                    }\n                    tokenScores.push([word, score, totalLength]);\n                    isFirstWord = false;\n                }\n                totalLength += word.length + 1;\n            }\n            totalLength += 1;\n        }\n\n        if (tokenScores.length === 0) {\n            return text.length > SNIPPET_LENGTH\n                ? text.substring(0, SNIPPET_LENGTH) + '…'\n                : text;\n        }\n\n        const scores = [];\n        let windowScore = 0;\n\n        for (var i = 0; i < Math.min(tokenScores.length, WINDOW_SIZE); i++) {\n            windowScore += tokenScores[i][1];\n        }\n        scores.push(windowScore);\n\n        // Slide the window and update the score.\n        for (var i = 1; i <= tokenScores.length - WINDOW_SIZE; i++) {\n            windowScore -= tokenScores[i - 1][1];\n            windowScore += tokenScores[i + WINDOW_SIZE - 1][1];\n            scores.push(windowScore);\n        }\n\n        let maxScoreIndex = 0;\n        let maxScore = 0;\n        for (var i = scores.length - 1; i >= 0; i--) {\n            if (maxScore < scores[i]) {\n                maxScore = scores[i];\n                maxScoreIndex = i;\n            }\n        }\n\n        const snippet = [];\n        // From my testing, the context is more clear if we start a few words back.\n        let start = adjustStartPos(\n            text,\n            tokenScores[maxScoreIndex][2],\n            PRE_MATCH_CONTEXT_WORDS\n        );\n\n        function adjustStartPos(text, matchStartIndex, numWordsBack) {\n            let spaceCount = 0;\n            let index = matchStartIndex - 1;\n            while (index >= 0 && spaceCount < numWordsBack) {\n                if (text[index] === ' ' && text[index - 1] !== '.') {\n                    spaceCount++;\n                } else if (text[index] === '.' && text[index + 1] === ' ') {\n                    // Stop if the match is at the start of a sentence.\n                    break;\n                }\n                index--;\n            }\n            return spaceCount === numWordsBack ? index + 1 : matchStartIndex;\n        }\n        const re = /^[\\x00-\\xff]+$/; // Regular expression for ASCII check.\n        for (\n            var i = maxScoreIndex;\n            i < maxScoreIndex + WINDOW_SIZE && i < tokenScores.length;\n            i++\n        ) {\n            const wordData = tokenScores[i];\n            if (start < wordData[2]) {\n                snippet.push(text.substring(start, wordData[2]));\n                start = wordData[2];\n            }\n\n            if (wordData[1] === HIGHLIGHT_SCORE) {\n                snippet.push('<b>');\n            }\n            const end = wordData[2] + wordData[0].length;\n            // Handle non-ASCII characters.\n            if (!re.test(wordData[0]) && wordData[0].length >= 12) {\n                const strBefore = text.substring(wordData[2], end);\n                const strAfter = substringByByte(strBefore, 12);\n                snippet.push(strAfter);\n            } else {\n                snippet.push(text.substring(wordData[2], end));\n            }\n\n            if (wordData[1] === HIGHLIGHT_SCORE) {\n                snippet.push('</b>');\n            }\n            start = end;\n        }\n\n        snippet.push('…');\n        const joinedSnippet = snippet.join('');\n        let truncatedSnippet = joinedSnippet;\n        if (joinedSnippet.replace(/<[^>]+>/g, '').length > SNIPPET_LENGTH) {\n            truncatedSnippet = joinedSnippet.substring(0, SNIPPET_LENGTH) + '…';\n        }\n\n        return truncatedSnippet;\n    }\n\n    // Handle input in the search box.\n    searchInput.addEventListener(\n        'input',\n        async function () {\n            const inputValue = this.value;\n            const searchTerm = inputValue.trim();\n            const searchIndex = await searchIndexPromise;\n            results.innerHTML = '';\n\n            // Use the raw input so the \"clear\" button appears even if there's only spaces.\n            clearSearchButton.style.display = inputValue.length > 0 ? 'block' : 'none';\n            resultsContainer.style.display = searchTerm.length > 0 ? 'block' : 'none';\n\n            // Perform the search and store the results.\n            const searchResults = searchIndex.search(searchTerm, {\n                bool: 'OR',\n                fields: {\n                    title: { boost: 3 },\n                    body: { boost: 2 },\n                    description: { boost: 1 },\n                    path: { boost: 1 },\n                },\n            });\n\n            // Update the number of results.\n            updateResultText(searchResults.length);\n\n            // Display the results.\n            let resultIdCounter = 0; // Counter to generate unique IDs.\n            searchResults.forEach(function (result) {\n                if (result.doc.title || result.doc.path || result.doc.id) {\n                    const resultDiv = document.createElement('div');\n                    resultDiv.setAttribute('role', 'option');\n                    resultDiv.id = 'result-' + resultIdCounter++;\n                    resultDiv.innerHTML = '<a href><span></span><span></span></a>';\n                    const linkElement = resultDiv.querySelector('a');\n                    const titleElement = resultDiv.querySelector('span:first-child');\n                    const snippetElement = resultDiv.querySelector('span:nth-child(2)');\n\n                    // Determine the text for the title.\n                    titleElement.textContent =\n                        result.doc.title || result.doc.path || result.doc.id;\n\n                    // Determine if the body or description is available for the snippet.\n                    let snippetText = result.doc.body\n                        ? generateSnippet(result.doc.body, searchTerm.split(/\\s+/))\n                        : result.doc.description\n                        ? result.doc.description\n                        : '';\n                    snippetElement.innerHTML = snippetText;\n\n                    // Create the hyperlink.\n                    let href = result.ref;\n                    if (result.doc.body) {\n                        // Include text fragment if body is available.\n                        const encodedSearchTerm = encodeURIComponent(searchTerm);\n                        href += `#:~:text=${encodedSearchTerm}`;\n                    }\n                    linkElement.href = href;\n\n                    results.appendChild(resultDiv);\n                }\n            });\n\n            searchInput.setAttribute(\n                'aria-expanded',\n                resultIdCounter > 0 ? 'true' : 'false'\n            );\n\n            if (results.firstChild) {\n                updateSelection(results.firstChild);\n            }\n\n            results.addEventListener('mouseover', function (event) {\n                if (event.target.closest('div[role=\"option\"]')) {\n                    updateSelection(event.target.closest('div[role=\"option\"]'));\n                }\n            });\n\n            results.addEventListener('click', function(event) {\n                const clickedElement = event.target.closest('a');\n                if (clickedElement) {\n                    const clickedHref = clickedElement.getAttribute('href');\n                    const currentPageUrl = window.location.href;\n\n                    // Normalise URLs by removing the text fragment and trailing slash.\n                    const normalizeUrl = (url) => url.split('#')[0].replace(/\\/$/, '');\n\n                    // Check if the clicked link matches the current page.\n                    // If using Ctrl+click or Cmd+click, don't close the modal.\n                    if (normalizeUrl(clickedHref) === normalizeUrl(currentPageUrl) &&\n                        !event.ctrlKey && !event.metaKey) {\n                        closeModal();\n                    }\n                }\n            });\n\n            // Add touch events to the results.\n            setupTouchEvents();\n        },\n        true\n    );\n\n    function updateResultText(count) {\n        // Determine the correct pluralization key based on count and language.\n        const pluralizationKey = getPluralizationKey(count, lang);\n\n        // Hide all result text spans.\n        Object.values(resultSpans).forEach((span) => {\n            if (span) span.style.display = 'none';\n        });\n\n        // Show the relevant result text span, replacing $NUMBER with the actual count.\n        const activeSpan = resultSpans[pluralizationKey];\n        if (activeSpan) {\n            activeSpan.style.display = 'inline';\n            activeSpan.textContent = activeSpan.textContent.replace(\n                '$NUMBER',\n                count.toString()\n            );\n        }\n    }\n\n    function getPluralizationKey(count, lang) {\n        let key = '';\n        const slavicLangs = ['uk', 'be', 'bs', 'hr', 'ru', 'sr'];\n\n        // Common cases: zero, one.\n        if (count === 0) {\n            key = 'zero_results';\n        } else if (count === 1) {\n            key = 'one_results';\n        } else {\n            // Arabic.\n            if (lang === 'ar') {\n                let modulo = count % 100;\n                if (count === 2) {\n                    key = 'two_results';\n                } else if (modulo >= 3 && modulo <= 10) {\n                    key = 'few_results';\n                } else {\n                    key = 'many_results';\n                }\n            } else if (slavicLangs.includes(lang)) {\n                // Slavic languages.\n                let modulo10 = count % 10;\n                let modulo100 = count % 100;\n                if (modulo10 === 1 && modulo100 !== 11) {\n                    key = 'one_results';\n                } else if (\n                    modulo10 >= 2 &&\n                    modulo10 <= 4 &&\n                    !(modulo100 >= 12 && modulo100 <= 14)\n                ) {\n                    key = 'few_results';\n                } else {\n                    key = 'many_results';\n                }\n            } else {\n                key = 'many_results'; // Default plural.\n            }\n        }\n\n        return key;\n    }\n\n    function setupTouchEvents() {\n        const resultDivs = document.querySelectorAll('#results > div');\n        resultDivs.forEach((div) => {\n            // Remove existing listener to avoid duplicates.\n            div.removeEventListener('touchstart', handleTouchStart);\n            div.addEventListener('touchstart', handleTouchStart, { passive: true });\n        });\n    }\n\n    function handleTouchStart() {\n        updateSelection(this);\n    }\n\n    // Handle keyboard navigation.\n    document.addEventListener('keydown', function (event) {\n        // Add handling for the modal open/close shortcut.\n        const isMac = navigator.userAgent.toLowerCase().includes('mac');\n        const MODAL_SHORTCUT_KEY = 'k';\n        const modalShortcutModifier = isMac ? event.metaKey : event.ctrlKey;\n\n        if (event.key === MODAL_SHORTCUT_KEY && modalShortcutModifier) {\n            event.preventDefault();\n            toggleModalVisibility();\n            return;\n        }\n\n        const activeElement = document.activeElement;\n        if (\n            event.key === 'Tab' &&\n            (activeElement === searchInput || activeElement === clearSearchButton)\n        ) {\n            event.preventDefault();\n            const nextFocusableElement =\n                activeElement === searchInput ? clearSearchButton : searchInput;\n            nextFocusableElement.focus();\n            return;\n        }\n\n        function updateResultSelection(newIndex, divsArray) {\n            updateSelection(divsArray[newIndex]);\n            divsArray[newIndex].scrollIntoView({ block: 'nearest', inline: 'start' });\n        }\n\n        const resultDivs = results.querySelectorAll('#results > div');\n        if (resultDivs.length === 0) return;\n\n        const divsArray = Array.from(resultDivs);\n        let activeDiv = results.querySelector('[aria-selected=\"true\"]');\n        let activeDivIndex = divsArray.indexOf(activeDiv);\n\n        if (\n            ['ArrowUp', 'ArrowDown', 'Home', 'End', 'PageUp', 'PageDown'].includes(\n                event.key\n            ) &&\n            !event.isComposing\n        ) {\n            event.preventDefault();\n            let newIndex = activeDivIndex;\n\n            switch (event.key) {\n                case 'ArrowUp':\n                    newIndex = Math.max(activeDivIndex - 1, 0);\n                    break;\n                case 'ArrowDown':\n                    newIndex = Math.min(activeDivIndex + 1, divsArray.length - 1);\n                    break;\n                case 'Home':\n                    newIndex = 0;\n                    break;\n                case 'End':\n                    newIndex = divsArray.length - 1;\n                    break;\n                case 'PageUp':\n                    newIndex = Math.max(activeDivIndex - 3, 0);\n                    break;\n                case 'PageDown':\n                    newIndex = Math.min(activeDivIndex + 3, divsArray.length - 1);\n                    break;\n            }\n\n            if (newIndex !== activeDivIndex) {\n                updateResultSelection(newIndex, divsArray);\n            }\n        }\n\n        if (event.key === 'Enter' && activeDiv && !event.isComposing) {\n            event.preventDefault();\n            event.stopImmediatePropagation();\n            const anchorTag = activeDiv.querySelector('a');\n            if (anchorTag) {\n                window.location.href = anchorTag.getAttribute('href');\n            }\n            closeModal(); // Necessary when linking to the current page.\n        }\n    });\n};\n"
  },
  {
    "path": "static/js/sortTable.js",
    "content": "// Select the table and table headers.\nvar table = document.querySelector('#sitemapTable');\nvar headers = Array.from(table.querySelectorAll('th'));\n\n// Create and append the live region for accessibility announcements.\nvar liveRegion = document.createElement('div');\nliveRegion.setAttribute('aria-live', 'polite');\nliveRegion.setAttribute('aria-atomic', 'true');\nliveRegion.classList.add('visually-hidden');\ndocument.body.appendChild(liveRegion);\n\n// Initialise headers with click and keyboard listeners.\ninitializeHeaders();\naddSortText(); // Add text for screen readers for initial sort direction.\nupdateSortIndicators(headers[0], 'asc'); // Set initial sort indicators.\n\nfunction updateSortIndicators(header, direction) {\n    removeSortArrows(header);\n    var arrow = document.createElement('span');\n    arrow.classList.add('sort-arrow');\n    arrow.textContent = direction === 'asc' ? ' ▲' : ' ▼';\n    arrow.setAttribute('aria-hidden', 'true');\n    header.appendChild(arrow);\n}\n\nfunction removeSortArrows(header) {\n    var arrows = header.querySelectorAll('.sort-arrow');\n    arrows.forEach(function (arrow) {\n        arrow.remove();\n    });\n}\n\nfunction initializeHeaders() {\n    headers.forEach(function (header, index) {\n        header.classList.add('sortable');\n        header.setAttribute('tabindex', '0');\n        header.sortDirection = 'asc'; // Default sort direction.\n        var sortAttribute = index === 0 ? 'ascending' : 'none';\n        header.setAttribute('aria-sort', sortAttribute);\n        header.addEventListener('click', function () {\n            sortTable(index);\n        });\n        header.addEventListener('keydown', function (e) {\n            if (e.key === 'Enter' || e.key === ' ') {\n                e.preventDefault();\n                sortTable(index);\n            }\n        });\n    });\n}\n\nfunction announceSort(header, direction) {\n    var columnTitle = header.querySelector('.columntitle').textContent;\n    liveRegion.textContent =\n        'Column ' + columnTitle + ' is now sorted in ' + direction + ' order';\n}\n\nfunction sortTable(index) {\n    var header = headers[index];\n    var direction = header.sortDirection === 'asc' ? 'desc' : 'asc';\n    var tbody = table.querySelector('tbody');\n    var rows = Array.from(tbody.querySelectorAll('tr'));\n    sortRows(rows, index, direction);\n    refreshTableBody(tbody, rows);\n    updateHeaderAttributes(header, direction);\n    announceSort(header, direction === 'asc' ? 'ascending' : 'descending');\n}\n\nfunction sortRows(rows, index, direction) {\n    rows.sort(function (rowA, rowB) {\n        var cellA = rowA.querySelectorAll('td')[index].textContent;\n        var cellB = rowB.querySelectorAll('td')[index].textContent;\n        return direction === 'asc'\n            ? cellA.localeCompare(cellB)\n            : cellB.localeCompare(cellA);\n    });\n}\n\nfunction refreshTableBody(tbody, rows) {\n    tbody.innerHTML = ''; // Clear existing rows.\n    rows.forEach(function (row) {\n        tbody.appendChild(row);\n    });\n}\n\nfunction updateHeaderAttributes(header, direction) {\n    headers.forEach(function (otherHeader) {\n        if (otherHeader !== header) {\n            otherHeader.setAttribute('aria-sort', 'none');\n            removeSortArrows(otherHeader);\n        }\n    });\n    header.setAttribute('aria-sort', direction === 'asc' ? 'ascending' : 'descending');\n    header.sortDirection = direction;\n    updateSortIndicators(header, direction);\n    updateAnnounceText(header);\n}\n\n// Update screen reader text for sorting.\nfunction updateAnnounceText(header) {\n    var span = header.querySelector('.visually-hidden');\n    span.textContent =\n        'Click to sort in ' +\n        (header.sortDirection === 'asc' ? 'descending' : 'ascending') +\n        ' order';\n}\n\n// Add text for screen readers regarding sort order.\nfunction addSortText() {\n    headers.forEach(function (header) {\n        var span = document.createElement('span');\n        span.classList.add('visually-hidden');\n        span.textContent = 'Click to sort in descending order';\n        header.appendChild(span);\n    });\n}\n\nheaders[0].sortDirection = 'asc';\nheaders[0].setAttribute('aria-sort', 'ascending');\n"
  },
  {
    "path": "static/js/themeSwitcher.js",
    "content": "// Get the theme switcher button elements.\nconst themeSwitcher = document.querySelector('.theme-switcher');\nconst themeResetter = document.querySelector('.theme-resetter');\nconst defaultTheme = document.documentElement.getAttribute('data-default-theme');\n\nfunction getSystemThemePreference() {\n    return window.matchMedia('(prefers-color-scheme: dark)').matches ? 'dark' : 'light';\n}\n\n// Determine the initial theme.\nlet currentTheme =\n    localStorage.getItem('theme') ||\n    document.documentElement.getAttribute('data-theme') ||\n    getSystemThemePreference();\n\nfunction setTheme(theme, saveToLocalStorage = false) {\n    document.documentElement.setAttribute('data-theme', theme);\n    currentTheme = theme;\n    themeSwitcher.setAttribute('aria-pressed', theme === 'dark');\n\n    if (saveToLocalStorage) {\n        localStorage.setItem('theme', theme);\n        themeResetter.classList.add('has-custom-theme');\n    } else {\n        localStorage.removeItem('theme');\n        themeResetter.classList.remove('has-custom-theme');\n    }\n\n    // Dispatch a custom event for comment systems.\n    window.dispatchEvent(new CustomEvent('themeChanged', { detail: { theme } }));\n}\n\nfunction resetTheme() {\n    setTheme(defaultTheme || getSystemThemePreference());\n}\n\n// Function to switch between dark and light themes.\nfunction switchTheme() {\n    setTheme(currentTheme === 'dark' ? 'light' : 'dark', true);\n}\n\n// Initialize the theme switcher button.\nthemeSwitcher.addEventListener('click', switchTheme);\nthemeResetter.addEventListener('click', resetTheme);\n\n// Update the theme based on system preference if necessary.\nif (!defaultTheme) {\n    window\n        .matchMedia('(prefers-color-scheme: dark)')\n        .addEventListener('change', (e) => {\n            setTheme(e.matches ? 'dark' : 'light');\n        });\n}\n\n// Set initial ARIA attribute and custom theme class.\nthemeSwitcher.setAttribute('aria-pressed', currentTheme === 'dark');\nif (localStorage.getItem('theme')) {\n    themeResetter.classList.add('has-custom-theme');\n}\n\n// Function to handle keydown event on theme toggler buttons.\nfunction handleThemeTogglerKeydown(event) {\n    if (event.key === 'Enter' || event.key === ' ') {\n        event.preventDefault();\n        if (event.target === themeSwitcher) {\n            switchTheme();\n        } else if (event.target === themeResetter) {\n            resetTheme();\n        }\n    }\n}\n\nthemeSwitcher.addEventListener('keydown', handleThemeTogglerKeydown);\nthemeResetter.addEventListener('keydown', handleThemeTogglerKeydown);\n"
  },
  {
    "path": "static/js/utterances.js",
    "content": "function setUtterancesTheme(newTheme) {\n    // Get the frame with class \"utterances-frame\".\n    const frame = document.querySelector('.utterances-frame');\n\n    if (frame) {\n        // If the iframe exists, send a message to set the theme.\n        frame.contentWindow.postMessage(\n            { type: 'set-theme', theme: newTheme },\n            'https://utteranc.es'\n        );\n    }\n}\n\nfunction initUtterances() {\n    // Get the comments div.\n    const commentsDiv = document.querySelector('.comments');\n\n    // Check if the comments div exists.\n    if (commentsDiv) {\n        // Get all necessary attributes for initializing Utterances.\n        const repo = commentsDiv.getAttribute('data-repo');\n        const issueTerm = commentsDiv.getAttribute('data-issue-term');\n        const label = commentsDiv.getAttribute('data-label');\n        const lightTheme = commentsDiv.getAttribute('data-light-theme');\n        const darkTheme = commentsDiv.getAttribute('data-dark-theme');\n        const lazyLoading = commentsDiv.getAttribute('data-lazy-loading');\n\n        // Create a new script element.\n        const script = document.createElement('script');\n        script.src = 'https://utteranc.es/client.js';\n        script.async = true;\n        script.setAttribute('repo', repo);\n        script.setAttribute('issue-term', issueTerm);\n        script.setAttribute('label', label);\n\n        // Set the initial theme.\n        const currentTheme =\n            document.documentElement.getAttribute('data-theme') || 'light';\n        const selectedTheme = currentTheme === 'dark' ? darkTheme : lightTheme;\n        script.setAttribute('theme', selectedTheme);\n\n        script.setAttribute('crossorigin', 'anonymous');\n\n        // Enable lazy loading if specified.\n        if (lazyLoading === 'true') {\n            script.setAttribute('data-loading', 'lazy');\n        }\n\n        // Append the script to the comments div.\n        commentsDiv.appendChild(script);\n\n        // Listen for themeChanged event to update the theme.\n        window.addEventListener('themeChanged', (event) => {\n            // Determine the new theme based on the event detail.\n            const selectedTheme =\n                event.detail.theme === 'dark' ? darkTheme : lightTheme;\n            // Set the new theme.\n            setUtterancesTheme(selectedTheme);\n        });\n    }\n}\n\n// Initialize Utterances.\ninitUtterances();\n"
  },
  {
    "path": "static/js/webmention.js",
    "content": "/* webmention.js\n\nSimple thing for embedding webmentions from webmention.io into a page, client-side.\n\n(c)2018-2022 fluffy (http://beesbuzz.biz)\n2025 mmai (https://misc.rhumbs.fr)\n\nPermission is hereby granted, free of charge, to any person obtaining a copy\nof this software and associated documentation files (the \"Software\"), to deal\nin the Software without restriction, including without limitation the rights\nto use, copy, modify, merge, publish, distribute, sublicense, and/or sell\ncopies of the Software, and to permit persons to whom the Software is\nfurnished to do so, subject to the following conditions:\n\nThe above copyright notice and this permission notice shall be included in all\ncopies or substantial portions of the Software.\n\nTHE SOFTWARE IS PROVIDED \"AS IS\", WITHOUT WARRANTY OF ANY KIND, EXPRESS OR\nIMPLIED, INCLUDING BUT NOT LIMITED TO THE WARRANTIES OF MERCHANTABILITY,\nFITNESS FOR A PARTICULAR PURPOSE AND NONINFRINGEMENT. IN NO EVENT SHALL THE\nAUTHORS OR COPYRIGHT HOLDERS BE LIABLE FOR ANY CLAIM, DAMAGES OR OTHER\nLIABILITY, WHETHER IN AN ACTION OF CONTRACT, TORT OR OTHERWISE, ARISING FROM,\nOUT OF OR IN CONNECTION WITH THE SOFTWARE OR THE USE OR OTHER DEALINGS IN THE\nSOFTWARE.\n\nBasic usage:\n\n<script src=\"/path/to/webmention.js\" data-param=\"val\" ... async />\n<div id=\"webmentions\"></div>\n\nAllowed parameters:\n\n    page-url:\n\n        The base URL to use for this page. Defaults to window.location\n\n    add-urls:\n\n        Additional URLs to check, separated by |s\n\n    id:\n\n        The HTML ID for the object to fill in with the webmention data.\n        Defaults to \"webmentions\"\n\n    wordcount:\n\n        The maximum number of words to render in reply mentions.\n\n    max-webmentions:\n\n        The maximum number of mentions to retrieve. Defaults to 30.\n\n    prevent-spoofing:\n\n        By default, Webmentions render using the mf2 'url' element, which plays\n        nicely with webmention bridges (such as brid.gy and telegraph)\n        but allows certain spoofing attacks. If you would like to prevent\n        spoofing, set this to a non-empty string (e.g. \"true\").\n\n    sort-by:\n\n        What to order the responses by; defaults to 'published'. See\n        https://github.com/aaronpk/webmention.io#api\n\n    sort-dir:\n\n        The order to sort the responses by; defaults to 'up' (i.e. oldest\n        first). See https://github.com/aaronpk/webmention.io#api\n\n    comments-are-reactions:\n\n        If set to a non-empty string (e.g. \"true\"), will display comment-type responses\n        (replies/mentions/etc.) as being part of the reactions\n        (favorites/bookmarks/etc.) instead of in a separate comment list.\n\nA more detailed example:\n\n<!-- If you want to translate the UI -->\n<script src=\"/path/to/umd/i18next.js\"></script>\n<script>\n    // Setup i18next as described in https://www.i18next.com/overview/getting-started#basic-sample\n</script>\n<!-- Otherwise, only using the following is fine -->\n<script src=\"/path/to/webmention.min.js\"\n    data-id=\"webmentionContainer\"\n    data-wordcount=\"30\"\n    data-prevent-spoofing=\"true\"\n    data-comments-are-reactions=\"true\"\n    />\n\n*/\n\n// Begin LibreJS code licensing\n// @license magnet:?xt=urn:btih:d3d9a9a6595521f9666a5e94cc830dab83b65699&dn=expat.txt\n\n(function () {\n  \"use strict\";\n\n  // Shim i18next\n  window.i18next = window.i18next || {\n    t: function t(/** @type {string} */key) { return key; }\n  }\n  const t = window.i18next.t.bind(window.i18next);\n\n  /**\n   * Read the configuration value.\n   *\n   * @param {string} key The configuration key.\n   * @param {string} dfl The default value.\n   * @returns {string}\n   */\n  function getCfg(key, dfl) {\n    return document.currentScript.getAttribute(\"data-\" + key) || dfl;\n  }\n\n  const refurl = getCfg(\"page-url\", window.location.href.replace(/#.*$/, \"\"));\n  const addurls = getCfg(\"add-urls\", undefined);\n  const containerID = getCfg(\"id\", \"webmentions\");\n  /** @type {Number} */\n  const textMaxWords = getCfg(\"wordcount\");\n  const maxWebmentions = getCfg(\"max-webmentions\", 30);\n  const mentionSource = getCfg(\"prevent-spoofing\") ? \"wm-source\" : \"url\";\n  const sortBy = getCfg(\"sort-by\", \"published\");\n  const sortDir = getCfg(\"sort-dir\", \"up\");\n\n  /**\n   * Strip the protocol off a URL.\n   *\n   * @param {string} url The URL to strip protocol off.\n   * @returns {string}\n   */\n  function stripurl(url) {\n    return url.substr(url.indexOf('//'));\n  }\n\n  /**\n   * Deduplicate multiple mentions from the same source URL.\n   *\n   * @param {Array<Reaction>} mentions Mentions of the source URL.\n   * @return {Array<Reaction>}\n   */\n  function dedupe(mentions) {\n    /** @type {Array<Reaction>} */\n    const filtered = [];\n    /** @type {Record<string, boolean>} */\n    const seen = {};\n\n    mentions.forEach(function (r) {\n      // Strip off the protocol (i.e. treat http and https the same)\n      const source = stripurl(r.url);\n      if (!seen[source]) {\n        filtered.push(r);\n        seen[source] = true;\n      }\n    });\n\n    return filtered;\n  }\n\n  /**\n   * Format comments as HTML.\n   *\n   * @param {Array<Reaction>} comments The comments to format.\n   * @returns string\n   */\n  function formatComments(type, comments) {\n\n    let html = `\n  <div class=\"webcomments\">\n    <h3 id=\"replies-header\">` + getIcon('comment') + `&nbsp;<span>` + comments.length + `</span> ` + type + `s </h3>\n    <ol aria-labelledby=\"replies-header\" role=\"list\">`;\n    comments.forEach(function (comment) {\n      let content = '';\n      if (comment.hasOwnProperty('content')) {\n        if (comment.content.hasOwnProperty('html')) {\n          content = comment.content.html;\n        } else if (comment.content.hasOwnProperty('text')) {\n          content = comment.content.text;\n        }\n      }\n\n      html += `\n  <li class=\"comment h-entry\">\n    <div>\n      <a class=\"comment_author u-author\"\n         href=\"`+ comment.author.url + `\"\n         target=\"_blank\"\n         title=\"`+ comment.author.name + `\"\n         rel=\"noreferrer\">\n        <img\n          src=\"`+ comment.author.photo + `\"\n          alt=\"\"\n          class=\"u-photo\"\n          loading=\"lazy\"\n          decoding=\"async\"\n          width=\"48\"\n          height=\"48\"\n        >\n        <span class=\"p-author\">`+ comment.author.name + `</span>\n      </a>`;\n      if (comment.published) {\n        const published = new Date(comment.published);\n        html += `\n      <time class=\"dt-published\" datetime=\"`+ comment.published + `\">` + published.toLocaleString(undefined, { dateStyle: \"medium\", timeStyle: \"short\" }) + `</time>`;\n      }\n      html += `\n    </div>\n\n    <p class=\"e-entry\">`+ content + `\n      <a class=\"u-url\"\n         href=\"`+ comment.url + `\"\n         target=\"_blank\"\n         rel=\"noreferrer\">\n        source\n      </a>\n    </p>\n  </li>\n`;\n    });\n    html += `\n    </ol >\n  </div >\n      `;\n\n    return html;\n  }\n\n  /**\n   * @typedef {Object} Reaction\n   * @property {string}      url\n   * @property {Object?}     author\n   * @property {string?}     author.name\n   * @property {string?}     author.photo\n   * @property {Object?}     content\n   * @property {string?}     content.text\n   * @property {RSVPEmoji?}  rsvp\n   * @property {MentionType?} wm-property\n   * @property {string?}     wm-source\n   */\n\n  function getIcon(name) {\n\n    if (name == 'like') {\n      return `<svg focusable = \"false\" width = \"24\" height = \"24\" viewBox = \"0 0 192 192\" xmlns = \"http://www.w3.org/2000/svg\" > <path d=\"M95.997 41.986l-.026-.035C85.746 28.36 68.428 21.423 51.165 24.881 30.138 29.094 15.004 47.558 15 69.003c0 24.413 14.906 47.964 39.486 70.086 8.43 7.586 17.437 14.468 26.444 20.533.728.49 1.444.967 2.148 1.43l1.39.909 1.355.872 1.317.835.645.403 1.259.78 1.194.726 1.032.619 1.38.807.418.236a6 6 0 005.864 0l1.138-.654 1.154-.684 1.118-.675.614-.376 1.26-.779a212 212 0 00.644-.403l1.317-.835 1.355-.872 1.39-.909c.704-.463 1.42-.94 2.148-1.43 9.007-6.065 18.015-12.947 26.444-20.533C162.094 116.967 177 93.416 177 69.004c-.004-21.446-15.138-39.91-36.165-44.123-17.07-3.42-34.174 3.323-44.43 16.568l-.408.537zm42.48-5.338c15.421 3.09 26.52 16.63 26.523 32.357 0 19.607-12.438 39.847-33.532 59.357l-1.316 1.205c-.22.201-.443.402-.666.603-7.977 7.18-16.548 13.727-25.118 19.498l-.745.5c-.74.494-1.466.973-2.177 1.437l-1.402.906-1.359.864-.662.416-1.292.8-.732.446-.73-.446-1.292-.8-.662-.416-1.36-.864-1.4-.906a235.406 235.406 0 01-2.923-1.937c-8.57-5.77-17.14-12.319-25.118-19.498l-.666-.603-1.316-1.205C39.438 108.852 27 88.612 27 69.004c.003-15.726 11.102-29.267 26.523-32.356 15.253-3.056 30.565 4.954 36.756 19.208l.204.478c2.084 4.878 9.009 4.85 11.053-.045 6.062-14.511 21.52-22.73 36.941-19.641z\" fill=\"currentColor\" /></svg> `;\n    } else if (name == 'repost') {\n      return `<svg focusable = \"false\" width = \"24\" height = \"24\" viewBox = \"0 0 192 192\" xmlns = \"http://www.w3.org/2000/svg\" > <path d=\"M18.472 146.335l-.075-.184a5.968 5.968 0 01-.216-.684l-.014-.056a5.643 5.643 0 01-.082-.397l-.013-.083a5.886 5.886 0 01-.072-.96V144c0-.157.006-.313.018-.467l.006-.075c.012-.132.028-.261.048-.39l.016-.095c.008-.05.017-.1.027-.149.005-.019.008-.038.012-.058.028-.133.06-.264.096-.393l.026-.088a5.86 5.86 0 01.482-1.159l.043-.077a5.642 5.642 0 01.31-.49l.015-.022.076-.104.044-.059a3.856 3.856 0 01.165-.208l.052-.061c.102-.12.21-.236.321-.348l18-18a6 6 0 018.661 8.303l-.175.183L38.484 138H120c23.196 0 42-18.804 42-42a6 6 0 0112 0c0 29.525-23.696 53.516-53.107 53.993L120 150H38.486l7.757 7.757a6 6 0 01.175 8.303l-.175.183a6 6 0 01-8.303.175l-.183-.175-18-18-.145-.151a6.036 6.036 0 01-.829-1.125l-.058-.105a4.08 4.08 0 01-.06-.114l-.04-.077a4.409 4.409 0 01-.139-.3l-.014-.036zM154.06 25.582l.183.175 18 18a6.036 6.036 0 01.974 1.276l.058.105c.02.035.038.07.056.105l.043.086a4.411 4.411 0 01.14.3l.014.036a5.965 5.965 0 01.291.868l.014.056c.032.13.059.263.082.397l.013.083a5.886 5.886 0 01.067.692v.014a6.11 6.11 0 01-.013.692l-.006.075a5.856 5.856 0 01-.048.39l-.016.095c-.008.05-.017.1-.027.149-.005.019-.008.038-.012.058-.028.133-.06.264-.096.393l-.026.088a5.86 5.86 0 01-.482 1.159l-.043.077-.052.09-.029.048a6.006 6.006 0 01-.32.478l-.044.059a3.857 3.857 0 01-.165.208l-.052.061a6.34 6.34 0 01-.176.197l-.145.15-18 18a6 6 0 01-8.661-8.302l.175-.183L153.514 54H72c-23.196 0-42 18.804-42 42a6 6 0 11-12 0c0-29.525 23.696-53.516 53.107-53.993L72 42h81.516l-7.759-7.757a6 6 0 01-.175-8.303l.175-.183a6 6 0 018.303-.175z\" fill=\"currentColor\" /></svg> `;\n    } else if (name == 'comment') {\n      return `<svg width = \"24\" height = \"24\" viewBox = \"0 0 150 150\" xmlns = \"http://www.w3.org/2000/svg\" > <path d=\"M75-.006a75 75 0 0174.997 74.31l.003.69c0 41.422-33.579 75-75 75H11.75c-6.49 0-11.75-5.26-11.75-11.75v-63.25a75 75 0 0175-75zm0 12a63 63 0 00-63 63v63h63c34.446 0 62.435-27.645 62.992-61.93l.008-1.041-.003-.633A63 63 0 0075 11.994zm21 72a6 6 0 01.225 11.996l-.225.004H51a6 6 0 01-.225-11.996l.225-.004h45zm0-24a6 6 0 01.225 11.996l-.225.004H51a6 6 0 01-.225-11.996l.225-.004h45z\" fill=\"currentColor\" /></svg> `;\n    } else if (name == 'bookmark') {\n      return `<svg xmlns=\"http://www.w3.org/2000/svg\" x=\"0px\" y=\"0px\" width=\"24\" height=\"24\" viewBox=\"0 0 24 24\">\n<path d=\"M 6.0097656 2 C 4.9143111 2 4.0097656 2.9025988 4.0097656 3.9980469 L 4 22 L 12 19 L 20 22 L 20 20.556641 L 20 4 C 20 2.9069372 19.093063 2 18 2 L 6.0097656 2 z M 6.0097656 4 L 18 4 L 18 19.113281 L 12 16.863281 L 6.0019531 19.113281 L 6.0097656 4 z\" fill=\"currentColor\"></path>\n</svg>`\n    }\n  }\n\n  /**\n   * Formats a list of reactions as HTML.\n   *\n   * @param {Array<Reaction>} reacts List of reactions to format\n   * @returns string\n   */\n  function formatReactions(type, reacts) {\n    let html = `\n      <div class=\"color--primary\" >\n        <h3 id=`+ type + ` - header\"> ` + getIcon(type) + `&nbsp; <span>` + reacts.length + `</span> ` + type + `s </h3>\n\n          <ol class=\"likes\" role = \"list\" aria - labelledby=\"`+ type + `-header\"> `;\n\n    reacts.forEach(function (react) {\n      html += `\n            <li class=\"h-card\">\n              <a class=\"u-url\"\n                href=\"`+ react.author.url + `\n       target=\"_blank\"\n    rel = \"noreferrer\"\n    title = \"`+ react.author.name + `\" >\n      <img\n        alt=\"\"\n        class=\"lazy mentions__image u-photo\"\n        src=\"`+ react.author.photo + `\"\n        loading=\"lazy\"\n        decoding=\"async\"\n        width=\"48\"\n        height=\"48\"\n      >\n        <span class=\"p-author visually-hidden\" aria-hidden=\"true\">{{ author }}</span>\n      </a>\n  </li>\n      `;\n    });\n\n    html += `\n    </ol >\n  </div >\n      `;\n    return html;\n  }\n\n  /**\n   * @typedef WebmentionResponse\n   * @type {Object}\n   * @property {Array<Reaction>} children\n   */\n\n  /**\n   * Register event listener.\n   */\n  window.addEventListener(\"load\", async function () {\n    const container = document.getElementById(containerID);\n    if (!container) {\n      // no container, so do nothing\n      return;\n    }\n\n    const pages = [stripurl(refurl)];\n    if (!!addurls) {\n      addurls.split('|').forEach(function (url) {\n        pages.push(stripurl(url));\n      });\n    }\n\n    let apiURL = `https://webmention.io/api/mentions.jf2?per-page=${maxWebmentions}&sort-by=${sortBy}&sort-dir=${sortDir}`;\n\n    pages.forEach(function (path) {\n      apiURL += `&target[]=${encodeURIComponent('http:' + path)}&target[]=${encodeURIComponent('https:' + path)}`;\n    });\n\n    // apiURL = 'http://127.0.0.1:1111/test_webmentions.jf2';\n    /** @type {WebmentionResponse} */\n    let json = {};\n    try {\n      // const response = await window.fetch(apiURL);\n      const response = await window.fetch(apiURL);\n      if (response.status >= 200 && response.status < 300) {\n        json = await response.json();\n      } else {\n        console.error(\"Could not parse response\");\n        new Error(response.statusText);\n      }\n    } catch (error) {\n      // Purposefully not escalate further, i.e. no UI update\n      console.error(\"Request failed\", error);\n    }\n\n    /** @type {Array<Reaction>} */\n    let comments = [];\n    /** @type {Array<Reaction>} */\n    let mentions = [];\n    /** @type {Array<Reaction>} */\n    const bookmarks = [];\n    /** @type {Array<Reaction>} */\n    const likes = [];\n    /** @type {Array<Reaction>} */\n    const reposts = [];\n    /** @type {Array<Reaction>} */\n    const follows = [];\n\n    /** @type {Record<MentionType, Array<Reaction>>} */\n    const mapping = {\n      \"in-reply-to\": comments,\n      \"like-of\": likes,\n      \"repost-of\": reposts,\n      \"bookmark-of\": bookmarks,\n      \"follow-of\": follows,\n      \"mention-of\": mentions,\n      \"rsvp\": comments\n    };\n\n    json.children.forEach(function (child) {\n      // Map each mention into its respective container\n      const store = mapping[child['wm-property']];\n      if (store) {\n        store.push(child);\n      }\n    });\n\n    // format the comment-type things\n    let formattedMentions = '';\n    if (mentions.length > 0) {\n      formattedMentions = formatComments('mention', dedupe(mentions));\n    }\n\n    let formattedComments = '';\n    if (comments.length > 0) {\n      formattedComments = formatComments('comment', dedupe(comments));\n    }\n\n    // format likes\n    let likesStr = '';\n    if (likes.length > 0) {\n      likesStr = formatReactions('like', dedupe(likes));\n    }\n\n    // format reposts\n    let repostsStr = '';\n    if (reposts.length > 0) {\n      repostsStr = formatReactions('repost', dedupe(reposts));\n    }\n\n    // format bookmarks\n    let bookmarksStr = '';\n    if (bookmarks.length > 0) {\n      bookmarksStr = formatReactions('bookmark', dedupe(bookmarks));\n    }\n\n    container.innerHTML = `<div id=\"webmentions\">${repostsStr}${likesStr}${bookmarksStr}${formattedComments}${formattedMentions}</div>`;\n  });\n}());\n\n// End-of-file marker for LibreJS\n// @license-end\n"
  },
  {
    "path": "static/no_js.css",
    "content": ".js{display:none}\n"
  },
  {
    "path": "static/sitemap_style.xsl",
    "content": "<?xml version=\"1.0\" encoding=\"UTF-8\"?>\n<xsl:stylesheet version=\"2.0\"\n        xmlns:xsl=\"http://www.w3.org/1999/XSL/Transform\"\n        xmlns:sitemap=\"http://www.sitemaps.org/schemas/sitemap/0.9\">\n    <xsl:output method=\"html\" encoding=\"UTF-8\" indent=\"yes\" />\n\n    <!-- The base URL is assumed to be the first URL in the sitemap. -->\n    <xsl:variable name=\"baseUrl\" select=\"(sitemap:urlset/sitemap:url)[1]/sitemap:loc\"/>\n    <!-- Remove http[s]:// -->\n    <xsl:variable name=\"baseUrlWithoutProtocol\" select=\"substring-after($baseUrl, '://')\"/>\n    <!-- Remove trailing slash -->\n    <xsl:variable name=\"clean_base_url\" select=\"substring-before($baseUrlWithoutProtocol, '/')\"/>\n\n    <xsl:template match=\"/sitemap:urlset\">\n        <html>\n            <head>\n                <title>Sitemap • <xsl:value-of select=\"$clean_base_url\"/></title>\n                <link rel=\"stylesheet\" href=\"{$baseUrl}main.css\"/>\n                <script src=\"{$baseUrl}js/sortTable.min.js\" defer=\"defer\"></script>\n            </head>\n            <body>\n                <div class=\"full-width\">\n                    <h1>Sitemap of <xsl:value-of select=\"$clean_base_url\"/></h1>\n                    <p>Number of URLs: <xsl:value-of select=\"count(sitemap:url)\"/></p>\n                    <table id=\"sitemapTable\" class=\"sitemap-table\" aria-label=\"URLs on the site and their last modification dates\">\n                        <thead>\n                            <tr>\n                                <th><span class=\"columntitle\">URL</span></th>\n                                <th><span class=\"columntitle\">Last modification</span></th>\n                            </tr>\n                        </thead>\n                        <tbody>\n                            <xsl:for-each select=\"sitemap:url\">\n                                <tr>\n                                    <td>\n                                        <a href=\"{sitemap:loc}\">\n                                            <xsl:value-of select=\"sitemap:loc\"/>\n                                        </a>\n                                    </td>\n                                    <td>\n                                        <xsl:value-of select=\"sitemap:lastmod\"/>\n                                    </td>\n                                </tr>\n                            </xsl:for-each>\n                        </tbody>\n                    </table>\n                </div>\n            </body>\n        </html>\n    </xsl:template>\n</xsl:stylesheet>\n"
  },
  {
    "path": "static/social_icons/LICENSE",
    "content": "Most icons in this directory are downloaded from [FontAwesome](https://fontawesome.com/). They are part of the [free offer](https://fontawesome.com/license/free) and are licensed under [CC BY 4.0](https://creativecommons.org/licenses/by/4.0/).\n\nExceptions:\n- nostr, by Andrea Nicolini, aka Bembureda, licensed under [CC0](https://creativecommons.org/publicdomain/zero/1.0/). Source: https://github.com/mbarulli/nostr-logo\n- Keybase, licensed under [CC0](https://creativecommons.org/publicdomain/zero/1.0/) by [Simple Icons](https://simpleicons.org/). Source: https://github.com/simple-icons/simple-icons/blob/develop/icons/keybase.svg \n"
  },
  {
    "path": "sumi.toml",
    "content": "# git-sumi ~ configuration file\n# Config: https://sumi.rs/docs/configuration\n# Rules: https://sumi.rs/docs/rules\n\n# Suppresses progress messages.\nquiet = false\n\n# Displays parsed commit message.\ndisplay = true\n\n# Sets display format: cli, json, table, toml.\nformat = \"cli\"\n\n# Processes each non-empty line as an individual commit.\nsplit_lines = false\n\n# Rule: Include one valid Gitmoji.\n# See https://gitmoji.dev/.\ngitmoji = true\n\n# Rule: Description must start with the specified case.\n# Options: 'any', 'lower', 'upper'.\ndescription_case = \"lower\"\n\n# Rule: Use the imperative mood in the description.\n# Example: 'Fix bug' instead of 'Fixed bug'.\nimperative = true\n\n# Rule: Do not end commit header with a period.\nno_period = true\n\n# Rule: Header length limit.\n# A value of 0 disables the rule.\nmax_header_length = 55\n\n# Rule: Body line length limit.\n# A value of 0 disables the rule.\nmax_body_length = 80\n\n# Rule: No leading, trailing, or consecutive spaces.\nwhitespace = true\n\n# Rule: Follow Conventional Commits format.\n# See https://www.conventionalcommits.org/.\nconventional = true\n\n# Rule: List of allowed commit scopes.\n# An empty list allows all scopes. Example: [\"docs\", \"cli\"].\nscopes_allowed = []\n\n# Rule: List of allowed commit types.\n# An empty list allows all types. Example: [\"feat\", \"fix\", \"docs\"].\ntypes_allowed = [\"feat\", \"fix\", \"docs\", \"refactor\", \"test\", \"chore\", \"misc\", \"style\"]\n\n# Rule: Header must match regex pattern.\n# Example: '^JIRA-\\d+:'.\nheader_pattern = '^([\\p{Emoji_Presentation}\\p{Extended_Pictographic}](?:\\u{FE0F})?\\u{200D}?) \\w' # The first character must be an emoji.\n"
  },
  {
    "path": "templates/404.html",
    "content": "{% extends \"page.html\" %}\n\n{% macro display_404_message(language_name) %}\n    {%- set language_strings = load_data(path=\"i18n/\" ~ language_name ~ '.toml', required=false) -%}\n    {%- if not language_strings -%}\n        {%- set language_strings = load_data(path=\"themes/tabi/i18n/\" ~ language_name ~ '.toml', required=false) -%}\n    {%- endif -%}\n    <p>{{ macros_translate::translate(key=\"page_missing\", default=\"The page you've requested seems to be missing\", force_lang=language_name, language_strings=language_strings) }}\n    {%- if config.languages | length > 0 -%}\n        &nbsp;{{ macros_translate::translate(key=\"translation_missing\", default=\"or hasn't been translated into your language yet\", force_lang=language_name, language_strings=language_strings) }}{{ macros_translate::translate(key=\"full_stop\", default=\".\", force_lang=language_name, language_strings=language_strings) }}\n    {%- else %}.\n    {%- endif %}<br>\n    {{ macros_translate::translate(key=\"check_url\", default=\"Check the URL for errors or\", force_lang=language_name, language_strings=language_strings) }}\n    <a href=\"{{ config.base_url }}{% if language_name != config.default_language %}/{{ language_name }}{% endif %}/\">\n    {{ macros_translate::translate(key=\"go_home\", default=\"go back to the homepage\", force_lang=language_name, language_strings=language_strings) }}</a>{{ macros_translate::translate(key=\"full_stop\", default=\".\", force_lang=language_name, language_strings=language_strings) }}</p>\n{% endmacro %}\n\n{% block main_content %}\n    <main class=\"centered-text\">\n        {{ macros_page_header::page_header(title=\"404\")}}\n        <div class=\"subheader\">{{ macros_translate::translate(key=\"not_found\") }}</div>\n\n        {# 404 message for base language #}\n        {{ self::display_404_message(language_name=config.default_language, is_multilingual=is_multilingual) }}\n\n        {#- Iterate through each extra language, to display the localised 404 message -#}\n        {%- for language_name, language in config.languages -%}\n            {%- if language_name == config.default_language -%}\n                {%- continue -%} {#- We've already displayed the 404 message for the base language -#}\n            {%- endif -%}\n            {{ self::display_404_message(language_name=language_name, is_multilingual=is_multilingual) }}\n        {%- endfor -%}\n    </main>\n{% endblock main_content %}\n"
  },
  {
    "path": "templates/anchor-link.html",
    "content": "<a class=\"header-anchor no-hover-padding\" href=\"#{{ id }}\" aria-label=\"Anchor link for: {{ id }}\"><span class=\"link-icon\" aria-hidden=\"true\"></span></a>\n"
  },
  {
    "path": "templates/archive.html",
    "content": "{% extends \"base.html\" %}\n\n{% block main_content %}\n\n{{ macros_page_header::page_header(title=section.title) }}\n\n{# Set locale for date #}\n{% set date_locale = macros_translate::translate(key=\"date_locale\", default=\"en_GB\", language_strings=language_strings) %}\n\n{#- Check for language-specific date formats -#}\n{%- set language_format = \"\" -%}\n{%- if config.extra.date_formats -%}\n    {%- for format_config in config.extra.date_formats -%}\n        {%- if format_config.lang == lang -%}\n            {%- if format_config.archive -%}\n                {%- set_global language_format = format_config.archive -%}\n            {%- endif -%}\n        {%- endif -%}\n    {%- endfor -%}\n{%- endif -%}\n\n<div class=\"archive\">\n    <ul class=\"list-with-title\">\n        {%- set source_paths = section.extra.section_path | default(value=\"blog/\") -%}\n        {%- if source_paths is iterable -%}\n            {%- set paths = source_paths -%}\n        {%- else -%}\n            {%- set paths = [source_paths] -%}\n        {%- endif %}\n        {%- set all_posts = [] -%}\n        {%- for path in paths -%}\n            {%- if lang == config.default_language %}\n                {%- set section_item = get_section(path=path ~ \"_index.md\") -%}\n            {%- else %}\n                {%- set section_item = get_section(path=path ~ \"_index.\" ~ lang ~ \".md\") -%}\n            {%- endif %}\n            {%- set_global all_posts = all_posts | concat(with=section_item.pages) -%}\n        {%- endfor %}\n\n        {# Sort all posts by date #}\n        {%- set archive_reverse = section.extra.archive_reverse | default(value=false) -%}\n        {%- set all_posts = all_posts | sort(attribute=\"date\") -%}\n        {%- if not archive_reverse -%}\n            {%- set all_posts = all_posts | reverse -%}\n        {%- endif -%}\n\n        {# Group posts by year. #}\n        {% set posts_by_year = all_posts | group_by(attribute=\"year\") %}\n        {% set years = [] %}\n        {% for year, ignored in posts_by_year %}\n            {% set_global years = years | concat(with=[year]) %}\n        {% endfor %}\n\n        {# Iterate over years #}\n        {% set years = years | sort %}\n        {%- if not archive_reverse -%}\n            {%- set years = years | reverse -%}\n        {%- endif -%}\n\n        {% for year in years %}\n            {% set posts = posts_by_year[year] %}\n            {% if posts | length > 0 %}\n            <li>\n                <h2 class=\"listing-title\">{{ year }}</h2>\n                <ul class=\"listing\">\n                    {% for post in posts %}\n                    <li class=\"listing-item\">\n                        <div class=\"post-time\">\n                            <span class=\"date\">\n                                {%- if language_format -%}\n                                    {{ post.date | date(format=language_format, locale=date_locale) }}\n                                {%- elif config.extra.archive_date_format -%}\n                                    {{ post.date | date(format=config.extra.archive_date_format, locale=date_locale) }}\n                                {%- else -%}\n                                    {{ post.date | date(format=\"%d %b\", locale=date_locale) }}\n                                {%- endif -%}\n                            </span>\n                        </div>\n                        <a href=\"{{ post.permalink }}\" title=\"{{ post.title }}\">{{ post.title | markdown(inline=true) | safe }}</a>\n                    </li>\n                    {% endfor %}\n                </ul>\n            </li>\n            {% endif %}\n        {% endfor %}\n    </ul>\n</div>\n\n{% endblock main_content %}\n"
  },
  {
    "path": "templates/atom.xml",
    "content": "{%- import \"macros/translate.html\" as macros_translate -%}\n{%- import \"macros/settings.html\" as macros_settings -%}\n{#- Load the internationalisation data -#}\n{%- set language_strings = load_data(path=\"i18n/\" ~ lang ~ '.toml', required=false) -%}\n{%- if not language_strings -%}\n    {%- set language_strings = load_data(path=\"themes/tabi/i18n/\" ~ lang ~ \".toml\", required=false) -%}\n{%- endif -%}\n<?xml version=\"1.0\" encoding=\"utf-8\"?>\n<?xml-stylesheet href=\"{{ get_url(path='/feed_style.xsl', trailing_slash=false) | safe }}\" type=\"text/xsl\"?>\n<feed xmlns=\"http://www.w3.org/2005/Atom\" xml:lang=\"{{ lang }}\">\n    <tabi:metadata xmlns:tabi=\"https://github.com/welpo/tabi\">\n        <tabi:base_url>{{ config.base_url }}</tabi:base_url>\n        <tabi:separator>\n            {{  config.extra.separator | default(value=\"•\") }}\n        </tabi:separator>\n        <tabi:about_feeds>\n            {{- macros_translate::translate(key=\"about_feeds\", default=\"This is a web feed, also known as an Atom feed. Subscribe by copying the URL from the address bar into your newsreader\", language_strings=language_strings) -}}\n        </tabi:about_feeds>\n        <tabi:visit_the_site>\n            {{- macros_translate::translate(key=\"visit_the_site\", default=\"Visit website\", language_strings=language_strings) -}}\n        </tabi:visit_the_site>\n        <tabi:recent_posts>\n            {{- macros_translate::translate(key=\"recent_posts\", default=\"Recent posts\", language_strings=language_strings) -}}\n        </tabi:recent_posts>\n        <tabi:last_updated_on>\n            {{- macros_translate::translate(key=\"last_updated_on\", default=\"Updated on $DATE\", language_strings=language_strings) -}}\n        </tabi:last_updated_on>\n        <tabi:default_theme>\n            {{- config.extra.default_theme | default(value=\"\") -}}\n        </tabi:default_theme>\n        <tabi:post_listing_date>\n            {{- config.extra.post_listing_date | default(value=\"date\") -}}\n        </tabi:post_listing_date>\n        <tabi:current_section>\n            {%- if term -%}\n                {{ term.name }}\n            {%- elif section.title -%}\n                {{ section.title }}\n            {%- else -%}\n                {{ config.title }}\n            {%- endif -%}\n        </tabi:current_section>\n    </tabi:metadata>\n\n    {#- Load extra CSS (skin) if set in config.toml -#}\n    {%- if config.extra.skin -%}\n        <link rel=\"extra-stylesheet\" href=\"{{ get_url(path='skins/' ~ config.extra.skin ~ '.css', cachebust=true) | safe }}\" />\n    {%- endif -%}\n\n    <title>{{ config.title | striptags | safe }}\n    {%- if term %} - {{ term.name }}\n    {%- elif section.title %} - {{ section.title }}\n    {%- endif -%}\n    </title>\n    {%- if config.description %}\n        <subtitle>{{ config.description }}</subtitle>\n    {%- endif %}\n    <link href=\"{{ feed_url | safe }}\" rel=\"self\" type=\"application/atom+xml\"/>\n    <link href=\"\n        {%- if term -%}\n            {{ term.permalink | escape_xml | safe }}\n        {%- elif section -%}\n            {{ section.permalink | escape_xml | safe }}\n        {%- else -%}\n            {{ get_url(path=\"/\", lang=lang) | escape_xml | safe }}\n        {%- endif -%}\n    \" rel=\"alternate\" type=\"text/html\"/>\n    <generator uri=\"https://www.getzola.org/\">Zola</generator>\n    {%- if last_updated -%}\n    <updated>{{ last_updated | date(format=\"%+\") }}</updated>\n    {%- endif -%}\n    <id>{{ feed_url | safe }}</id>\n    {%- for page in pages %}\n    {%- if macros_settings::evaluate_setting_priority(setting=\"hide_from_feed\", page=page, default_global_value=false) == \"true\" -%}\n        {%- continue -%}\n    {%- endif -%}\n    {#- Skip if hide_from_main_feed is true and this is the main feed -#}\n    {%- if macros_settings::evaluate_setting_priority(setting=\"hide_from_main_feed\", page=page, default_global_value=false) == \"true\"\n         and not section\n         and not term -%}\n        {%- continue -%}\n    {%- endif -%}\n    <entry xml:lang=\"{{ page.lang }}\">\n        <title>{{ page.title }}</title>\n        <published>{{ page.date | date(format=\"%+\") }}</published>\n        <updated>{{ page.updated | default(value=page.date) | date(format=\"%+\") }}</updated>\n        <author>\n            <name>\n            {%- if page.authors -%}\n                {{ page.authors[0] }}\n            {%- elif config.author -%}\n                {{ config.author }}\n            {%- else -%}\n                Unknown\n            {%- endif -%}\n            </name>\n        </author>\n        <link rel=\"alternate\" href=\"{{ page.permalink | safe }}\" type=\"text/html\"/>\n        <id>{{ page.permalink | safe }}</id>\n        {% if config.extra.full_content_in_feed %}\n            <content type=\"html\">{{ page.content }}</content>\n        {% endif -%}\n        {% if page.description -%}\n            <summary type=\"html\">{{ page.description }}</summary>\n        {% elif page.summary -%}\n            <summary type=\"html\">{{ page.summary | striptags | trim_end_matches(pat=\".\") | safe }}…</summary>\n        {% endif -%}\n    </entry>\n    {%- endfor %}\n</feed>\n"
  },
  {
    "path": "templates/base.html",
    "content": "{% import \"macros/feed_utils.html\" as feed_utils %}\n{% import \"macros/format_date.html\" as macros_format_date %}\n{% import \"macros/list_posts.html\" as macros_list_posts %}\n{% import \"macros/page_header.html\" as macros_page_header %}\n{% import \"macros/rel_attributes.html\" as macros_rel_attributes %}\n{% import \"macros/series_page.html\" as macros_series_page %}\n{% import \"macros/settings.html\" as macros_settings %}\n{% import \"macros/table_of_contents.html\" as macros_toc %}\n{% import \"macros/target_attribute.html\" as macros_target_attribute %}\n{% import \"macros/translate.html\" as macros_translate %}\n\n{# Load the internationalisation data for the current language from\nthe .toml files in the user's '/i18n' folder, falling back to the theme's.\nThis variable will hold all the text strings for the language #}\n{%- set language_strings = load_data(path=\"i18n/\" ~ lang ~ '.toml', required=false) -%}\n{%- if not language_strings -%}\n    {%- set language_strings = load_data(path=\"themes/tabi/i18n/\" ~ lang ~ \".toml\", required=false) -%}\n{%- endif -%}\n{% set rtl_languages = [\"ar\", \"arc\", \"az\", \"dv\", \"ff\", \"he\", \"ku\", \"nqo\", \"fa\", \"rhg\", \"syc\", \"ur\"] %}\n\n{#- Necessary for the hierarchy macro -#}\n{%- if page -%}\n    {%- set current_page = page -%}\n{%- else -%}\n    {%- set current_page = \"\"-%}\n{%- endif -%}\n\n<!DOCTYPE html>\n<html lang=\"{{ lang }}\" {% if config.extra.default_theme -%}\n    data-theme=\"{{config.extra.default_theme}}\"\n{%- endif -%}{% if macros_settings::evaluate_setting_priority(setting=\"force_codeblock_ltr\", page=current_page, default_global_value=true) == \"false\" -%}\n    data-code-direction=\"inherit\"{% endif %}>\n\n{% include \"partials/header.html\" %}\n\n<body{% if lang in rtl_languages %} dir=\"rtl\"{% endif %}{% if config.extra.override_serif_with_sans %} class=\"use-sans-serif\"{% endif %}>\n    <a href=\"#main-content\" id=\"skip-link\">{{ macros_translate::translate(key=\"skip_to_content\", default=\"Skip to content\", language_strings=language_strings) }}</a>\n    {% include \"partials/nav.html\" %}\n    <div class=\"content\" id=\"main-content\">\n\n        {# Post page is the default #}\n        {% block main_content %}\n            Nothing here?!\n        {% endblock main_content %}\n    </div>\n    {% include \"partials/footer.html\" %}\n\n    {# Users can optionally provide this template to add content to the body element. #}\n    {% include \"tabi/extend_body.html\" ignore missing %}\n</body>\n\n</html>\n"
  },
  {
    "path": "templates/cards.html",
    "content": "{% extends \"base.html\" %}\n\n{% block main_content %}\n    {% if section.extra.section_path -%}\n        {% set section = get_section(path=section.extra.section_path) %}\n    {% endif -%}\n\n    {{ macros_page_header::page_header(title=section.title) }}\n\n    <main>\n        {% if section.content -%}\n            <div id=\"page-content\">{{ section.content | safe }}</div>\n        {% endif %}\n\n        {%- if paginator %}\n            {%- set show_pages = paginator.pages -%}\n        {% else %}\n            {%- set show_pages = section.pages -%}\n        {% endif -%}\n\n        {%- if macros_settings::evaluate_setting_priority(setting=\"enable_cards_tag_filtering\", page=section, default_global_value=true) == \"true\" -%}\n            {%- include \"partials/filter_card_tags.html\" -%}\n        {%- endif -%}\n\n\n        {%- include \"partials/cards_pages.html\" -%}\n    </main>\n\n    {% if paginator %}\n        {%- include \"partials/paginate.html\" -%}\n    {% endif %}\n{% endblock main_content %}\n"
  },
  {
    "path": "templates/index.html",
    "content": "{% extends \"section.html\" %}\n"
  },
  {
    "path": "templates/info-page.html",
    "content": "{# Template for non-articles (About Me, Privacy…) #}\n\n{% extends \"base.html\" %}\n\n{%- block main_content %}\n\n{%- set page_or_section = page | default(value=section) -%}\n\n{{ macros_page_header::page_header(title=page_or_section.title) }}\n\n<div id=\"page-content\">\n    <main>\n        {# The replace pattern is used to enable arbitrary locations for the Table of Contents #}\n        {# This is Philipp Oppermann's workaround: https://github.com/getzola/zola/issues/584#issuecomment-474329637 #}\n        {{ page_or_section.content | replace(from=\"<!-- toc -->\", to=macros_toc::toc(page=page_or_section, header=false, language_strings=language_strings)) | safe }}\n    </main>\n</div>\n\n{%- include \"partials/extra_features.html\" -%}\n\n{%- endblock main_content %}\n"
  },
  {
    "path": "templates/internal/alias.html",
    "content": "<!doctype html>\n<html lang=\"en\">\n<head>\n    <meta charset=\"utf-8\">\n    <link rel=\"canonical\" href=\"{{ url | safe }}\">\n    <link rel=\"stylesheet\" href=\"{{ get_url(path=\"main.css\", cachebust=true) | safe }}\" />\n    <meta http-equiv=\"refresh\" content=\"0; url={{ url | safe }}\">\n    <title>Redirect</title>\n</head>\n<body class=\"center-content\">\n    <h1><a href=\"{{ url | safe }}\">Click here</a> to be redirected.</h1>\n</body>\n</html>\n"
  },
  {
    "path": "templates/macros/feed_utils.html",
    "content": "{#- Feed utility macros -#}\n\n{#- Zola 0.19.0 uses `generate_feeds`. Prior versions use `generate_feed` -#}\n{%- macro get_generate_feed() -%}\n    {{- config.generate_feeds | default(value=config.generate_feed) -}}\n{%- endmacro get_generate_feed -%}\n\n{%- macro get_feed_url() -%}\n    {{- config.feed_filenames[0] | default(value=(config.feed_filename)) -}}\n{%- endmacro get_feed_url -%}\n\n{#- Check footer feed icon conditions -#}\n{%- macro should_show_footer_feed_icon() -%}\n    {%- set generate_feed = feed_utils::get_generate_feed() == \"true\" -%}\n    {%- set feed_url = feed_utils::get_feed_url() -%}\n    {{- generate_feed and config.extra.feed_icon and feed_url -}}\n{%- endmacro should_show_footer_feed_icon -%}\n\n{#- Get feed icon name: string value or \"rss\" for true, empty for false -#}\n{%- macro get_feed_icon_name() -%}\n    {%- if config.extra.feed_icon is string -%}\n        {{- config.extra.feed_icon -}}\n    {%- elif config.extra.feed_icon -%}\n        {{- \"rss\" -}}\n    {%- endif -%}\n{%- endmacro get_feed_icon_name -%}\n"
  },
  {
    "path": "templates/macros/format_date.html",
    "content": "{%- macro format_date(date, short, language_strings=\"\") -%}\n\n{#- Set locale -#}\n{%- set date_locale = macros_translate::translate(key=\"date_locale\", default=\"en_GB\", language_strings=language_strings) -%}\n\n{#- Check for language-specific date formats -#}\n{%- set language_format = \"\" -%}\n{%- if config.extra.date_formats -%}\n    {%- for format_config in config.extra.date_formats -%}\n        {%- if format_config.lang == lang -%}\n            {%- if short and format_config.short -%}\n                {%- set_global language_format = format_config.short -%}\n            {%- elif not short and format_config.long -%}\n                {%- set_global language_format = format_config.long -%}\n            {%- endif -%}\n        {%- endif -%}\n    {%- endfor -%}\n{%- endif -%}\n\n{%- if language_format -%}\n    {{ date | date(format=language_format, locale=date_locale) }}\n{%- elif config.extra.short_date_format and short -%}\n    {{ date | date(format=config.extra.short_date_format, locale=date_locale) }}\n{%- elif config.extra.long_date_format and not short -%}\n    {{ date | date(format=config.extra.long_date_format, locale=date_locale) }}\n{%- elif not config.extra.short_date_format and date_locale == \"en_GB\" -%}\n    {%- set day = date | date(format='%-d') | int -%}\n\n    {%- if day in [11, 12, 13] -%}\n        {%- set suffix = \"th\" -%}\n    {%- else -%}\n        {%- set last_digit = day % 10 -%}\n        {%- if last_digit == 1 -%}\n            {%- set suffix = \"st\" -%}\n        {%- elif last_digit == 2 -%}\n            {%- set suffix = \"nd\" -%}\n        {%- elif last_digit == 3 -%}\n            {%- set suffix = \"rd\" -%}\n        {%- else -%}\n            {%- set suffix = \"th\" -%}\n        {%- endif -%}\n    {%- endif -%}\n\n    {#- Return the date. -#}\n    {{ date | date(format=\"%-d\") }}{{ suffix }}\n    {%- if short == true -%}\n        {{ date | date(format=\" %b %Y\") }}\n    {%- else -%}\n        {{ date | date(format=\" %B %Y\") }}\n    {%- endif -%}\n{%- else -%}\n    {%- if short -%}\n        {{ date | date(format=\"%-d %b %Y\", locale=date_locale) }}\n    {%- else -%}\n        {{ date | date(format=\"%d %b %Y\", locale=date_locale) }}\n    {%- endif -%}\n{%- endif -%}\n\n{%- endmacro -%}\n"
  },
  {
    "path": "templates/macros/list_posts.html",
    "content": "{# `metadata` can be \"dates\", \"indexes\" or both (e.g. \"dates indexes\" or \"indexes dates\"). #}\n{# If both, the order doesn't matter and indexes will always be displayed before dates. #}\n{# It would also work with arrays (e.g. [\"dates\"] or [\"indexes\"] or even [\"indexes\",\"dates\"]). #}\n{# Nevertheless, arrays cannot be used as a default value for a macro parameter in Tera (see https://github.com/Keats/tera/issues/710). #}\n{# `paginator` is only used to compute indexes metadata and can be let empty otherwise. #}\n{% macro list_posts(posts, all_posts=\"\", max=999999, metadata=\"dates\", language_strings=\"\", section_path=\"\", paginator=\"\", pinned_first=false, current_page=1) %}\n\n{%- set separator = config.extra.separator | default(value=\"•\") -%}\n\n{# Separate pinned and regular posts from all_posts if available, otherwise from posts #}\n{% if pinned_first %}\n    {% set source_posts = all_posts | default(value=posts) %}\n    {% set pinned_posts = [] %}\n    {% set regular_posts = [] %}\n    {% for post in source_posts %}\n        {% if post.extra.pinned %}\n            {% set_global pinned_posts = pinned_posts | concat(with=post) %}\n        {% else %}\n            {% set_global regular_posts = regular_posts | concat(with=post) %}\n        {% endif %}\n    {% endfor %}\n\n    {# On page 1 or when no pagination, show pinned then regular #}\n    {% if current_page == 1 %}\n        {% if paginator %}\n            {# With pagination: pinned + current page's posts #}\n            {% set display_posts = pinned_posts | concat(with=posts) %}\n        {% else %}\n            {# Without pagination: pinned + regular (no duplicates) #}\n            {% set display_posts = pinned_posts | concat(with=regular_posts) %}\n        {% endif %}\n    {% else %}\n        {% set display_posts = posts %}\n    {% endif %}\n{% else %}\n    {% set display_posts = posts %}\n{% endif %}\n\n<div class=\"bloglist-container\">\n    {# Display posts #}\n    {% for post in display_posts %}\n        {% if loop.index <= max %}\n            {% if loop.index == max or loop.last %}\n                {% set bottom_divider = false %}\n            {% else %}\n                {% set bottom_divider = true %}\n            {% endif %}\n\n            <section class=\"bloglist-meta {% if bottom_divider -%}bottom-divider{%- endif -%}\">\n                <ul>\n                    {%- if \"indexes\" in metadata -%}\n                        {%- set post_index = loop.index -%}\n                        {%- set number_of_posts = posts | length -%}\n                        {# in case we have a pager, the index has been computed for the current page. #}\n                        {%- if paginator -%}\n                            {%- set number_of_posts = paginator.total_pages -%}\n                            {%- set number_of_other_pages = paginator.current_index - 1 -%}\n                            {%- set posts_per_page = paginator.paginate_by -%}\n                            {%- set posts_in_other_pages = number_of_other_pages * posts_per_page -%}\n                            {%- set post_index = posts_in_other_pages + post_index -%}\n                        {%- endif -%}\n                        {%- if macros_settings::evaluate_setting_priority(setting=\"post_listing_index_reversed\", page=section, default_global_value=false) == \"true\" -%}\n                            {# index starts at 1 instead of 0 #}\n                            {%- set post_index = number_of_posts + 1 - post_index -%}\n                        {%- endif -%}\n                        <li class=\"index-label\">{{ post_index }}</li>\n                    {%- endif -%}\n\n                    {%- if \"dates\" in metadata -%}\n                        {%- set allowed_post_listing_dates = [\"date\", \"updated\", \"both\"] -%}\n                        {#- Calling the hierarchy macro here causes an error due to the \"get parents\" part of the macro. -#}\n                        {#- This seems cleaner. -#}\n                        {%- set post_listing_date = section.extra.post_listing_date | default(value=config.extra.post_listing_date) | default(value=\"date\") -%}\n                        {%- if post_listing_date not in allowed_post_listing_dates -%}\n                            {{ throw(message=\"ERROR: Invalid value for config.extra.post_listing_date. Allowed values are 'date', 'updated', or 'both'.\") }}\n                        {%- endif -%}\n\n                        {%- set show_date = post.date and post_listing_date == \"date\" or post.date and post_listing_date == \"both\" or post.date and post_listing_date == \"updated\" and not post.updated -%}\n                        {%- set show_updated = post.updated and post_listing_date == \"updated\" or post.updated and post_listing_date == \"both\" -%}\n\n                        {%- if show_date or show_updated -%}\n                            {%- if show_date -%}\n                                <li class=\"date\">{{- macros_format_date::format_date(date=post.date, short=false, language_strings=language_strings) -}}</li>\n                            {%- endif -%}\n                            {%- if show_date and show_updated -%}\n                                <li class=\"mobile-only separator\">{{- separator -}}</li>\n                            {%- endif -%}\n                            {%- if show_updated -%}\n                                {%- set last_updated_str = macros_translate::translate(key=\"last_updated_on\", default=\"Updated on $DATE\", language_strings=language_strings) -%}\n                                {%- set formatted_date = macros_format_date::format_date(date=post.updated, short=true, language_strings=language_strings) -%}\n                                {%- set updated_str = last_updated_str | replace(from=\"$DATE\", to=formatted_date) -%}\n                                <li class=\"date\">{{ updated_str }}</li>\n                            {%- endif -%}\n                        {%- endif -%}\n                    {%- endif -%}\n\n                    {% if post.extra.local_image or post.extra.remote_image %}\n                        <li class=\"post-thumbnail\">\n                            <a href=\"{{ post.permalink }}\">\n                                {% if post.extra.local_image_dark and not post.extra.local_image %}\n                                    {{ throw(message=\"ERROR: `local_image_dark` requires `local_image` to be set in \" ~ post.path) }}\n                                {% endif %}\n                                {% if post.extra.remote_image_dark and not post.extra.remote_image %}\n                                    {{ throw(message=\"ERROR: `remote_image_dark` requires `remote_image` to be set in \" ~ post.path) }}\n                                {% endif %}\n                                {% if post.extra.local_image %}\n                                    {% if post.extra.local_image_dark %}\n                                        {# Show light image only in light mode, dark image only in dark mode #}\n                                        {% set meta_dark = get_image_metadata(path=post.extra.local_image_dark, allow_missing=true) %}\n                                        <img class=\"thumbnail-image img-dark\"\n                                            alt=\"{{ post.extra.local_image_dark }}\"\n                                            src=\"{{ get_url(path=post.extra.local_image_dark) }}\"\n                                            {% if meta_dark.width %}width=\"{{ meta_dark.width }}\"{% endif %}\n                                            {% if meta_dark.height %}height=\"{{ meta_dark.height }}\"{% endif %}>\n                                    {% endif %}\n                                    {% set meta = get_image_metadata(path=post.extra.local_image, allow_missing=true) %}\n                                    <img class=\"thumbnail-image{% if post.extra.local_image_dark %} img-light{% endif %}\"\n                                        alt=\"{{ post.extra.local_image }}\"\n                                        src=\"{{ get_url(path=post.extra.local_image) }}\"\n                                        {% if meta.width %}width=\"{{ meta.width }}\"{% endif %}\n                                        {% if meta.height %}height=\"{{ meta.height }}\"{% endif %}>\n                                {% elif post.extra.remote_image %}\n                                    {% if post.extra.remote_image_dark %}\n                                        {# Show light image only in light mode, dark image only in dark mode #}\n                                        {% set meta_dark = get_image_metadata(path=post.extra.remote_image_dark, allow_missing=true) %}\n                                        <img class=\"thumbnail-image img-dark\"\n                                            alt=\"{{ post.extra.remote_image_dark }}\"\n                                            src=\"{{ get_url(path=post.extra.remote_image_dark) }}\">\n                                    {% endif %}\n                                    <img class=\"thumbnail-image{% if post.extra.remote_image_dark %} img-light{% endif %}\"\n                                        alt=\"{{ post.extra.remote_image }}\"\n                                        src=\"{{ post.extra.remote_image }}\">\n                                {% endif %}\n                            </a>\n                        </li>\n                    {% endif %}\n\n                    {% if post.draft %}\n                        <li class=\"draft-label\">{{ macros_translate::translate(key=\"draft\", default=\"DRAFT\", language_strings=language_strings) }}</li>\n                    {% endif %}\n                </ul>\n            </section>\n\n            <section class=\"bloglist-content {% if bottom_divider -%}bottom-divider{%- endif -%}\">\n                <div>\n                    {% if pinned_first and post.extra.pinned %}\n                    <div class=\"pinned-label\">\n                        <svg aria-hidden=\"true\" xmlns=\"http://www.w3.org/2000/svg\" viewBox=\"0 0 16 16\"><path fill=\"currentColor\" fill-rule=\"evenodd\" d=\"M10.5 2.255v-.01c.003-.03.013-.157-.361-.35C9.703 1.668 8.967 1.5 8 1.5s-1.703.169-2.138.394c-.375.194-.365.32-.362.351v.01c-.003.03-.013.157.362.35C6.297 2.832 7.033 3 8 3s1.703-.169 2.139-.394c.374-.194.364-.32.361-.351M12 2.25c0 .738-.433 1.294-1.136 1.669l.825 2.31c1.553.48 2.561 1.32 2.561 2.52c0 1.854-2.402 2.848-5.5 2.985V15a.75.75 0 0 1-1.5 0v-3.266c-3.098-.136-5.5-1.131-5.5-2.984c0-1.2 1.008-2.04 2.561-2.52l.825-2.311C4.433 3.544 4 2.988 4 2.25C4 .75 5.79 0 8 0s4 .75 4 2.25\" clip-rule=\"evenodd\"/></svg>\n                        <span>{{ macros_translate::translate(key=\"pinned\", default=\"Pinned\", language_strings=language_strings) }}</span>\n                    </div>\n                    {% endif %}\n\n                    <h2 class=\"bloglist-title\">\n                        <a href=\"{{ post.permalink }}\">{{ post.title }}</a>\n                    </h2>\n\n                    {% if post.taxonomies.tags %}\n                        <div class=\"bloglist-tags\">\n                            {% for tag in post.taxonomies.tags %}\n                                <a class=\"tag\" href=\"{{ get_taxonomy_url(kind='tags', name=tag, lang=lang) | safe }}\">{{ tag }}</a>\n                            {% endfor %}\n                        </div>\n                    {% endif %}\n\n                    <div class=\"description\">\n                        {% if post.description %}\n                            <p>{{ post.description | markdown(inline=true) | safe  }}</p>\n                        {% elif post.summary %}\n                            <p>{{ post.summary | markdown(inline=true) | trim_end_matches(pat=\".\") | safe }}…</p>\n                        {% endif %}\n                    </div>\n                    <a class=\"readmore\" href=\"{{ post.permalink }}\">{{ macros_translate::translate(key=\"read_more\", default=\"Read more\", language_strings=language_strings) }}&nbsp;<span class=\"arrow\">→</span></a>\n                </div>\n            </section>\n        {% endif %}\n        {% if not loop.last %}\n            {% if loop.index == max %}\n                <div class=\"all-posts\">\n                    <a href=\"{{ get_url(path=section_path, lang=lang) }}/\">{{ macros_translate::translate(key=\"all_posts\", default=\"All posts\", language_strings=language_strings) }}&nbsp;<span class=\"arrow\">⟶</span></a>\n                </div>\n            {% endif %}\n        {% endif %}\n    {% endfor %}\n</div>\n{% endmacro %}\n"
  },
  {
    "path": "templates/macros/page_header.html",
    "content": "{% macro page_header(title, show_feed_icon=false) %}\n\n{% set rel_attributes = macros_rel_attributes::rel_attributes() | trim %}\n\n\n{%- set blank_target = macros_target_attribute::target_attribute(new_tab=config.markdown.external_links_target_blank) -%}\n\n<h1 class=\"title-container section-title bottom-divider\">\n    {{ title -}}\n    {% if show_feed_icon %}\n        {%- set feed_url = feed_utils::get_feed_url() -%}\n        {%- set feed_icon_name = feed_utils::get_feed_icon_name() -%}\n        <a class=\"no-hover-padding social\" rel=\"{{ rel_attributes }}\" {{ blank_target }} href=\"{{ get_url(path=term.path ~ feed_url, lang=lang, trailing_slash=false) | safe }}\">\n            <img loading=\"lazy\" alt=\"feed\" title=\"feed\" src=\"{{ get_url(path='/social_icons/' ~ feed_icon_name ~ '.svg') }}\">\n        </a>\n    {% endif %}\n</h1>\n\n{% endmacro page_header %}\n"
  },
  {
    "path": "templates/macros/rel_attributes.html",
    "content": "{% macro rel_attributes() %}\n\n{%- set rel_attributes = [] -%}\n{%- if config.markdown.external_links_target_blank -%}\n    {%- set rel_attributes = rel_attributes | concat(with=\"noopener\")  -%}\n{%- endif -%}\n{# https://developer.mozilla.org/en-US/docs/Web/HTML/Attributes/rel#nofollow #}\n{# This is ignored, as it doesn't make sense to set `nofollow` on projects or social links. #}\n{# {%- if config.markdown.external_links_no_follow -%}\n    {%- set rel_attributes = rel_attributes | concat(with=\"nofollow\")  -%}\n{%- endif -%} #}\n{%- if config.markdown.external_links_no_referrer -%}\n    {%- set rel_attributes = rel_attributes | concat(with=\"noreferrer\")  -%}\n{%- endif -%}\n\n{# Return the array of rel attributes joined by a space #}\n{{- rel_attributes | join(sep=\" \") -}}\n\n{% endmacro rel_attributes %}\n"
  },
  {
    "path": "templates/macros/series_page.html",
    "content": "{#\nThose macros deal with introduction and navigation for series pages.\nUsing macros have been prefered over partial inclusion or inline code to make sure series_ordered_pages is forced to be used.\nA section's pages natural order is invalid in case of reversed pagination which would lead to invalid series' pages order.\nTo prevent this, pages are ordered correctly in a separate variable which must be used instead of the series section pages.\n#}\n\n{#\nComputes the introduction of a series's page.\n\nParameters:\n    - `page`: The page object being part of the series.\n\t\t- `series_section`: The series' section the page belongs to.\n\t\t- `series_ordered_pages`: The series' pages properly ordered (see at the top of this file for an explanation).\n\t\t- `language_strings`: A dictionary containing the translation strings.\n#}\n{% macro process_series_template(template_type, page, series_section, series_ordered_pages, language_strings) %}\n    {%- if \"series\" in series_section.extra and series_section.extra.series -%}\n        {# Prepare variables for substitution #}\n        {%- set series_title = series_section.title -%}\n        {%- set series_permalink = series_section.permalink -%}\n        {%- set series_html_link = '<a href=\"' ~ series_section.permalink ~ '\" aria_label=\"' ~ series_section.title ~ '\">' ~ series_section.title ~ '</a>' -%}\n\t\t{# Build series pages list #}\n\t\t{%- set series_pages_list = [] -%}\n\t\t{%- for series_page in series_ordered_pages -%}\n\t\t\t{%- if series_page.relative_path == page.relative_path -%}\n\t\t\t\t{%- set series_pages_list_item = '<li>' ~ series_page.title ~ '</li>' -%}\n\t\t\t{%- else -%}\n\t\t\t\t{%- set series_pages_list_item = '<li><a href=\"' ~ series_page.permalink ~ '\" aria_label=\"' ~ series_page.title ~ '\">' ~ series_page.title ~ '</a></li>' -%}\n\t\t\t{%- endif -%}\n\t\t\t{%- set_global series_pages_list = series_pages_list | concat(with=series_pages_list_item) -%}\n\t\t{%- endfor -%}\n\t\t{%- set series_pages_list = series_pages_list | join(sep=\"\") -%}\n\t\t{%- if macros_settings::evaluate_setting_priority(setting=\"post_listing_index_reversed\", page=series_section, default_global_value=false) == \"true\" -%}\n\t\t\t{%- set series_pages_ordered_list = '<ol reversed>' ~ series_pages_list ~ '</ol>' -%}\n\t\t{%- else -%}\n\t\t\t{%- set series_pages_ordered_list = '<ol>' ~ series_pages_list ~ '</ol>' -%}\n\t\t{%- endif -%}\n\t\t{%- set series_pages_unordered_list = '<ul>' ~ series_pages_list ~ '</ul>' -%}\n\n        {# Get page position and navigation info #}\n        {%- set series_pages_number = 0 -%}\n        {%- set series_page_index = 0 -%}\n        {%- set first_page = series_ordered_pages | first -%}\n        {%- set is_found = false -%}\n\n        {%- for series_page in series_ordered_pages -%}\n            {%- set_global series_pages_number = series_pages_number + 1 -%}\n            {%- if series_page.relative_path == page.relative_path -%}\n                {%- set_global series_page_index = series_pages_number -%}\n                {%- set_global is_found = true -%}\n            {%- else -%}\n                {%- if not is_found -%}\n                    {%- set_global prev_page = series_page -%}\n                {%- elif not next_page is defined -%}\n                    {%- set_global next_page = series_page -%}\n                {%- endif -%}\n            {%- endif -%}\n        {%- endfor -%}\n\n\t\t{# Determine template to use based on available navigation #}\n\t\t{%- set position = \"middle\" -%}\n\t\t{%- if prev_page is defined and not next_page is defined -%}\n\t\t\t{%- set_global position = \"prev_only\" -%}\n\t\t{%- elif next_page is defined and not prev_page is defined -%}\n\t\t\t{%- set_global position = \"next_only\" -%}\n\t\t{%- endif -%}\n\n        {# Get template from config #}\n        {%- set templates_key = \"series_\" ~ template_type ~ \"_templates\" -%}\n        {%- set template = \"\" -%}\n        {%- if series_section.extra[templates_key] is defined -%}\n            {%- if series_section.extra[templates_key][position] is defined -%}\n                {%- set_global template = series_section.extra[templates_key][position] -%}\n            {%- elif series_section.extra[templates_key].default is defined -%}\n                {%- set_global template = series_section.extra[templates_key].default -%}\n            {%- endif -%}\n        {%- endif -%}\n\n\t\t{# Prepare navigation variables #}\n\t\t{%- if prev_page is defined -%}\n\t\t\t{%- set prev_title = prev_page.title -%}\n\t\t\t{%- set prev_permalink = prev_page.permalink -%}\n\t\t\t{%- set prev_html_link = '<a href=\"' ~ prev_page.permalink ~ '\" aria_label=\"' ~ prev_page.title ~ '\">' ~ prev_page.title ~ '</a>' -%}\n\t\t\t{%- set prev_description = prev_page.description | default(value=\"\") -%}\n\t\t{%- endif -%}\n\t\t{%- if next_page is defined -%}\n\t\t\t{%- set next_title = next_page.title -%}\n\t\t\t{%- set next_permalink = next_page.permalink -%}\n\t\t\t{%- set next_html_link = '<a href=\"' ~ next_page.permalink ~ '\" aria_label=\"' ~ next_page.title ~ '\">' ~ next_page.title ~ '</a>' -%}\n\t\t\t{%- set next_description = next_page.description | default(value=\"\") -%}\n\t\t{%- endif -%}\n\n\t\t{# Replace standard variables #}\n\t\t{%- set template = template\n\t\t\t| replace(from=\"$SERIES_TITLE\", to=series_title)\n\t\t\t| replace(from=\"$SERIES_PERMALINK\", to=series_permalink)\n\t\t\t| replace(from=\"$SERIES_HTML_LINK\", to=series_html_link)\n\t\t\t| replace(from=\"$FIRST_TITLE\", to=first_page.title)\n\t\t\t| replace(from=\"$FIRST_HTML_LINK\", to='<a href=\"' ~ first_page.permalink ~ '\">' ~ first_page.title ~ '</a>')\n\t\t\t| replace(from=\"$SERIES_PAGES_NUMBER\", to=series_pages_number | as_str)\n\t\t\t| replace(from=\"$SERIES_PAGE_INDEX\", to=series_page_index | as_str)\n\t\t\t| replace(from=\"$SERIES_PAGES_OLIST\", to=series_pages_ordered_list)\n\t\t\t| replace(from=\"$SERIES_PAGES_ULIST\", to=series_pages_unordered_list)\n\t\t-%}\n\n        {# Replace navigation variables if they exist #}\n\t\t{%- if prev_page is defined -%}\n\t\t{%- set template = template\n\t\t\t| replace(from=\"$PREV_TITLE\", to=prev_title)\n\t\t\t| replace(from=\"$PREV_PERMALINK\", to=prev_permalink)\n\t\t\t| replace(from=\"$PREV_HTML_LINK\", to=prev_html_link)\n\t\t\t| replace(from=\"$PREV_DESCRIPTION\", to=prev_description)\n\t\t-%}\n\t\t{%- endif -%}\n\n\t\t{%- if next_page is defined -%}\n\t\t{%- set template = template\n\t\t\t| replace(from=\"$NEXT_TITLE\", to=next_title)\n\t\t\t| replace(from=\"$NEXT_PERMALINK\", to=next_permalink)\n\t\t\t| replace(from=\"$NEXT_HTML_LINK\", to=next_html_link)\n\t\t\t| replace(from=\"$NEXT_DESCRIPTION\", to=next_description)\n\t\t-%}\n\t\t{%- endif -%}\n\n\t\t{# Custom placeholders #}\n\t\t{%- if series_section.extra.series_template_placeholders is defined -%}\n\t\t\t{%- set missing_vars = [] -%}\n\t\t\t{%- for placeholder in series_section.extra.series_template_placeholders -%}\n\t\t\t\t{%- if placeholder in template -%}\n\t\t\t\t\t{%- set var_name = placeholder | replace(from=\"$\", to=\"\") | lower -%}\n\t\t\t\t\t{%- if page.extra.series_template_variables is defined and page.extra.series_template_variables[var_name] is defined -%}\n\t\t\t\t\t\t{%- set_global template = template | replace(from=placeholder, to=page.extra.series_template_variables[var_name]) -%}\n\t\t\t\t\t{%- else -%}\n\t\t\t\t\t\t{%- set_global missing_vars = missing_vars | concat(with=var_name) -%}\n\t\t\t\t\t{%- endif -%}\n\t\t\t\t{%- endif -%}\n\t\t\t{%- endfor -%}\n\t\t\t{%- if missing_vars | length > 0 -%}\n\t\t\t\t{%- set missing_vars_str = missing_vars | join(sep=\", \") -%}\n\t\t\t\t{{ throw(message=\"ERROR: The following variables are included in this page's series templates (`series_template_placeholders`) but have not been set in the `series_template_variables` of this page: \" ~ missing_vars_str) }}\n\t\t\t{%- endif -%}\n\t\t{%- endif -%}\n\n        {# Output the processed template if not empty #}\n        {%- if template | length > 0 -%}\n            <section class=\"series-page-{{ template_type }}\">\n                {{ template | markdown | safe }}\n            </section>\n        {%- endif -%}\n    {%- endif -%}\n{% endmacro %}\n\n{# Macro for series introduction #}\n{% macro get_introduction(page, series_section, series_ordered_pages, language_strings) %}\n    {{ macros_series_page::process_series_template(template_type=\"intro\", page=page, series_section=series_section, series_ordered_pages=series_ordered_pages, language_strings=language_strings) }}\n{% endmacro %}\n\n{# Macro for series outro #}\n{% macro get_outro(page, series_section, series_ordered_pages, language_strings) %}\n    {{ macros_series_page::process_series_template(template_type=\"outro\", page=page, series_section=series_section, series_ordered_pages=series_ordered_pages, language_strings=language_strings) }}\n{% endmacro %}\n"
  },
  {
    "path": "templates/macros/settings.html",
    "content": "{#\nEvaluates the priority of a particular setting across different scopes.\n\nThe priority is as follows: page > section > config.\n\nParameters:\n    - setting: The name of the setting to evaluate.\n    - page: The page object containing settings.\n    - default_global_value: The setting's default value.\n#}\n\n{% macro evaluate_setting_priority(setting, page, section=\"\", default_global_value=\"\") %}\n\n{%- if section -%}\n    {%- set current_section = section -%}\n{%- elif page -%}\n    {%- set current_section = \"\" -%}\n    {#- Retrieve last ancestor to determine current section, if applicable -#}\n    {%- if page.ancestors | length > 0 -%}\n        {%- set last_ancestor = page.ancestors | slice(start=-1) -%}\n        {%- set_global current_section = get_section(path=last_ancestor.0, metadata_only=true) -%}\n    {%- else -%}\n        {#- We're likely in a nested page. Try to find the parent page or nearest section. -#}\n        {%- set components = page.components -%}\n        {%- for i in range(start=1, end=components | length) -%}\n            {%- if lang == config.default_language -%}\n                {%- set potential_path = components | slice(end=components | length - i) | join(sep=\"/\") -%}\n                {%- set potential_page = potential_path ~ \"/index.md\" -%}\n                {%- set potential_section = potential_path ~ \"/_index.md\" -%}\n            {%- else -%}\n                {%- set potential_path = components | slice(start=1, end=components | length - i) | join(sep=\"/\") -%}\n                {%- set potential_page = potential_path ~ \"/index.\" ~ lang ~ \".md\" -%}\n                {%- set potential_section = potential_path ~ \"/_index.\" ~ lang ~ \".md\" -%}\n            {%- endif -%}\n            {#- Check for parent page first. -#}\n            {%- set page_data = load_data(path=potential_page, required=false) -%}\n            {%- if page_data -%}\n                {%- set_global current_section = get_page(path=potential_page) -%}\n                {%- break -%}\n            {%- endif -%}\n            {#- No parent page, check for section. -#}\n            {%- set section_data = load_data(path=potential_section, required=false) -%}\n            {%- if section_data -%}\n                {%- set_global current_section = get_section(path=potential_section, metadata_only=true) -%}\n                {%- break -%}\n            {%- endif -%}\n        {%- endfor -%}\n    {%- endif -%}\n{%- endif -%}\n\n{%- set priority_order = [\n    page.extra[setting] | default(value=\"\"),\n    current_section.extra[setting] | default(value=\"\"),\n    config.extra[setting] | default(value=\"\")\n] -%}\n\n{%- set output = default_global_value -%}\n\n{%- for value in priority_order -%}\n    {%- if value != \"\" -%}\n        {%- set_global output = value -%}\n        {%- break -%}\n    {%- endif -%}\n{%- endfor -%}\n\n{{- output -}}\n\n{% endmacro %}\n"
  },
  {
    "path": "templates/macros/table_of_contents.html",
    "content": "{% macro toc(page, header, language_strings=\"\") %}\n\n{%- set toc_levels = page.extra.toc_levels | default(value=3) -%}\n\n{% if page.extra.toc_ignore_pattern %}\n    {%- set toc_ignore_pattern = page.extra.toc_ignore_pattern -%}\n{% endif %}\n\n<div class=\"toc-container\">\n    {% if header %}\n        <h3>{{ macros_translate::translate(key=\"table_of_contents\", default=\"Table of Contents\", language_strings=language_strings) }}</h3>\n    {% endif %}\n\n    <ul>\n        {% for h1 in page.toc %}\n            {# Only render headers if there's no ignore pattern, or if the header text doesn't match the pattern. #}\n            {% if not toc_ignore_pattern or not (h1.title is matching(toc_ignore_pattern)) %}\n                <li><a href=\"{{ h1.permalink | safe }}\">{{ h1.title }}</a>\n                    {% if h1.children and toc_levels > 1 %}\n                        <ul>\n                            {% for h2 in h1.children %}\n                                {% if not toc_ignore_pattern or not (h2.title is matching(toc_ignore_pattern)) %}\n                                    <li><a href=\"{{ h2.permalink | safe }}\">{{ h2.title }}</a>\n                                        {% if h2.children and toc_levels > 2 %}\n                                            <ul>\n                                                {% for h3 in h2.children %}\n                                                    {% if not toc_ignore_pattern or not (h3.title is matching(toc_ignore_pattern)) %}\n                                                        <li><a href=\"{{ h3.permalink | safe }}\">{{ h3.title }}</a>\n                                                            {% if h3.children and toc_levels > 3 %}\n                                                                <ul>\n                                                                    {% for h4 in h3.children %}\n                                                                        {% if not toc_ignore_pattern or not (h4.title is matching(toc_ignore_pattern)) %}\n                                                                            <li><a href=\"{{ h4.permalink | safe }}\">{{ h4.title }}</a></li>\n                                                                        {% endif %}\n                                                                    {% endfor %}\n                                                                </ul>\n                                                            {% endif %}\n                                                        </li>\n                                                    {% endif %}\n                                                {% endfor %}\n                                            </ul>\n                                        {% endif %}\n                                    </li>\n                                {% endif %}\n                            {% endfor %}\n                        </ul>\n                    {% endif %}\n                </li>\n            {% endif %}\n        {% endfor %}\n    </ul>\n</div>\n\n{% endmacro toc %}\n"
  },
  {
    "path": "templates/macros/target_attribute.html",
    "content": "{% macro target_attribute(new_tab) %}\n \n{%- set blank_target = \"\" -%}\n\n{%- if new_tab -%}\n    {%- set blank_target = \"target=_blank\" -%}\n{%- endif -%}\n\n{{ blank_target }}\n\n{% endmacro target_attribute %}\n"
  },
  {
    "path": "templates/macros/translate.html",
    "content": "{#- Dynamically selects the appropriate translation key based on the provided `number` and `lang` context.\nIf a `number` is provided, the macro will attempt to pluralize the translation key based on the language's rules.\n\nParameters:\n- `key`: The base key for the translation string, matching the i18n files. Example: `results` to get `zero_results`, `one_results`, `many_results`, etc.\n- `number`: Optional. The numerical value associated with the key.\n- `language_strings`: A dictionary containing the translation strings.\n- `default`: A default string to use if no translation is found for the key.\n- `replace`: Optional. If `true`, the macro will replace the `$NUMBER` placeholder in the translation string with the provided `number`.\n\nThe macro supports special pluralization rules for:\n- Arabic (`ar`): Has unique forms for zero, one, two, few, and many.\n- Slavic languages: Pluralization depends on the last digit of the number, with exceptions for numbers ending in 11-14.\n\nNOTE: If the logic for pluralization is modified, it needs to be replicated on the JavaScript search.\nFiles: static/js/searchElasticlunr.js and its minified version at static/js/searchElasticlunr.min.js\nFunction name: getPluralizationKey -#}\n{% macro translate(key, number=-1, language_strings=\"\", default=\"\", replace=true) %}\n    {%- set slavic_plural_languages = [\"uk\", \"be\", \"bs\", \"hr\", \"ru\", \"sr\"] -%}\n\n    {%- set key_prefix = \"\" -%}\n    {#- `zero_` and `one_` are common cases. We handle \"many\" (plural) later, after language-specific pluralization -#}\n    {%- if number == 0 -%}\n        {%- set key_prefix = \"zero_\" -%}\n    {%- elif number == 1 -%}\n        {%- set key_prefix = \"one_\" -%}\n    {%- endif -%}\n\n    {#- Pluralization -#}\n    {%- if number != -1 and key_prefix == \"\" -%}\n        {#- Arabic -#}\n        {%- if lang == \"ar\" -%}\n            {%- set modulo = number % 100 -%}\n            {%- if number == 2 -%}\n                {%- set key_prefix = \"two_\" -%}\n            {%- elif modulo >= 3 and modulo <= 10 -%}\n                {%- set key_prefix = \"few_\" -%}\n            {%- else -%}\n                {%- set key_prefix = \"many_\" -%}\n            {%- endif -%}\n        {#- Slavic languages like Russian or Ukrainian -#}\n        {%- elif lang in slavic_plural_languages -%}\n            {%- set modulo10 = number % 10 -%}\n            {%- set modulo100 = number % 100 -%}\n            {%- if modulo10 == 1 and modulo100 != 11 -%}\n                {%- set key_prefix = \"one_\" -%}\n            {%- elif modulo10 >= 2 and modulo10 <= 4 -%}\n                {%- if modulo100 >= 12 and modulo100 <= 14 -%}\n                    {%- set key_prefix = \"many_\" -%}\n                {%- else -%}\n                    {%- set key_prefix = \"few_\" -%}\n                {%- endif -%}\n            {%- else -%}\n                {%- set key_prefix = \"many_\" -%}\n            {%- endif -%}\n        {%- else -%}\n            {#- Default pluralization -#}\n            {#- Zero and one are already handled -#}\n            {%- set key_prefix = \"many_\" -%}\n        {%- endif -%}\n    {%- endif -%}\n\n    {#- Translated string -#}\n    {%- set final_key = key_prefix ~ key -%}\n    {%- set translated_text = language_strings[final_key] | default(value=default) | safe -%}\n\n    {#- Replace $NUMBER with the number -#}\n    {%- if replace -%}\n        {%- set translated_text = translated_text | replace(from=\"$NUMBER\", to=number | as_str) -%}\n    {%- endif -%}\n    {{- translated_text -}}\n{% endmacro %}\n"
  },
  {
    "path": "templates/page.html",
    "content": "{% extends \"base.html\" %}\n\n{% block main_content %}\n{%- set separator = config.extra.separator | default(value=\"•\") -%}\n\n{%- set rel_attributes = macros_rel_attributes::rel_attributes() | trim -%}\n\n{%- set blank_target = macros_target_attribute::target_attribute(new_tab=config.markdown.external_links_target_blank) -%}\n\n{# Debugging #}\n{# <div><pre>\n    Page path: {{ page.path }}\n    Page components: {{ page.components | join(sep=\", \") }}\n    Page ancestors: {{ page.ancestors | join(sep=\", \") }}\n    Current language: {{ lang }}\n    Default language: {{ config.default_language }}\n    Current section: {% if current_section %}{{ current_section.path }}{% else %}None{% endif %}\n    Page extra: {{ page.extra | json_encode() }}\n    {% if section -%}\n        {%- set current_section = section -%}\n    {%- elif page -%}\n        {%- set current_section = \"\" -%}\n        {%- if page.ancestors | length > 0 -%}\n            {%- set last_ancestor = page.ancestors | slice(start=-1) -%}\n            {%- set_global current_section = get_section(path=last_ancestor.0, metadata_only=true) -%}\n        {%- else -%}\n            {%- set components = page.components -%}\n            {%- for i in range(start=1, end=components | length) -%}\n                {%- if lang == config.default_language -%}\n                    {%- set potential_path = components | slice(end=components | length - i) | join(sep=\"/\") -%}\n                    {%- set potential_page = potential_path ~ \"/index.md\" -%}\n                    {%- set potential_section = potential_path ~ \"/_index.md\" -%}\n                {%- else -%}\n                    {%- set potential_path = components | slice(start=1, end=components | length - i) | join(sep=\"/\") -%}\n                    {%- set potential_page = potential_path ~ \"/index.\" ~ lang ~ \".md\" -%}\n                    {%- set potential_section = potential_path ~ \"/_index.\" ~ lang ~ \".md\" -%}\n                {%- endif -%}\n                Checking parent page: {{ potential_page }}\n                {%- set page_data = load_data(path=potential_page, required=false) -%}\n                {%- if page_data -%}\n                    {%- set_global current_section = get_page(path=potential_page) %}\n    Parent page found: {{ current_section.path }}\n                    {%- break -%}\n                {%- endif -%}\n                Checking section: {{ potential_section }}\n                {%- set section_data = load_data(path=potential_section, required=false) -%}\n                {%- if section_data -%}\n                    {%- set_global current_section = get_section(path=potential_section, metadata_only=true) -%}\n                    Section found: {{ current_section.path }}\n                    {%- break -%}\n                {%- endif -%}\n            {%- endfor -%}\n        {%- endif -%}\n    {%- endif %}\nFound nearest parent/section: {% if current_section %}{{ current_section.path }}{% else %}None{% endif %}\nCurrent section extra: {% if current_section %}{{ current_section.extra | json_encode() }}{% else %}None{% endif %}\n</pre></div>\n\n    {% set settings_to_test = [\n        \"iine\",\n        \"iine_icon\",\n        \"enable_cards_tag_filtering\",\n        \"footnote_backlinks\",\n        \"add_src_to_code_block\",\n        \"force_codeblock_ltr\",\n        \"copy_button\",\n        \"katex\",\n        \"quick_navigation_buttons\",\n        \"show_reading_time\",\n        \"show_date\",\n        \"show_author\",\n        \"show_remote_changes\",\n        \"toc\",\n        \"show_previous_next_article_links\",\n        \"invert_previous_next_article_links\",\n        \"previous_next_article_links_full_width\",\n        \"enable_csp\",\n    ] %}\n\n    <table>\n        <thead>\n            <tr>\n                <th>setting</th>\n                <th>page</th>\n                <th>section</th>\n                <th>config</th>\n                <th>macro output</th>\n            </tr>\n        </thead>\n        <tbody>\n            {% for setting in settings_to_test %}\n                <tr>\n                    <td><code>{{ setting }}</code></td>\n                    <td>{{ page.extra[setting] | default(value=\"⬛\") }}</td>\n                    <td>{{ current_section.extra[setting] | default(value=\"⬛\") }}</td>\n                    <td>{{ config.extra[setting] | default(value=\"⬛\") }}</td>\n                    <td>{{ macros_settings::evaluate_setting_priority(setting=setting, page=page) }}</td>\n                </tr>\n            {% endfor %}\n        </tbody>\n    </table>\n</div> #}\n\n{# {{ __tera_context }} #}\n{# End debugging #}\n\n<main>\n    <article class=\"h-entry\">\n        <h1 class=\"p-name article-title\">\n            {{ page.title | markdown(inline=true) | safe }}\n        </h1>\n        <a class=\"u-url u-uid\" href=\"{{ page.permalink | safe }}\"></a>\n\n        <ul class=\"meta\">\n            {#- Draft indicator -#}\n            {% if page.draft %}\n                <li class=\"draft-label\">{{ macros_translate::translate(key=\"draft\", default=\"DRAFT\", language_strings=language_strings) }}</li>\n            {% endif %}\n\n         {#- Author(s) -#}\n            {%- if page.authors or config.author and macros_settings::evaluate_setting_priority(setting=\"show_author\", page=page, default_global_value=false) == \"true\" -%}\n                {%- if page.authors -%}\n                    {%- set author_list = page.authors -%}\n                {%- else -%}\n                    {%- set author_list = [config.author] -%}\n                {%- endif -%}\n\n                {%- if author_list | length == 1 -%}\n                    {%- set author_string = '<span class=\"p-author\">' ~ author_list.0 ~ '</span>' -%}\n                {%- else -%}\n                    {%- set last_author = author_list | last -%}\n                    {%- set other_authors = author_list | slice(end=-1) -%}\n                    {%- set author_separator = macros_translate::translate(key=\"author_separator\", default=\", \", language_strings=language_strings) -%}\n                   {%- set author_separator = '</span>' ~ author_separator ~ '<span class=\"p-author\">' -%}\n                    {%- set conjunction = macros_translate::translate(key=\"author_conjunction\", default=\" and \", language_strings=language_strings) -%}\n                    {%- set conjunction = '</span>' ~ conjunction ~ '<span class=\"p-author\">' -%}\n                    {%- set author_string = other_authors | join(sep=author_separator) -%}\n                    {%- set author_string = author_string ~ conjunction ~ last_author -%}\n                    {%- set author_string = '<span class=\"p-author\">' ~ author_string ~ '</span>' -%}\n                {%- endif -%}\n\n                {%- set by_author = macros_translate::translate(key=\"by_author\", default=\"By $AUTHOR\", language_strings=language_strings) -%}\n                <li>{{ by_author | replace(from=\"$AUTHOR\", to=author_string) | safe }}</li>\n                {%- set previous_visible = true -%}\n            {%- endif -%}\n\n            {%- if config.extra.hcard and config.extra.hcard.enable and ( not author_list or author_list is containing(config.author)) -%}\n                {% include \"partials/hcard_small.html\" %}\n            {%- endif -%}\n\n            {%- set separator_with_class = \"<span class='separator' aria-hidden='true'>\" ~ separator ~ \"</span>\"-%}\n\n            {#- Date -#}\n            {%- if page.date and macros_settings::evaluate_setting_priority(setting=\"show_date\", page=page, default_global_value=true) == \"true\" -%}\n                <li><time class=\"dt-published\" datetime=\"{{ page.date }}\">{%- if previous_visible -%}{{ separator_with_class | safe }}{%- endif -%}{{ macros_format_date::format_date(date=page.date, short=true, language_strings=language_strings) }}</time></li>\n                {#- Variable to keep track of whether we've shown a section, to avoid separators as the first element -#}\n                {%- set previous_visible = true -%}\n            {%- endif -%}\n\n            {#- Reading time -#}\n            {%- if macros_settings::evaluate_setting_priority(setting=\"show_reading_time\", page=page, default_global_value=true) == \"true\" -%}\n                <li title=\"{{ macros_translate::translate(key=\"words\", number=page.word_count, default=\"$NUMBER words\", language_strings=language_strings) }}\">{%- if previous_visible -%}{{ separator_with_class | safe }}{%- endif -%}{{ macros_translate::translate(key=\"min_read\", number=page.reading_time, default=\"$NUMBER min read\", language_strings=language_strings) }}</li>\n                {%- set previous_visible = true -%}\n            {%- endif -%}\n\n            {#- Tags -#}\n            {%- if page.taxonomies and page.taxonomies.tags -%}\n                <li class=\"tag\">{%- if previous_visible -%}{{ separator_with_class | safe }}{%- endif -%}{{- macros_translate::translate(key=\"tags\", default=\"tags\", language_strings=language_strings) | capitalize -}}:&nbsp;</li>\n                {%- for tag in page.taxonomies.tags -%}\n                    <li class=\"tag\"><a class=\"p-category\" href=\"{{ get_taxonomy_url(kind='tags', name=tag, lang=lang) | safe }}\">{{ tag }}</a>\n                    {%- if not loop.last -%}\n                        ,&nbsp;\n                    {%- endif -%}\n                    </li>\n                {%- endfor -%}\n                {%- set previous_visible = true -%}\n            {%- endif -%}\n\n            {#- Last updated on -#}\n            {% if page.updated %}\n                {%- set last_updated_str = macros_translate::translate(key=\"last_updated_on\", default=\"Updated on $DATE\", language_strings=language_strings) -%}\n                {%- set formatted_date = macros_format_date::format_date(date=page.updated, short=true, language_strings=language_strings) -%}\n                {%- set updated_str = last_updated_str | replace(from=\"$DATE\", to=formatted_date) -%}\n                {%- set previous_visible = true -%}\n                </ul><ul class=\"meta last-updated\"><li><time class=\"dt-updated\" datetime=\"{{ page.updated }}\">{{ updated_str }}</time></li>\n                {#- Show link to remote changes if enabled -#}\n                {%- if config.extra.remote_repository_url and macros_settings::evaluate_setting_priority(setting=\"show_remote_changes\", page=page, default_global_value=true) == \"true\" -%}\n                    <li>{%- if previous_visible -%}{{ separator_with_class | safe }}{%- endif -%}<a class=\"external\" href=\"{% include \"partials/history_url.html\" %}\" {{ blank_target }} rel=\"{{ rel_attributes }}\">{{ macros_translate::translate(key=\"see_changes\", default=\"See changes\", language_strings=language_strings) }}</a></li>\n                {%- endif -%}\n            {% endif %}\n        </ul>\n\n        {#- A page is part of a series if one of the sections above (whether it is transparent or not) has the `extra.series` parameter set to true. -#}\n        {#- As a series might be a transparent section in order to mix up its articles with those of the section just above or the root, -#}\n        {#- there is no other way but to compute the potential path of each ancestor section related to the page and look for the first one being a series. -#}\n        {#- Using the `ancestors` field of a section is not possible because transparent sections are not present in this field. -#}\n        {%- set series_path_components = [] -%}\n        {%- set section_paths = [] -%}\n        {%- for path in page.relative_path | split(pat=\"/\") | slice(end=-1) -%}\n            {%- set_global series_path_components = series_path_components | concat(with=path) -%}\n            {%- set section_path = series_path_components | concat(with=\"_index.md\") | join(sep=\"/\") -%}\n            {%- set_global section_paths = section_paths | concat(with=section_path) -%}\n        {%- endfor -%}\n        {#- The series the page is part of is the closest section flagged as a series, if any -#}\n        {%- for section_path in section_paths | reverse -%}\n            {%- set section_file_exists = load_data(path=section_path, required=false) -%}\n            {%- if section_file_exists -%}\n                {%- set loaded_section = get_section(path=section_path,lang=lang) -%}\n                {%- if \"series\" in loaded_section.extra and loaded_section.extra.series -%}\n                    {%- set_global series_section = loaded_section -%}\n                    {%- set_global series_ordered_pages = loaded_section.pages -%}\n                    {%- if loaded_section.paginated | default(value=0) > 0 and loaded_section.paginate_reversed -%}\n                        {%- set_global series_ordered_pages = loaded_section.pages | reverse -%}\n                    {%- endif -%}\n                    {%- break -%}\n                {%- endif -%}\n            {%- endif -%}\n        {%- endfor -%}\n\n        {% if page.extra.tldr %}\n            <div class=\"admonition note\">\n                <div class=\"admonition-icon admonition-icon-note\"></div>\n                <div class=\"admonition-content\">\n                    <strong class=\"admonition-title\">\n                        <span title=\"Too long; didn't read (summary)\">TL;DR</span>\n                    </strong>\n                    <p>{{ page.extra.tldr | markdown | safe }}</p>\n                </div>\n            </div>\n        {% endif %}\n\n        {#- Optional table of contents below the header -#}\n        {% if page.toc and macros_settings::evaluate_setting_priority(setting=\"toc\", page=page, default_global_value=false) == \"true\" %}\n            {{ macros_toc::toc(page=page, header=true, language_strings=language_strings) }}\n        {% endif %}\n\n        {#- Optional Summary paragraph for readers -#}\n        {% if page.description %}\n            <p class=\"p-summary\" hidden>{{ page.description }}</p>\n        {%- endif -%}\n\n\n        <section class=\"e-content body\">\n            {#- Replace series_intro placeholder -#}\n            {%- set content_with_intro = page.content -%}\n            {%- if \"<!-- series_intro -->\" in page.content -%}\n                {%- set series_intro_html = macros_series_page::get_introduction(page=page, series_section=series_section, series_ordered_pages=series_ordered_pages, language_strings=language_strings) -%}\n                {%- set content_with_intro = content_with_intro | replace(from=\"<!-- series_intro -->\", to=series_intro_html) -%}\n            {%- elif series_section -%}\n                {%- set series_intro_html = macros_series_page::get_introduction(page=page, series_section=series_section, series_ordered_pages=series_ordered_pages, language_strings=language_strings) -%}\n                {%- set content_with_intro = series_intro_html ~ content_with_intro -%}\n            {%- endif -%}\n\n            {#- Handle series_outro placeholder -#}\n            {%- set processed_content = content_with_intro -%}\n            {%- if \"<!-- series_outro -->\" in content_with_intro -%}\n                {%- set series_outro_html = macros_series_page::get_outro(page=page, series_section=series_section, series_ordered_pages=series_ordered_pages, language_strings=language_strings) -%}\n                {%- set processed_content = processed_content | replace(from=\"<!-- series_outro -->\", to=series_outro_html) -%}\n            {%- elif series_section -%}\n                {%- set series_outro_html = macros_series_page::get_outro(page=page, series_section=series_section, series_ordered_pages=series_ordered_pages, language_strings=language_strings) -%}\n                {#- We want the outro at the end of the article, but before footnote definitions -#}\n                {%- set footnotes_marker = '<hr><ol class=\"footnotes-list\">' -%}\n                {%- if footnotes_marker in content_with_intro -%}\n                    {%- set content_sections = processed_content | split(pat=footnotes_marker) -%}\n                    {%- set main_content = content_sections | first -%}\n                    {%- set footnotes_content = content_sections | slice(start=1) | join(sep=footnotes_marker) -%}\n                    {%- set processed_content = main_content ~ series_outro_html ~ footnotes_marker ~ footnotes_content -%}\n                {%- else -%}\n                    {%- set processed_content = processed_content ~ series_outro_html -%}\n                {%- endif -%}\n            {%- endif -%}\n\n            {#- Replace TOC and render final content -#}\n            {#- The replace pattern is used to enable arbitrary locations for the Table of Contents -#}\n            {#- This is Philipp Oppermann's workaround: https://github.com/getzola/zola/issues/584#issuecomment-474329637 -#}\n            {{ processed_content | replace(from=\"<!-- toc -->\", to=macros_toc::toc(page=page, header=false, language_strings=language_strings)) | safe }}\n        </section>\n\n        {#- iine button -#}\n        {%- if macros_settings::evaluate_setting_priority(setting=\"iine\", page=page, default_global_value=false) == \"true\" -%}\n            {% include \"partials/iine_button.html\" %}\n        {%- endif -%}\n\n        {% if macros_settings::evaluate_setting_priority(setting=\"show_previous_next_article_links\", page=page, default_global_value=true) == \"true\" %}\n            {%- if page.lower or page.higher -%}\n                {% set next_label = macros_translate::translate(key=\"next\", default=\"Next\", language_strings=language_strings) %}\n                {% set prev_label = macros_translate::translate(key=\"prev\", default=\"Prev\", language_strings=language_strings) %}\n                {% if macros_settings::evaluate_setting_priority(setting=\"invert_previous_next_article_links\", page=page, default_global_value=true) == \"true\" %}\n                    {% if page.higher %}\n                        {% set left_link = page.higher.permalink %}\n                        {% set left_label = prev_label %}\n                        {% set left_title = page.higher.title %}\n                    {% endif %}\n                    {% if page.lower %}\n                        {% set right_link = page.lower.permalink %}\n                        {% set right_label = next_label %}\n                        {% set right_title = page.lower.title %}\n                    {% endif %}\n                {% else %}\n                    {% if page.lower %}\n                        {% set left_link = page.lower.permalink %}\n                        {% set left_label = next_label %}\n                        {% set left_title = page.lower.title %}\n                    {% endif %}\n                    {% if page.higher %}\n                        {% set right_link = page.higher.permalink %}\n                        {% set right_label = prev_label %}\n                        {% set right_title = page.higher.title %}\n                    {% endif %}\n                {% endif %}\n                {% if macros_settings::evaluate_setting_priority(setting=\"previous_next_article_links_full_width\", page=page, default_global_value=true) == \"true\" %}\n                    {%- set full_width_class = \"full-width\" -%}\n                {% endif %}\n            <nav class=\"{{ full_width_class | default(value=\"\") }} article-navigation\">\n                <div>\n                {%- if left_link and left_label and left_title -%}\n                <a href=\"{{ left_link | safe }}\" aria-label=\"{{ left_label }}\" aria-describedby=\"left_title\"><span class=\"arrow\">←</span>&nbsp;{{ left_label }}</a>\n                <p aria-hidden=\"true\" id=\"left_title\">{{ left_title | truncate(length=100, end=\"…\") | markdown(inline=true) | safe }}</p>\n                {%- endif -%}\n                </div>\n                <div>\n                {%- if right_link and right_label and right_title -%}\n                <a href=\"{{ right_link | safe }}\" aria-label=\"{{ right_label }}\" aria-describedby=\"right_title\">{{ right_label }}&nbsp;<span class=\"arrow\">→</span></a>\n                <p aria-hidden=\"true\" id=\"right_title\">{{ right_title | truncate(length=100, end=\"…\") | markdown(inline=true) | safe }}</p>\n                {%- endif -%}\n                </div>\n            </nav>\n            {%- endif -%}\n        {%- endif -%}\n\n        {#- Comments -#}\n        {#- Check if comments are enabled, checking that they are not disabled on the specific page -#}\n        {% set systems = [\"giscus\", \"utterances\", \"hyvortalk\", \"isso\"] %}\n        {% set enabled_systems = 0 %}\n        {% set comment_system = \"\" %}\n\n        {% for system in systems %}\n            {% set global_enabled = config.extra[system].enabled_for_all_posts | default(value=false) %}\n            {% set page_enabled = page.extra[system] | default(value=global_enabled) %}\n            {% set is_enabled = global_enabled and page_enabled != false or page_enabled == true %}\n\n            {% if is_enabled %}\n                {% set_global comment_system = system %}\n                {% set_global enabled_systems = enabled_systems + 1 %}\n            {% endif %}\n        {% endfor %}\n        {#- Ensure only one comment system is enabled -#}\n        {% if enabled_systems > 1 %}\n            {{ throw(message=\"ERROR: Multiple comment systems have been enabled for the same page. Check your config.toml and individual page settings to ensure only one comment system is activated at a time.\") }}\n        {% endif %}\n        {% if comment_system %}\n            {% include \"partials/comments.html\" %}\n        {% endif %}\n\n        {#- Webmentions -#}\n        {%- set global_webmentions_enabled = config.extra.webmentions.enable | default(value=false) -%}\n        {%- set page_webmentions_enabled = page.extra.webmentions | default(value=global_webmentions_enabled) -%}\n        {%- set webmentions_enabled = global_webmentions_enabled and page_webmentions_enabled != false or page_webmentions_enabled == true -%}\n        {%- if webmentions_enabled -%}\n            {%- include \"partials/webmentions.html\" -%}\n        {%- endif -%}\n\n    </article>\n</main>\n\n{%- include \"partials/extra_features.html\" -%}\n\n{% endblock main_content %}\n"
  },
  {
    "path": "templates/partials/analytics.html",
    "content": "{% set analytics_service = config.extra.analytics.service %}\n{% set analytics_id = config.extra.analytics.id | default(value=\"\") %}\n{% set self_hosted_url = config.extra.analytics.self_hosted_url | default(value=\"\") %}\n\n{% if analytics_service == \"goatcounter\" %}\n    {# Prevent non-demo sites from using the demo analytics account #}\n    {% if self_hosted_url == \"https://tabi-stats.osc.garden\" and config.base_url == \"https://welpo.github.io/tabi\" or self_hosted_url != \"https://tabi-stats.osc.garden\" %}\n    <script async\n    {% if self_hosted_url %}\n        data-goatcounter=\"{{ self_hosted_url ~ '/count' }}\"\n        src=\"{{ self_hosted_url ~ '/count.js' }}\"\n    {% else %}\n        data-goatcounter=\"https://{{ analytics_id }}.goatcounter.com/count\"\n        src=\"https://gc.zgo.at/count.js\"\n        {% endif %}\n    ></script>\n    {% endif %}\n\n{% elif analytics_service == \"umami\" %}\n    <script async defer\n    {% if self_hosted_url %}\n        data-website-id=\"{{ analytics_id }}\"\n        src=\"{{ self_hosted_url ~ '/script.js' }}\"\n    {% else %}\n        data-website-id=\"{{ analytics_id }}\"\n        src=\"https://cloud.umami.is/script.js\"\n    {% endif %}\n    {% if config.extra.analytics.do_not_track %}data-do-not-track=\"true\"{% endif %}>\n    </script>\n\n{% elif analytics_service == \"plausible\" %}\n{% if analytics_id is matching(\"^pa-[^\\.]+$\") %}\n    {# Use new script available in Plausible 3.1.0 or later #}\n    <script defer src=\"{{ get_url(path='js/initializePlausible.min.js', trailing_slash=false) | safe }}\"></script>\n    <script defer src=\"{% if self_hosted_url %}{{ self_hosted_url ~ '/js/' }}{% else %}https://plausible.io/js/{% endif %}{{ analytics_id }}.js\"></script>\n{% else %}\n    <script\n        defer\n        data-domain=\"{{ analytics_id }}\"\n        src=\"{% if self_hosted_url %}{{ self_hosted_url ~ '/js/plausible.js' }}{% else %}https://plausible.io/js/script.js{% endif %}\"\n    ></script>\n{% endif %}\n\n{% endif %}\n"
  },
  {
    "path": "templates/partials/cards_pages.html",
    "content": "{%- set rel_attributes = macros_rel_attributes::rel_attributes() | trim -%}\n{%- set max_projects = max_projects | default(value=999999) -%}\n<div class=\"cards\">\n    {%- for page in show_pages %}\n        {# Used only in main page #}\n        {% if loop.index > max_projects %}\n            {% break %}\n        {% endif %}\n        {# Determine which URL to use, default is page.permalink #}\n        {%- set blank_target = macros_target_attribute::target_attribute(new_tab=config.markdown.external_links_target_blank and page.extra.link_to) -%}\n\n        {% set target_url = page.extra.link_to | default(value=page.permalink) %}\n\n        <a rel=\"{{ rel_attributes }}\"\n            {{ blank_target }}\n            href=\"{{ target_url }}\"\n            class=\"card\"\n            {% if page.taxonomies %}\n            data-tags=\"{% for tax_name, terms in page.taxonomies %}{% for term in terms | unique %}{{ term | lower }}{% if not loop.last %},{% endif %}{% endfor %}{% endfor %}\"\n            {% endif %}>\n            {% if page.extra.invertible_image %}{% set card_image_class = \"card-image invertible-image\" %}{% else %}{% set card_image_class = \"card-image\" %}{% endif %}\n            {% if page.extra.local_image_dark and not page.extra.local_image %}\n                {{ throw(message=\"ERROR: `local_image_dark` requires `local_image` to be set in \" ~ page.path) }}\n            {% endif %}\n            {% if page.extra.remote_image_dark and not page.extra.remote_image %}\n                {{ throw(message=\"ERROR: `remote_image_dark` requires `remote_image` to be set in \" ~ page.path) }}\n            {% endif %}\n            {% if page.extra.local_image %}\n                {% if page.extra.local_image_dark %}\n                    {# Show light image only in light mode, dark image only in dark mode #}\n                    {% set meta_dark = get_image_metadata(path=page.extra.local_image_dark, allow_missing=true) %}\n                    <img class=\"card-image img-dark\"\n                        alt=\"{{ page.extra.local_image_dark }}\"\n                        src=\"{{ get_url(path=page.extra.local_image_dark) }}\"\n                        {% if meta_dark.width %}width=\"{{ meta_dark.width }}\"{% endif %}\n                        {% if meta_dark.height %}height=\"{{ meta_dark.height }}\"{% endif %}>\n                {% endif %}\n                {% set meta = get_image_metadata(path=page.extra.local_image, allow_missing=true) %}\n                <img class=\"{{ card_image_class }}{% if page.extra.local_image_dark %} img-light{% endif %}\"\n                    alt=\"{{ page.extra.local_image }}\"\n                    src=\"{{ get_url(path=page.extra.local_image) }}\"\n                    {% if meta.width %}width=\"{{ meta.width }}\"{% endif %}\n                    {% if meta.height %}height=\"{{ meta.height }}\"{% endif %}>\n            {% elif page.extra.remote_image %}\n                {% if page.extra.remote_image_dark %}\n                    {# Show light image only in light mode, dark image only in dark mode #}\n                    {% set meta_dark = get_image_metadata(path=page.extra.remote_image_dark, allow_missing=true) %}\n                    <img class=\"card-image img-dark\"\n                        alt=\"{{ page.extra.remote_image_dark }}\"\n                        src=\"{{ page.extra.remote_image_dark }}\">\n                {% endif %}\n                {% set meta = get_image_metadata(path=page.extra.remote_image, allow_missing=true) %}\n                <img class=\"{{ card_image_class }}{% if page.extra.remote_image_dark %} img-light{% endif %}\"\n                    alt=\"{{ page.extra.remote_image }}\"\n                    src=\"{{ page.extra.remote_image }}\">\n            {% else %}\n                <div class=\"card-image-placeholder\"></div>\n            {% endif %}\n\n            <div class=\"card-info\">\n                <h2 class=\"card-title\">{{ page.title | markdown(inline=true) | safe }}</h2>\n                <div class=\"card-description\">\n                    {% if page.description %}\n                        {{ page.description | markdown(inline=true) | safe }}\n                    {% endif %}\n                </div>\n            </div>\n        </a>\n    {% endfor -%}\n</div>\n"
  },
  {
    "path": "templates/partials/comments.html",
    "content": "{% set automatic_loading = config.extra[comment_system].automatic_loading %}\n\n<div id=\"comments\" class=\"comments\"\n\n{% if comment_system == \"giscus\" %}\n    data-repo=\"{{ config.extra.giscus.repo }}\"\n    data-repo-id=\"{{ config.extra.giscus.repo_id }}\"\n    data-category=\"{{ config.extra.giscus.category }}\"\n    data-category-id=\"{{ config.extra.giscus.category_id }}\"\n    {% if config.extra.giscus.mapping == \"slug\" %}\n        data-mapping=\"specific\"\n        data-term=\"{{ page.slug }}\"\n    {% else %}\n        data-mapping=\"{{ config.extra.giscus.mapping }}\"\n    {% endif %}\n    data-strict=\"{{ config.extra.giscus.strict_title_matching }}\"\n    data-reactions-enabled=\"{{ config.extra.giscus.enable_reactions }}\"\n    {% if config.extra.giscus.comment_box_above_comments %}\n        data-input-position=\"top\"\n    {% else %}\n        data-input-position=\"bottom\"\n    {% endif %}\n    data-light-theme=\"{{ config.extra.giscus.light_theme }}\"\n    data-dark-theme=\"{{ config.extra.giscus.dark_theme }}\"\n    {% if config.extra.giscus.lang  %}\n        data-lang=\"{{ config.extra.giscus.lang }}\"\n    {% else %}\n        data-lang=\"{{ lang }}\"\n    {% endif %}\n    data-lazy-loading=\"{{ config.extra.giscus.lazy_loading }}\"\n    >\n\n{% elif comment_system == \"utterances\" %}\n    data-repo=\"{{ config.extra.utterances.repo }}\"\n    {% if config.extra.utterances.issue_term == \"slug\" %}\n        data-issue-term=\"{{ page.slug }}\"\n    {% else %}\n        data-issue-term=\"{{ config.extra.utterances.issue_term }}\"\n    {% endif %}\n    data-label=\"{{ config.extra.utterances.label }}\"\n    data-light-theme=\"{{ config.extra.utterances.light_theme }}\"\n    data-dark-theme=\"{{ config.extra.utterances.dark_theme }}\"\n    data-lazy-loading=\"{{ config.extra.utterances.lazy_loading }}\"\n    >\n\n{% elif comment_system == \"hyvortalk\" %}\n    data-website-id=\"{{ config.extra.hyvortalk.website_id }}\"\n    {% if config.extra.hyvortalk.page_id_is_slug %}\n        data-page-id=\"{{ page.slug }}\"\n    {% else %}\n        data-page-id=\"{{ current_url }}\"\n    {% endif %}\n    {% if config.extra.hyvortalk.lang  %}\n        data-page-language=\"{{ config.extra.hyvortalk.lang }}\"\n    {% else %}\n        data-page-language=\"{{ lang }}\"\n    {% endif %}\n    data-page-author=\"{{ config.extra.hyvortalk.page_author }}\"\n    {% if config.extra.hyvortalk.lazy_loading  %}\n        data-loading=\"lazy\"\n    {% else %}\n        data-loading=\"default\"\n    {% endif %}\n    >\n\n{% elif comment_system == \"isso\" %}\n    data-endpoint-url=\"{{ config.extra.isso.endpoint_url }}\"\n    {% if config.extra.isso.page_id_is_slug %}\n    {%- set default_lang_url = current_path | replace(from='/' ~ lang ~ '/', to = '/') -%}\n        data-isso-id=\"{{ default_lang_url }}\"\n        data-title=\"{{ default_lang_url }}\"\n    {% endif %}\n    {% if config.extra.isso.lang  %}\n        data-page-language=\"{{ config.extra.isso.lang }}\"\n    {% else %}\n        data-page-language=\"{{ lang }}\"\n    {% endif %}\n    data-max-comments-top=\"{{ config.extra.isso.max_comments_top }}\"\n    data-max-comments-nested=\"{{ config.extra.isso.max_comments_nested }}\"\n    data-avatar=\"{{ config.extra.isso.avatar }}\"\n    data-voting=\"{{ config.extra.isso.voting }}\"\n    data-page-author-hashes=\"{{ config.extra.isso.page_author_hashes }}\"\n    data-lazy-loading=\"{{ config.extra.isso.lazy_loading }}\"\n    >\n{% endif %}\n\n{% if automatic_loading %}\n    <script src=\"{{ get_url(path='js/' ~ comment_system ~ '.min.js', trailing_slash=false) | safe }}\" async></script>\n{% else %}\n    <button id=\"load-comments\" class=\"load-comments-button\" data-script-src=\"{{ get_url(path='js/' ~ comment_system ~ '.min.js', trailing_slash=false) | safe }}\">{{ macros_translate::translate(key=\"load_comments\", default=\"Load comments\", language_strings=language_strings) }}</button>\n    <script src=\"{{ get_url(path='js/loadComments.min.js', trailing_slash=false) | safe }}\" async></script>\n{% endif %}\n\n<noscript>You need JavaScript to view the comments.</noscript>\n</div>\n"
  },
  {
    "path": "templates/partials/content_security_policy.html",
    "content": "<meta http-equiv=\"Content-Security-Policy\"\ncontent=\"default-src 'self'\n{%- if config.extra.allowed_domains -%}\n;\n    {#- Check if a comment system is enabled to allow the necessary domains and directives -#}\n    {%- set utterances_enabled = config.extra.utterances.enabled_for_all_posts or page.extra.utterances -%}\n    {%- set giscus_enabled = config.extra.giscus.enabled_for_all_posts or page.extra.giscus -%}\n    {%- set hyvortalk_enabled = config.extra.hyvortalk.enabled_for_all_posts or page.extra.hyvortalk -%}\n    {%- set isso_enabled = config.extra.isso.enabled_for_all_posts or page.extra.isso -%}\n    {%- if page -%}\n        {%- set iine_enabled = macros_settings::evaluate_setting_priority(setting=\"iine\", page=page, default_global_value=false) == \"true\" -%}\n    {%- endif -%}\n    {%- if page -%}\n        {%- set mermaid_enabled = macros_settings::evaluate_setting_priority(setting=\"mermaid\", page=page, default_global_value=false) == \"true\" -%}\n    {%- endif -%}\n    {%- set serve_local_mermaid = config.extra.serve_local_mermaid | default(value=true) -%}\n\n    {#- Initialise a base script-src directive -#}\n    {%- set script_src = \"script-src 'self'\" -%}\n\n    {#- Initialise a base connect-src directive -#}\n    {%- set connect_src = \"connect-src 'self'\" -%}\n\n    {# Base logic for appending analytics domains #}\n    {%- set analytics_url = config.extra.analytics.self_hosted_url | default(value=\"\") %}\n    {%- if analytics_url -%}\n        {%- set script_src = script_src ~ \" \" ~ analytics_url -%}\n        {%- set connect_src = connect_src ~ \" \" ~ analytics_url -%}\n    {%- else -%}\n        {%- if config.extra.analytics.service -%}\n            {%- if config.extra.analytics.service == \"goatcounter\" -%}\n                {%- set script_src = script_src ~ \" gc.zgo.at\" -%}\n                {%- set connect_src = connect_src ~ \" \" ~ config.extra.analytics.id ~ \".goatcounter.com/count\" -%}\n            {%- elif config.extra.analytics.service == \"umami\" -%}\n                {%- set script_src = script_src ~ \" cloud.umami.is\" -%}\n                {%- set connect_src = connect_src ~ \" *.umami.dev\" ~ \" cloud.umami.is\" -%}\n            {%- elif config.extra.analytics.service == \"plausible\" -%}\n                {%- set script_src = script_src ~ \" plausible.io\" -%}\n                {%- set connect_src = connect_src ~ \" plausible.io\" -%}\n            {%- endif -%}\n        {%- endif -%}\n    {%- endif -%}\n\n    {%- if hyvortalk_enabled -%}\n        {%- set connect_src = connect_src ~ \" talk.hyvor.com\" -%}\n        {%- set script_src = script_src ~ \" talk.hyvor.com\" -%}\n    {%- elif isso_enabled -%}\n        {%- set connect_src = connect_src ~ \" \" ~ config.extra.isso.endpoint_url -%}\n        {%- set script_src = script_src ~ \" \" ~ config.extra.isso.endpoint_url -%}\n    {%- elif giscus_enabled -%}\n        {%- set script_src = script_src ~ \" \" ~ \" giscus.app\" -%}\n    {%- elif utterances_enabled -%}\n        {%- set script_src = script_src ~ \" \" ~ \" utteranc.es\" -%}\n    {%- endif -%}\n\n    {%- if (mermaid_enabled and not serve_local_mermaid) or iine_enabled -%}\n        {%- set script_src = script_src ~ \" \" ~ \" cdn.jsdelivr.net\" -%}\n    {%- endif -%}\n\n    {#- Check if a webmention system is enabled to allow the necessary domains and directives -#}\n    {%- if config.extra.webmentions.enable -%}\n        {%- set connect_src = connect_src ~ \" webmention.io\" -%}\n    {%- endif -%}\n\n    {#- Check if iine like buttons are enabled to allow the necessary domains -#}\n    {%- if iine_enabled -%}\n        {%- set connect_src = connect_src ~ \" vhiweeypifbwacashxjz.supabase.co\" -%}\n    {%- endif -%}\n\n    {#- Append WebSocket for Zola serve mode -#}\n    {%- if config.mode == \"serve\" -%}\n        {%- set connect_src = connect_src ~ \" ws:\" -%}\n    {%- endif -%}\n\n    {%- for domain in config.extra.allowed_domains -%}\n        {%- if domain.directive == \"connect-src\" -%}\n            {%- set configured_connect_src = domain.domains | join(sep=' ') | safe -%}\n            {%- set_global connect_src = connect_src ~ \" \" ~ configured_connect_src  -%}\n            {%- continue -%}\n        {%- endif -%}\n\n        {%- if domain.directive == \"script-src\" -%}\n            {%- set configured_script_src = domain.domains | join(sep=' ') | safe -%}\n            {%- set_global script_src = script_src ~ \" \" ~ configured_script_src  -%}\n            {%- continue -%}\n        {%- endif -%}\n\n        {#- Handle directives that are not connect-src -#}\n        {{ domain.directive }} {{ domain.domains | join(sep=' ') | safe -}}\n\n        {%- if domain.directive == \"style-src\" -%}\n            {%- if utterances_enabled or hyvortalk_enabled or mermaid_enabled %} 'unsafe-inline'\n            {%- endif -%}\n        {%- endif -%}\n\n        {%- if domain.directive == \"font-src\" -%}\n            {%- if mermaid_enabled %} 'self'\n            {%- endif -%}\n        {%- endif -%}\n\n        {%- if domain.directive == \"frame-src\" -%}\n            {%- if giscus_enabled %} giscus.app\n            {%- elif utterances_enabled %} utteranc.es\n            {%- elif hyvortalk_enabled %} talk.hyvor.com\n            {%- endif %}\n        {%- endif -%}\n\n        {%- if not loop.last -%}\n        ;\n        {%- endif -%}\n    {%- endfor -%}\n\n    {#- Insert the generated connect-src -#}\n    {{ \";\" ~ connect_src }}\n\n    {#- Insert the generated script-src -#}\n    {{ \";\" ~ script_src }}\n\n{%- endif -%}\">\n"
  },
  {
    "path": "templates/partials/copyright.html",
    "content": "{%- if config.extra.copyright -%}\n    {% set copyright = config.extra.copyright %}\n    {# Try to look for a language-specific copyright notice in the new config setup #}\n    {%- if config.extra.copyright_translations -%}\n        {%- if lang in config.extra.copyright_translations -%}\n            {% set copyright = config.extra.copyright_translations[lang] %}\n        {%- endif -%}\n    {%- elif config.extra.translate_copyright -%}\n        {# Old way to translate the copyright through toml files #}\n        {{ throw(message=\"ERROR: The 'translate_copyright' feature has been deprecated. Please set 'copyright_translations' in config.toml. See the documentation: https://welpo.github.io/tabi/blog/mastering-tabi-settings/#copyright\") }}\n    {%- endif -%}\n\n    {# Check for missing variables in the notice #}\n    {% set copyright_placeholders = [\"$AUTHOR\", \"$TITLE\"] %}\n    {% set missing_vars = [] %}\n    {% for placeholder in copyright_placeholders %}\n        {% if placeholder in copyright %}\n            {# Attempt to retrieve the corresponding variable by trimming the $ sign and converting to lowercase #}\n            {% set var_name = placeholder | replace(from=\"$\", to=\"\") | lower %}\n            {% if not config[var_name] %}\n                {# Append the variable name to the list of missing variables #}\n                {% set_global missing_vars = missing_vars | concat(with=var_name) %}\n            {% endif %}\n        {% endif %}\n    {% endfor %}\n\n    {% if missing_vars | length > 0 %}\n        {% set missing_vars_str = missing_vars | join(sep=\", \") %}\n        {{ throw(message=\"ERROR: The following variables are included in `copyright` but have not been set in the config.toml: \" ~ missing_vars_str) }}\n    {% endif %}\n\n    {# At this point, we know that all variables needed defined, so we can safely override the missing ones #}\n    {% set author = config.author | default(value=\"\") %}\n    {% set title = config.title | default(value=\"\") %}\n\n    {# Render the copyright notice, replacing the variables #}\n    {% set current_year = now() | date(format=\"%Y\") %}\n    <p>{{ copyright | replace(from=\"$AUTHOR\", to=author) | replace(from=\"$TITLE\", to=title) | replace(from=\"$CURRENT_YEAR\", to=current_year) | replace(from=\"$SEPARATOR\", to=separator) | markdown | safe }}</p>\n{%- endif -%}\n"
  },
  {
    "path": "templates/partials/extra_features.html",
    "content": "{%- set page_or_section = page | default(value=section) -%}\n\n{# prepare parameters for evaluate_setting_priority macro #}\n{%- set page_s = page | default(value=\"\") -%}\n{%- set section_s = section | default(value=\"\") -%}\n\n{# Quick navigation buttons #}\n{% if macros_settings::evaluate_setting_priority(setting=\"quick_navigation_buttons\", page=page_s, section=section_s, default_global_value=false) == \"true\" %}\n    <div id=\"button-container\">\n        {# Button to go show a floating Table of Contents #}\n        {% if page_or_section.toc %}\n            <div id=\"toc-floating-container\">\n                <input type=\"checkbox\" id=\"toc-toggle\" class=\"toggle\"/>\n                <label for=\"toc-toggle\" id=\"toc-button\" class=\"button\" title=\"{{ macros_translate::translate(key=\"toggle_toc\", default=\"Toggle Table of Contents\", language_strings=language_strings) }}\">\n                    <svg xmlns=\"http://www.w3.org/2000/svg\" viewBox=\"0 -960 960 960\"><path d=\"M414.82-193.094q-18.044 0-30.497-12.32-12.453-12.319-12.453-30.036t12.453-30.086q12.453-12.37 30.497-12.37h392.767q17.237 0 29.927 12.487 12.69 12.486 12.69 30.203 0 17.716-12.69 29.919t-29.927 12.203H414.82Zm0-244.833q-18.044 0-30.497-12.487Q371.87-462.9 371.87-480.45t12.453-29.92q12.453-12.369 30.497-12.369h392.767q17.237 0 29.927 12.511 12.69 12.512 12.69 29.845 0 17.716-12.69 30.086-12.69 12.37-29.927 12.37H414.82Zm0-245.167q-18.044 0-30.497-12.32t-12.453-30.037q0-17.716 12.453-30.086 12.453-12.369 30.497-12.369h392.767q17.237 0 29.927 12.486 12.69 12.487 12.69 30.203 0 17.717-12.69 29.92-12.69 12.203-29.927 12.203H414.82ZM189.379-156.681q-32.652 0-55.878-22.829t-23.226-55.731q0-32.549 23.15-55.647 23.151-23.097 55.95-23.097 32.799 0 55.313 23.484 22.515 23.484 22.515 56.246 0 32.212-22.861 54.893-22.861 22.681-54.963 22.681Zm0-245.167q-32.652 0-55.878-23.134-23.226-23.135-23.226-55.623 0-32.487 23.467-55.517t56.12-23.03q32.102 0 54.721 23.288 22.62 23.288 22.62 55.775 0 32.488-22.861 55.364-22.861 22.877-54.963 22.877Zm-.82-244.833q-32.224 0-55.254-23.288-23.03-23.289-23.03-55.623 0-32.333 23.271-55.364 23.272-23.03 55.495-23.03 32.224 0 55.193 23.288 22.969 23.289 22.969 55.622 0 32.334-23.21 55.364-23.21 23.031-55.434 23.031Z\"/></svg>\n                </label>\n                <div class=\"toc-content\">\n                    {{ macros_toc::toc(page=page_or_section, header=false, language_strings=language_strings) }}\n                </div>\n            </div>\n        {% endif %}\n\n        {# Button to go to the comment/webmentions section #}\n        {% if comment_system or config.extra.webmentions.enable %}\n            {%- if comment_system -%}\n                {#- Comments are shown above webmentions -#}\n                {%- set comments_id = \"comments\" -%}\n            {%- else -%}\n                {%- set comments_id = \"webmentions\" -%}\n            {%- endif -%}\n\n            <a href=\"#{{- comments_id -}}\" id=\"comments-button\" class=\"no-hover-padding\" title=\"{{ macros_translate::translate(key=\"go_to_comments\", default=\"Go to comments section\", language_strings=language_strings) }}\">\n                <svg viewBox=\"0 0 20 20\" fill=\"currentColor\"><path d=\"M18 10c0 3.866-3.582 7-8 7a8.841 8.841 0 01-4.083-.98L2 17l1.338-3.123C2.493 12.767 2 11.434 2 10c0-3.866 3.582-7 8-7s8 3.134 8 7zM7 9H5v2h2V9zm8 0h-2v2h2V9zM9 9h2v2H9V9z\" clip-rule=\"evenodd\" fill-rule=\"evenodd\"/></svg>\n            </a>\n        {% endif %}\n\n        {# Button to go to the top of the page #}\n        <a href=\"#\" id=\"top-button\" class=\"no-hover-padding\" title=\"{{ macros_translate::translate(key=\"go_to_top\", default=\"Go to the top of the page\", language_strings=language_strings) }}\">\n            <svg viewBox=\"0 0 20 20\" fill=\"currentColor\"><path d=\"M3.293 9.707a1 1 0 010-1.414l6-6a1 1 0 011.414 0l6 6a1 1 0 01-1.414 1.414L11 5.414V17a1 1 0 11-2 0V5.414L4.707 9.707a1 1 0 01-1.414 0z\"/></svg>\n        </a>\n    </div>\n{% endif %}\n\n{# Add KaTeX functionality #}\n{%- if macros_settings::evaluate_setting_priority(setting=\"katex\", page=page_s, section=section_s, default_global_value=false) == \"true\" -%}\n    <link rel=\"stylesheet\" href=\"{{ get_url(path='katex.min.css', trailing_slash=false) | safe }}\">\n    <script defer src=\"{{ get_url(path='js/katex.min.js', trailing_slash=false) | safe }}\"></script>\n    {# Load mhchem extension if chemistry formulas detected #}\n    {%- if page and page.content and (\"\\ce{\" in page.content or \"\\pu{\" in page.content) -%}\n        <script defer src=\"{{ get_url(path='js/mhchem.min.js', trailing_slash=false) | safe }}\"></script>\n    {%- endif -%}\n{%- endif -%}\n\n{# Load mermaid.js #}\n{%- if macros_settings::evaluate_setting_priority(setting=\"mermaid\", page=page_s, section=section_s, default_global_value=false) == \"true\" -%}\n    {%- if config.extra.serve_local_mermaid | default(value=true) -%}\n        <script defer src=\"{{ get_url(path='js/mermaid.min.js', trailing_slash=false) | safe }}\"></script>\n    {%- else -%}\n        <script defer src=\"https://cdn.jsdelivr.net/npm/mermaid@latest/dist/mermaid.min.js\"></script>\n    {%- endif -%}\n{%- endif -%}\n\n{# Add copy button to code blocks #}\n{%- if macros_settings::evaluate_setting_priority(setting=\"copy_button\", page=page_s, section=section_s, default_global_value=true) == \"true\" -%}\n    {#- Add hidden HTML elements with the translated strings for the button's interactions -#}\n    <span id=\"copy-success\" class=\"hidden\">\n        {{ macros_translate::translate(key=\"copied\", default=\"Copied!\", language_strings=language_strings) }}\n    </span>\n    <span id=\"copy-init\" class=\"hidden\">\n        {{ macros_translate::translate(key=\"copy_code_to_clipboard\", default=\"Copy code to clipboard\", language_strings=language_strings) }}\n    </span>\n    <script defer src=\"{{ get_url(path='js/copyCodeToClipboard.min.js', trailing_slash=false) | safe }}\"></script>\n{%- endif -%}\n\n{# JavaScript to make code block names clickable when they are URLs: https://welpo.github.io/tabi/blog/shortcodes/#show-source-or-path #}\n{# Note: \"add_src_to_code_block\" is deprecated in favor of \"code_block_name_links\". It will be removed in a future release. #}\n{# See https://github.com/welpo/tabi/pull/489 #}\n{%- if macros_settings::evaluate_setting_priority(setting=\"code_block_name_links\", page=page_s, section=section_s, default_global_value=false) == \"true\"\n    or macros_settings::evaluate_setting_priority(setting=\"add_src_to_code_block\", page=page_s, section=section_s, default_global_value=false) == \"true\" -%}\n    <script defer src=\"{{ get_url(path='js/codeBlockNameLinks.min.js', trailing_slash=false) | safe }}\"></script>\n{%- endif -%}\n\n{# Add backlinks to footnotes #}\n{%- if macros_settings::evaluate_setting_priority(setting=\"footnote_backlinks\", page=page_s, section=section_s, default_global_value=false) == \"true\" -%}\n    <script defer src=\"{{ get_url(path='js/footnoteBacklinks.min.js', trailing_slash=false | safe )}}\"></script>\n{%- endif -%}\n\n{# Add iine.js for the like button #}\n{%- if page and macros_settings::evaluate_setting_priority(setting=\"iine\", page=page_s, section=section_s, default_global_value=false) == \"true\" -%}\n    <script defer src=\"https://cdn.jsdelivr.net/gh/welpo/iine@main/iine.mini.js\"></script>\n{%- endif -%}\n"
  },
  {
    "path": "templates/partials/filter_card_tags.html",
    "content": "{#- Collect all terms. -#}\n{#- We don't use `get_taxonomy` so users aren't forced to use 'tags' -#}\n{% set all_terms = [] %}\n{% for page in show_pages %}\n    {% if page.taxonomies %}\n        {% for tax_name, terms in page.taxonomies %}\n            {% for term in terms %}\n                {% set_global all_terms = all_terms | concat(with=term) %}\n            {% endfor %}\n        {% endfor %}\n    {% endif %}\n{% endfor %}\n\n{#- Display unique terms -#}\n{% set unique_terms = all_terms | unique | sort %}\n{%- if unique_terms | length >= 2 -%}\n    <ul class=\"filter-controls\" role=\"group\" aria-label=\"{{ macros_translate::translate(key='project_filters', default='Project filters', language_strings=language_strings) }}\">\n        <li class=\"taxonomy-item no-hover-padding\">\n            <a id=\"all-projects-filter\" class=\"no-hover-padding active\"\n            href=\"{{ get_url(path=\"projects\", lang=lang) }}\"\n            data-filter=\"all\">\n                {{- macros_translate::translate(key=\"all_projects\", default=\"All projects\", language_strings=language_strings) -}}\n            </a>\n        </li>\n        {% for term in unique_terms %}\n            <li class=\"taxonomy-item no-hover-padding\">\n                <a class=\"no-hover-padding\"\n                href=\"{{ get_taxonomy_url(kind=\"tags\", name=term, lang=lang) }}\"\n                data-filter=\"{{ term | lower }}\">{{ term }}</a>\n            </li>\n        {% endfor %}\n    </ul>\n    {#- Load the script -#}\n    <script src=\"{{ get_url(path='js/filterCards.min.js', trailing_slash=false, cachebust=true) | safe }}\" defer></script>\n{% endif %}\n"
  },
  {
    "path": "templates/partials/footer.html",
    "content": "{%- set separator = config.extra.separator | default(value=\"•\") -%}\n\n{%- set rel_attributes = macros_rel_attributes::rel_attributes() | trim -%}\n\n{%- set blank_target = macros_target_attribute::target_attribute(new_tab=config.markdown.external_links_target_blank) -%}\n\n{#- Feed icon -#}\n{%- set feed_url = feed_utils::get_feed_url() -%}\n{%- set feed_icon_name = feed_utils::get_feed_icon_name() -%}\n{%- set should_show_feed = feed_utils::should_show_footer_feed_icon() == \"true\" -%}\n\n{%- set should_show_footer_icons = should_show_feed or config.extra.socials or config.extra.email -%}\n\n<footer>\n    <section>\n        <nav class=\"socials nav-navs\">\n            {%- if should_show_footer_icons -%}\n                <ul>\n                    {%- if should_show_feed -%}\n                    <li>\n                        <a class=\"nav-links no-hover-padding social\" rel=\"{{ rel_attributes }}\" {{ blank_target }} href=\"{{ get_url(path=feed_url, lang=lang, trailing_slash=false) | safe }}\">\n                        <img loading=\"lazy\" alt=\"feed\" title=\"feed\" src=\"{{ get_url(path='/social_icons/' ~ feed_icon_name ~ '.svg') }}\">\n                        </a>\n                    </li>\n                    {%- endif -%}\n\n                    {# Mail icon #}\n                    {%- if config.extra.email -%}\n                        {%- set email_already_encoded = (config.extra.email is not containing(\"@\")) -%}\n                        {%- set email_needs_decoding = email_already_encoded or config.extra.encode_plaintext_email -%}\n\n                        {%- if email_already_encoded -%}\n                            {%- set encoded_email = config.extra.email -%}\n                            {# Verify the pre-encoded e-mail is valid (i.e. contains an '@') #}\n                            {%- set decoded_email = encoded_email | base64_decode -%}\n                            {%- if '@' not in decoded_email -%}\n                                {{ throw(message=\"ERROR: The provided e-mail appears to be base64-encoded, but does not decode to a valid e-mail address.\")}}\n                            {%- endif -%}\n                        {%- elif config.extra.encode_plaintext_email -%}\n                            {%- set encoded_email = config.extra.email | base64_encode -%}\n                        {%- endif -%}\n\n                        <li class=\"{% if email_needs_decoding %}js{% endif %}\">\n                            {%- if email_needs_decoding -%}\n                                <a class=\"nav-links no-hover-padding social\" href=\"#\" data-encoded-email=\"{{ encoded_email | safe }}\">\n                            {%- else -%}\n                                <a class=\"nav-links no-hover-padding social\" href=\"mailto:{{ config.extra.email | safe }}\">\n                            {%- endif -%}\n                                <img loading=\"lazy\" alt=\"email\" title=\"email\" src=\"{{ get_url(path='social_icons/email.svg') }}\">\n                            </a>\n                        </li>\n                    {%- endif -%}\n\n                    {%- if config.extra.socials %}\n                        {% for social in config.extra.socials %}\n                            <li>\n                                <a class=\"nav-links no-hover-padding social\" rel=\"{{ rel_attributes }} me\" {{ blank_target }} href=\"{{ social.url | safe }}\">\n                                    <img loading=\"lazy\" alt=\"{{ social.name }}\" title=\"{{ social.name }}\" src=\"{{ get_url(path='social_icons/' ~ social.icon ~ '.svg') }}\">\n                                </a>\n                            </li>\n                        {% endfor %}\n                    {% endif %}\n                </ul>\n            {% endif %}\n        </nav>\n\n        {# Footer menu #}\n        <nav class=\"nav-navs\">\n            {%- if config.extra.footer_menu %}\n                <small>\n                    <ul>\n                        {% for menu in config.extra.footer_menu %}\n                        <li>\n                            {%- set trailing_slash = menu.trailing_slash | default(value=true) -%}\n\n                            {%- if menu.url == \"sitemap.xml\" -%}\n                                {%- set url = get_url(path=menu.url, trailing_slash=trailing_slash) -%}\n                            {%- elif menu.url is starting_with(\"http\") -%}\n                                {%- if menu.trailing_slash -%}\n                                    {%- set url = menu.url ~ \"/\" -%}\n                                {%- else -%}\n                                    {%- set url = menu.url -%}\n                                {%- endif -%}\n                            {%- else -%}\n                                {%- set url = get_url(path=menu.url, lang=lang, trailing_slash=trailing_slash) -%}\n                            {%- endif -%}\n\n                            <a class=\"nav-links no-hover-padding\" href=\"{{ url }}\">\n                                {{ macros_translate::translate(key=menu.name, default=menu.name, language_strings=language_strings) }}\n                            </a>\n                        </li>\n                    {% endfor %}\n                    </ul>\n                </small>\n        {% endif %}\n        </nav>\n\n        <div class=\"credits\">\n            <small>\n                {% include \"partials/copyright.html\" %}\n\n                {# Shows \"Powered by Zola & tabi\" notice #}\n                {{ macros_translate::translate(key=\"powered_by\", default=\"Powered by\", language_strings=language_strings) }}\n                <a rel=\"{{ rel_attributes }}\" {{ blank_target }} href=\"https://www.getzola.org\">Zola</a>\n                {{ macros_translate::translate(key=\"and\", default=\"&\", language_strings=language_strings) }}\n                <a rel=\"{{ rel_attributes }}\" {{ blank_target }} href=\"https://github.com/welpo/tabi\">tabi</a>\n\n                {# Shows link to remote repository #}\n                {%- if config.extra.remote_repository_url and config.extra.show_remote_source | default(value=true) -%}\n                {{ separator }}\n                    <a rel=\"{{ rel_attributes }}\" {{ blank_target }} href=\"{{ config.extra.remote_repository_url }}\">\n                        {{ macros_translate::translate(key=\"site_source\", default=\"Site source\", language_strings=language_strings) }}\n                    </a>\n                {%- endif -%}\n            </small>\n        </div>\n    </section>\n\n    {# Load the decoding script if email is encoded #}\n    {%- if email_needs_decoding -%}\n        <script src=\"{{ get_url(path='js/decodeMail.min.js') }}\" async></script>\n    {%- endif -%}\n\n    {# Modal structure for search #}\n    {%- if config.build_search_index -%}\n    {% include \"partials/search_modal.html\" %}\n    {%- endif -%}\n</footer>\n"
  },
  {
    "path": "templates/partials/hcard.html",
    "content": "{%- set hcard = config.extra.hcard -%}\n\n{% set full_name = config.author %}\n{% if hcard.full_name %}\n  {% set full_name = hcard.full_name %}\n{% endif %}\n\n{%- set homepage = config.base_url -%}\n{% if hcard.homepage %}\n  {%- set homepage = hcard.homepage -%}\n{% endif %}\n\n{% if hcard.enable %}\n<div class=\"h-card hidden\">\n  <div>\n    {%- if hcard.avatar -%}\n    <img\n      class=\"u-photo\"\n      src=\"{{ get_url(path=hcard.avatar, cachebust=true) }}\"\n      width=\"200\"\n      height=\"200\"\n      alt=\"{{ full_name }}\"\n    />\n    {%- endif -%}\n\n    <span class=\"p-name\" rel=\"me\">{{ full_name }}</span>\n\n    {% if hcard.p_nickname %}\n    ( <span class=\"p-nickname\">{{ hcard.p_nickname }}</span> )\n    {% endif %}\n  </div>\n\n  {% if hcard.biography %}\n  <p class=\"p-note\">{{ hcard.biography }}</p>\n  {% endif %}\n\n  {# links #}\n  <div>\n    {%- if hcard.with_mail and config.extra.email and not config.extra.encode_plaintext_email -%}\n    <span>\n      <a class=\"u-email\" href=\"mailto:{{ config.extra.email | safe }}\">email</a>\n    </span> -\n    {%- endif -%}\n\n    <span>\n      <a class=\"u-url u-id\" href=\"{{ homepage }}\">homepage</a>\n    </span> -\n\n    {%- if hcard.with_social_links and config.extra.socials %}\n      {% for social in config.extra.socials %}\n      <span>\n        <a class=\"p-url\" rel=\"me\" href=\"{{ social.url | safe }}\">{{ social.name }}</a>\n      </span> -\n      {% endfor %}\n    {% endif %}\n  </div>\n\n  {# additional properties #}\n  {% set dl_started = false %}\n  {% for key, value in hcard %}\n    {% if key not in ['enable', 'with_mail', 'with_social_links', 'homepage', 'full_name', 'avatar', 'biography', 'p_nickname'] %}\n      {% if not dl_started %}\n        <dl>\n        {% set_global dl_started = true %}\n      {% endif %}\n      <dt>{{ key | replace(from=\"p_\", to=\"\") | replace(from=\"u_\", to=\"\") | replace(from=\"dt_\", to=\"\") | replace(from=\"_\", to=\" \") | capitalize }}</dt>\n      <dd class=\"{{ key | replace(from=\"_\", to=\"-\") }}\">{{ value }}</dd>\n    {% endif %}\n  {% endfor %}\n  {% if dl_started %}\n    </dl>\n  {% endif %}\n\n  </div>\n{% endif %}\n"
  },
  {
    "path": "templates/partials/hcard_small.html",
    "content": "{%- set hcard = config.extra.hcard -%}\n\n{%- set full_name = config.author -%}\n{%- if hcard.full_name -%}\n  {%- set full_name = hcard.full_name -%}\n{%- endif -%}\n\n{%- set homepage = config.base_url -%}\n{%- if hcard.homepage -%}\n  {%- set homepage = hcard.homepage -%}\n{%- endif -%}\n\n{%- set icon_attr = \"\" -%}\n{%- if hcard.avatar -%}\n    {%- set icon_attr = \"author-icon\" -%}\n{%- endif -%}\n\n<span class=\"hidden p-author h-card\">\n<a rel=\"author\" href=\"{{ homepage }}\" class=\"u-url {{ icon_attr }}\" title=\"{{ full_name }}\">\n    {%- if hcard.avatar -%}\n    <img class=\"u-photo\" src=\"{{ get_url(path=hcard.avatar, cachebust=true) }}\" alt=\"{{ full_name }}\" />\n    {%- else -%}\n    {{ full_name }}\n    {%- endif -%}\n</a>\n</span>\n"
  },
  {
    "path": "templates/partials/header.html",
    "content": "<head>\n    <meta charset=\"UTF-8\">\n\n    {%- if macros_settings::evaluate_setting_priority(setting=\"enable_csp\", page=page | default(value=\"\"), section=section | default(value=\"\"), default_global_value=\"true\") == \"true\" -%}\n        {%- include \"partials/content_security_policy.html\" -%}\n    {%- endif -%}\n\n    <meta name=\"viewport\" content=\"width=device-width, initial-scale=1.0\">\n    <meta name=\"base\" content=\"{{ config.base_url | safe }}\">\n\n    {# Site title #}\n    <title>{%- include \"partials/title.html\" -%}</title>\n\n    {# Favicon #}\n    {% if config.extra.favicon %}\n        <link rel=\"icon\" type=\"image/png\" href=\"{{ get_url(path=config.extra.favicon) }}\"/>\n    {% endif %}\n    {% if config.extra.favicon_emoji %}\n        <link rel=icon href='data:image/svg+xml,<svg xmlns=\"http://www.w3.org/2000/svg\" viewBox=\"0 0 100 100\"><text y=\"50%\" x=\"50%\" dominant-baseline=\"central\" text-anchor=\"middle\" font-size=\"88\">{{ config.extra.favicon_emoji }}</text></svg>'>\n    {% endif %}\n\n    {# Feeds #}\n    {% if config.generate_feeds | default(value=config.generate_feed) %}\n        {% if config.feed_filenames %}\n            {# Zola 0.19 and newer #}\n            {% for feed in config.feed_filenames %}\n                {% if feed == \"atom.xml\" %}\n                    <link rel=\"alternate\" type=\"application/atom+xml\" title=\"{{ config.title | safe }} - Atom Feed\" href=\"{{ get_url(path=feed, trailing_slash=false) | safe }}\">\n                {% elif feed == \"rss.xml\" %}\n                    <link rel=\"alternate\" type=\"application/rss+xml\" title=\"{{ config.title | safe }} - RSS Feed\" href=\"{{ get_url(path=feed, trailing_slash=false) | safe }}\">\n                {% else %}\n                    <link rel=\"alternate\" href=\"{{ get_url(path=feed, trailing_slash=false) | safe }}\">\n                {% endif %}\n            {% endfor %}\n        {% else %}\n            {# Older Zola versions #}\n            {% set feed_url = config.feed_filename | default(value=\"atom.xml\") %}\n            <link rel=\"alternate\" type=\"application/atom+xml\" title=\"{{ config.title | safe }}\" href=\"{{ get_url(path=feed_url, trailing_slash=false) | safe }}\">\n        {% endif %}\n    {% endif %}\n\n    {# CSS #}\n    {# Load subset of glyphs for header. Avoids flashing issue in Firefox #}\n    {% if config.extra.enable_subset %}\n        {% if config.extra.custom_subset == true %}\n            <link rel=\"stylesheet\" href=\"{{ get_url(path=\"custom_subset.css\" , cachebust=true) }}\">\n        {% elif lang == 'en' %}\n            <link rel=\"stylesheet\" href=\"{{ get_url(path=\"inter_subset_en.css\", cachebust=true ) }}\">\n        {% elif lang == 'es' %}\n            <link rel=\"stylesheet\" href=\"{{ get_url(path=\"inter_subset_es.css\", cachebust=true ) }}\">\n        {% endif %}\n    {% endif %}\n\n    {# Define array of CSS files to load. main.css is always loaded. #}\n    {%- set stylesheets = [ \"main.css\" ] -%}\n\n    {# Load extra CSS files from config.toml #}\n    {%- if config.extra.stylesheets -%}\n        {%- set stylesheets = stylesheets | concat(with=config.extra.stylesheets) -%}\n    {%- endif -%}\n\n    {# Load extra CSS files from page metadata #}\n    {%- if page.extra.stylesheets -%}\n        {%- set stylesheets = stylesheets | concat(with=page.extra.stylesheets) -%}\n    {%- endif -%}\n\n    {# Load extra CSS for custom skin #}\n    {%- if config.extra.skin -%}\n        {%- set stylesheets = stylesheets | concat(with='skins/' ~ config.extra.skin ~ '.css') -%}\n    {%- endif -%}\n\n    {# Load all stylesheets #}\n    {%- for stylesheet in stylesheets %}\n        <link rel=\"stylesheet\" href=\"{{ get_url(path=stylesheet, cachebust=true) | safe }}\" />\n    {%- endfor %}\n\n    <meta name=\"color-scheme\" content=\"{%- if config.extra.theme_switcher -%}light dark{%- elif config.extra.default_theme -%}{{config.extra.default_theme}}{%- else -%}light{%- endif -%}\" />\n\n    {%- if config.extra.browser_theme_color and config.extra.browser_theme_color is iterable -%}\n        {# Handle array values: theme_color[0] for light mode, theme_color[1] for dark mode #}\n        <meta name=\"theme-color\" media=\"(prefers-color-scheme: light)\" content=\"{{ config.extra.browser_theme_color[0] }}\" />\n        <meta name=\"theme-color\" media=\"(prefers-color-scheme: dark)\" content=\"{{ config.extra.browser_theme_color[1] }}\" />\n    {%- elif config.extra.browser_theme_color -%}\n        {# Handle single value #}\n        <meta name=\"theme-color\" content=\"{{ config.extra.browser_theme_color }}\" />\n    {%- endif -%}\n\n    {%- if page.description %}\n        <meta name=\"description\" content=\"{{ page.description }}\" />\n        <meta property=\"og:description\" content=\"{{ page.description }}\" />\n    {%- elif section.description %}\n        <meta name=\"description\" content=\"{{ section.description }}\" />\n        <meta property=\"og:description\" content=\"{{ section.description }}\" />\n    {%- elif page.summary %}\n        <meta name=\"description\" content=\"{{ page.summary | striptags | trim_end_matches(pat=\".\") | safe }}…\" />\n        <meta property=\"og:description\" content=\"{{ page.summary | striptags | trim_end_matches(pat=\".\") | safe }}…\" />\n    {%- else %}\n        <meta name=\"description\" content=\"{{ config.description }}\" />\n        <meta property=\"og:description\" content=\"{{ config.description }}\" />\n    {%- endif %}\n\n    {% if is_404 %}\n        <meta name=\"robots\" content=\"noindex, follow\" />\n    {% endif %}\n\n    <meta property=\"og:title\" content=\"{{ page.title | default(value=config.title) | safe }}\" />\n    <meta property=\"og:type\" content=\"article\" />\n\n    {# Image for social media sharing #}\n    {%- include \"partials/social_media_images.html\" -%}\n\n    {# Add og:locale and hreflang tags for multilingual sites #}\n    {%- if config.languages | length > 0 and current_url %}\n        {%- include \"partials/multilingual_tags.html\" -%}\n    {%- else -%}\n        <meta property=\"og:locale\" content=\"{{ macros_translate::translate(key=\"date_locale\", default=\"en_GB\", language_strings=language_strings) }}\" />\n    {%- endif %}\n\n    {# Set canonical URL #}\n    {%- if current_url -%}\n        {%- if page.extra.canonical_url or section.extra.canonical_url -%}\n            {%- set canonical_url = page.extra.canonical_url | default(value=section.extra.canonical_url) -%}\n        {%- elif config.extra.base_canonical_url -%}\n            {%- set canonical_url = current_url | replace(from=config.base_url, to=config.extra.base_canonical_url) -%}\n        {%- endif -%}\n    {%- endif -%}\n\n    {# Add canonical URL, if set #}\n    {%- if canonical_url -%}\n        <link rel=\"canonical\" href=\"{{ canonical_url }}\" />\n        <meta property=\"og:url\" content=\"{{ canonical_url }}\" />\n    {%- elif current_url -%}\n        <meta property=\"og:url\" content=\"{{ current_url }}\" />\n    {%- endif -%}\n\n    <meta property=\"og:site_name\" content=\"{{ config.title }}\">\n\n    {%- if config.extra.theme_switcher and config.extra.theme_switcher == true -%}\n        {# If JavaScript is disabled, hide the button. #}\n        <noscript><link rel=\"stylesheet\" href=\"{{ get_url(path='no_js.css') | safe }}\"/></noscript>\n        <script type=\"text/javascript\" src=\"{{ get_url(path='js/initializeTheme.min.js') | safe }}\"></script>\n        <script defer src=\"{{ get_url(path='js/themeSwitcher.min.js', trailing_slash=false) | safe }}\"></script>\n    {%- endif -%}\n\n    {%- if config.extra.analytics.service -%}\n        {%- include \"partials/analytics.html\" -%}\n    {%- endif -%}\n\n    {# Socials #}\n    {%- if config.extra.fediverse_creator -%}\n        <meta name=\"fediverse:creator\" content=\"@{{ config.extra.fediverse_creator[\"handle\"] }}@{{ config.extra.fediverse_creator[\"domain\"]}}\" />\n    {%- endif -%}\n\n    {# Search #}\n    {%- if config.build_search_index -%}\n        {%- if config.search.index_format -%}\n            {%- set search_index_format = config.search.index_format -%}\n        {%- elif config.extra.index_format -%}\n            {# Necessary to support Zola 0.17.X, as it doesn't have access to config.search.index_format #}\n            {# See: https://github.com/getzola/zola/issues/2165 #}\n            {%- set search_index_format = config.extra.index_format -%}\n        {%- else -%}\n            {%- set search_index_format = \"elasticlunr_json\" -%}\n        {%- endif -%}\n\n        {%- if search_index_format == \"elasticlunr_javascript\" -%}\n            <script defer src=\"{{ get_url(path='search_index.' ~ lang ~ '.js', cachebust=true) | safe }}\"></script>\n        {%- endif -%}\n\n        {# Main search script #}\n        <script defer src=\"{{ get_url(path='js/searchElasticlunr.min.js', cachebust=true) | safe }}\"></script>\n\n        {# Support correct stemming and stop word filtering in non-English search #}\n        {%- if lang != \"en\" -%}\n            <script defer src=\"{{ get_url(path='js/lunr/lunrStemmerSupport.min.js') | safe }}\"></script>\n            {%- if lang == \"ja\" -%}\n                <script defer src=\"{{ get_url(path='js/lunr/tinyseg.min.js') | safe }}\"></script>\n            {%- endif -%}\n            <script defer src=\"{{ get_url(path='js/lunr/lunr.' ~ lang ~ '.min.js') | safe }}\"></script>\n        {%- endif -%}\n    {%- endif -%}\n\n    {# Users can optionally provide this template to add content to the head element. #}\n    {% include \"tabi/extend_head.html\" ignore missing %}\n</head>\n"
  },
  {
    "path": "templates/partials/history_url.html",
    "content": "{%- set relative_path = page.relative_path -%}\n{%- set repository_url = config.extra.remote_repository_url | trim_end_matches(pat='/') -%}\n{%- set branch = config.extra.remote_repository_branch | default(value=\"main\") -%}\n{%- set git_platform = config.extra.remote_repository_git_platform | default(value=\"auto\") -%}\n\n{# Auto-detect the git platform based on the URL#}\n{%- if git_platform == \"auto\" %}\n    {%- if repository_url is containing(\"github.\") -%}\n        {%- set git_platform = \"github\" -%}\n    {%- elif repository_url is containing(\"gitlab.\") -%}\n        {%- set git_platform = \"gitlab\" -%}\n    {%- elif repository_url is matching(\"(gitea\\.|codeberg\\.)\") -%}\n        {%- set git_platform = \"gitea\" -%}\n    {%- endif -%}\n{%- endif -%}\n\n{# Generate the commit history URL based on the git platform #}\n{%- if git_platform == \"github\" -%}\n    {{ repository_url ~ '/commits/' ~ branch ~ '/content/' }}{{ relative_path | urlencode }}\n{%- elif git_platform == \"gitlab\" -%}\n    {{ repository_url ~ '/-/commits/' ~ branch ~ '/content/' }}{{ relative_path | urlencode }}\n{%- elif git_platform in [\"gitea\", \"codeberg\"] -%}\n    {{ repository_url ~ '/commits/branch/' ~ branch ~ '/content/' }}{{ relative_path | urlencode }}\n{%- else -%}\n    {{ throw(message=\"ERROR: Unknown, unsupported, or unspecified git platform. If you're using a custom domain, please specify the 'git_platform' in the config. If you think this is a bug, please report it: https://github.com/welpo/tabi/issues/new?assignees=&labels=bug&template=bug_report.md&title=Unsupported%20Git%20Platform%20Detected\") }}\n{%- endif -%}\n"
  },
  {
    "path": "templates/partials/home_banner.html",
    "content": "{%- set header = section.extra.header-%}\n<div id=\"banner-container-home\">\n    <div id=\"home-banner-text\">\n        <h1 id=\"home-banner-header\">{{ header.title }}</h1>\n        <section id=\"banner-home-subtitle\">\n            {{ section.content | safe }}\n        </section>\n    </div>\n    {%- if header.img -%}\n        {%- if header.img is containing(\"$BASE_URL\") -%}\n            {# Conversion no longer supported in favour of proper path. #}\n            {{ throw(message=\"ERROR: The image path for the header should not contain '$BASE_URL'. Please remove it and use the proper image path.\") }}\n        {%- else -%}\n            {%- set image_path = get_url(path=header.img, trailing_slash=false) | safe -%}\n        {%- endif -%}\n        <div id=\"image-container-home\">\n            <img alt=\"{{ header.img_alt | default(value=\"the owner\") }}\" id=\"banner-home-img\" src=\"{{ image_path }}\" />\n        </div>\n    {%- endif -%}\n</div>\n"
  },
  {
    "path": "templates/partials/iine_button.html",
    "content": "{% import \"macros/settings.html\" as macros_settings %}\n{%- set button_icon = button_icon | default(value=macros_settings::evaluate_setting_priority(setting=\"iine_icon\", page=page | default(value=\"\"), section=section | default(value=\"\"), default_global_value=\"heart\")) -%}\n{%- if config.extra.iine_unified_languages and lang != config.default_language -%}\n    {%- set unified_slug = page.path | replace(from='/' ~ lang ~ '/', to='/') -%}\n    {%- set slug = slug | default(value=unified_slug) -%}\n{%- else -%}\n    {%- set slug = slug | default(value=page.path) -%}\n{%- endif -%}\n{%- if label -%}\n    {%- set final_label = label -%}\n{%- elif language_strings -%}\n    {%- set final_label = macros_translate::translate(key=\"like_this_post\", default=\"Like this post\", language_strings=language_strings) -%}\n{%- else -%}\n    {%- set final_label = \"Like this post\" -%}\n{%- endif -%}\n\n{%- if button_icon == \"heart\" -%}\n  {%- set icon_display = \"♥️\" -%}\n{%- elif button_icon == \"thumbs_up\" -%}\n  {%- set icon_display = \"👍\" -%}\n{%- elif button_icon == \"upvote\" -%}\n  {%- set icon_display = \"⬆️\" -%}\n{%- else -%}\n  {%- set icon_display = button_icon -%}\n{%- endif -%}\n\n<form method=\"post\" action=\"https://vhiweeypifbwacashxjz.supabase.co/rest/v1/rpc/increment_hits?apikey=sb_publishable_EoB7MFJhCmb6PiAk-GPJ4w_PGhQ44Ru\" class=\"iine-form\">\n  <input type=\"hidden\" name=\"page_slug\" value=\"{%- if slug -%}{{ slug }}{%- else -%}{{ current_url | default(value=page.path) }}{%- endif -%}\">\n  <button class=\"iine-button\" type=\"submit\"\n    {%- if slug %} data-slug=\"{{ slug }}\"{% endif %}\n    {%- if button_icon %} data-icon=\"{{ button_icon }}\"{% endif %}\n    aria-label=\"{{ final_label }}\" title=\"{{ final_label }}\">{{ icon_display }}</button>\n</form>\n"
  },
  {
    "path": "templates/partials/language_switcher.html",
    "content": "<li class=\"language-switcher\">\n    <details class=\"dropdown\">\n        <summary role=\"button\" aria-haspopup=\"true\" title=\"{{ macros_translate::translate(key=\"language_selection\", default=\"Language selection\", language_strings=language_strings) }}\" aria-label=\"{{ macros_translate::translate(key=\"language_selection\", default=\"Language selection\", language_strings=language_strings) }}\">\n            {%- if config.extra.show_selected_language_code_in_language_switcher -%}\n            <div class=\"language-switcher-icon language-switcher-icon-with-code\"></div>\n            <div class=\"language-switcher-icon-code\">{{lang}}</div>\n            {%- else -%}\n            <div class=\"language-switcher-icon\"></div>\n            {%- endif -%}\n        </summary>\n        <div class=\"dropdown-content\" role=\"menu\">\n            {#- Display the current language first in the dropdown -#}\n            {{ macros_translate::translate(key=\"language_name\", default=lang, language_strings=language_strings) }}\n            {#- Loop through all the available languages in the config -#}\n            {%- for lcode, ldetails in config.languages -%}\n                {#- Skip the current language to avoid linking to the current page -#}\n                {%- if lang == lcode -%}\n                    {%- continue -%}\n                {%- endif -%}\n                {#- Dynamically load the language strings for each language -#}\n                {%- set other_language_strings = load_data(path=\"i18n/\" ~ lcode ~ \".toml\", required=false) -%}\n                {%- if not other_language_strings -%}\n                    {%- set other_language_strings = load_data(path=\"themes/tabi/i18n/\" ~ lcode ~ \".toml\", required=false) -%}\n                {%- endif -%}\n                {#- Use the loaded language strings to get the language name -#}\n                {% set language_name = macros_translate::translate(key=\"language_name\", default=lcode,\n                language_strings=other_language_strings) %}\n                {#- Check if the language code matches the default language -#}\n                {%- if lcode == config.default_language -%}\n                    {#- If it does, link to the root path (no language code in URL) -#}\n                    <a role=\"menuitem\" lang=\"{{ lcode }}\" aria-label=\"{{ language_name }}\" href=\"{{ current_url | replace(from='/' ~ lang ~ '/', to = '/') }}\">{{ language_name }}</a>\n                    {#- Check if the current language is the default language -#}\n                    {#- If it is, append the language code to the base URL -#}\n                {%- elif lang == config.default_language -%}\n                    <a role=\"menuitem\" lang=\"{{ lcode }}\" aria-label=\"{{ language_name }}\" href=\"{{ config.base_url }}/{{ lcode }}{{ current_path | default(value=\"/\") | safe }}\">{{ language_name }}</a>\n                {%- else -%}\n                    {#- If it's not, replace the current language code in the URL with the new one -#}\n                    <a role=\"menuitem\" lang=\"{{ lcode }}\" aria-label=\"{{ language_name }}\" href=\"{{ current_url | replace(from='/' ~ lang ~ '/', to='/' ~ lcode ~ '/') }}\">{{ language_name }}</a>\n                {%- endif -%}\n            {%- endfor -%}\n        </div>\n    </details>\n</li>\n"
  },
  {
    "path": "templates/partials/main_page_posts_list.html",
    "content": "{%- if paginator or extra_section -%}\n    <div id=\"posts-list\">\n        <div>\n            {{ macros_page_header::page_header(title=section.title) }}\n        </div>\n\n        {# Check if both paginate_by and section_path are set #}\n        {%- set both_settings_set = paginator and extra_section -%}\n        {%- set paginator_has_no_pages = paginator and paginator.pages | length == 0 -%}\n        {%- set extra_section_has_pages = extra_section and extra_section.pages | length > 0 -%}\n\n        {# Display warning if both settings are set #}\n        {%- if both_settings_set and paginator_has_no_pages and extra_section_has_pages -%}\n            <div class=\"admonition warning\">\n                <div class=\"admonition-icon admonition-icon-warning\"></div>\n                <div class=\"admonition-content\">\n                    <strong class=\"admonition-title\">WARNING: Conflicting Configuration</strong>\n                    <p>\n                        No posts are displayed due to conflicting settings in your <code>_index.md</code>:\n                    </p>\n                    <ul>\n                        <li><code>paginate_by</code> is set, but there are no posts to paginate in the current section.</li>\n                        <li><code>section_path</code> is set, and posts are available in that section.</li>\n                    </ul>\n                    <p>\n                        <strong>Solution:</strong> Remove <code>paginate_by</code> from your <code>_index.md</code>.\n                        To limit the number of displayed posts, use <code>max_posts</code> in the <code>[extra]</code> section instead.\n                    </p>\n                </div>\n            </div>\n        {%- endif -%}\n\n        {# Get all posts for pinning if we're in root section with pagination #}\n        {%- if paginator and is_root_section -%}\n            {%- set root_section = get_section(path=\"_index.md\") -%}\n            {%- set all_posts = root_section.pages -%}\n            {%- set pages = paginator.pages -%}\n        {%- elif paginator -%}\n            {%- set all_posts = paginator.pages -%}\n            {%- set pages = paginator.pages -%}\n        {%- else -%}\n            {%- set all_posts = extra_section.pages -%}\n            {%- set pages = extra_section.pages -%}\n        {%- endif -%}\n\n        {% set max_posts = section.extra.max_posts | default(value=999999) %}\n        {{ macros_list_posts::list_posts(\n            posts=pages,\n            all_posts=all_posts,\n            max=max_posts,\n            language_strings=language_strings,\n            section_path=extra_section.path | default(value=\"blog\"),\n            pinned_first=is_root_section,\n            current_page=paginator.current_index | default(value=1)\n        ) }}\n    </div>\n\n    {% if paginator and paginator.pages | length > 0 %}\n        {%- include \"partials/paginate.html\" -%}\n    {% endif %}\n{%- endif -%}\n"
  },
  {
    "path": "templates/partials/main_page_projects_list.html",
    "content": "{% if section.extra.projects_path %}\n    {%- set projects_section = get_section(path=section.extra.projects_path) -%}\n    {%- if projects_section -%}\n        <div id=\"featured-projects\" class=\"list\">\n            {{ macros_page_header::page_header(title=macros_translate::translate(key=\"featured_projects\", default=\"Featured projects\", language_strings=language_strings)) }}\n        </div>\n        {%- set show_pages = projects_section.pages -%}\n        {%- set max_projects = section.extra.max_projects | default(value=3) -%}\n        {%- include \"partials/cards_pages.html\" -%}\n    {%- endif -%}\n    {%- if show_pages | length > max_projects -%}\n        <div class=\"all-posts\" id=\"all-projects\">\n            <a href=\"{{ get_url(path=projects_section.path, lang=lang) }}/\">{{ macros_translate::translate(key=\"all_projects\", default=\"All projects\", language_strings=language_strings) }}&nbsp;<span class=\"arrow\">⟶</span></a>\n        </div>\n    {%- endif -%}\n{% endif %}\n"
  },
  {
    "path": "templates/partials/multilingual_tags.html",
    "content": "{%- if section.translations -%}\n    {%- set current_translations = section.translations -%}\n{%- elif page.translations -%}\n    {%- set current_translations = page.translations -%}\n{%- endif -%}\n\n{%- if current_translations -%}\n\n    {%- for translation in current_translations -%}\n        {%- set lcode = translation.lang | default(value = config.default_language) -%}\n\n        {#- Dynamically load the language strings for each language -#}\n        {%- set other_language_strings = load_data(path=\"i18n/\" ~ lcode ~ \".toml\", required=false) -%}\n        {%- if not other_language_strings -%}\n            {%- set other_language_strings = load_data(path=\"themes/tabi/i18n/\" ~ lcode ~ \".toml\", required=false) -%}\n        {%- endif -%}\n        <meta property=\"og:locale:alternate\" content=\"{{ macros_translate::translate(key=\"date_locale\", default=\"en_GB\", language_strings=other_language_strings) }}\" />\n\n        {# Construct href for hreflang #}\n        {%- set href = translation.permalink -%}\n        {%- if lcode == config.default_language -%}\n            {%- set href = href | replace(from='/' ~ lang ~ '/', to = '/') -%}\n        {%- else -%}\n            {%- set href = href | replace(from='/' ~ lang ~ '/', to='/' ~ lcode ~ '/') -%}\n        {%- endif -%}\n        <link rel=\"alternate\" hreflang=\"{{ lcode }}\" href=\"{{ href | safe }}\" />\n    {%- endfor -%}\n\n{%- endif -%}\n"
  },
  {
    "path": "templates/partials/nav.html",
    "content": "<header>\n    <nav class=\"navbar\">\n        <div class=\"nav-title\">\n            <a class=\"home-title\" href=\"{{ get_url(path='/', lang=lang, trailing_slash=(lang == config.default_language)) }}\">{{ config.title }}</a>\n        </div>\n\n        {%- if config.extra.menu %}\n            <div class=\"nav-navs\">\n                <ul>\n                    {%- if config.extra.menu %}\n                        {% for menu in config.extra.menu %}\n                            <li>\n                                {% set trailing_slash = menu.trailing_slash | default(value=true) %}\n                                {%- if menu.url is starting_with(\"http\") -%}\n                                    {%- if trailing_slash -%}\n                                        <a class=\"nav-links no-hover-padding\" href=\"{{ menu.url }}/\">\n                                    {%- else -%}\n                                        <a class=\"nav-links no-hover-padding\" href=\"{{ menu.url }}\">\n                                    {%- endif -%}\n                                {%- else -%}\n                                    <a class=\"nav-links no-hover-padding\" href=\"{{ get_url(path=menu.url, lang=lang, trailing_slash=trailing_slash) }}\">\n                                {%- endif -%}\n                                {{ macros_translate::translate(key=menu.name, default=menu.name, language_strings=language_strings) }}\n                                </a>\n                            </li>\n                        {% endfor %}\n                    {%- endif -%}\n\n                    {#- Wrap the icons to keep them all together -#}\n                    <li class=\"menu-icons-container\">\n                        <ul class=\"menu-icons-group\">\n                            {# Search #}\n                            {%- if config.build_search_index %}\n                            {%- set search_icon_title = macros_translate::translate(key='search_icon_title', default='Press $SHORTCUT to open search', language_strings=language_strings) -%}\n                            <li class=\"js menu-icon\">\n                                <div role=\"button\" tabindex=\"0\" id=\"search-button\" class=\"search-icon interactive-icon\" title=\"{{ search_icon_title }}\" aria-label=\"{{ search_icon_title }}\">\n                                    <svg xmlns=\"http://www.w3.org/2000/svg\" viewBox=\"0 -960 960 960\">\n                                        <path d=\"M784-120 532-372q-30 24-69 38t-83 14q-109 0-184.5-75.5T120-580q0-109 75.5-184.5T380-840q109 0 184.5 75.5T640-580q0 44-14 83t-38 69l252 252-56 56ZM380-400q75 0 127.5-52.5T560-580q0-75-52.5-127.5T380-760q-75 0-127.5 52.5T200-580q0 75 52.5 127.5T380-400Z\"/>\n                                    </svg>\n                                </div>\n                            </li>\n                            {%- endif %}\n\n                            {# Language switcher #}\n                            {# Displayed only if more than one language is available #}\n                            {%- if config.languages | length > 0 %}\n                            {% include \"partials/language_switcher.html\" %}\n                            {%- endif %}\n\n                            {# Theme switcher #}\n                            {%- if config.extra.theme_switcher and config.extra.theme_switcher == true -%}\n                            {%- include \"partials/theme_switcher.html\" -%}\n                            {%- endif -%}\n                        </ul>\n                    </li>\n                </ul>\n            </div>\n        {% endif %}\n    </nav>\n</header>\n"
  },
  {
    "path": "templates/partials/paginate.html",
    "content": "{% if paginator and paginator.number_pagers > 1 %}\n    <ul class=\"pagination\">\n        {% if paginator.previous %}\n            <li class=\"page-item page-prev\">\n                <a href=\"{{ paginator.previous }}\" class=\"page-link\" aria-label=\"{{ macros_translate::translate(key=\"prev\", default=\"Prev\", language_strings=language_strings) }}\"><span class=\"arrow\">←</span> {{ macros_translate::translate(key=\"prev\", default=\"Prev\", language_strings=language_strings) }}</a>\n            </li>\n        {% else %}\n            <li class=\"page-item page-prev\">\n                <span class=\"page-link disabled\" aria-disabled=\"true\" aria-label=\"{{ macros_translate::translate(key=\"prev\", default=\"Prev\", language_strings=language_strings) }} (disabled)\"><span class=\"arrow\">←</span> {{ macros_translate::translate(key=\"prev\", default=\"Prev\", language_strings=language_strings) }}</span>\n            </li>\n        {% endif %}\n\n        <li class=\"page-item page-numbers\">\n            {{ paginator.current_index }} {{ macros_translate::translate(key=\"of\", default=\"of\", language_strings=language_strings) }} {{ paginator.number_pagers }}\n        </li>\n\n        {% if paginator.next %}\n            <li class=\"page-item page-next\">\n                <a href=\"{{ paginator.next }}\" class=\"page-link\" aria-label=\"{{ macros_translate::translate(key=\"next\", default=\"Next\", language_strings=language_strings) }}\">{{ macros_translate::translate(key=\"next\", default=\"Next\", language_strings=language_strings) }} <span class=\"arrow\">→</span></a>\n            </li>\n        {% else %}\n            <li class=\"page-item page-next\">\n                <span class=\"page-link disabled\" aria-disabled=\"true\" aria-label=\"{{ macros_translate::translate(key=\"next\", default=\"Next\", language_strings=language_strings) }} (disabled)\">{{ macros_translate::translate(key=\"next\", default=\"Next\", language_strings=language_strings) }} <span class=\"arrow\">→</span></span>\n            </li>\n        {% endif %}\n    </ul>\n{% endif %}\n"
  },
  {
    "path": "templates/partials/search_modal.html",
    "content": "<div id=\"searchModal\" class=\"search-modal js\" role=\"dialog\" aria-labelledby=\"modalTitle\">\n    <h1 id=\"modalTitle\" class=\"visually-hidden\">{{ macros_translate::translate(key='search', default='Search', language_strings=language_strings) }}</h1>\n    <div id=\"modal-content\">\n        <div id=\"searchBar\">\n            <div class=\"search-icon\" aria-hidden=\"true\">\n                <svg xmlns=\"http://www.w3.org/2000/svg\" viewBox=\"0 -960 960 960\">\n                    <path d=\"M784-120 532-372q-30 24-69 38t-83 14q-109 0-184.5-75.5T120-580q0-109 75.5-184.5T380-840q109 0 184.5 75.5T640-580q0 44-14 83t-38 69l252 252-56 56ZM380-400q75 0 127.5-52.5T560-580q0-75-52.5-127.5T380-760q-75 0-127.5 52.5T200-580q0 75 52.5 127.5T380-400Z\"/>\n                </svg>\n            </div>\n            <input id=\"searchInput\" role=\"combobox\" autocomplete=\"off\" spellcheck=\"false\" aria-expanded=\"false\" aria-controls=\"results-container\" placeholder=\"{{ macros_translate::translate(key='search', default='Search', language_strings=language_strings) }}…\"/>\n            <div id=\"clear-search\" class=\"close-icon interactive-icon\" tabindex=\"0\" role=\"button\" title=\"{{ macros_translate::translate(key='clear_search', default='Clear search', language_strings=language_strings) }}\">\n                <svg xmlns=\"http://www.w3.org/2000/svg\" viewBox=\"0 -960 960 960\">\n                <path d=\"m256-200-56-56 224-224-224-224 56-56 224 224 224-224 56 56-224 224 224 224-56 56-224-224-224 224Z\"/>\n                </svg>\n            </div>\n        </div>\n        <div id=\"results-container\">\n            <div id=\"results-info\">\n                {#- Add the strings here so JavaScript can grab them -#}\n                {#- These are used in all languages -#}\n                <span id=\"zero_results\"> {{ macros_translate::translate(key='results', number=0, default='No results', language_strings=language_strings, replace=false) }}</span>\n                <span id=\"one_results\"> {{ macros_translate::translate(key='results', number=1, default='1 result', language_strings=language_strings, replace=false) }}</span>\n                <span id=\"many_results\"> {{ macros_translate::translate(key='results', number=11, default='$NUMBER results', language_strings=language_strings, replace=false) }}</span>\n                {#- Strings for specific languages -#}\n                <span id=\"two_results\"> {{ macros_translate::translate(key='results', number=2, default='$NUMBER results', language_strings=language_strings, replace=false) }}</span>\n                <span id=\"few_results\"> {{ macros_translate::translate(key='results', number=2, default='$NUMBER results', language_strings=language_strings, replace=false) }}</span>\n            </div>\n            <div id=\"results\" role=\"listbox\"></div>\n        </div>\n    </div>\n</div>\n"
  },
  {
    "path": "templates/partials/social_media_images.html",
    "content": "{%- set social_media_card = macros_settings::evaluate_setting_priority(setting=\"social_media_card\", page=page | default(value=\"\"), section=section | default(value=\"\"), default_global_value=\"\") -%}\n{% if social_media_card %}\n    {# Get base path from page/section #}\n    {% set base_path = \"\" %}\n    {% if section and section.path %}\n        {% set base_path = section.path | trim_end_matches(pat=\"/_index.md\") %}\n        {% if base_path and not social_media_card is starting_with(\"/\") %}\n            {% set base_path = base_path ~ \"/\" %}\n        {% endif %}\n    {% else %}\n        {% set base_path = page.colocated_path | default(value=\"\") %}\n    {% endif %}\n\n    {% set current_path = base_path ~ social_media_card | trim_start_matches(pat=\"/\") %}\n\n    {# Try parent path by removing the last directory component #}\n    {% set parent_path = base_path | split(pat=\"/\") | slice(end=-2) | join(sep=\"/\") %}\n    {% if parent_path and not social_media_card is starting_with(\"/\") %}\n        {% set parent_path = parent_path ~ \"/\" %}\n    {% endif %}\n    {% set parent_relative_path = parent_path ~ social_media_card | trim_start_matches(pat=\"/\") %}\n\n    {# Check all possible locations #}\n    {%- set current_meta = get_image_metadata(path=current_path, allow_missing=true) -%}\n    {%- set parent_meta = get_image_metadata(path=parent_relative_path, allow_missing=true) -%}\n    {%- set absolute_meta = get_image_metadata(path=social_media_card, allow_missing=true) -%}\n\n    {% if current_meta %}\n        {% set final_path = current_path %}\n        {% set meta = current_meta %}\n    {% elif parent_meta %}\n        {% set final_path = parent_relative_path %}\n        {% set meta = parent_meta %}\n    {% elif absolute_meta %}\n        {% set final_path = social_media_card %}\n        {% set meta = absolute_meta %}\n    {% else %}\n        {{ throw(message=\"Could not find social media card image. Tried:\n            1. Current page path: '\" ~ current_path ~ \"'\n            2. Parent page path: '\" ~ parent_relative_path ~ \"'\n            3. Absolute path: '\" ~ social_media_card ~ \"'\n            Please ensure the file exists at one of these locations.\") }}\n    {% endif %}\n\n    <meta property=\"og:image\" content=\"{{ get_url(path=final_path, cachebust=true) }}\" />\n    <meta property=\"og:image:width\" content=\"{{ meta.width }}\" />\n    <meta property=\"og:image:height\" content=\"{{ meta.height }}\" />\n    <meta name=\"twitter:image\" content=\"{{ get_url(path=final_path, cachebust=true) }}\" />\n    <meta name=\"twitter:card\" content=\"summary_large_image\" />\n{% endif %}\n"
  },
  {
    "path": "templates/partials/theme_switcher.html",
    "content": "<li class=\"theme-switcher-wrapper js\">\n    {#- Create the localised strings for the title and aria-label attributes -#}\n    {%- set toggle_str = macros_translate::translate(key='toggle_mode', default='Toggle $MODE mode', language_strings=language_strings) -%}\n    {%- set dark_str = macros_translate::translate(key='dark', default='dark', language_strings=language_strings) -%}\n    {%- set light_str = macros_translate::translate(key='light', default='light', language_strings=language_strings) -%}\n\n    {%- set combined_mode_str = dark_str ~ \"/\" ~ light_str -%}\n    {%- set title_label = toggle_str | replace(from=\"$MODE\", to=combined_mode_str) -%}\n    {%- set aria_label = toggle_str | replace(from=\"$MODE\", to=dark_str) -%}\n\n    <div\n        title=\"{{ title_label }}\"\n        class=\"theme-switcher\"\n        tabindex=\"0\"\n        role=\"button\"\n        aria-label=\"{{ aria_label }}\"\n        aria-pressed=\"false\">\n    </div>\n\n    {%- set reset_str = macros_translate::translate(key='reset_mode', default='Reset mode to default', language_strings=language_strings) -%}\n\n    <div\n        title=\"{{ reset_str }}\"\n        class=\"theme-resetter arrow\"\n        tabindex=\"0\"\n        role=\"button\"\n        aria-hidden=\"true\"\n        aria-label=\"{{ reset_str }}\">\n    </div>\n\n</li>\n"
  },
  {
    "path": "templates/partials/title.html",
    "content": "{#- Setup -#}\n{% if not config.title %}\n{{ throw(message=\"ERROR: No `title` set in `config.toml`. tabi requires a title to function.\") }}\n{% endif %}\n{%- set prefix = config.title | safe -%}\n{%- set custom_separator = config.extra.separator | default(value=\"•\") -%}\n{%- set separator = \" \" ~ custom_separator ~ \" \" -%}\n\n{#- Get the base path for the current language -#}\n{%- if lang != config.default_language %}\n    {%- set base_path = \"/\" ~ lang ~ \"/\" %}\n{%- else -%}\n    {%- set base_path = \"/\" %}\n{%- endif %}\n\n{%- if current_path and current_path == base_path -%}\n    {%- set suffix = \"\" -%}\n    {%- set separator = \"\" -%}\n{% elif title %}\n    {%- set suffix =  title -%}\n{% elif section.title -%}\n    {%- set suffix = section.title -%}\n{% elif page.title %}\n    {%- set suffix = page.title -%}\n{% elif term.name %}\n    {#- Individual tags -#}\n    {%- set suffix = term.name -%}\n{% elif taxonomy.name %}\n    {#- List of tags -#}\n    {%- set suffix = macros_translate::translate(key=taxonomy.name, language_strings=language_strings) | capitalize -%}\n{% else %}\n    {%- set suffix = \"404\" %}\n{%- endif -%}\n\n{#- Return the final concatenated string -#}\n{%- if config.extra.invert_title_order -%}\n    {{- suffix ~ separator ~ prefix -}}\n{%- else -%}\n    {{- prefix ~ separator ~ suffix -}}\n{%- endif -%}\n"
  },
  {
    "path": "templates/partials/webmentions.html",
    "content": "{# Incorporate webmention.io links and script into the page head.\n1. Provide the link to the webmention data in the at webmention.io.\n2. Link to the stylesheet for styling webmentions on a page.\n3. Add and configure the javascript to fetch and display the webmentions collected at webmention.io. #}\n\n<link rel=\"webmention\" href=\"https://webmention.io/{{ config.extra.webmentions.domain }}/webmention\" />\n\n{# Calculate the configured data for the script, if any #}\n\n{% set script_data = \"\" %}\n\n{% if config.extra.webmentions.id %}\n{% set script_data = script_data ~ \"data-id=\" ~ config.extra.webmentions.id %}\n{% endif %}\n\n{% if config.extra.webmentions.page_url %}\n{% set script_data = script_data ~ \" data-page-url=\" ~ config.extra.webmentions.page_url %}\n{% endif %}\n\n{% if config.extra.webmentions.add_urls %}\n{% set script_data = script_data ~ \"data-add-urls=\" ~ config.extra.webmentions.add_urls %}\n{% endif %}\n\n{% if config.extra.webmentions.wordcount %}\n{% set script_data = script_data ~ \" data-wordcount=\" ~ config.extra.webmentions.wordcount %}\n{% endif %}\n\n{% if config.extra.webmentions.max_webmentions %}\n{% set script_data = script_data ~ \"data-max-webmentions=\" ~ config.extra.webmentions.max_webmentions %}\n{% endif %}\n\n{% if config.extra.webmentions.prevent_spoofing %}\n{% set script_data = script_data ~ \"data-prevent-spoofing=\" ~ config.extra.webmentions.prevent_spoofing %}\n{% endif %}\n\n{% if config.extra.webmentions.sort_by %}\n{% set script_data = script_data ~ \"data-sort-by=\" ~ config.extra.webmentions.sort_by %}\n{% endif %}\n\n{% if config.extra.webmentions.sort_dir %}\n{% set script_data = script_data ~ \"data-sort-dir=\" ~ config.extra.webmentions.sort_dir %}\n{% endif %}\n\n{% if config.extra.webmentions.comments_are_reactions %}\n{% set script_data = script_data ~ \" data-comments-are-reactions=\" ~ config.extra.webmentions.comments_are_reactions %}\n{% endif %}\n\n<script async src=\"{{ get_url(path='js/webmention.min.js', trailing_slash=false, cachebust=true) | safe }}\" {{ script_data }}>\n</script>\n\n<div class=\"webmentions-container\" id=\"webmentions\"></div>\n"
  },
  {
    "path": "templates/section.html",
    "content": "{% extends \"base.html\" %}\n\n{% block main_content %}\n\n{# We'll only pin posts in the root section. #}\n{# Right now both the main page and blog/ use the same `section.html` template. #}\n{# To avoid using different templates, we do this. #}\n{%- if lang == config.default_language -%}\n    {%- set expected_root = \"/\" -%}\n{%- else -%}\n    {%- set expected_root = \"/\" ~ lang ~ \"/\" -%}\n{%- endif -%}\n{%- set is_root_section = current_path == expected_root -%}\n\n{%- set show_projects_first = section.extra.show_projects_first | default(value=false) -%}\n{%- if show_projects_first -%}\n    {%- set first_section = \"projects\" -%}\n{%- else -%}\n    {%- set first_section = \"posts\" -%}\n{%- endif -%}\n\n{%- if section.extra.section_path or paginator and projects_path -%}\n    {%- set more_than_one_section_shown = true -%}\n{%- endif -%}\n\n<main {% if more_than_one_section_shown %}class=\"{{ first_section }}-first\"{% endif %}>\n{%- if config.extra.hcard %}\n    {%- include \"partials/hcard.html\" -%}\n{% endif -%}\n{%- if section.extra.header %}\n    {%- include \"partials/home_banner.html\" -%}\n{%- elif section.content -%}\n<section>\n    {{ section.content | safe }}\n</section>\n{% endif -%}\n\n{% if section.extra.section_path -%}\n    {% set extra_section = get_section(path=section.extra.section_path) %}\n{% endif -%}\n\n{% if section.extra.section_path -%}\n    {% set path = section.extra.section_path | replace(from=\"/_index.md\", to = \"\") %}\n{% else -%}\n    {% set path = section.path %}\n{% endif -%}\n\n{%- if show_projects_first -%}\n    {%- include \"partials/main_page_projects_list.html\" -%}\n    {%- include \"partials/main_page_posts_list.html\" -%}\n{%- else -%}\n    {%- include \"partials/main_page_posts_list.html\" -%}\n    {%- include \"partials/main_page_projects_list.html\" -%}\n{%- endif -%}\n</main>\n\n{%- include \"partials/extra_features.html\" -%}\n\n{% endblock main_content %}\n"
  },
  {
    "path": "templates/series.html",
    "content": "{% extends \"base.html\" %}\n\n{% block main_content %}\n\n{# Throw an error if the section is not flagged as a series. #}\n{# This page would be displayed properly but it would become impossible for the series' child pages to reference their series. #}\n{%- if \"series\" not in section.extra or not section.extra.series -%}\n    {{ throw(message=\"Section is not flagged as a series. Set `section.extra.series` to `true` if you want to use `series.html` template.\") }}\n{%- endif -%}\n\n<main>\n{%- if section.extra.header %}\n    {%- include \"partials/home_banner.html\" -%}\n{% endif -%}\n\n    {%- set show_jump = false -%}\n    {%- set show_jump_hierarchy = macros_settings::evaluate_setting_priority(setting=\"show_jump_to_posts\", page=section) -%}\n    {%- if show_jump_hierarchy == \"true\" -%}\n        {%- set show_jump = true -%}\n    {%- elif show_jump_hierarchy != \"false\" -%}\n        {#- Default to true if the content is long and var is unset #}\n        {%- if section.content | length > 2000 -%}\n            {%- set show_jump = true -%}\n        {%- endif -%}\n    {%- endif -%}\n\n    {%- if show_jump -%}\n    <div class=\"title-with-jump bottom-divider\">\n        <h1 class=\"title-container section-title\">{{ section.title }}</h1>\n        <a href=\"#posts-list\" class=\"jump-link\">{{ macros_translate::translate(key=\"jump_to_posts\", default=\"Jump to posts\", language_strings=language_strings) }} ↓</a>\n    </div>\n    {%- else -%}\n        {{ macros_page_header::page_header(title=section.title) }}\n    {%- endif -%}\n\n    <section class=\"body\">\n        {{ section.content | safe }}\n    </section>\n\n    <div id=\"posts-list\">\n        <h2 class=\"bottom-divider\">\n            {{ macros_translate::translate(key=\"all_posts\", default=\"All posts\", language_strings=language_strings) }}\n        </h2>\n        {%- if paginator %}\n            {%- set pages = paginator.pages -%}\n        {% else %}\n            {%- set pages = section.pages -%}\n        {% endif -%}\n\n        {% set max_posts = section.extra.max_posts | default(value=999999) %}\n        {{ macros_list_posts::list_posts(posts=pages, max=max_posts, metadata=\"indexes\", language_strings=language_strings, section_path=section.path, paginator=paginator | default(value=\"\")) }}\n    </div>\n\n    {% if paginator %}\n        {%- include \"partials/paginate.html\" -%}\n    {% endif %}\n\n</main>\n\n{%- include \"partials/extra_features.html\" -%}\n\n{% endblock main_content %}\n"
  },
  {
    "path": "templates/shortcodes/add_src_to_code_block.html",
    "content": "<span class=\"code-source hidden\" data-source=\"{{ src | safe }}\"></span>\n"
  },
  {
    "path": "templates/shortcodes/admonition.html",
    "content": "{%- set type = type | default(value=\"info\") -%}\n{%- set title = title | default(value=type | upper) -%}\n{%- set icon = icon | default(value=type) -%}\n{%- set text = text | default(value=body) -%}\n\n<div class=\"admonition {{ type }}\">\n    <div class=\"admonition-icon admonition-icon-{{ icon }}\"></div>\n    <div class=\"admonition-content\">\n        <strong class=\"admonition-title\">{{ title | safe }}</strong>\n        {{ text | markdown | safe }}\n    </div>\n</div>\n"
  },
  {
    "path": "templates/shortcodes/aside.html",
    "content": "{%- set text = text | default(value=body) -%}\n\n<aside {% if position %}data-position=\"{{ position }}\"{% endif %}>\n    {{ text | markdown | safe }}\n</aside>\n"
  },
  {
    "path": "templates/shortcodes/dimmable_image.html",
    "content": "{#- Determine image path based on whether the src is remote or local -#}\n{%- if src is starting_with(\"http\") or raw_path -%}\n    {%- set image_url = src -%}\n{%- else -%}\n    {%- set colocated_path = page.colocated_path | default(value=\"\") -%}\n    {%- set relative_path = colocated_path ~ src -%}\n    {%- set meta = get_image_metadata(path=relative_path, allow_missing=true) -%}\n\n    {#- Fallback to absolute path if relative path doesn't work -#}\n    {%- if not meta -%}\n        {%- set meta = get_image_metadata(path=src, allow_missing=true) -%}\n        {%- set image_url = get_url(path=src, cachebust=true) -%}\n    {%- else -%}\n        {%- set image_url = get_url(path=relative_path, cachebust=true) -%}\n    {%- endif -%}\n{%- endif -%}\n\n{%- set lazy_loading = lazy_loading | default(value=true) -%}\n\n{%- set class_list = \"dimmable-image\" -%}\n{%- if inline -%}\n    {%- set class_list = class_list ~ \" inline\" -%}\n{%- endif -%}\n\n{%- if full_width -%}\n    <div class=\"full-width\">\n{%- endif -%}\n<img class=\"{{ class_list }}\" src=\"{{ image_url }}\"{% if lazy_loading %} loading=\"lazy\"{% endif %}{% if alt %} alt=\"{{ alt }}\"{% endif %}{% if meta.width %} width=\"{{ meta.width }}\"{% endif %}{% if meta.height %} height=\"{{ meta.height }}\" {% endif %}/>\n{%- if full_width -%}\n    </div>\n{%- endif -%}\n"
  },
  {
    "path": "templates/shortcodes/dual_theme_image.html",
    "content": "{%- set colocated_path = page.colocated_path | default(value=\"\") -%}\n{%- set lazy_loading = lazy_loading | default(value=true) -%}\n{%- set inline = inline | default(value=false) -%}\n\n{%- set light_class_list = \"img-light\" -%}\n{%- set dark_class_list = \"img-dark\" -%}\n{%- if inline -%}\n    {%- set light_class_list = light_class_list ~ \" inline\" -%}\n    {%- set dark_class_list = dark_class_list ~ \" inline\" -%}\n{%- endif -%}\n\n{# Handling for light mode image #}\n{%- if light_src is starting_with(\"http\") or raw_path -%}\n    {%- set light_image_url = light_src -%}\n{%- else -%}\n    {%- set relative_light_path = colocated_path ~ light_src -%}\n    {%- set light_meta = get_image_metadata(path=relative_light_path, allow_missing=true) -%}\n    {%- if not light_meta -%}\n        {%- set light_image_url = get_url(path=light_src, cachebust=true) -%}\n    {%- else -%}\n        {%- set light_image_url = get_url(path=relative_light_path, cachebust=true) -%}\n    {%- endif -%}\n{%- endif -%}\n\n{# Handling for dark mode image #}\n{%- if dark_src is starting_with(\"http\") or raw_path -%}\n    {%- set dark_image_url = dark_src -%}\n{%- else -%}\n    {%- set relative_dark_path = colocated_path ~ dark_src -%}\n    {%- set dark_meta = get_image_metadata(path=relative_dark_path, allow_missing=true) -%}\n    {%- if not dark_meta -%}\n        {%- set dark_image_url = get_url(path=dark_src, cachebust=true) -%}\n    {%- else -%}\n        {%- set dark_image_url = get_url(path=relative_dark_path, cachebust=true) -%}\n    {%- endif -%}\n{%- endif -%}\n\n{%- if full_width -%}\n    <div class=\"full-width\">\n{%- endif -%}\n<img class=\"{{ light_class_list }}\" src=\"{{ light_image_url }}\"{% if lazy_loading %} loading=\"lazy\"{% endif %}{% if alt %} alt=\"{{ alt }}\"{% endif %}{% if light_meta.width %} width=\"{{ light_meta.width }}\"{% endif %}{% if light_meta.height %} height=\"{{ light_meta.height }}\" {% endif %}>\n<img class=\"{{ dark_class_list }}\" src=\"{{ dark_image_url }}\"{% if lazy_loading %} loading=\"lazy\"{% endif %}{% if alt %} alt=\"{{ alt }}\"{% endif %}{% if dark_meta.width %} width=\"{{ dark_meta.width }}\"{% endif %}{% if dark_meta.height %} height=\"{{ dark_meta.height }}\" {% endif %}>\n{%- if full_width -%}\n    </div>\n{%- endif -%}\n"
  },
  {
    "path": "templates/shortcodes/force_text_direction.html",
    "content": "{%- set direction = direction | default(value=\"ltr\") -%}\n\n<div data-force-text-direction=\"{{ direction }}\">\n    {{ body | markdown | safe }}\n</div>\n"
  },
  {
    "path": "templates/shortcodes/full_width_image.html",
    "content": "{#- Set paths based on whether the src is remote or local -#}\n{%- if src is starting_with(\"http\") or raw_path -%}\n    {%- set image_url = src -%}\n{%- else -%}\n    {%- set colocated_path = page.colocated_path | default(value=\"\") -%}\n    {%- set relative_path = colocated_path ~ src -%}\n    {%- set meta = get_image_metadata(path=relative_path, allow_missing=true) -%}\n\n    {#- Fallback to absolute path if relative path doesn't work -#}\n    {%- if not meta -%}\n        {%- set meta = get_image_metadata(path=src, allow_missing=true) -%}\n        {%- set image_url = get_url(path=src, cachebust=true) -%}\n    {%- else %}\n        {%- set image_url = get_url(path=relative_path, cachebust=true) -%}\n    {%- endif -%}\n{%- endif -%}\n\n{%- set lazy_loading = lazy_loading | default(value=true) -%}\n\n<div class=\"full-width\">\n    <img src=\"{{ image_url }}\"{% if alt %} alt=\"{{ alt }}\"{% endif %}{% if meta.width %} width=\"{{ meta.width }}\"{% endif %}{% if meta.height %} height=\"{{ meta.height }}\"{% endif %}{% if lazy_loading %} loading=\"lazy\"{% endif %}/>\n</div>\n"
  },
  {
    "path": "templates/shortcodes/iine.html",
    "content": "{% set button_icon = icon | default(value=\"heart\") %}\n{% set label = label | default(value=\"Like this post\") %}\n{% set slug = slug | default(value=page.path) %}\n\n{% include \"partials/iine_button.html\" %}\n"
  },
  {
    "path": "templates/shortcodes/image_hover.html",
    "content": "{%- set colocated_path = page.colocated_path | default(value=\"\") -%}\n{%- set lazy_loading = lazy_loading | default(value=true) -%}\n{%- set inline = inline | default(value=false) -%}\n\n{#- Determine the tag for container elements -#}\n{#- Necessary for inline images -#}\n{%- set tag = \"div\" -%}\n{%- if inline -%}\n    {%- set tag = \"span\" -%}\n{%- endif -%}\n\n{%- set img_class_list = \"\" -%}\n{%- if inline -%}\n    {%- set img_class_list = img_class_list ~ \" inline\" -%}\n{%- endif -%}\n\n{#- Direct or relative URL handling for default image -#}\n{%- if default_src is starting_with(\"http\") or raw_path -%}\n    {%- set default_image_url = default_src -%}\n{%- else -%}\n    {%- set relative_default_path = colocated_path ~ default_src -%}\n    {%- set default_meta = get_image_metadata(path=relative_default_path, allow_missing=true) -%}\n    {%- if not default_meta -%}\n        {%- set default_image_url = get_url(path=default_src, cachebust=true) -%}\n    {%- else -%}\n        {%- set default_image_url = get_url(path=relative_default_path, cachebust=true) -%}\n    {%- endif -%}\n{%- endif -%}\n\n{#- Direct or relative URL handling for hovered image -#}\n{%- if hovered_src is starting_with(\"http\") or raw_path -%}\n    {%- set hovered_image_url = hovered_src -%}\n{%- else -%}\n    {%- set relative_hovered_path = colocated_path ~ hovered_src -%}\n    {%- set hovered_meta = get_image_metadata(path=relative_hovered_path, allow_missing=true) -%}\n    {%- if not hovered_meta -%}\n        {%- set hovered_image_url = get_url(path=hovered_src, cachebust=true) -%}\n    {%- else -%}\n        {%- set hovered_image_url = get_url(path=relative_hovered_path, cachebust=true) -%}\n    {%- endif -%}\n{%- endif -%}\n\n<{{ tag }} class=\"image-hover-container{% if full_width %} full-width{% endif %}\">\n    <{{ tag }} class=\"image-default\">\n        <img class=\"{{ img_class_list }}\" src=\"{{ default_image_url }}\"{% if lazy_loading %} loading=\"lazy\"{% endif %}{% if default_alt %} alt=\"{{ default_alt }}\"{% endif %}{% if default_meta.width %} width=\"{{ default_meta.width }}\"{% endif %}{% if default_meta.height %} height=\"{{ default_meta.height }}\"{% endif %}>\n    </{{ tag }}>\n    <{{ tag }} class=\"image-hovered\">\n        <img class=\"{{ img_class_list }}\" src=\"{{ hovered_image_url }}\"{% if lazy_loading %} loading=\"lazy\"{% endif %}{% if hovered_alt %} alt=\"{{ hovered_alt }}\"{% endif %}{% if hovered_meta.width %} width=\"{{ hovered_meta.width }}\"{% endif %}{% if hovered_meta.height %} height=\"{{ hovered_meta.height }}\"{% endif %}>\n    </{{ tag }}>\n</{{ tag }}>\n"
  },
  {
    "path": "templates/shortcodes/image_toggler.html",
    "content": "{# The `random_id` ensures that each instance of the shortcode has a \"unique\" id #}\n{# allowing individual interactive elements (like toggles) to function correctly. #}\n{# This avoids conflicts when multiple instances of the shortcode are used. #}\n{%- set random_id = get_random(end=100000) -%}\n{%- set colocated_path = page.colocated_path | default(value=\"\") -%}\n{%- set lazy_loading = lazy_loading | default(value=true) -%}\n{%- set inline = inline | default(value=false) -%}\n\n{#- Determine the class for the images -#}\n{#- Necessary for inline images -#}\n{%- set tag = \"div\" -%}\n{%- if inline -%}\n    {%- set tag = \"span\" -%}\n{%- endif -%}\n\n{%- set img_class_list = \"\" -%}\n{%- if inline -%}\n    {%- set img_class_list = img_class_list ~ \" inline\" -%}\n{%- endif -%}\n\n{# Direct or relative URL handling for default image #}\n{%- if default_src is starting_with(\"http\") or raw_path -%}\n    {%- set default_image_url = default_src -%}\n{%- else -%}\n    {%- set relative_default_path = colocated_path ~ default_src -%}\n    {%- set default_meta = get_image_metadata(path=relative_default_path, allow_missing=true) -%}\n    {%- if not default_meta -%}\n        {%- set default_image_url = get_url(path=default_src, cachebust=true) -%}\n    {%- else -%}\n        {%- set default_image_url = get_url(path=relative_default_path, cachebust=true) -%}\n    {%- endif -%}\n{%- endif -%}\n\n{# Direct or relative URL handling for toggled image #}\n{%- if toggled_src is starting_with(\"http\") or raw_path -%}\n    {%- set toggled_image_url = toggled_src -%}\n{%- else -%}\n    {%- set relative_toggled_path = colocated_path ~ toggled_src -%}\n    {%- set toggled_meta = get_image_metadata(path=relative_toggled_path, allow_missing=true) -%}\n    {%- if not toggled_meta -%}\n        {%- set toggled_image_url = get_url(path=toggled_src, cachebust=true) -%}\n    {%- else -%}\n        {%- set toggled_image_url = get_url(path=relative_toggled_path, cachebust=true) -%}\n    {%- endif -%}\n{%- endif -%}\n\n<{{ tag }} class=\"image-toggler-container {% if full_width %}full-width{% endif %}\">\n    <input type=\"checkbox\" id=\"toggle-img-{{ random_id }}\" class=\"image-toggler-toggle\">\n    <label for=\"toggle-img-{{ random_id }}\" class=\"image-label\">\n        <{{ tag }} class=\"image-default\">\n            <img class=\"{{ img_class_list }}\" src=\"{{ default_image_url }}\"{% if lazy_loading %} loading=\"lazy\"{% endif %}{% if default_alt %} alt=\"{{ default_alt }}\"{% endif %}{% if default_meta.width %} width=\"{{ default_meta.width }}\"{% endif %}{% if default_meta.height %} height=\"{{ default_meta.height }}\"{% endif %}>\n        </{{ tag }}>\n        <{{ tag }} class=\"image-toggled\">\n            <img class=\"{{ img_class_list }}\" src=\"{{ toggled_image_url }}\"{% if lazy_loading %} loading=\"lazy\"{% endif %}{% if toggled_alt %} alt=\"{{ toggled_alt }}\"{% endif %}{% if toggled_meta.width %} width=\"{{ toggled_meta.width }}\"{% endif %}{% if toggled_meta.height %} height=\"{{ toggled_meta.height }}\"{% endif %}>\n        </{{ tag }}>\n    </label>\n</{{ tag }}>\n"
  },
  {
    "path": "templates/shortcodes/invertible_image.html",
    "content": "{#- Determine if src is a remote URL or a local path -#}\n{%- if src is starting_with(\"http\") or raw_path -%}\n    {%- set image_url = src -%}\n{%- else -%}\n    {%- set colocated_path = page.colocated_path | default(value=\"\") -%}\n    {%- set relative_path = colocated_path ~ src -%}\n    {%- set meta = get_image_metadata(path=relative_path, allow_missing=true) -%}\n\n    {#- Fallback to absolute path if relative path doesn't work -#}\n    {%- if not meta -%}\n        {%- set meta = get_image_metadata(path=src, allow_missing=true) -%}\n        {%- set image_url = get_url(path=src, cachebust=true) -%}\n    {%- else %}\n        {%- set image_url = get_url(path=relative_path, cachebust=true) -%}\n    {%- endif -%}\n{%- endif -%}\n\n{%- set lazy_loading = lazy_loading | default(value=true) -%}\n{%- set inline = inline | default(value=false) -%}\n\n{%- set class_list = \"invertible-image\" -%}\n{%- if inline -%}\n    {%- set class_list = class_list ~ \" inline\" -%}\n{%- endif -%}\n\n{%- if full_width -%}\n    <div class=\"full-width\">\n{%- endif -%}\n<img class=\"{{ class_list }}\" src=\"{{ image_url }}\"{% if lazy_loading %} loading=\"lazy\"{% endif %}{% if alt %} alt=\"{{ alt }}\"{% endif %}{% if meta.width %} width=\"{{ meta.width }}\"{% endif %}{% if meta.height %} height=\"{{ meta.height }}\" {% endif %}/>\n{%- if full_width -%}\n    </div>\n{%- endif -%}\n"
  },
  {
    "path": "templates/shortcodes/mermaid.html",
    "content": "{% set invertible = invertible | default(value=true) %}\n{% set full_width = full_width | default(value=false) %}\n<noscript>\n    <strong>⚠️ JavaScript is required to render the diagram.</strong>\n</noscript>\n<pre class=\"mermaid{% if invertible %} invertible-image{% endif %}{% if full_width %} full-width{% endif %}\">\n    {{ body | safe }}\n</pre>\n"
  },
  {
    "path": "templates/shortcodes/multilingual_quote.html",
    "content": "{%- import \"macros/translate.html\" as macros_translate -%}\n{# Load internationalisation data #}\n{%- set language_strings = load_data(path=\"i18n/\" ~ lang ~ '.toml', required=false) -%}\n{%- if not language_strings -%}\n    {%- set language_strings = load_data(path=\"themes/tabi/i18n/\" ~ lang ~ \".toml\", required=false) -%}\n{%- endif -%}\n\n{%- set open_quote = macros_translate::translate(key=\"open_quotation_mark\", default=\"“\", language_strings=language_strings) -%}\n{%- set close_quote = macros_translate::translate(key=\"close_quotation_mark\", default=\"”\", language_strings=language_strings) -%}\n\n{#- The `random_id` ensures that each instance of the shortcode has a \"unique\" id -#}\n{#- allowing individual interactive elements (like toggles) to function correctly. -#}\n{#- This avoids conflicts when multiple instances of the shortcode are used. -#}\n{#- More context: https://github.com/welpo/tabi/issues/82 -#}\n{%- set random_id = get_random(end=100000) -%}\n\n<div class=\"quote-container\">\n    <input type=\"checkbox\" id=\"toggle-{{ random_id }}\" class=\"quote-toggle\">\n    <div class=\"quote\">\n        <div class=\"translated\">\n            <blockquote>\n                <p>{{ open_quote ~ translated ~ close_quote }}</p>\n                <p>{% if author %}— {{ author | safe }}{% endif %} <label for=\"toggle-{{ random_id }}\" class=\"quote-label quote-label-original\">\n                    ({{- macros_translate::translate(key=\"show_original_quote\", default=\"Show original quote\", language_strings=language_strings) -}})\n                </label></p>\n            </blockquote>\n        </div>\n        <div class=\"original\">\n            <blockquote>\n                <p>{{ open_quote ~ original ~ close_quote }}</p>\n                <p>{% if author %}— {{ author | safe }}{% endif %} <label for=\"toggle-{{ random_id }}\" class=\"quote-label quote-label-translate\">\n                    ({{- macros_translate::translate(key=\"show_translation\", default=\"Show translation\", language_strings=language_strings) -}})\n                </label></p>\n            </blockquote>\n        </div>\n    </div>\n</div>\n"
  },
  {
    "path": "templates/shortcodes/references.html",
    "content": "<div class=\"references\">\n    {{ body | markdown | safe }}\n</div>\n"
  },
  {
    "path": "templates/shortcodes/remote_text.html",
    "content": "{%- set start = start | default(value=1) -%}\n{%- set end = end | default(value=0) -%}\n\n{#- load_data uses different arguments based on whether it's a remote or local file -#}\n{%- if src is starting_with(\"http\") -%}\n    {%- set response = load_data(url=src, format=\"plain\") -%}\n{%- else -%}\n    {#- Try to load the file from a relative path -#}\n    {%- set colocated_path = page.colocated_path | default(value=\"\") -%}\n    {%- set relative_path = colocated_path ~ src -%}\n    {%- set response = load_data(path=relative_path, format=\"plain\", required=false) -%}\n    {#- If relative path fails, try absolute path -#}\n    {%- if not response -%}\n        {%- set response = load_data(path=src, format=\"plain\") -%}\n    {%- endif -%}\n{%- endif -%}\n\n{%- set lines = response | trim_end | split(pat=\"\\n\") -%}\n\n{%- if start > 0 -%}\n    {%- set start = start - 1 -%}\n{%- endif -%}\n\n{%- if end == 0 or end > lines | length -%}\n    {%- set end = lines | length -%}\n{%- endif -%}\n\n{%- set lines = lines | slice(start=start, end=end) -%}\n\n{{- lines | join(sep=\"\\n\") | safe -}}\n"
  },
  {
    "path": "templates/shortcodes/spoiler.html",
    "content": "{# The `random_id` ensures that each instance of the shortcode has a \"unique\" id #}\n{# allowing individual interactive elements (like toggles) to function correctly. #}\n{# This avoids conflicts when multiple instances of the shortcode are used. #}\n{# More context: https://github.com/welpo/tabi/issues/82 #}\n{%- set random_id = get_random(end=100000) -%}\n{%- set fixed_blur = fixed_blur | default(value=false) -%}\n\n<label class=\"spoiler-container{% if fixed_blur %} fixed-blur{% endif %}\">\n    <input type=\"checkbox\" id=\"spoiler-{{random_id}}\" class=\"spoiler-toggle\" role=\"button\" aria-pressed=\"false\" />\n    <span class=\"spoiler-content\" title=\"reveal spoiler\" tabindex=\"0\">\n        <span class=\"spoiler-hidden\">{{ text | markdown | trim_start_matches(pat=\"<p>\") | trim_start_matches(pat=\"</p>\") | safe }}</span>\n    </span>\n</label>\n"
  },
  {
    "path": "templates/shortcodes/toc.html",
    "content": "{# Inserts special string to add the Table of Contents anywhere on a post #}\n<!-- toc -->\n"
  },
  {
    "path": "templates/shortcodes/wide_container.html",
    "content": "<div class=\"full-width\">\n    {{ body | markdown | safe }}\n</div>\n"
  },
  {
    "path": "templates/sitemap.xml",
    "content": "<?xml version=\"1.0\" encoding=\"UTF-8\"?>\n<?xml-stylesheet href=\"{{ get_url(path='/sitemap_style.xsl', trailing_slash=false) | safe }}\" type=\"text/xsl\"?>\n<urlset xmlns=\"http://www.sitemaps.org/schemas/sitemap/0.9\">\n    {%- for sitemap_entry in entries %}\n    <url>\n        <loc>{{ sitemap_entry.permalink | escape_xml | safe }}</loc>\n        {%- if sitemap_entry.updated %}\n        <lastmod>{{ sitemap_entry.updated }}</lastmod>\n        {%- endif %}\n    </url>\n    {%- endfor %}\n</urlset>\n"
  },
  {
    "path": "templates/tags/list.html",
    "content": "{% extends \"index.html\" %}\n\n{% block main_content %}\n\n{%- set title = macros_translate::translate(key=\"all_tags\", default=\"All tags\", language_strings=language_strings) -%}\n\n{{ macros_page_header::page_header(title=title)}}\n\n{% set tag_count = terms | length %}\n{% if config.extra.tag_sorting == \"frequency\" %}\n    {% set terms = terms | sort(attribute=\"pages\") | reverse %}\n{% elif config.extra.tag_sorting != \"name\" %}\n    {{ throw (message=\"Invalid tag_sorting option: \" ~ config.extra.tag_sorting ~ \". Valid options are 'name' and 'frequency'.\") }}\n{% endif %}\n<div id=\"tag-cloud\" class=\"{% if tag_count > 16 %}three-columns{% elif tag_count > 8 %}two-columns{% endif %}\">\n    <ul class=\"tags\">\n        {%- for term in terms -%}\n            <li class=\"tags-item\">\n                {%- set number_of_posts = term.pages | length -%}\n                {% if config.extra.compact_tags %}\n                    {# Shows the number of posts per tag as a superscript #}\n                    <a href=\"{{ term.permalink | safe }}\"\n                       aria-label=\"{{ term.name }} –\n                        {{ term.pages | length }}\n                        {{- macros_translate::translate(key=\"posts\", number=number_of_posts, default=\"$NUMBER posts\", language_strings=language_strings) -}}\n                    \">\n                        {{ term.name }}\n                    </a> <sup>{{ number_of_posts }}</sup>\n                {% else %}\n                    <a href=\"{{ term.permalink | safe }}\">\n                        {{ term.name }} </a>\n                    <span> – </span>\n                    {{- macros_translate::translate(key=\"posts\", number=number_of_posts, default=\"$NUMBER posts\", language_strings=language_strings) -}}\n                {% endif %}\n            </li>\n        {%- endfor -%}\n    </ul>\n</div>\n\n{% endblock main_content %}\n"
  },
  {
    "path": "templates/tags/single.html",
    "content": "{% extends \"index.html\" %}\n\n{% block main_content %}\n\n{#- Feed icon -#}\n{%- set generate_feed = feed_utils::get_generate_feed() == \"true\" -%}\n{%- set feed_url = feed_utils::get_feed_url() -%}\n{%- set feed_pre_conditions = generate_feed and feed_url and taxonomy.feed -%}\n{%- set show_feed_icon = feed_pre_conditions and term.pages | filter(attribute=\"date\") -%}\n\n{{ macros_page_header::page_header(title=term.name, show_feed_icon=show_feed_icon) }}\n\n{% set max = section.extra.max_posts | default(value=999999) %}\n{{ macros_list_posts::list_posts(posts=term.pages, max=max, language_strings=language_strings) }}\n\n<ul class=\"pagination\">\n    <li class=\"page-item\">\n        <a class=\"all-tags\" href=\"{{ get_url(path=\"tags\", lang=lang) }}/\"><span class=\"arrow\">←</span>&nbsp;{{- macros_translate::translate(key=\"all_tags\", default=\"All tags\", language_strings=language_strings) -}}</a>\n    </li>\n</ul>\n\n{% endblock main_content %}\n"
  },
  {
    "path": "templates/taxonomy_list.html",
    "content": "{% extends \"index.html\" %}\n\n{% block main_content %}\n\n{%- set title = macros_translate::translate(key=taxonomy.name, default=taxonomy.name, language_strings=language_strings) -%}\n\n{{ macros_page_header::page_header(title=title)}}\n\n{% set tag_count = terms | length %}\n{% if config.extra.tag_sorting == \"frequency\" %}\n    {% set terms = terms | sort(attribute=\"pages\") | reverse %}\n{% elif config.extra.tag_sorting != \"name\" %}\n    {{ throw (message=\"Invalid tag_sorting option: \" ~ config.extra.tag_sorting ~ \". Valid options are 'name' and 'frequency'.\") }}\n{% endif %}\n<div id=\"tag-cloud\" class=\"{% if tag_count > 16 %}three-columns{% elif tag_count > 8 %}two-columns{% endif %}\">\n    <ul class=\"tags\">\n        {%- for term in terms -%}\n            <li class=\"tags-item\">\n                {%- set number_of_posts = term.pages | length -%}\n                {% if config.extra.compact_tags %}\n                    {# Shows the number of posts per tag as a superscript #}\n                    <a href=\"{{ term.permalink | safe }}\"\n                       aria-label=\"{{ term.name }} –\n                        {{ term.pages | length }}\n                        {{- macros_translate::translate(key=\"posts\", number=number_of_posts, default=\"$NUMBER posts\", language_strings=language_strings) -}}\n                    \">\n                        {{ term.name }}\n                    </a> <sup>{{ number_of_posts }}</sup>\n                {% else %}\n                    <a href=\"{{ term.permalink | safe }}\">\n                        {{ term.name }} </a>\n                    <span> – </span>\n                    {{- macros_translate::translate(key=\"posts\", number=number_of_posts, default=\"$NUMBER posts\", language_strings=language_strings) -}}\n                {% endif %}\n            </li>\n        {%- endfor -%}\n    </ul>\n</div>\n\n{% endblock main_content %}\n"
  },
  {
    "path": "templates/taxonomy_single.html",
    "content": "{% extends \"index.html\" %}\n\n{% block main_content %}\n\n{#- Feed icon -#}\n{%- set generate_feed = feed_utils::get_generate_feed() == \"true\" -%}\n{%- set feed_url = feed_utils::get_feed_url() -%}\n{%- set feed_pre_conditions = generate_feed and feed_url and taxonomy.feed -%}\n{%- set show_feed_icon = feed_pre_conditions and term.pages | filter(attribute=\"date\") | length > 0 -%}\n\n{{ macros_page_header::page_header(title=term.name, show_feed_icon=show_feed_icon) }}\n\n{% set max = section.extra.max_posts | default(value=999999) %}\n{{ macros_list_posts::list_posts(posts=term.pages, max=max, language_strings=language_strings) }}\n\n<ul class=\"pagination\">\n    <li class=\"page-item\">\n        <a class=\"all-tags\" href=\"{{ get_url(path=\"tags\", lang=lang) }}/\"><span class=\"arrow\">←</span>&nbsp;{{- macros_translate::translate(key=taxonomy.name, default=taxonomy.name, language_strings=language_strings) -}}</a>\n    </li>\n</ul>\n\n{% endblock main_content %}\n"
  },
  {
    "path": "theme.toml",
    "content": "name = \"tabi\"\ndescription = \"tabi is an accessible Zola theme with search, multi-language support, optional JavaScript, a perfect Lighthouse score, and comprehensive documentation. Crafted for personal websites and blogs.\"\nlicense = \"MIT\"\nhomepage = \"https://github.com/welpo/tabi\"\n\n# The minimum version of Zola required\nmin_version = \"0.17.0\"\n\n# An optional live demo URL\ndemo = \"https://welpo.github.io/tabi\"\n\n# The theme author info: you!\n[author]\nname = \"Óscar Fernández\"\nhomepage = \"https://osc.garden\"\n\n# Any variable there can be overridden in the end user `config.toml`\n# You don't need to prefix variables by the theme name but as this will\n# be merged with user data, some kind of prefix or nesting is preferable\n# Use snake_casing to be consistent with the rest of Zola\n[extra]\n# Check out the documentation (or the comments below) to learn how to customise tabi:\n# https://welpo.github.io/tabi/blog/mastering-tabi-settings/\n\n# Search index format.\n# For Zola 0.17.X users only.\n# This MUST MATCH the setting in [search].index_format.\n# Example: If [search].index_format is \"elasticlunr_javascript\", set:\n# index_format = \"elasticlunr_javascript\" here.\n# index_format = \"\"\n\n# Use sans-serif font everywhere.\n# By default, the serif font is only used in articles.\noverride_serif_with_sans = false\n\n# Enable JavaScript theme toggler to allow users to switch between dark/light mode.\n# If disabled, your site will use the theme specified in the `default_theme` variable.\ntheme_switcher = true\n\n# This setting determines the default theme on load (\"light\" or \"dark\").\n# To follow the user's OS theme, leave it empty or unset.\ndefault_theme = \"\"\n\n# Choose the colourscheme (skin) for the theme. Default is \"teal\".\n# Skin available: blue, lavender, mint, red, sakura, teal, monochrome, lowcontrast_orange, lowcontrast_peach, lowcontrast_pink, indigo_ingot, evangelion\n# See them live and learn how to create your own: https://welpo.github.io/tabi/blog/customise-tabi/#skins\n# WARNING! \"lowcontrast\" skins, while aesthetically pleasing, may not provide optimal\n# contrast (in light theme) for readability and might not be suitable for all users.\n# Furthermore, low contrasting elements will affect your Google Lighthouse rating.\n# All other skins have optimal contrast.\nskin = \"\"\n\n# Set browser theme colour. Can be a single colour or [light, dark].\n# Note: Bright colors may be ignored in dark mode.\n# More details: https://developer.mozilla.org/en-US/docs/Web/HTML/Element/meta/name/theme-color\n# browser_theme_color = \"#087e96\"  # Example of single value.\n# browser_theme_color = [\"#ffffff\", \"#000000\"]  # Example of light/dark colours.\n\n# Show current language code on the language switcher\nshow_selected_language_code_in_language_switcher = false\n\n# List additional stylesheets to load site-wide.\n# These stylesheets should be located in your site's `static` directory.\n# Example: stylesheets = [\"extra1.css\", \"path/extra2.css\"]\n# You can load a stylesheet for a single post by adding it to the [extra] section of the post's front matter, following this same format.\nstylesheets = []\n\n# Sets the default canonical URL for all pages.\n# Individual pages can override this in the [extra] section using canonical_url.\n# Example: \"$base_url/blog/post1\" will get the canonical URL \"https://example.com/blog/post1\".\n# Note: To ensure accuracy in terms of matching content, consider setting 'canonical_url' individually per page.\n# base_canonical_url = \"https://example.com\"\n\n# Remote repository for your Zola site.\n# Used for `show_remote_changes` and `show_remote_source` (see below).\n# Supports GitHub, GitLab, Gitea, and Codeberg.\n# remote_repository_url = \"https://github.com/welpo/tabi\"\n# Set this to \"auto\" to try and auto-detect the platform based on the repository URL.\n# Accepted values are \"github\", \"gitlab\", \"gitea\", and \"codeberg\".\nremote_repository_git_platform = \"auto\"  # Defaults to \"auto\".\n# Branch in the repo hosting the Zola site.\nremote_repository_branch = \"main\"  # Defaults to \"main\".\n# Show a link to the commit history of updated posts, right next to the last updated date.\n# Can be set at page or section levels, following the hierarchy: page > section > config. See: https://welpo.github.io/tabi/blog/mastering-tabi-settings/#settings-hierarchy\nshow_remote_changes = true  # Defaults to true.\n# Show a link to the repository of the site, right next to the \"Powered by Zola & tabi\" text.\nshow_remote_source = true  # Defaults to true.\n\n# Add a \"copy\" button to codeblocks (loads ~700 bytes of JavaScript).\n# Can be set at page or section levels, following the hierarchy: page > section > config. See: https://welpo.github.io/tabi/blog/mastering-tabi-settings/#settings-hierarchy\ncopy_button = true\n\n# Make code block names clickable if they are URLs (loads ~400 bytes of JavaScript).\n# Can be set at page or section levels, following the hierarchy: page > section > config. See: https://welpo.github.io/tabi/blog/mastering-tabi-settings/#settings-hierarchy\ncode_block_name_links = false\n\n# Force left-to-right (LTR) direction for code blocks.\n# Set to false to allow code to follow the document's natural direction.\n# Can be set at page or section levels. See: https://welpo.github.io/tabi/blog/mastering-tabi-settings/#settings-hierarchy\nforce_codeblock_ltr = true\n\n# Show the author(s) of a page.\n# Can be set at page or section levels, following the hierarchy: page > section > config. See: https://welpo.github.io/tabi/blog/mastering-tabi-settings/#settings-hierarchy\nshow_author = false\n\n# Show the reading time of a page.\n# Can be set at page or section levels, following the hierarchy: page > section > config. See: https://welpo.github.io/tabi/blog/mastering-tabi-settings/#settings-hierarchy\nshow_reading_time = true\n\n# Show the date of a page below its title.\n# Can be set at page or section levels, following the hierarchy: page > section > config. See: https://welpo.github.io/tabi/blog/mastering-tabi-settings/#settings-hierarchy\nshow_date = true\n\n# Determines how dates are displayed in the post listing (e.g. front page or /blog). Options:\n# \"date\" - Show only the original date of the post (default if unset).\n# \"updated\" - Show only the last updated date of the post. If there is no last updated date, it shows the original date.\n# \"both\" - Show both the original date and the last updated date.\npost_listing_date = \"date\"\n\n# Enable iine like buttons on all posts: https://iine.to/\n# Can be set at page or section levels, following the hierarchy: page > section > config. See: https://welpo.github.io/tabi/blog/mastering-tabi-settings/#settings-hierarchy\niine = false\niine_icon = \"heart\"  # See https://iine.to/#customise\n# Unify like counts across all language versions of the same page.\n# When enabled, likes on /es/blog/hello/ will count towards /blog/hello/ (default language).\niine_unified_languages = true\n\n# Show \"Jump to posts\" link next to series' title.\n# By default, the link appears automatically when a series description exceeds 2000 characters.\n# Can be set at page or section levels, following the hierarchy: page > section > config. See: https://welpo.github.io/tabi/blog/mastering-tabi-settings/#settings-hierarchy\n# show_jump_to_posts = true\n\n# Determines if indexes should be increasing (false) or decreasing (true) in series' posts list.\n# It has only effect if the section uses indexes metadata (which is only the case for series as of now).\n# Can be set at section levels, following the hierarchy: section > config. See: https://welpo.github.io/tabi/blog/mastering-tabi-settings/#settings-hierarchy\npost_listing_index_reversed = false  # Defaults to false.\n\n# DEPRECATED!\n# Use Zola's built-in `bottom_footnotes = true` in the [markdown] section instead. (Available since v0.19.0)\n# Adds backlinks to footnotes (loads ~500 bytes of JavaScripts).\n# Can be set at page or section levels, following the hierarchy: page > section > config. See: https://welpo.github.io/tabi/blog/mastering-tabi-settings/#settings-hierarchy\nfootnote_backlinks = false\n\n# Enable KaTeX for all posts.\n# Can be set at page or section levels, following the hierarchy: page > section > config. See: https://welpo.github.io/tabi/blog/mastering-tabi-settings/#settings-hierarchy\nkatex = false\n\n# Enable Mermaid diagrams for all posts.\n# Loads ~2.5MB of JavaScript.\n# Can be set at page or section levels, following the hierarchy: page > section > config. See: https://welpo.github.io/tabi/blog/mastering-tabi-settings/#settings-hierarchy\nmermaid = false\n\n# Serve Mermaid JavaScript locally. Version bundled with tabi.\n# If set to false, it will load the latest version from JSDelivr.\n# Only relevant when `mermaid = true`.\nserve_local_mermaid = true\n\n# Show links to previous and next articles at the bottom of posts.\n# Can be set at page or section levels, following the hierarchy: page > section > config. See: https://welpo.github.io/tabi/blog/mastering-tabi-settings/#settings-hierarchy\nshow_previous_next_article_links = false\n\n# Invert order of the links to previous and next articles at the bottom of posts.\n# By default, next articles are on the left side of the page and previous articles are on the right side.\n# To reverse the order (next articles on the right and previous articles on the left), set it to true.\n# Can be set at page or section levels, following the hierarchy: page > section > config. See: https://welpo.github.io/tabi/blog/mastering-tabi-settings/#settings-hierarchy\ninvert_previous_next_article_links = false\n\n# Whether the navigation for previous/next article should match the full width of the site (same as the navigation bar at the top) or the article width.\n# To match the navigation bar at the top, set it to true.\nprevious_next_article_links_full_width = true\n\n# Quick navigation buttons.\n# Adds \"go up\" and \"go to comments\" buttons on the bottom right (hidden for mobile).\n# Can be set at page or section levels, following the hierarchy: page > section > config. See: https://welpo.github.io/tabi/blog/mastering-tabi-settings/#settings-hierarchy\nquick_navigation_buttons = false\n\n# Add a Table of Contents to all posts, right below the title and metadata.\n# Can be set at page or section levels, following the hierarchy: page > section > config. See: https://welpo.github.io/tabi/blog/mastering-tabi-settings/#settings-hierarchy\n# toc = false\n\n# Date format used when listing posts (main page, /blog section, tag posts list…)\n# Default is \"6th July 2049\" in English and \"%d %B %Y\" in other languages.\n# long_date_format = \"%d %B %Y\"\n\n# Date format used for blog posts.\n# Default is \"6th July 2049\" in English and \"%-d %B %Y\" in other languages.\nshort_date_format = \"\"\n\n# Date format used for the archive page.\n# Default is \"06 July\" in English and \"%d %b\" in other languages.\n# archive_date_format = \"\"\n\n# Per-language date format overrides.\n# Examples: Spanish uses \"3 de febrero de 2024\", German uses \"3. Februar 2024\"\n# date_formats = [\n#     { lang = \"es\", long = \"%d de %B de %Y\", short = \"%d %b %Y\" },\n#     { lang = \"de\", long = \"%d. %B %Y\", short = \"%d.%m.%Y\" },\n# ]\n\n# Custom separator used in title tag and posts metadata (between date, time to read, and tags).\nseparator = \"•\"\n\n# Use a shorter layout for All tags listing.\n# Default: tag_name – n post[s]\n# Compact: tag_name^n (superscript number)\ncompact_tags = false\n\n# How tags are sorted in a Tags listing based on templates/tags/list.html.\n# \"name\" for alphabetical, \"frequency\" for descending count of posts.\n# Default: \"name\".\ntag_sorting = \"name\"\n\n# Show clickable tags above cards.html template (e.g. projects/) to filter the displayed items.\n# Loads JS to filter. If JS is disabled, the buttons are links to the tag's page.\n# Can be set at the section or config.toml level, following the hierarchy: section > config. See: https://welpo.github.io/tabi/blog/mastering-tabi-settings/#settings-hierarchy\n# Default: true\nenable_cards_tag_filtering = true\n\n# Invert the order of the site title and page title in the browser tab.\n# Example: true => \"Blog • ~/tabi\", false => \"~/tabi • Blog\"\ninvert_title_order = false\n\n# Full path after the base URL required. So if you were to place it in \"static\" it would be \"/favicon.ico\"\n# favicon = \"\"\n\n# Add an emoji here to use it as favicon.\n# Compatibility: https://caniuse.com/link-icon-svg\n# favicon_emoji = \"🌱\"\n\n# Path to the fallback image for social media cards (the preview image shown when sharing a link on WhatsApp, LinkedIn…).\n# Can be set at page or section levels, following the hierarchy: page > section > config. See: https://welpo.github.io/tabi/blog/mastering-tabi-settings/#settings-hierarchy\n# Learn how to create these images in batch and automatically:\n# https://osc.garden/blog/automating-social-media-cards-zola/\n# social_media_card = \"img/social_cards/index.jpg\"\n\nmenu = [\n    { name = \"blog\", url = \"blog\", trailing_slash = true },\n    { name = \"archive\", url = \"archive\", trailing_slash = true },\n    { name = \"tags\", url = \"tags\", trailing_slash = true },\n    { name = \"projects\", url = \"projects\", trailing_slash = true },\n]\n\n# The RSS icon will be shown if (1) it's enabled and (2) the following variable is set.\n# Set to true to use the default \"rss\" icon, or specify a custom icon name (e.g. \"square-rss\").\n# The icon must exist in static/social_icons/ (without the .svg extension).\n# Note for Zola 0.19.X users: when `feed_filenames` has two filenames, only the first one will be linked in the footer.\nfeed_icon = true\n\n# Show the full post content in the Atom feed.\n# If it's set to false, only the description or summary will be shown.\nfull_content_in_feed = false\n\n# Email address for footer's social section.\n# Protect against spambots:\n# 1. Use base64 for email (convert at https://www.base64encode.org/ or `printf 'your@email.com' | base64`).\n# 2. Or, set 'encode_plaintext_email' to true for auto-encoding (only protects on site, not in public repos).\n# email = \"bWFpbEBleGFtcGxlLmNvbQ==\"  # base64 encoded version of \"mail@example.com\"\n# Decoding requires ~400 bytes of JavaScript. If JS is disabled, the email won't be displayed.\nencode_plaintext_email = true  # Setting is ignored if email is already encoded.\n\n# Social media links for the footer.\n# Built-in icons: https://github.com/welpo/tabi/tree/main/static/social_icons\n# To use a custom icon, add it to your site's `static/social_icons` directory.\n# socials = [\n#     { name = \"github\", url = \"https://github.com/welpo/\", icon = \"github\" },\n#     { name = \"soundcloud\", url = \"https://soundcloud.com/oskerwyld\", icon = \"soundcloud\" },\n#     { name = \"instagram\", url = \"https://instagram.com/oskerwyld\", icon = \"instagram\" },\n#     { name = \"youtube\", url = \"https://youtube.com/@oskerwyld\", icon = \"youtube\" },\n#     { name = \"spotify\", url = \"https://open.spotify.com/artist/5Hv2bYBhMp1lUHFri06xkE\", icon = \"spotify\" },\n# ]\n\n# Fediverse profile.\n# Adds metadata to feature the author's profile in Mastodon link previews.\n# Example: for @username@example.com, use:\n# fediverse_creator = { handle = \"username\", domain = \"example.com\" }\n\n# Extra menu to show on the footer, below socials section.\n# footer_menu = [\n#     {url = \"about\", name = \"about\", trailing_slash = true},\n#     {url = \"privacy\", name = \"privacy\", trailing_slash = true},\n#     {url = \"sitemap.xml\", name = \"sitemap\", trailing_slash = false},\n#     {url = \"https://example.com\", name = \"external link\", trailing_slash = true},\n# ]\n\n# Enable a copyright notice for the footer, shown between socials and the \"Powered by\" text.\n# $TITLE will be replaced by the website's title.\n# $CURRENT_YEAR will be replaced by the current year.\n# $AUTHOR will be replaced by the `author` variable.\n# $SEPARATOR will be replaced by the `separator` variable.\n# Markdown is supported (links, emphasis, etc).\n# copyright = \"$TITLE © $CURRENT_YEAR $AUTHOR $SEPARATOR Unless otherwise noted, the content in this website is available under the [CC BY-SA 4.0](https://creativecommons.org/licenses/by-sa/4.0/) license.\"\n\n# For multi-language sites, you can set a different copyright for each language.\n# The old way of setting `translated_copyright = true` and using i18n files is deprecated.\n# If a translation is missing for language, the `copyright` value will be used.\n# copyright_translations.es = \"$TITLE © $CURRENT_YEAR $AUTHOR $SEPARATOR A menos que se indique lo contrario, el contenido de esta web está disponible bajo la licencia [CC BY-SA 4.0](https://creativecommons.org/licenses/by-sa/4.0/).\"\n\n# Custom security headers. What urls should your website be able to connect to?\n# You need to specify the CSP and the URLs associated with the directive.\n# Useful if you want to load remote content safely (embed YouTube videos, which needs frame-src, for example).\n# Default directive is self.\n# Default config, allows for https remote images and embedding YouTube and Vimeo content.\n# This configuration (along with the right webserver settings) gets an A+ in Mozilla's Observatory: https://observatory.mozilla.org\n# Note: to use a Zola built-in syntax highlighting theme, allow unsafe-inline for style-src.\nallowed_domains = [\n    { directive = \"font-src\", domains = [\"'self'\", \"data:\"] },\n    { directive = \"img-src\", domains = [\"'self'\", \"https://*\", \"data:\"] },\n    { directive = \"media-src\", domains = [\"'self'\"] },\n    { directive = \"script-src\", domains = [\"'self'\"] },\n    { directive = \"style-src\", domains = [\"'self'\"] },\n    { directive = \"frame-src\", domains = [\"player.vimeo.com\", \"https://www.youtube-nocookie.com\"] },\n]\n\n# Enable the CSP directives configured (or default).\n# Can be set at page or section levels, following the hierarchy: page > section > config. See: https://welpo.github.io/tabi/blog/mastering-tabi-settings/#settings-hierarchy\nenable_csp = true\n\n# Font subsetting configuration.\n# This feature helps prevent text flashing in Firefox when using custom fonts.\n# See: https://welpo.github.io/tabi/blog/custom-font-subset/\n# Enable or disable font subsetting completely, both built-in and custom subsets.\nenable_subset = true\n# Use a custom subset of characters for the header.\n# If true, tabi will load the `static/custom_subset.css` file.\n# If false, tabi will use the default language-specific subset (English or Spanish).\ncustom_subset = true\n\n[extra.analytics]\n# Specify which analytics service you want to use.\n# Supported options: [\"goatcounter\", \"umami\", \"plausible\"]\n# service = \"umami\"\n\n# Unique identifier for tracking.\n# For GoatCounter, this is the code you choose during signup.\n# For Umami, this is the website ID.\n# For Plausible, this is the random script name without the extension (e.g. \"pa-XXXXXX\") or the domain name (legacy mode, e.g. \"example.com\").\n# Note: Leave this field empty if you're self-hosting GoatCounter.\n# id = \"yourID\"\n\n# Optional: Specify the URL for self-hosted analytics instances.\n# For GoatCounter: Base URL like \"https://stats.example.com\"\n# For Umami: Base URL like \"https://umami.example.com\"\n# For Plausible: Base URL like \"https://plausible.example.com\"\n# Leave this field empty if you're using the service's default hosting.\n# self_hosted_url = \"\"\n\n# Optional: For Umami, enable this option to respect users' Do Not Track (DNT) settings. The default is true.\ndo_not_track = true\n\n# giscus support for comments. https://giscus.app\n# Setup instructions: https://welpo.github.io/tabi/blog/comments/#setup\n[extra.giscus]\n# enabled_for_all_posts = false  # Enables giscus on all posts. It can be enabled on individual posts by setting `giscus = true` in the [extra] section of a post's front matter.\n# automatic_loading = true  # If set to false, a \"Load comments\" button will be shown.\n# repo = \"welpo/tabi-comments\"\n# repo_id = \"R_kgDOJ59Urw\"  # Find this value in https://giscus.app/\n# category = \"Announcements\"\n# category_id = \"DIC_kwDOJ59Ur84CX0QG\"  # Find this value in https://giscus.app/\n# mapping = \"slug\"  # Available: pathname; url; title; slug. \"slug\" will use the post's filename (slug); this is the only way to share comments between languages.\n# strict_title_matching = 1  # 1 to enable, 0 to disable. https://github.com/giscus/giscus/blob/main/ADVANCED-USAGE.md#data-strict\n# enable_reactions = 1  # 1 to enable, 0 to disable.\n# comment_box_above_comments = false\n# light_theme = \"noborder_light\"\n# dark_theme = \"noborder_dark\"\n# lang = \"\"  # Leave blank to match the page's language.\n# lazy_loading = true\n\n# utterances support for comments. https://utteranc.es\n# Setup instructions: https://welpo.github.io/tabi/blog/comments/#setup\n[extra.utterances]\n# enabled_for_all_posts = false  # Enables utterances on all posts. It can be enabled on individual posts by setting `utterances = true` in the [extra] section of a post's front matter.\n# automatic_loading = true  # If set to false, a \"Load comments\" button will be shown.\n# repo = \"yourGithubUsername/yourRepo\"  # https://utteranc.es/#heading-repository\n# issue_term = \"slug\"  # Available: pathname; url; title; slug. \"slug\" will use the post's filename (slug); this is the only way to share comments between languages. https://utteranc.es/#heading-mapping\n# label = \"💬\"  # https://utteranc.es/#heading-issue-label\n# light_theme = \"github-light\"  # https://utteranc.es/#heading-theme\n# dark_theme = \"photon-dark\"  # https://utteranc.es/#heading-theme\n# lazy_loading = true\n\n# Hyvor Talk support for comments. https://talk.hyvor.com\n[extra.hyvortalk]\n# enabled_for_all_posts = false  # Enables hyvortalk on all posts. It can be enabled on individual posts by setting `hyvortalk = true` in the [extra] section of a post's front matter.\n# automatic_loading = true  # If set to false, a \"Load comments\" button will be shown.\n# website_id = \"1234\"\n# page_id_is_slug = true  # If true, it will use the post's filename (slug) as id; this is the only way to share comments between languages. If false, it will use the entire url as id.\n# lang = \"\"  # Leave blank to match the page's language.\n# page_author = \"\"  # Email (or base64 encoded email) of the author.\n# lazy_loading = true\n\n# Isso support for comments. https://isso-comments.de/\n# You need to self-host the backend first: https://blog.phusion.nl/isso-simple-self-hosted-commenting-system/\n# More info on some settings: https://isso-comments.de/docs/reference/client-config/\n[extra.isso]\n# enabled_for_all_posts = false  # Enables Isso on all posts. It can be enabled on individual posts by setting `isso = true` in the [extra] section of a post's front matter.\n# automatic_loading = true  # If set to false, a \"Load comments\" button will be shown.\n# endpoint_url = \"\"  # Accepts relative paths like \"/comments/\" or \"/isso/\", as well as full urls like \"https://example.com/comments/\". Include the trailing slash.\n# page_id_is_slug = true  # If true, it will use the relative path for the default language as id; this is the only way to share comments between languages. If false, it will use the entire url as id.\n# lang = \"\"  # Leave blank to match the page's language.\n# max_comments_top = \"inf\"  # Number of top level comments to show by default. If some comments are not shown, an “X Hidden” link is shown.\n# max_comments_nested = \"5\"  # Number of nested comments to show by default. If some comments are not shown, an “X Hidden” link is shown.\n# avatar = true\n# voting = true\n# page_author_hashes = \"\"  # hash (or list of hashes) of the author.\n# lazy_loading = true  # Loads when the comments are in the viewport (using the Intersection Observer API).\n\n[extra.webmentions]\n# To disable for a specific section or page, set webmentions = false in that page/section's front matter's [extra] section.\nenable = false\n# Specify the domain registered with webmention.io.\n# domain = \"\"\n\n# The HTML ID for the object to fill in with the webmention data.\n# Defaults to \"webmentions\"\n# id = \"webmentions\"\n\n# data configuration for the webmention.min.js script\n# The base URL to use for this page. Defaults to window.location\n# page_url =\n\n# Additional URLs to check, separated by |s\n# add_urls\n\n# The maximum number of words to render in reply mentions.\n# wordcount = 20\n\n# The maximum number of mentions to retrieve. Defaults to 30.\n# max_webmentions = 30\n\n# By default, Webmentions render using the mf2 'url' element, which plays\n# nicely with webmention bridges (such as brid.gy and telegraph)\n# but allows certain spoofing attacks. If you would like to prevent\n# spoofing, set this to a non-empty string (e.g. \"true\").\n# prevent_spoofing\n\n# What to order the responses by; defaults to 'published'. See\n# https://github.com/aaronpk/webmention.io#api\n# sort_by\n\n# The order to sort the responses by; defaults to 'up' (i.e. oldest\n# first). See https://github.com/aaronpk/webmention.io#api\n# sort_dir\n\n# If set to a non-empty string (e.g. \"true\"), will display comment-type responses\n# (replies/mentions/etc.) as being part of the reactions\n# (favorites/bookmarks/etc.) instead of in a separate comment list.\n# comments_are_reactions = \"true\"\n\n# h-card configuration\n# Will identify you on the indieweb (see https://microformats.org/wiki/h-card)\n[extra.hcard]\n# Enable home page h-card.\nenable = true\n# Add your email to the card if extra.email is set and not encoded.\n# with_mail = true\n# Add your social links ('socials' config) to the card.\nwith_social_links = true\n# Homepage url. Defaults to the value of 'base_url'.\n# homepage = \"https://myhomepage.net\"\n# avatar = \"img/profile.webp\"\n# Display name, default to the value of 'author'.\n# full_name = \"John Doe\"\n# Small bio, as shown on social media profiles.\n# biography = \"Fond of the indieweb\"\n#\n# You can add any property from https://microformats.org/wiki/h-card#Properties\n# Make sure to replace all '-' characters by '_'\n# Examples:\n# p_nickname = \"nickname\"\n# p_locality = \"Bordeaux\"\n# p_country_name = \"France\"\n"
  }
]