Compare commits
88 Commits
| Author | SHA1 | Date | |
|---|---|---|---|
|
559ff6a7b3
|
|||
|
3e3bb67884
|
|||
|
d0935a474f
|
|||
|
a57bc8e3b5
|
|||
| 08622bd092 | |||
| a8b2df43bd | |||
|
b249ae9d46
|
|||
| b4badfbd15 | |||
| cb0fe96654 | |||
| 6b04b75314 | |||
|
6076b5f6f2
|
|||
|
5036ff79f7
|
|||
|
50ca8ab5bf
|
|||
|
3f427dfa32
|
|||
|
9b7c3dc697
|
|||
|
c732b49339
|
|||
|
9829a74f95
|
|||
|
372b960261
|
|||
|
c671c2a4bc
|
|||
|
8088b2e69c
|
|||
|
e4d63de680
|
|||
|
0e8287e4a8
|
|||
|
48950d1d82
|
|||
|
e1d74c3091
|
|||
|
f57d81f508
|
|||
|
67a9058b7d
|
|||
|
79e34d7e43
|
|||
|
0cda1706fa
|
|||
|
85467f4f1e
|
|||
|
00ccb96581
|
|||
|
5227beb02a
|
|||
|
22f9cb4e00
|
|||
|
dbf3e5c27a
|
|||
|
d9ca40bf77
|
|||
|
b8cbc49647
|
|||
|
b80e98a4ec
|
|||
|
350f41f439
|
|||
|
00b5020642
|
|||
|
492ea8755f
|
|||
|
39da474db6
|
|||
|
44bcd5607f
|
|||
|
5ff44a1912
|
|||
|
4241f3965a
|
|||
|
002e9c62db
|
|||
| cf494401c8 | |||
|
2a0323e579
|
|||
|
7e23b3625a
|
|||
|
bce51dc3d6
|
|||
| 57250b3adc | |||
| 08895902ec | |||
| 0107c4d38c | |||
| 6595cd11d6 | |||
| 08b53eb2ac | |||
| d71b6afd7f | |||
| ec32c24af3 | |||
| 122117f125 | |||
| 3b93fb8719 | |||
| b601b67b6e | |||
| 03a9e46baa | |||
|
0a39ecd637
|
|||
|
bc4fefa3f1
|
|||
|
156a886ac1
|
|||
| f6cf89a37f | |||
|
cc783ee30e
|
|||
|
c429cd06c8
|
|||
|
35f2ea7536
|
|||
| 0ec663ae68 | |||
| 0973868782 | |||
| cd06c526af | |||
| d155b05798 | |||
| 3352894e4d | |||
| d743ede95d | |||
| 9403e84d78 | |||
| 961d79754e | |||
| 383dd59851 | |||
| 549c15ca6c | |||
| 57d949677f | |||
| cc6ad14506 | |||
| cace0f8593 | |||
| 0484200fad | |||
| 74d97dac2a | |||
| 6c636905e2 | |||
| c79473d646 | |||
|
3cb5269985
|
|||
|
3bde09aebc
|
|||
|
effc05826a
|
|||
|
8ec9701aa9
|
|||
| a95a6e1722 |
4
.github/workflows/build-release.yml
vendored
@@ -12,7 +12,7 @@ jobs:
|
|||||||
- name: Install Dependencies
|
- name: Install Dependencies
|
||||||
run: pip install -r requirements.txt
|
run: pip install -r requirements.txt
|
||||||
- name: Build Package
|
- name: Build Package
|
||||||
run: pyinstaller builder.py modules/*.py -n StaticGalleryBuilder -F --add-data files:files --add-data templates:templates --add-data .version:.
|
run: pyinstaller builder.py modules/*.py -n StaticGalleryBuilder-$(cat .version)-linux -F --add-data files:files --add-data templates:templates --add-data .version:.
|
||||||
- name: Release
|
- name: Release
|
||||||
uses: softprops/action-gh-release@v2
|
uses: softprops/action-gh-release@v2
|
||||||
if: startsWith(github.ref, 'refs/tags/')
|
if: startsWith(github.ref, 'refs/tags/')
|
||||||
@@ -20,4 +20,4 @@ jobs:
|
|||||||
make_latest: true
|
make_latest: true
|
||||||
generate_release_notes: true
|
generate_release_notes: true
|
||||||
files: |
|
files: |
|
||||||
dist/StaticGalleryBuilder
|
dist/StaticGalleryBuilder*
|
||||||
|
|||||||
5
.gitignore
vendored
@@ -164,6 +164,9 @@ cython_debug/
|
|||||||
test/.static
|
test/.static
|
||||||
test/.thumbnails
|
test/.thumbnails
|
||||||
test/**/index.html
|
test/**/index.html
|
||||||
test/**/.sizelist.json
|
test/**/license.html
|
||||||
|
test/**/.metadata.json
|
||||||
test/manifest.json
|
test/manifest.json
|
||||||
themes/previews
|
themes/previews
|
||||||
|
logs
|
||||||
|
.lock
|
||||||
10
.hintrc
@@ -3,6 +3,14 @@
|
|||||||
"development"
|
"development"
|
||||||
],
|
],
|
||||||
"hints": {
|
"hints": {
|
||||||
"apple-touch-icons": "off"
|
"apple-touch-icons": "off",
|
||||||
|
"compat-api/css": [
|
||||||
|
"default",
|
||||||
|
{
|
||||||
|
"ignore": [
|
||||||
|
"-webkit-app-region"
|
||||||
|
]
|
||||||
|
}
|
||||||
|
]
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
@@ -1 +1 @@
|
|||||||
3.12
|
StaticGalleryBuilder
|
||||||
|
|||||||
28
README.md
@@ -20,19 +20,19 @@ Python script to generate static HTML files for website galleries.
|
|||||||
## Requirements
|
## Requirements
|
||||||
|
|
||||||
- Python 3.x
|
- Python 3.x
|
||||||
- `numpy` library
|
|
||||||
- `tqdm` library
|
- `tqdm` library
|
||||||
- `Jinja2` library
|
- `Jinja2` library
|
||||||
- `Pillow` library
|
- `Pillow` library
|
||||||
- `rich_argparse` library
|
- `rich_argparse` library
|
||||||
- `cairosvg` library (for SVG to PNG icon conversion)
|
- `cairosvg` library (for SVG to PNG icon conversion)
|
||||||
|
- `python-json-logger` library (for logging)
|
||||||
|
|
||||||
## Installation
|
## Installation
|
||||||
|
|
||||||
Install the required libraries using pip:
|
Install the required libraries using pip:
|
||||||
|
|
||||||
```sh
|
```sh
|
||||||
pip install numpy tqdm Jinja2 Pillow rich-argparse cairosvg
|
pip install tqdm Jinja2 Pillow rich-argparse cairosvg python-json-logger
|
||||||
```
|
```
|
||||||
|
|
||||||
## Usage
|
## Usage
|
||||||
@@ -43,20 +43,22 @@ The script supports several command-line options to customize its behavior. Belo
|
|||||||
|
|
||||||
### Options
|
### Options
|
||||||
|
|
||||||
- `-h, --help`: Show the help message and exit.
|
|
||||||
- `-p ROOT, --root-directory ROOT`: Specify the root folder where the images are stored. This option is required.
|
|
||||||
- `-w URL, --web-root-url URL`: Specify the base URL for the web root of the image hosting site. This option is required.
|
|
||||||
- `-t TITLE, --site-title TITLE`: Specify the title of the image hosting site. This option is required.
|
|
||||||
- `-r, --regenerate-thumbnails`: Regenerate thumbnails even if they already exist.
|
|
||||||
- `-n, --non-interactive-mode`: Run in non-interactive mode, disabling progress bars.
|
|
||||||
- `-l LICENSE, --license-type LICENSE`: Specify the license type for the images. Choices are `cc-zero`, `cc-by`, `cc-by-sa`, `cc-by-nd`, `cc-by-nc`, `cc-by-nc-sa`, and `cc-by-nc-nd`.
|
|
||||||
- `-a AUTHOR, --author-name AUTHOR`: Specify the name of the author of the images. Default is "Author".
|
- `-a AUTHOR, --author-name AUTHOR`: Specify the name of the author of the images. Default is "Author".
|
||||||
- `-e EXTENSION, --file-extensions EXTENSION`: Specify the file extensions to include. This option can be specified multiple times.
|
- `-e EXTENSION, --file-extensions EXTENSION`: Specify the file extensions to include. This option can be specified multiple times.
|
||||||
|
- `-l LICENSE, --license-type LICENSE`: Specify the license type for the images. Choices are `cc-zero`, `cc-by`, `cc-by-sa`, `cc-by-nd`, `cc-by-nc`, `cc-by-nc-sa`, and `cc-by-nc-nd`.
|
||||||
|
- `-m, --web-manifest`: Generate a web manifest file.
|
||||||
|
- `-n, --non-interactive-mode`: Run in non-interactive mode, disabling progress bars.
|
||||||
|
- `-p ROOT, --root-directory ROOT`: Specify the root folder where the images are stored. **(This option is required)**.
|
||||||
|
- `-t TITLE, --site-title TITLE`: Specify the title of the image hosting site. **(This option is required)**.
|
||||||
|
- `-w URL, --web-root-url URL`: Specify the base URL for the web root of the image hosting site. **(This option is required)**.
|
||||||
|
- `--exclude-folder FOLDER`: Specify folders to exclude from processing. This option can be specified multiple times.
|
||||||
|
- `--ignore-other-files`: Ignore files that do not match the specified extensions.
|
||||||
|
- `--regenerate-thumbnails`: Regenerate thumbnails even if they already exist.
|
||||||
|
- `--reread-metadata`: Reread image metadata if it already exists.
|
||||||
|
- `--reread-sidecar`: Reread sidecar file data.
|
||||||
|
- `--reverse-sort`: Sort images by reverse name order.
|
||||||
- `--theme-path PATH`: Specify the path to the CSS theme file. Default is the provided default theme.
|
- `--theme-path PATH`: Specify the path to the CSS theme file. Default is the provided default theme.
|
||||||
- `--use-fancy-folders`: Enable fancy folder view instead of the default Apache directory listing.
|
- `--use-fancy-folders`: Enable fancy folder view instead of the default Apache directory listing.
|
||||||
- `--ignore-other-files`: Ignore files that do not match the specified extensions.
|
|
||||||
- `--exclude-folder FOLDER`: Specify folders to exclude from processing. This option can be specified multiple times.
|
|
||||||
- `-m, --web-manifest`: Generate a web manifest file.
|
|
||||||
|
|
||||||
### Examples
|
### Examples
|
||||||
|
|
||||||
@@ -95,6 +97,8 @@ To generate a web manifest file:
|
|||||||
- The root and web root paths must point to the same folder, one on the filesystem and one on the web server. Use absolute paths.
|
- The root and web root paths must point to the same folder, one on the filesystem and one on the web server. Use absolute paths.
|
||||||
- The script generates the preview thumbnails in a `.thumbnails` subdirectory within the root folder.
|
- The script generates the preview thumbnails in a `.thumbnails` subdirectory within the root folder.
|
||||||
- The `.lock` file prevents multiple instances of the script from running simultaneously. Make sure to remove it if the script terminates unexpectedly.
|
- The `.lock` file prevents multiple instances of the script from running simultaneously. Make sure to remove it if the script terminates unexpectedly.
|
||||||
|
- Add a `info` file into any directory containing pictures and it will be read and displayed as a tooltip on the website.
|
||||||
|
- Add tags to the Image xmp `subject` or to `.metadata.json` to tag images for filtering.
|
||||||
|
|
||||||
## License
|
## License
|
||||||
|
|
||||||
|
|||||||
@@ -1,24 +1,22 @@
|
|||||||
{
|
{
|
||||||
"extensions": {
|
"extensions": {
|
||||||
"recommendations": [
|
"recommendations": [
|
||||||
|
"charliermarsh.ruff",
|
||||||
"esbenp.prettier-vscode",
|
"esbenp.prettier-vscode",
|
||||||
"ms-edgedevtools.vscode-edge-devtools",
|
"ms-edgedevtools.vscode-edge-devtools",
|
||||||
"ms-python.black-formatter",
|
|
||||||
"ms-python.debugpy",
|
"ms-python.debugpy",
|
||||||
"ms-python.pylint",
|
|
||||||
"ms-python.python",
|
"ms-python.python",
|
||||||
"ms-python.vscode-pylance",
|
"ms-python.vscode-pylance",
|
||||||
"samuelcolvin.jinjahtml",
|
"samuelcolvin.jinjahtml",
|
||||||
"vscode.css-language-features",
|
"vscode.css-language-features",
|
||||||
"vscode.html-language-features",
|
"vscode.html-language-features"
|
||||||
"waderyan.gitblame",
|
]
|
||||||
],
|
|
||||||
},
|
},
|
||||||
"folders": [
|
"folders": [
|
||||||
{
|
{
|
||||||
"name": "StaticGalleryBuilder",
|
"name": "StaticGalleryBuilder",
|
||||||
"path": "./",
|
"path": "./"
|
||||||
},
|
}
|
||||||
],
|
],
|
||||||
"launch": {
|
"launch": {
|
||||||
"version": "0.2.0",
|
"version": "0.2.0",
|
||||||
@@ -39,14 +37,17 @@
|
|||||||
"cc-by-nc-sa",
|
"cc-by-nc-sa",
|
||||||
"-n",
|
"-n",
|
||||||
"-m",
|
"-m",
|
||||||
"-r"
|
"--reverse-sort",
|
||||||
|
"--regenerate-thumbnails",
|
||||||
|
"--reread-metadata",
|
||||||
|
"--folderthumbnails"
|
||||||
],
|
],
|
||||||
"console": "integratedTerminal",
|
"console": "integratedTerminal",
|
||||||
"name": "Testfolder",
|
"name": "Testfolder",
|
||||||
"postDebugTask": "Delete Lockfile",
|
"postDebugTask": "Delete Lockfile",
|
||||||
"program": "${workspaceFolder}/builder.py",
|
"program": "${workspaceFolder}/builder.py",
|
||||||
"request": "launch",
|
"request": "launch",
|
||||||
"type": "debugpy",
|
"type": "debugpy"
|
||||||
},
|
},
|
||||||
{
|
{
|
||||||
"args": [
|
"args": [
|
||||||
@@ -57,66 +58,90 @@
|
|||||||
"-t",
|
"-t",
|
||||||
"Pictures",
|
"Pictures",
|
||||||
"--theme",
|
"--theme",
|
||||||
"themes/catpuccin.css",
|
"themes/default.css",
|
||||||
"--use-fancy-folders",
|
"--use-fancy-folders",
|
||||||
"--web-manifest",
|
"--web-manifest",
|
||||||
"-n",
|
"-n",
|
||||||
"-m",
|
"-m",
|
||||||
"-r",
|
// "--regenerate-thumbnails",
|
||||||
|
// "--reread-metadata",
|
||||||
|
"--folderthumbnails"
|
||||||
],
|
],
|
||||||
"console": "integratedTerminal",
|
"console": "integratedTerminal",
|
||||||
"name": "woek",
|
"name": "woek",
|
||||||
"postDebugTask": "Delete Lockfile 2",
|
"postDebugTask": "Delete Lockfile 2",
|
||||||
"program": "${workspaceFolder}/builder.py",
|
"program": "${workspaceFolder}/builder.py",
|
||||||
"request": "launch",
|
"request": "launch",
|
||||||
"type": "debugpy",
|
"type": "debugpy"
|
||||||
|
},
|
||||||
|
{
|
||||||
|
"args": [
|
||||||
|
"--use-fancy-folders",
|
||||||
|
"-p",
|
||||||
|
"/mnt/nfs/pictures/",
|
||||||
|
"-w",
|
||||||
|
"https://pictures.sorogon.eu/",
|
||||||
|
"-t",
|
||||||
|
"Sorogon's Pictures",
|
||||||
|
"--theme",
|
||||||
|
"/home/user/git/github.com/greflm13/simple-picture-server/themes/alpenglow.css",
|
||||||
|
"-m",
|
||||||
|
"--exclude-folder",
|
||||||
|
"Scans",
|
||||||
|
"--exclude-folder",
|
||||||
|
"*/Galleries/*",
|
||||||
|
"--exclude-folder",
|
||||||
|
"*/Wallpaper/*",
|
||||||
|
"--folderthumbnails"
|
||||||
|
],
|
||||||
|
"console": "integratedTerminal",
|
||||||
|
"name": "production",
|
||||||
|
"program": "${workspaceFolder}/builder.py",
|
||||||
|
"request": "launch",
|
||||||
|
"type": "debugpy"
|
||||||
},
|
},
|
||||||
{
|
{
|
||||||
"args": [
|
"args": [
|
||||||
"${workspaceFolder}/themes",
|
"${workspaceFolder}/themes",
|
||||||
"https://pictures.sorogon.eu/Analog/Example/"
|
"https://pictures.sorogon.eu/public/Example/"
|
||||||
],
|
],
|
||||||
"console": "integratedTerminal",
|
"console": "integratedTerminal",
|
||||||
"name": "Generate Themes previews",
|
"name": "Generate Themes previews",
|
||||||
"program": "${workspaceFolder}/generate_previews.py",
|
"program": "${workspaceFolder}/generate_previews.py",
|
||||||
"request": "launch",
|
"request": "launch",
|
||||||
"type": "debugpy",
|
"type": "debugpy"
|
||||||
}
|
}
|
||||||
],
|
]
|
||||||
},
|
},
|
||||||
"settings": {
|
"settings": {
|
||||||
"[css]": {
|
"[css]": {
|
||||||
"editor.defaultFormatter": "vscode.css-language-features",
|
"editor.defaultFormatter": "vscode.css-language-features"
|
||||||
},
|
},
|
||||||
"[jinja-css]": {
|
"[jinja-css]": {
|
||||||
"editor.defaultFormatter": "vscode.css-language-features",
|
"editor.defaultFormatter": "vscode.css-language-features"
|
||||||
},
|
},
|
||||||
"[jinja-html]": {
|
"[jinja-html]": {
|
||||||
"editor.defaultFormatter": "vscode.html-language-features",
|
"editor.defaultFormatter": "vscode.html-language-features"
|
||||||
},
|
},
|
||||||
"[jinja-js]": {
|
"[jinja-js]": {
|
||||||
"editor.defaultFormatter": "esbenp.prettier-vscode",
|
"editor.defaultFormatter": "esbenp.prettier-vscode"
|
||||||
},
|
},
|
||||||
"[python]": {
|
"[python]": {
|
||||||
"editor.defaultFormatter": "ms-python.black-formatter",
|
"editor.defaultFormatter": "charliermarsh.ruff"
|
||||||
},
|
},
|
||||||
"black-formatter.args": [
|
"black-formatter.args": ["-l 260"],
|
||||||
"-l 140",
|
"black-formatter.interpreter": ["/usr/bin/python3"],
|
||||||
],
|
|
||||||
"black-formatter.interpreter": [
|
|
||||||
"/usr/bin/python3",
|
|
||||||
],
|
|
||||||
"editor.formatOnSave": false,
|
"editor.formatOnSave": false,
|
||||||
"emmet.includeLanguages": {
|
"emmet.includeLanguages": {
|
||||||
"jinja-css": "css",
|
"jinja-css": "css",
|
||||||
"jinja-html": "html",
|
"jinja-html": "html",
|
||||||
"jinja-js": "javascript",
|
"jinja-js": "javascript",
|
||||||
"jinja-json": "json",
|
"jinja-json": "json"
|
||||||
},
|
},
|
||||||
"files.associations": {
|
"files.associations": {
|
||||||
"**/*.css.j2": "jinja-css",
|
"**/*.css.j2": "jinja-css",
|
||||||
"**/*.css": "css",
|
"**/*.css": "css",
|
||||||
"**/*.html.j2": "jinja-html",
|
"**/*.html.j2": "jinja-html"
|
||||||
},
|
},
|
||||||
"gitblame.inlineMessageEnabled": true,
|
"gitblame.inlineMessageEnabled": true,
|
||||||
"gitblame.inlineMessageFormat": "${author.name}, ${time.ago} • ${commit.summary}",
|
"gitblame.inlineMessageFormat": "${author.name}, ${time.ago} • ${commit.summary}",
|
||||||
@@ -132,21 +157,25 @@
|
|||||||
"json.schemaDownload.enable": true,
|
"json.schemaDownload.enable": true,
|
||||||
"json.schemas": [
|
"json.schemas": [
|
||||||
{
|
{
|
||||||
"fileMatch": [
|
"fileMatch": ["manifest.json.j2"],
|
||||||
"manifest.json.j2",
|
"url": "https://json.schemastore.org/web-manifest-combined.json"
|
||||||
],
|
}
|
||||||
"url": "https://json.schemastore.org/web-manifest-combined.json",
|
|
||||||
},
|
|
||||||
],
|
],
|
||||||
"prettier.htmlWhitespaceSensitivity": "css",
|
"prettier.htmlWhitespaceSensitivity": "css",
|
||||||
"pylint.args": [
|
"python-envs.pythonProjects": [
|
||||||
"--disable=C0111",
|
{
|
||||||
"--disable=C0301",
|
"path": "",
|
||||||
"--good-names-rgxs=^[_a-z][_a-z0-9]?$",
|
"envManager": "ms-python.python:pyenv",
|
||||||
|
"packageManager": "ms-python.python:pip"
|
||||||
|
}
|
||||||
],
|
],
|
||||||
"python.analysis.inlayHints.callArgumentNames": "off",
|
"python.analysis.inlayHints.callArgumentNames": "off",
|
||||||
"python.analysis.inlayHints.functionReturnTypes": false,
|
"python.analysis.inlayHints.functionReturnTypes": false,
|
||||||
"python.analysis.inlayHints.variableTypes": false,
|
"python.analysis.inlayHints.variableTypes": false,
|
||||||
|
"yaml.schemas": {
|
||||||
|
"https://raw.githubusercontent.com/pamburus/hl/master/schema/json/config.schema.json": "file:///home/user/git/github.com/greflm13/StaticGalleryBuilder/hl_config.yaml"
|
||||||
|
},
|
||||||
|
"ruff.lineLength": 180
|
||||||
},
|
},
|
||||||
"tasks": {
|
"tasks": {
|
||||||
"version": "2.0.0",
|
"version": "2.0.0",
|
||||||
@@ -157,6 +186,14 @@
|
|||||||
"label": "Delete Lockfile",
|
"label": "Delete Lockfile",
|
||||||
"problemMatcher": [],
|
"problemMatcher": [],
|
||||||
"type": "shell",
|
"type": "shell",
|
||||||
|
"presentation": {
|
||||||
|
"echo": false,
|
||||||
|
"reveal": "never",
|
||||||
|
"focus": false,
|
||||||
|
"panel": "shared",
|
||||||
|
"showReuseMessage": false,
|
||||||
|
"clear": true
|
||||||
|
}
|
||||||
},
|
},
|
||||||
{
|
{
|
||||||
"command": "rm -f /home/user/woek/Pictures/.lock",
|
"command": "rm -f /home/user/woek/Pictures/.lock",
|
||||||
@@ -164,7 +201,81 @@
|
|||||||
"label": "Delete Lockfile 2",
|
"label": "Delete Lockfile 2",
|
||||||
"problemMatcher": [],
|
"problemMatcher": [],
|
||||||
"type": "shell",
|
"type": "shell",
|
||||||
|
"presentation": {
|
||||||
|
"echo": false,
|
||||||
|
"reveal": "never",
|
||||||
|
"focus": false,
|
||||||
|
"panel": "shared",
|
||||||
|
"showReuseMessage": false,
|
||||||
|
"clear": true
|
||||||
|
}
|
||||||
},
|
},
|
||||||
],
|
{
|
||||||
|
"command": "pyinstaller builder.py modules/*.py -n StaticGalleryBuilder-$(cat .version)-linux -F --add-data files:files --add-data templates:templates --add-data .version:.",
|
||||||
|
"isBackground": false,
|
||||||
|
"label": "Build",
|
||||||
|
"problemMatcher": [],
|
||||||
|
"type": "shell",
|
||||||
|
"presentation": {
|
||||||
|
"echo": true,
|
||||||
|
"reveal": "always",
|
||||||
|
"focus": false,
|
||||||
|
"panel": "shared",
|
||||||
|
"showReuseMessage": false,
|
||||||
|
"clear": false
|
||||||
},
|
},
|
||||||
|
"group": {
|
||||||
|
"kind": "build",
|
||||||
|
"isDefault": true
|
||||||
|
},
|
||||||
|
"dependsOn": ["Clean"]
|
||||||
|
},
|
||||||
|
{
|
||||||
|
"command": "rm -rf build dist",
|
||||||
|
"isBackground": true,
|
||||||
|
"label": "Clean",
|
||||||
|
"problemMatcher": [],
|
||||||
|
"type": "shell",
|
||||||
|
"presentation": {
|
||||||
|
"echo": true,
|
||||||
|
"reveal": "never",
|
||||||
|
"focus": false,
|
||||||
|
"panel": "shared",
|
||||||
|
"showReuseMessage": false,
|
||||||
|
"clear": true
|
||||||
|
},
|
||||||
|
"group": "build"
|
||||||
|
},
|
||||||
|
{
|
||||||
|
"command": "LESS=-SR hl logs/latest.jsonl --config hl_config.yaml",
|
||||||
|
"isBackground": false,
|
||||||
|
"label": "View Latest Log",
|
||||||
|
"problemMatcher": [],
|
||||||
|
"type": "shell",
|
||||||
|
"presentation": {
|
||||||
|
"echo": false,
|
||||||
|
"reveal": "always",
|
||||||
|
"focus": true,
|
||||||
|
"panel": "dedicated",
|
||||||
|
"showReuseMessage": false,
|
||||||
|
"clear": true
|
||||||
|
}
|
||||||
|
},
|
||||||
|
{
|
||||||
|
"command": "COLUMNS=120 ./builder.py --generate-help-preview help.svg",
|
||||||
|
"isBackground": false,
|
||||||
|
"label": "Create help svg",
|
||||||
|
"problemMatcher": [],
|
||||||
|
"type": "shell",
|
||||||
|
"presentation": {
|
||||||
|
"echo": false,
|
||||||
|
"reveal": "always",
|
||||||
|
"focus": true,
|
||||||
|
"panel": "dedicated",
|
||||||
|
"showReuseMessage": false,
|
||||||
|
"clear": true
|
||||||
|
}
|
||||||
|
}
|
||||||
|
]
|
||||||
|
}
|
||||||
}
|
}
|
||||||
139
builder.py
@@ -1,19 +1,20 @@
|
|||||||
#!/usr/bin/env python3
|
#!/usr/bin/env python3
|
||||||
import os
|
import os
|
||||||
import re
|
import re
|
||||||
|
import sys
|
||||||
import shutil
|
import shutil
|
||||||
import fnmatch
|
import urllib.error
|
||||||
import urllib.parse
|
import urllib.parse
|
||||||
|
import urllib.request
|
||||||
from multiprocessing import Pool, freeze_support
|
from multiprocessing import Pool, freeze_support
|
||||||
from pathlib import Path
|
from pathlib import Path
|
||||||
from typing import Dict, List, Tuple
|
|
||||||
|
|
||||||
from tqdm.auto import tqdm
|
from tqdm.auto import tqdm
|
||||||
from PIL import Image, ImageOps
|
from PIL import Image, ImageOps
|
||||||
|
from jsmin import jsmin
|
||||||
|
|
||||||
from modules.argumentparser import parse_arguments, Args
|
from modules.argumentparser import parse_arguments, Args
|
||||||
from modules.svg_handling import icons, webmanifest, extract_colorscheme
|
|
||||||
from modules.generate_html import list_folder, EXCLUDES
|
|
||||||
|
|
||||||
# fmt: off
|
# fmt: off
|
||||||
# Constants
|
# Constants
|
||||||
@@ -34,10 +35,19 @@ IMG_EXTENSIONS = [".jpg", ".jpeg", ".png"]
|
|||||||
NOT_LIST = ["*/Galleries/*", "Archives"]
|
NOT_LIST = ["*/Galleries/*", "Archives"]
|
||||||
# fmt: on
|
# fmt: on
|
||||||
|
|
||||||
pbardict: Dict[str, tqdm] = {}
|
args = parse_arguments(VERSION)
|
||||||
|
|
||||||
|
lock_file = os.path.join(args.root_directory, ".lock")
|
||||||
|
if os.path.exists(lock_file):
|
||||||
|
print("Another instance of this program is running.")
|
||||||
|
sys.exit()
|
||||||
|
else:
|
||||||
|
from modules.logger import logger
|
||||||
|
from modules.svg_handling import icons, webmanifest, extract_colorscheme
|
||||||
|
from modules.generate_html import list_folder
|
||||||
|
|
||||||
|
|
||||||
def init_globals(_args: Args, raw: List[str]) -> Tuple[Args, List[str]]:
|
def init_globals(_args: Args, raw: list[str]) -> tuple[Args, list[str]]:
|
||||||
"""
|
"""
|
||||||
Initialize global variables and set default values for arguments.
|
Initialize global variables and set default values for arguments.
|
||||||
|
|
||||||
@@ -45,12 +55,12 @@ def init_globals(_args: Args, raw: List[str]) -> Tuple[Args, List[str]]:
|
|||||||
-----------
|
-----------
|
||||||
_args : Args
|
_args : Args
|
||||||
Parsed command-line arguments.
|
Parsed command-line arguments.
|
||||||
raw : List[str]
|
raw : list[str]
|
||||||
List of raw file extensions.
|
list of raw file extensions.
|
||||||
|
|
||||||
Returns:
|
Returns:
|
||||||
--------
|
--------
|
||||||
Tuple[Args, List[str]]
|
tuple[Args, list[str]]
|
||||||
Updated arguments and raw file extensions.
|
Updated arguments and raw file extensions.
|
||||||
"""
|
"""
|
||||||
if not _args.file_extensions:
|
if not _args.file_extensions:
|
||||||
@@ -77,10 +87,13 @@ def copy_static_files(_args: Args) -> None:
|
|||||||
static_dir = os.path.join(_args.root_directory, ".static")
|
static_dir = os.path.join(_args.root_directory, ".static")
|
||||||
if os.path.exists(static_dir):
|
if os.path.exists(static_dir):
|
||||||
print("Removing existing .static folder...")
|
print("Removing existing .static folder...")
|
||||||
|
logger.info("removing existing .static folder")
|
||||||
shutil.rmtree(static_dir)
|
shutil.rmtree(static_dir)
|
||||||
|
|
||||||
print("Copying static files...")
|
print("Copying static files...")
|
||||||
|
logger.info("copying static files")
|
||||||
shutil.copytree(STATIC_FILES_DIR, static_dir, dirs_exist_ok=True)
|
shutil.copytree(STATIC_FILES_DIR, static_dir, dirs_exist_ok=True)
|
||||||
|
logger.info("reading theme file", extra={"theme": _args.theme_path})
|
||||||
with open(_args.theme_path, "r", encoding="utf-8") as f:
|
with open(_args.theme_path, "r", encoding="utf-8") as f:
|
||||||
theme = f.read()
|
theme = f.read()
|
||||||
split = theme.split(".foldericon {")
|
split = theme.split(".foldericon {")
|
||||||
@@ -92,33 +105,42 @@ def copy_static_files(_args: Args) -> None:
|
|||||||
for match in re.finditer(r"content: (.*);", foldericon):
|
for match in re.finditer(r"content: (.*);", foldericon):
|
||||||
foldericon = match[1]
|
foldericon = match[1]
|
||||||
foldericon = foldericon.replace('"', "")
|
foldericon = foldericon.replace('"', "")
|
||||||
|
logger.info("found foldericon", extra={"foldericon": foldericon})
|
||||||
break
|
break
|
||||||
if "url" in foldericon:
|
if "url" in foldericon:
|
||||||
|
logger.info("foldericon in theme file, using it")
|
||||||
shutil.copyfile(_args.theme_path, os.path.join(static_dir, "theme.css"))
|
shutil.copyfile(_args.theme_path, os.path.join(static_dir, "theme.css"))
|
||||||
return
|
return
|
||||||
with open(os.path.join(SCRIPTDIR, foldericon), "r", encoding="utf-8") as f:
|
with open(os.path.join(SCRIPTDIR, foldericon), "r", encoding="utf-8") as f:
|
||||||
|
logger.info("Reading foldericon svg")
|
||||||
svg = f.read()
|
svg = f.read()
|
||||||
if "svg.j2" in foldericon:
|
if "svg.j2" in foldericon:
|
||||||
|
logger.info("foldericon in theme file is a jinja2 template")
|
||||||
colorscheme = extract_colorscheme(_args.theme_path)
|
colorscheme = extract_colorscheme(_args.theme_path)
|
||||||
svg = svg.replace("{{ color1 }}", colorscheme["color1"])
|
for color_key, color_value in colorscheme.items():
|
||||||
svg = svg.replace("{{ color2 }}", colorscheme["color2"])
|
svg = svg.replace(f"{{{{ {color_key} }}}}", color_value)
|
||||||
svg = svg.replace("{{ color3 }}", colorscheme["color3"])
|
logger.info("replaced colors in svg")
|
||||||
svg = svg.replace("{{ color4 }}", colorscheme["color4"])
|
|
||||||
svg = urllib.parse.quote(svg)
|
svg = urllib.parse.quote(svg)
|
||||||
with open(os.path.join(static_dir, "theme.css"), "x", encoding="utf-8") as f:
|
with open(os.path.join(static_dir, "theme.css"), "x", encoding="utf-8") as f:
|
||||||
|
logger.info("writing theme file")
|
||||||
f.write(themehead + '\n.foldericon {\n content: url("data:image/svg+xml,' + svg + '");\n}\n' + themetail)
|
f.write(themehead + '\n.foldericon {\n content: url("data:image/svg+xml,' + svg + '");\n}\n' + themetail)
|
||||||
|
logger.info("minifying javascript")
|
||||||
|
with open(os.path.join(SCRIPTDIR, "templates", "functionality.js"), "r", encoding="utf-8") as js_file:
|
||||||
|
with open(os.path.join(static_dir, "functionality.min.js"), "w+", encoding="utf-8") as min_file:
|
||||||
|
min_file.write(jsmin(js_file.read()))
|
||||||
|
|
||||||
|
|
||||||
def generate_thumbnail(arguments: Tuple[str, str, str]) -> None:
|
def generate_thumbnail(arguments: tuple[str, str, str]) -> None:
|
||||||
"""
|
"""
|
||||||
Generate a thumbnail for a given image.
|
Generate a thumbnail for a given image.
|
||||||
|
|
||||||
Parameters:
|
Parameters:
|
||||||
-----------
|
-----------
|
||||||
arguments : Tuple[str, str, str, bool]
|
arguments : tuple[str, str, str, bool]
|
||||||
A tuple containing the folder, item, root directory, and regenerate thumbnails flag.
|
A tuple containing the folder, item, root directory, and regenerate thumbnails flag.
|
||||||
"""
|
"""
|
||||||
folder, item, root_directory = arguments
|
folder, item, root_directory = arguments
|
||||||
|
image = os.path.join(folder, item)
|
||||||
path = os.path.join(root_directory, ".thumbnails", folder.removeprefix(root_directory), item) + ".jpg"
|
path = os.path.join(root_directory, ".thumbnails", folder.removeprefix(root_directory), item) + ".jpg"
|
||||||
oldpath = os.path.join(root_directory, ".thumbnails", folder.removeprefix(root_directory), os.path.splitext(item)[0]) + ".jpg"
|
oldpath = os.path.join(root_directory, ".thumbnails", folder.removeprefix(root_directory), os.path.splitext(item)[0]) + ".jpg"
|
||||||
if os.path.exists(oldpath):
|
if os.path.exists(oldpath):
|
||||||
@@ -127,64 +149,54 @@ def generate_thumbnail(arguments: Tuple[str, str, str]) -> None:
|
|||||||
except FileNotFoundError:
|
except FileNotFoundError:
|
||||||
pass
|
pass
|
||||||
if not os.path.exists(path):
|
if not os.path.exists(path):
|
||||||
|
logger.info("generating thumbnail for %s", item, extra={"path": image})
|
||||||
try:
|
try:
|
||||||
with Image.open(os.path.join(folder, item)) as imgfile:
|
with Image.open(image) as imgfile:
|
||||||
imgrgb = imgfile.convert("RGB")
|
imgrgb = imgfile.convert("RGB")
|
||||||
img = ImageOps.exif_transpose(imgrgb)
|
img = ImageOps.exif_transpose(imgrgb)
|
||||||
img.thumbnail((512, 512))
|
img.thumbnail((512, 512))
|
||||||
img.save(path, "JPEG", quality=75, optimize=True, mode="RGB")
|
img.save(path, "JPEG", quality=75, optimize=True, mode="RGB")
|
||||||
except OSError:
|
except OSError:
|
||||||
print(f"Failed to generate thumbnail for {os.path.join(folder, item)}")
|
logger.error("Failed to generate thumbnail for %s", item, extra={"path": image})
|
||||||
|
print(f"Failed to generate thumbnail for {image}")
|
||||||
|
return
|
||||||
|
else:
|
||||||
|
logger.debug("thumbnail already exists for %s", item, extra={"path": image})
|
||||||
|
|
||||||
|
|
||||||
def get_total_folders(folder: str, _args: Args, _total: int = 0) -> int:
|
def main(args) -> None:
|
||||||
"""
|
|
||||||
Recursively count the total number of folders to be processed.
|
|
||||||
|
|
||||||
Parameters:
|
|
||||||
-----------
|
|
||||||
folder : str
|
|
||||||
The current folder being processed.
|
|
||||||
_args : Args
|
|
||||||
Parsed command-line arguments.
|
|
||||||
_total : int, optional
|
|
||||||
The running total of folders, default is 0.
|
|
||||||
|
|
||||||
Returns:
|
|
||||||
--------
|
|
||||||
int
|
|
||||||
The total number of folders.
|
|
||||||
"""
|
|
||||||
_total += 1
|
|
||||||
pbardict["traversingbar"].desc = f"Traversing filesystem - {folder}"
|
|
||||||
pbardict["traversingbar"].update(1)
|
|
||||||
|
|
||||||
items = sorted(os.listdir(folder))
|
|
||||||
for item in items:
|
|
||||||
if item not in EXCLUDES and os.path.isdir(os.path.join(folder, item)) and not item.startswith("."):
|
|
||||||
if item not in _args.exclude_folders and not any(fnmatch.fnmatchcase(os.path.join(folder, item), exclude) for exclude in _args.exclude_folders):
|
|
||||||
_total = get_total_folders(os.path.join(folder, item), _args, _total)
|
|
||||||
return _total
|
|
||||||
|
|
||||||
|
|
||||||
def main() -> None:
|
|
||||||
"""
|
"""
|
||||||
Main function to process images and generate a static image hosting website.
|
Main function to process images and generate a static image hosting website.
|
||||||
"""
|
"""
|
||||||
thumbnails: List[Tuple[str, str, str, bool]] = []
|
thumbnails: list[tuple[str, str, str, bool]] = []
|
||||||
|
|
||||||
args = parse_arguments(VERSION)
|
|
||||||
args, raw = init_globals(args, RAW_EXTENSIONS)
|
args, raw = init_globals(args, RAW_EXTENSIONS)
|
||||||
|
|
||||||
lock_file = os.path.join(args.root_directory, ".lock")
|
|
||||||
if os.path.exists(lock_file):
|
|
||||||
print("Another instance of this program is running.")
|
|
||||||
exit()
|
|
||||||
|
|
||||||
try:
|
try:
|
||||||
Path(lock_file).touch()
|
Path(lock_file).touch()
|
||||||
|
logger.info("starting builder", extra={"version": VERSION, "arguments": args})
|
||||||
|
|
||||||
|
logger.info("getting logo from sorogon.eu")
|
||||||
|
req = urllib.request.Request("https://files.sorogon.eu/logo.svg")
|
||||||
|
try:
|
||||||
|
with urllib.request.urlopen(req, timeout=10) as res:
|
||||||
|
logo = res.read().decode()
|
||||||
|
|
||||||
|
if logo.startswith("<?xml"):
|
||||||
|
logo = re.sub(r"<\?xml.+\?>", "", logo).strip()
|
||||||
|
if logo.startswith("<!--"):
|
||||||
|
logo = re.sub(r"<!--.+-->", "", logo).strip()
|
||||||
|
logo = logo.replace("\n", " ")
|
||||||
|
logo = " ".join(logo.split())
|
||||||
|
except urllib.error.URLError:
|
||||||
|
logo = "</srgn>"
|
||||||
|
|
||||||
|
if args.reread_metadata:
|
||||||
|
logger.warning("reread metadata flag is set to true, all image metadata will be reread")
|
||||||
if args.regenerate_thumbnails:
|
if args.regenerate_thumbnails:
|
||||||
|
logger.warning("regenerate thumbnails flag is set to true, all thumbnails will be regenerated")
|
||||||
if os.path.exists(os.path.join(args.root_directory, ".thumbnails")):
|
if os.path.exists(os.path.join(args.root_directory, ".thumbnails")):
|
||||||
|
logger.info("removing old thumbnails folder")
|
||||||
shutil.rmtree(os.path.join(args.root_directory, ".thumbnails"))
|
shutil.rmtree(os.path.join(args.root_directory, ".thumbnails"))
|
||||||
os.makedirs(os.path.join(args.root_directory, ".thumbnails"), exist_ok=True)
|
os.makedirs(os.path.join(args.root_directory, ".thumbnails"), exist_ok=True)
|
||||||
|
|
||||||
@@ -196,21 +208,18 @@ def main() -> None:
|
|||||||
webmanifest(args)
|
webmanifest(args)
|
||||||
|
|
||||||
if args.non_interactive_mode:
|
if args.non_interactive_mode:
|
||||||
|
logger.info("generating HTML files")
|
||||||
print("Generating HTML files...")
|
print("Generating HTML files...")
|
||||||
thumbnails = list_folder(0, args.root_directory, args.site_title, args, raw, VERSION)
|
thumbnails = list_folder(args.root_directory, args.site_title, args, raw, VERSION, logo)
|
||||||
with Pool(os.cpu_count()) as pool:
|
with Pool(os.cpu_count()) as pool:
|
||||||
|
logger.info("generating thumbnails")
|
||||||
print("Generating thumbnails...")
|
print("Generating thumbnails...")
|
||||||
pool.map(generate_thumbnail, thumbnails)
|
pool.map(generate_thumbnail, thumbnails)
|
||||||
else:
|
else:
|
||||||
pbardict["traversingbar"] = tqdm(desc="Traversing filesystem", unit="folders", ascii=True, dynamic_ncols=True)
|
thumbnails = list_folder(args.root_directory, args.site_title, args, raw, VERSION, logo)
|
||||||
total = get_total_folders(args.root_directory, args)
|
|
||||||
pbardict["traversingbar"].desc = "Traversing filesystem"
|
|
||||||
pbardict["traversingbar"].update(0)
|
|
||||||
pbardict["traversingbar"].close()
|
|
||||||
|
|
||||||
thumbnails = list_folder(total, args.root_directory, args.site_title, args, raw, VERSION)
|
|
||||||
|
|
||||||
with Pool(os.cpu_count()) as pool:
|
with Pool(os.cpu_count()) as pool:
|
||||||
|
logger.info("generating thumbnails")
|
||||||
for _ in tqdm(
|
for _ in tqdm(
|
||||||
pool.imap_unordered(generate_thumbnail, thumbnails),
|
pool.imap_unordered(generate_thumbnail, thumbnails),
|
||||||
total=len(thumbnails),
|
total=len(thumbnails),
|
||||||
@@ -222,9 +231,9 @@ def main() -> None:
|
|||||||
pass
|
pass
|
||||||
finally:
|
finally:
|
||||||
os.remove(lock_file)
|
os.remove(lock_file)
|
||||||
return
|
logger.info("finished builder", extra={"version": VERSION})
|
||||||
|
|
||||||
|
|
||||||
if __name__ == "__main__":
|
if __name__ == "__main__":
|
||||||
freeze_support()
|
freeze_support()
|
||||||
main()
|
main(args)
|
||||||
|
|||||||
183
files/global.css
@@ -28,6 +28,7 @@ body {
|
|||||||
.folders figure {
|
.folders figure {
|
||||||
margin-bottom: 32px;
|
margin-bottom: 32px;
|
||||||
margin-top: 50px;
|
margin-top: 50px;
|
||||||
|
width: 180px;
|
||||||
}
|
}
|
||||||
|
|
||||||
.header h1 {
|
.header h1 {
|
||||||
@@ -42,11 +43,21 @@ body {
|
|||||||
}
|
}
|
||||||
|
|
||||||
.folders figcaption {
|
.folders figcaption {
|
||||||
width: 120px;
|
width: 100%;
|
||||||
|
overflow: hidden;
|
||||||
font-size: smaller;
|
font-size: smaller;
|
||||||
text-align: center;
|
text-align: center;
|
||||||
}
|
}
|
||||||
|
|
||||||
|
.folderthumb {
|
||||||
|
height: 40px;
|
||||||
|
width: 70px !important;
|
||||||
|
overflow: hidden;
|
||||||
|
aspect-ratio: 1 / 1;
|
||||||
|
object-fit: contain;
|
||||||
|
margin: 20px 20px 0px -90px;
|
||||||
|
}
|
||||||
|
|
||||||
.row {
|
.row {
|
||||||
display: -ms-flexbox;
|
display: -ms-flexbox;
|
||||||
display: flex;
|
display: flex;
|
||||||
@@ -55,10 +66,21 @@ body {
|
|||||||
padding: 0 2px;
|
padding: 0 2px;
|
||||||
}
|
}
|
||||||
|
|
||||||
|
.centerload {
|
||||||
|
margin-top: 100px;
|
||||||
|
display: flex;
|
||||||
|
justify-content: center;
|
||||||
|
align-items: center;
|
||||||
|
}
|
||||||
|
|
||||||
figure {
|
figure {
|
||||||
margin: 0;
|
margin: 0;
|
||||||
}
|
}
|
||||||
|
|
||||||
|
.licensefile {
|
||||||
|
padding: 30px;
|
||||||
|
}
|
||||||
|
|
||||||
.caption {
|
.caption {
|
||||||
padding-top: 4px;
|
padding-top: 4px;
|
||||||
text-align: center;
|
text-align: center;
|
||||||
@@ -90,7 +112,7 @@ figure {
|
|||||||
.footer a img {
|
.footer a img {
|
||||||
height: 22px !important;
|
height: 22px !important;
|
||||||
margin-left: 3px;
|
margin-left: 3px;
|
||||||
vertical-align: text-bottom
|
vertical-align: text-bottom;
|
||||||
}
|
}
|
||||||
|
|
||||||
.navbar {
|
.navbar {
|
||||||
@@ -116,14 +138,18 @@ figure {
|
|||||||
text-decoration: none;
|
text-decoration: none;
|
||||||
}
|
}
|
||||||
|
|
||||||
.navbar .title {
|
.navbar .navleft {
|
||||||
|
float: left;
|
||||||
|
}
|
||||||
|
|
||||||
|
.navbar .navcenter {
|
||||||
position: absolute;
|
position: absolute;
|
||||||
left: 50%;
|
left: 50%;
|
||||||
transform: translateX(-50%);
|
transform: translateX(-50%);
|
||||||
}
|
}
|
||||||
|
|
||||||
.navbar .license {
|
.navbar .navright {
|
||||||
float: right;
|
float: right
|
||||||
}
|
}
|
||||||
|
|
||||||
.navbar li .header {
|
.navbar li .header {
|
||||||
@@ -135,34 +161,116 @@ figure {
|
|||||||
|
|
||||||
.tooltip {
|
.tooltip {
|
||||||
position: relative;
|
position: relative;
|
||||||
|
cursor: pointer;
|
||||||
}
|
}
|
||||||
|
|
||||||
.tooltip .tooltiptext {
|
.tooltip a {
|
||||||
display: none;
|
cursor: pointer;
|
||||||
width: max-content;
|
}
|
||||||
|
|
||||||
|
.tagtoggle {
|
||||||
|
cursor: pointer;
|
||||||
|
float: right;
|
||||||
|
padding: 12px;
|
||||||
|
|
||||||
|
}
|
||||||
|
|
||||||
|
.tagflex {
|
||||||
|
display: flex;
|
||||||
|
}
|
||||||
|
|
||||||
|
.tag {
|
||||||
|
flex: 1;
|
||||||
|
}
|
||||||
|
|
||||||
|
input {
|
||||||
|
pointer-events: none;
|
||||||
|
}
|
||||||
|
|
||||||
|
.tooltiptext {
|
||||||
|
cursor: default;
|
||||||
position: absolute;
|
position: absolute;
|
||||||
z-index: 100;
|
z-index: 100;
|
||||||
opacity: 0;
|
opacity: 0;
|
||||||
transition: opacity 0.3s;
|
transition: opacity 0.3s;
|
||||||
|
float: left;
|
||||||
|
max-height: 286px;
|
||||||
|
overflow: scroll;
|
||||||
|
}
|
||||||
|
|
||||||
|
.tooltip .infotext {
|
||||||
padding: 12px;
|
padding: 12px;
|
||||||
|
width: max-content;
|
||||||
}
|
}
|
||||||
|
|
||||||
.tooltip:hover .tooltiptext {
|
.tooltiptext.tagdropdown {
|
||||||
|
width: max-content;
|
||||||
|
right: 0;
|
||||||
|
padding: 0;
|
||||||
|
margin: 0;
|
||||||
|
overflow: hidden;
|
||||||
|
transition: max-height 0.3s ease, opacity 0.3s ease;
|
||||||
|
max-height: 0;
|
||||||
|
opacity: 0;
|
||||||
|
}
|
||||||
|
|
||||||
|
.tooltiptext.tagdropdown.show {
|
||||||
|
max-height: 80vh;
|
||||||
|
overflow-y: scroll;
|
||||||
|
opacity: 1;
|
||||||
|
}
|
||||||
|
|
||||||
|
.tooltip:hover .infotext {
|
||||||
display: block;
|
display: block;
|
||||||
opacity: 1;
|
opacity: 1;
|
||||||
}
|
}
|
||||||
|
|
||||||
.tooltip:active .tooltiptext {
|
.tooltip:active .infotext {
|
||||||
display: block;
|
display: block;
|
||||||
opacity: 1;
|
opacity: 1;
|
||||||
}
|
}
|
||||||
|
|
||||||
/* Create eight equal columns that sits next to each other */
|
.tagentryparent {
|
||||||
|
width: 100%;
|
||||||
|
overflow: hidden;
|
||||||
|
transition: max-height 0.3s ease, opacity 0.3s ease;
|
||||||
|
max-height: 0;
|
||||||
|
opacity: 0;
|
||||||
|
}
|
||||||
|
|
||||||
|
.tagentryparent.show {
|
||||||
|
max-height: 80vh;
|
||||||
|
opacity: 1;
|
||||||
|
overflow-y: scroll;
|
||||||
|
}
|
||||||
|
|
||||||
|
.tagentry {
|
||||||
|
list-style: none;
|
||||||
|
width: 100%;
|
||||||
|
cursor: pointer;
|
||||||
|
margin: 0;
|
||||||
|
padding: 0;
|
||||||
|
display: block;
|
||||||
|
}
|
||||||
|
|
||||||
|
.tooltip .tooltiptext ol {
|
||||||
|
margin-left: 0;
|
||||||
|
padding-left: 0.5em;
|
||||||
|
}
|
||||||
|
|
||||||
|
.tooltip .tooltiptext .tagentry label {
|
||||||
|
cursor: pointer;
|
||||||
|
height: 100%;
|
||||||
|
padding: 12px;
|
||||||
|
display: block;
|
||||||
|
}
|
||||||
|
|
||||||
.column {
|
.column {
|
||||||
-ms-flex: 12.5%;
|
-ms-flex: 12.5%;
|
||||||
flex: 12.5%;
|
flex: 12.5%;
|
||||||
max-width: 12.5%;
|
max-width: 12.5%;
|
||||||
padding: 0 4px;
|
padding: 0 4px;
|
||||||
|
display: inline-block;
|
||||||
}
|
}
|
||||||
|
|
||||||
.column img {
|
.column img {
|
||||||
@@ -175,6 +283,14 @@ figure {
|
|||||||
cursor: pointer;
|
cursor: pointer;
|
||||||
}
|
}
|
||||||
|
|
||||||
|
.attribution svg {
|
||||||
|
height: calc(6.75pt + 12px);
|
||||||
|
width: fit-content;
|
||||||
|
position: relative;
|
||||||
|
bottom: -8px;
|
||||||
|
right: 0px;
|
||||||
|
}
|
||||||
|
|
||||||
#totop {
|
#totop {
|
||||||
display: none;
|
display: none;
|
||||||
position: fixed;
|
position: fixed;
|
||||||
@@ -187,7 +303,6 @@ figure {
|
|||||||
border-style: none;
|
border-style: none;
|
||||||
}
|
}
|
||||||
|
|
||||||
/* Responsive layout - makes a four column-layout instead of eight columns */
|
|
||||||
@media screen and (max-width: 1000px) {
|
@media screen and (max-width: 1000px) {
|
||||||
.column {
|
.column {
|
||||||
-ms-flex: 25%;
|
-ms-flex: 25%;
|
||||||
@@ -195,21 +310,32 @@ figure {
|
|||||||
max-width: 25%;
|
max-width: 25%;
|
||||||
}
|
}
|
||||||
|
|
||||||
|
.folders figure {
|
||||||
|
width: 160px;
|
||||||
|
}
|
||||||
|
|
||||||
.folders img {
|
.folders img {
|
||||||
width: 80px;
|
width: 80px;
|
||||||
}
|
}
|
||||||
|
|
||||||
.folders figcaption {
|
.folders figcaption {
|
||||||
width: 100px;
|
|
||||||
font-size: small;
|
font-size: small;
|
||||||
}
|
}
|
||||||
|
|
||||||
|
.folderthumb {
|
||||||
|
height: 30px;
|
||||||
|
width: 50px !important;
|
||||||
|
overflow: hidden;
|
||||||
|
aspect-ratio: 1 / 1;
|
||||||
|
object-fit: contain;
|
||||||
|
margin: 15px 20px 0px -70px;
|
||||||
|
}
|
||||||
|
|
||||||
.navbar {
|
.navbar {
|
||||||
font-size: medium;
|
font-size: medium;
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
|
||||||
/* Responsive layout - makes a two column-layout instead of four columns */
|
|
||||||
@media screen and (max-width: 800px) {
|
@media screen and (max-width: 800px) {
|
||||||
.column {
|
.column {
|
||||||
-ms-flex: 50%;
|
-ms-flex: 50%;
|
||||||
@@ -217,15 +343,27 @@ figure {
|
|||||||
max-width: 50%;
|
max-width: 50%;
|
||||||
}
|
}
|
||||||
|
|
||||||
|
.folders figure {
|
||||||
|
width: 140px;
|
||||||
|
}
|
||||||
|
|
||||||
.folders img {
|
.folders img {
|
||||||
width: 60px;
|
width: 60px;
|
||||||
}
|
}
|
||||||
|
|
||||||
.folders figcaption {
|
.folders figcaption {
|
||||||
width: 80px;
|
|
||||||
font-size: x-small;
|
font-size: x-small;
|
||||||
}
|
}
|
||||||
|
|
||||||
|
.folderthumb {
|
||||||
|
height: 25px;
|
||||||
|
width: 30px !important;
|
||||||
|
overflow: hidden;
|
||||||
|
aspect-ratio: 1 / 1;
|
||||||
|
object-fit: contain;
|
||||||
|
margin: 10px 20px 0px -50px;
|
||||||
|
}
|
||||||
|
|
||||||
.navbar {
|
.navbar {
|
||||||
font-size: small;
|
font-size: small;
|
||||||
}
|
}
|
||||||
@@ -239,7 +377,6 @@ figure {
|
|||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
|
||||||
/* Responsive layout - makes the two columns stack on top of each other instead of next to each other */
|
|
||||||
@media screen and (max-width: 600px) {
|
@media screen and (max-width: 600px) {
|
||||||
.column {
|
.column {
|
||||||
-ms-flex: 100%;
|
-ms-flex: 100%;
|
||||||
@@ -247,15 +384,27 @@ figure {
|
|||||||
max-width: 100%;
|
max-width: 100%;
|
||||||
}
|
}
|
||||||
|
|
||||||
|
.folders figure {
|
||||||
|
width: 120px;
|
||||||
|
}
|
||||||
|
|
||||||
.folders img {
|
.folders img {
|
||||||
width: 40px;
|
width: 40px;
|
||||||
}
|
}
|
||||||
|
|
||||||
.folders figcaption {
|
.folders figcaption {
|
||||||
width: 60px;
|
|
||||||
font-size: xx-small;
|
font-size: xx-small;
|
||||||
}
|
}
|
||||||
|
|
||||||
|
.folderthumb {
|
||||||
|
height: 15px;
|
||||||
|
width: 20px !important;
|
||||||
|
overflow: hidden;
|
||||||
|
aspect-ratio: 1 / 1;
|
||||||
|
object-fit: contain;
|
||||||
|
margin: 5px 10px 0px -35px;
|
||||||
|
}
|
||||||
|
|
||||||
.navbar {
|
.navbar {
|
||||||
font-size: xx-small;
|
font-size: xx-small;
|
||||||
}
|
}
|
||||||
|
|||||||
@@ -35,8 +35,7 @@
|
|||||||
transition: opacity 0.2s;
|
transition: opacity 0.2s;
|
||||||
-webkit-box-shadow: none;
|
-webkit-box-shadow: none;
|
||||||
box-shadow: none; }
|
box-shadow: none; }
|
||||||
.pswp__button:focus,
|
.pswp__button:focus, .pswp__button:hover {
|
||||||
.pswp__button:hover {
|
|
||||||
opacity: 1; }
|
opacity: 1; }
|
||||||
.pswp__button:active {
|
.pswp__button:active {
|
||||||
outline: none;
|
outline: none;
|
||||||
|
|||||||
@@ -1,131 +1 @@
|
|||||||
<?xml version="1.0" encoding="UTF-8" standalone="no"?>
|
<svg width="264" height="88" viewBox="0 0 264 88" xmlns="http://www.w3.org/2000/svg"><title>default-skin 2</title><g fill="none" fill-rule="evenodd"><g><path d="M67.002 59.5v3.768c-6.307.84-9.184 5.75-10.002 9.732 2.22-2.83 5.564-5.098 10.002-5.098V71.5L73 65.585 67.002 59.5z" id="Shape" fill="#fff"/><g fill="#fff"><path d="M13 29v-5h2v3h3v2h-5zM13 15h5v2h-3v3h-2v-5zM31 15v5h-2v-3h-3v-2h5zM31 29h-5v-2h3v-3h2v5z" id="Shape"/></g><g fill="#fff"><path d="M62 24v5h-2v-3h-3v-2h5zM62 20h-5v-2h3v-3h2v5zM70 20v-5h2v3h3v2h-5zM70 24h5v2h-3v3h-2v-5z"/></g><path d="M20.586 66l-5.656-5.656 1.414-1.414L22 64.586l5.656-5.656 1.414 1.414L23.414 66l5.656 5.656-1.414 1.414L22 67.414l-5.656 5.656-1.414-1.414L20.586 66z" fill="#fff"/><path d="M111.785 65.03L110 63.5l3-3.5h-10v-2h10l-3-3.5 1.785-1.468L117 59l-5.215 6.03z" fill="#fff"/><path d="M152.215 65.03L154 63.5l-3-3.5h10v-2h-10l3-3.5-1.785-1.468L147 59l5.215 6.03z" fill="#fff"/><g><path id="Rectangle-11" fill="#fff" d="M160.957 28.543l-3.25-3.25-1.413 1.414 3.25 3.25z"/><path d="M152.5 27c3.038 0 5.5-2.462 5.5-5.5s-2.462-5.5-5.5-5.5-5.5 2.462-5.5 5.5 2.462 5.5 5.5 5.5z" id="Oval-1" stroke="#fff" stroke-width="1.5"/><path fill="#fff" d="M150 21h5v1h-5z"/></g><g><path d="M116.957 28.543l-1.414 1.414-3.25-3.25 1.414-1.414 3.25 3.25z" fill="#fff"/><path d="M108.5 27c3.038 0 5.5-2.462 5.5-5.5s-2.462-5.5-5.5-5.5-5.5 2.462-5.5 5.5 2.462 5.5 5.5 5.5z" stroke="#fff" stroke-width="1.5"/><path fill="#fff" d="M106 21h5v1h-5z"/><path fill="#fff" d="M109.043 19.008l-.085 5-1-.017.085-5z"/></g></g></g></svg>
|
||||||
<svg
|
|
||||||
width="264"
|
|
||||||
height="88"
|
|
||||||
viewBox="0 0 264 88"
|
|
||||||
version="1.1"
|
|
||||||
id="svg12"
|
|
||||||
sodipodi:docname="default-skin.svg"
|
|
||||||
inkscape:version="1.3.2 (091e20ef0f, 2023-11-25, custom)"
|
|
||||||
xmlns:inkscape="http://www.inkscape.org/namespaces/inkscape"
|
|
||||||
xmlns:sodipodi="http://sodipodi.sourceforge.net/DTD/sodipodi-0.dtd"
|
|
||||||
xmlns="http://www.w3.org/2000/svg"
|
|
||||||
xmlns:svg="http://www.w3.org/2000/svg"
|
|
||||||
xmlns:rdf="http://www.w3.org/1999/02/22-rdf-syntax-ns#"
|
|
||||||
xmlns:cc="http://creativecommons.org/ns#"
|
|
||||||
xmlns:dc="http://purl.org/dc/elements/1.1/">
|
|
||||||
<defs
|
|
||||||
id="defs12" />
|
|
||||||
<sodipodi:namedview
|
|
||||||
id="namedview12"
|
|
||||||
pagecolor="#ffffff"
|
|
||||||
bordercolor="#000000"
|
|
||||||
borderopacity="0.25"
|
|
||||||
inkscape:showpageshadow="true"
|
|
||||||
inkscape:pageopacity="0.0"
|
|
||||||
inkscape:pagecheckerboard="true"
|
|
||||||
inkscape:deskcolor="#d1d1d1"
|
|
||||||
showborder="true"
|
|
||||||
borderlayer="true"
|
|
||||||
inkscape:zoom="3.0855569"
|
|
||||||
inkscape:cx="95.606729"
|
|
||||||
inkscape:cy="65.142212"
|
|
||||||
inkscape:window-width="1896"
|
|
||||||
inkscape:window-height="1016"
|
|
||||||
inkscape:window-x="1932"
|
|
||||||
inkscape:window-y="52"
|
|
||||||
inkscape:window-maximized="1"
|
|
||||||
inkscape:current-layer="g11" />
|
|
||||||
<title
|
|
||||||
id="title1">default-skin 2</title>
|
|
||||||
<g
|
|
||||||
fill="none"
|
|
||||||
fill-rule="evenodd"
|
|
||||||
id="g12">
|
|
||||||
<g
|
|
||||||
id="g11">
|
|
||||||
<g
|
|
||||||
fill="#fff"
|
|
||||||
id="g1">
|
|
||||||
<path
|
|
||||||
d="M13 29v-5h2v3h3v2h-5zM13 15h5v2h-3v3h-2v-5zM31 15v5h-2v-3h-3v-2h5zM31 29h-5v-2h3v-3h2v5z"
|
|
||||||
id="path1" />
|
|
||||||
</g>
|
|
||||||
<g
|
|
||||||
fill="#fff"
|
|
||||||
id="g2">
|
|
||||||
<path
|
|
||||||
d="M62 24v5h-2v-3h-3v-2h5zM62 20h-5v-2h3v-3h2v5zM70 20v-5h2v3h3v2h-5zM70 24h5v2h-3v3h-2v-5z"
|
|
||||||
id="path2" />
|
|
||||||
</g>
|
|
||||||
<path
|
|
||||||
d="M20.586 66l-5.656-5.656 1.414-1.414L22 64.586l5.656-5.656 1.414 1.414L23.414 66l5.656 5.656-1.414 1.414L22 67.414l-5.656 5.656-1.414-1.414L20.586 66z"
|
|
||||||
fill="#fff"
|
|
||||||
id="path3" />
|
|
||||||
<path
|
|
||||||
d="M111.785 65.03L110 63.5l3-3.5h-10v-2h10l-3-3.5 1.785-1.468L117 59l-5.215 6.03z"
|
|
||||||
fill="#fff"
|
|
||||||
id="path4" />
|
|
||||||
<path
|
|
||||||
d="M152.215 65.03L154 63.5l-3-3.5h10v-2h-10l3-3.5-1.785-1.468L147 59l5.215 6.03z"
|
|
||||||
fill="#fff"
|
|
||||||
id="path5" />
|
|
||||||
<g
|
|
||||||
id="g6">
|
|
||||||
<path
|
|
||||||
id="Rectangle-11"
|
|
||||||
fill="#fff"
|
|
||||||
d="M160.957 28.543l-3.25-3.25-1.413 1.414 3.25 3.25z" />
|
|
||||||
<path
|
|
||||||
d="M152.5 27c3.038 0 5.5-2.462 5.5-5.5s-2.462-5.5-5.5-5.5-5.5 2.462-5.5 5.5 2.462 5.5 5.5 5.5z"
|
|
||||||
id="Oval-1"
|
|
||||||
stroke="#fff"
|
|
||||||
stroke-width="1.5" />
|
|
||||||
<path
|
|
||||||
fill="#fff"
|
|
||||||
d="M150 21h5v1h-5z"
|
|
||||||
id="path6" />
|
|
||||||
</g>
|
|
||||||
<g
|
|
||||||
id="g10">
|
|
||||||
<path
|
|
||||||
d="M116.957 28.543l-1.414 1.414-3.25-3.25 1.414-1.414 3.25 3.25z"
|
|
||||||
fill="#fff"
|
|
||||||
id="path7" />
|
|
||||||
<path
|
|
||||||
d="M108.5 27c3.038 0 5.5-2.462 5.5-5.5s-2.462-5.5-5.5-5.5-5.5 2.462-5.5 5.5 2.462 5.5 5.5 5.5z"
|
|
||||||
stroke="#fff"
|
|
||||||
stroke-width="1.5"
|
|
||||||
id="path8" />
|
|
||||||
<path
|
|
||||||
fill="#fff"
|
|
||||||
d="M106 21h5v1h-5z"
|
|
||||||
id="path9" />
|
|
||||||
<path
|
|
||||||
fill="#fff"
|
|
||||||
d="M109.043 19.008l-.085 5-1-.017.085-5z"
|
|
||||||
id="path10" />
|
|
||||||
</g>
|
|
||||||
<g
|
|
||||||
id="g14">
|
|
||||||
<path
|
|
||||||
id="Shape"
|
|
||||||
style="fill:#ffffff"
|
|
||||||
d="M 63.708984 57.572266 L 63.708984 66.917969 L 62.773438 66.917969 L 59.173828 66.917969 L 65.089844 72.916016 L 71.173828 66.917969 L 67.40625 66.917969 L 66.640625 66.917969 L 66.640625 57.572266 L 63.708984 57.572266 z " />
|
|
||||||
<path
|
|
||||||
id="rect13"
|
|
||||||
style="fill:#ffffff;stroke:#ffffff;stroke-width:0.649;stroke-linejoin:round;stroke-dasharray:none;stroke-opacity:1"
|
|
||||||
d="m 56.24391,72.078606 v 3.51729 h 17.859837 v -3.51729 h -1.210961 v 2.324677 H 57.456705 v -2.324677 z" />
|
|
||||||
</g>
|
|
||||||
</g>
|
|
||||||
</g>
|
|
||||||
<metadata
|
|
||||||
id="metadata12">
|
|
||||||
<rdf:RDF>
|
|
||||||
<cc:Work
|
|
||||||
rdf:about="">
|
|
||||||
<dc:title>default-skin 2</dc:title>
|
|
||||||
</cc:Work>
|
|
||||||
</rdf:RDF>
|
|
||||||
</metadata>
|
|
||||||
</svg>
|
|
||||||
|
Before Width: | Height: | Size: 4.1 KiB After Width: | Height: | Size: 1.5 KiB |
6
files/pswp/photoswipe-ui-default.min.js
vendored
@@ -15,21 +15,14 @@
|
|||||||
touch-action: none;
|
touch-action: none;
|
||||||
z-index: 1500;
|
z-index: 1500;
|
||||||
-webkit-text-size-adjust: 100%;
|
-webkit-text-size-adjust: 100%;
|
||||||
text-size-adjust: 100%;
|
|
||||||
/* create separate layer, to avoid paint on window.onscroll in webkit/blink */
|
/* create separate layer, to avoid paint on window.onscroll in webkit/blink */
|
||||||
-webkit-backface-visibility: hidden;
|
-webkit-backface-visibility: hidden;
|
||||||
backface-visibility: hidden;
|
outline: none; }
|
||||||
outline: none;
|
.pswp * {
|
||||||
}
|
|
||||||
|
|
||||||
.pswp * {
|
|
||||||
-webkit-box-sizing: border-box;
|
-webkit-box-sizing: border-box;
|
||||||
box-sizing: border-box;
|
box-sizing: border-box; }
|
||||||
}
|
.pswp img {
|
||||||
|
max-width: none; }
|
||||||
.pswp img {
|
|
||||||
max-width: none;
|
|
||||||
}
|
|
||||||
|
|
||||||
/* style is added when JS option showHideOpacity is set to true */
|
/* style is added when JS option showHideOpacity is set to true */
|
||||||
.pswp--animate_opacity {
|
.pswp--animate_opacity {
|
||||||
@@ -38,33 +31,28 @@
|
|||||||
will-change: opacity;
|
will-change: opacity;
|
||||||
/* for open/close transition */
|
/* for open/close transition */
|
||||||
-webkit-transition: opacity 333ms cubic-bezier(0.4, 0, 0.22, 1);
|
-webkit-transition: opacity 333ms cubic-bezier(0.4, 0, 0.22, 1);
|
||||||
transition: opacity 333ms cubic-bezier(0.4, 0, 0.22, 1);
|
transition: opacity 333ms cubic-bezier(0.4, 0, 0.22, 1); }
|
||||||
}
|
|
||||||
|
|
||||||
.pswp--open {
|
.pswp--open {
|
||||||
display: block;
|
display: block; }
|
||||||
}
|
|
||||||
|
|
||||||
.pswp--zoom-allowed .pswp__img {
|
.pswp--zoom-allowed .pswp__img {
|
||||||
/* autoprefixer: off */
|
/* autoprefixer: off */
|
||||||
cursor: -webkit-zoom-in;
|
cursor: -webkit-zoom-in;
|
||||||
cursor: -moz-zoom-in;
|
cursor: -moz-zoom-in;
|
||||||
cursor: zoom-in;
|
cursor: zoom-in; }
|
||||||
}
|
|
||||||
|
|
||||||
.pswp--zoomed-in .pswp__img {
|
.pswp--zoomed-in .pswp__img {
|
||||||
/* autoprefixer: off */
|
/* autoprefixer: off */
|
||||||
cursor: -webkit-grab;
|
cursor: -webkit-grab;
|
||||||
cursor: -moz-grab;
|
cursor: -moz-grab;
|
||||||
cursor: grab;
|
cursor: grab; }
|
||||||
}
|
|
||||||
|
|
||||||
.pswp--dragging .pswp__img {
|
.pswp--dragging .pswp__img {
|
||||||
/* autoprefixer: off */
|
/* autoprefixer: off */
|
||||||
cursor: -webkit-grabbing;
|
cursor: -webkit-grabbing;
|
||||||
cursor: -moz-grabbing;
|
cursor: -moz-grabbing;
|
||||||
cursor: grabbing;
|
cursor: grabbing; }
|
||||||
}
|
|
||||||
|
|
||||||
/*
|
/*
|
||||||
Background is added as a separate element.
|
Background is added as a separate element.
|
||||||
@@ -78,10 +66,10 @@
|
|||||||
height: 100%;
|
height: 100%;
|
||||||
background: #000;
|
background: #000;
|
||||||
opacity: 0;
|
opacity: 0;
|
||||||
|
-webkit-transform: translateZ(0);
|
||||||
|
transform: translateZ(0);
|
||||||
-webkit-backface-visibility: hidden;
|
-webkit-backface-visibility: hidden;
|
||||||
backface-visibility: hidden;
|
will-change: opacity; }
|
||||||
will-change: opacity;
|
|
||||||
}
|
|
||||||
|
|
||||||
.pswp__scroll-wrap {
|
.pswp__scroll-wrap {
|
||||||
position: absolute;
|
position: absolute;
|
||||||
@@ -89,8 +77,7 @@
|
|||||||
top: 0;
|
top: 0;
|
||||||
width: 100%;
|
width: 100%;
|
||||||
height: 100%;
|
height: 100%;
|
||||||
overflow: hidden;
|
overflow: hidden; }
|
||||||
}
|
|
||||||
|
|
||||||
.pswp__container,
|
.pswp__container,
|
||||||
.pswp__zoom-wrap {
|
.pswp__zoom-wrap {
|
||||||
@@ -100,8 +87,7 @@
|
|||||||
left: 0;
|
left: 0;
|
||||||
right: 0;
|
right: 0;
|
||||||
top: 0;
|
top: 0;
|
||||||
bottom: 0;
|
bottom: 0; }
|
||||||
}
|
|
||||||
|
|
||||||
/* Prevent selection and tap highlights */
|
/* Prevent selection and tap highlights */
|
||||||
.pswp__container,
|
.pswp__container,
|
||||||
@@ -111,8 +97,7 @@
|
|||||||
-ms-user-select: none;
|
-ms-user-select: none;
|
||||||
user-select: none;
|
user-select: none;
|
||||||
-webkit-tap-highlight-color: transparent;
|
-webkit-tap-highlight-color: transparent;
|
||||||
-webkit-touch-callout: none;
|
-webkit-touch-callout: none; }
|
||||||
}
|
|
||||||
|
|
||||||
.pswp__zoom-wrap {
|
.pswp__zoom-wrap {
|
||||||
position: absolute;
|
position: absolute;
|
||||||
@@ -122,27 +107,22 @@
|
|||||||
transform-origin: left top;
|
transform-origin: left top;
|
||||||
/* for open/close transition */
|
/* for open/close transition */
|
||||||
-webkit-transition: -webkit-transform 333ms cubic-bezier(0.4, 0, 0.22, 1);
|
-webkit-transition: -webkit-transform 333ms cubic-bezier(0.4, 0, 0.22, 1);
|
||||||
transition: transform 333ms cubic-bezier(0.4, 0, 0.22, 1);
|
transition: transform 333ms cubic-bezier(0.4, 0, 0.22, 1); }
|
||||||
}
|
|
||||||
|
|
||||||
.pswp__bg {
|
.pswp__bg {
|
||||||
will-change: opacity;
|
will-change: opacity;
|
||||||
/* for open/close transition */
|
/* for open/close transition */
|
||||||
-webkit-transition: opacity 333ms cubic-bezier(0.4, 0, 0.22, 1);
|
-webkit-transition: opacity 333ms cubic-bezier(0.4, 0, 0.22, 1);
|
||||||
transition: opacity 333ms cubic-bezier(0.4, 0, 0.22, 1);
|
transition: opacity 333ms cubic-bezier(0.4, 0, 0.22, 1); }
|
||||||
}
|
|
||||||
|
|
||||||
.pswp--animated-in .pswp__bg,
|
.pswp--animated-in .pswp__bg,
|
||||||
.pswp--animated-in .pswp__zoom-wrap {
|
.pswp--animated-in .pswp__zoom-wrap {
|
||||||
-webkit-transition: none;
|
-webkit-transition: none;
|
||||||
transition: none;
|
transition: none; }
|
||||||
}
|
|
||||||
|
|
||||||
.pswp__container,
|
.pswp__container,
|
||||||
.pswp__zoom-wrap {
|
.pswp__zoom-wrap {
|
||||||
-webkit-backface-visibility: hidden;
|
-webkit-backface-visibility: hidden; }
|
||||||
backface-visibility: hidden;
|
|
||||||
}
|
|
||||||
|
|
||||||
.pswp__item {
|
.pswp__item {
|
||||||
position: absolute;
|
position: absolute;
|
||||||
@@ -150,40 +130,34 @@
|
|||||||
right: 0;
|
right: 0;
|
||||||
top: 0;
|
top: 0;
|
||||||
bottom: 0;
|
bottom: 0;
|
||||||
overflow: hidden;
|
overflow: hidden; }
|
||||||
}
|
|
||||||
|
|
||||||
.pswp__img {
|
.pswp__img {
|
||||||
position: absolute;
|
position: absolute;
|
||||||
width: auto;
|
width: auto;
|
||||||
height: auto;
|
height: auto;
|
||||||
top: 0;
|
top: 0;
|
||||||
left: 0;
|
left: 0; }
|
||||||
}
|
|
||||||
|
|
||||||
/*
|
/*
|
||||||
stretched thumbnail or div placeholder element (see below)
|
stretched thumbnail or div placeholder element (see below)
|
||||||
style is added to avoid flickering in webkit/blink when layers overlap
|
style is added to avoid flickering in webkit/blink when layers overlap
|
||||||
*/
|
*/
|
||||||
.pswp__img--placeholder {
|
.pswp__img--placeholder {
|
||||||
-webkit-backface-visibility: hidden;
|
-webkit-backface-visibility: hidden; }
|
||||||
backface-visibility: hidden;
|
|
||||||
}
|
|
||||||
|
|
||||||
/*
|
/*
|
||||||
div element that matches size of large image
|
div element that matches size of large image
|
||||||
large image loads on top of it
|
large image loads on top of it
|
||||||
*/
|
*/
|
||||||
.pswp__img--placeholder--blank {
|
.pswp__img--placeholder--blank {
|
||||||
background: #222;
|
background: #222; }
|
||||||
}
|
|
||||||
|
|
||||||
.pswp--ie .pswp__img {
|
.pswp--ie .pswp__img {
|
||||||
width: 100% !important;
|
width: 100% !important;
|
||||||
height: auto !important;
|
height: auto !important;
|
||||||
left: 0;
|
left: 0;
|
||||||
top: 0;
|
top: 0; }
|
||||||
}
|
|
||||||
|
|
||||||
/*
|
/*
|
||||||
Error message appears when image is not loaded
|
Error message appears when image is not loaded
|
||||||
@@ -198,10 +172,8 @@
|
|||||||
font-size: 14px;
|
font-size: 14px;
|
||||||
line-height: 16px;
|
line-height: 16px;
|
||||||
margin-top: -8px;
|
margin-top: -8px;
|
||||||
color: #CCC;
|
color: #CCC; }
|
||||||
}
|
|
||||||
|
|
||||||
.pswp__error-msg a {
|
.pswp__error-msg a {
|
||||||
color: #CCC;
|
color: #CCC;
|
||||||
text-decoration: underline;
|
text-decoration: underline; }
|
||||||
}
|
|
||||||
3788
files/pswp/photoswipe.js
Normal file
9
files/pswp/photoswipe.min.js
vendored
@@ -1,24 +1,20 @@
|
|||||||
import os
|
import os
|
||||||
import re
|
import re
|
||||||
import sys
|
import sys
|
||||||
import time
|
|
||||||
import shutil
|
import shutil
|
||||||
import base64
|
import base64
|
||||||
import logging
|
|
||||||
import fileinput
|
import fileinput
|
||||||
import urllib.parse
|
import urllib.parse
|
||||||
import urllib.request
|
import urllib.request
|
||||||
from typing import List
|
from typing import List
|
||||||
from selenium import webdriver
|
from selenium import webdriver
|
||||||
|
from selenium.webdriver.common.by import By
|
||||||
from selenium.webdriver.chrome.service import Service
|
from selenium.webdriver.chrome.service import Service
|
||||||
from selenium.webdriver.chrome.options import Options
|
from selenium.webdriver.chrome.options import Options
|
||||||
from selenium.webdriver.common.by import By
|
|
||||||
|
|
||||||
|
from modules.logger import consolelogger as logger
|
||||||
from modules.css_color import extract_colorscheme
|
from modules.css_color import extract_colorscheme
|
||||||
|
|
||||||
# Set up logging
|
|
||||||
logging.basicConfig(level=logging.INFO, format="%(asctime)s - %(levelname)s - %(message)s")
|
|
||||||
|
|
||||||
|
|
||||||
def replace_all(file, search_exp, replace_exp):
|
def replace_all(file, search_exp, replace_exp):
|
||||||
for line in fileinput.input(file, inplace=1):
|
for line in fileinput.input(file, inplace=1):
|
||||||
@@ -36,11 +32,14 @@ def take_screenshot(html_file_path: str, css_file: str, output_file: str, driver
|
|||||||
output_file (str): Path where the screenshot will be saved.
|
output_file (str): Path where the screenshot will be saved.
|
||||||
driver (webdriver.Chrome): The Chrome WebDriver instance.
|
driver (webdriver.Chrome): The Chrome WebDriver instance.
|
||||||
"""
|
"""
|
||||||
|
logger.info("taking screenshot for %s", css_file)
|
||||||
try:
|
try:
|
||||||
# Open the HTML file or URL
|
# Open the HTML file or URL
|
||||||
if html_file_path.startswith(("http://", "https://")):
|
if html_file_path.startswith(("http://", "https://")):
|
||||||
|
logger.info("opening URL: %s", html_file_path)
|
||||||
driver.get(html_file_path)
|
driver.get(html_file_path)
|
||||||
else:
|
else:
|
||||||
|
logger.info("opening file: %s", html_file_path)
|
||||||
driver.get(f"file://{os.path.abspath(html_file_path)}")
|
driver.get(f"file://{os.path.abspath(html_file_path)}")
|
||||||
|
|
||||||
# Remove current theme.css
|
# Remove current theme.css
|
||||||
@@ -52,9 +51,11 @@ def take_screenshot(html_file_path: str, css_file: str, output_file: str, driver
|
|||||||
}
|
}
|
||||||
});
|
});
|
||||||
"""
|
"""
|
||||||
|
logger.info("removing current theme.css")
|
||||||
driver.execute_script(remove_css_script)
|
driver.execute_script(remove_css_script)
|
||||||
|
|
||||||
with open(css_file, "r", encoding="utf-8") as f:
|
with open(css_file, "r", encoding="utf-8") as f:
|
||||||
|
logger.info("reading CSS file: %s", css_file)
|
||||||
css_content = f.read()
|
css_content = f.read()
|
||||||
|
|
||||||
# Extract folder icon content
|
# Extract folder icon content
|
||||||
@@ -65,20 +66,26 @@ def take_screenshot(html_file_path: str, css_file: str, output_file: str, driver
|
|||||||
folder_icon_content = re.sub(r"/\*.*\*/", "", folder_icon_content)
|
folder_icon_content = re.sub(r"/\*.*\*/", "", folder_icon_content)
|
||||||
|
|
||||||
for match in re.finditer(r"content: (.*);", folder_icon_content):
|
for match in re.finditer(r"content: (.*);", folder_icon_content):
|
||||||
|
logger.info("found foldericon", extra={"foldericon": folder_icon_content})
|
||||||
folder_icon_content = match.group(1).replace('"', "")
|
folder_icon_content = match.group(1).replace('"', "")
|
||||||
break
|
break
|
||||||
|
|
||||||
if "url" not in folder_icon_content:
|
if "url" not in folder_icon_content:
|
||||||
|
logger.info("Reading foldericon svg")
|
||||||
with open(folder_icon_content, "r", encoding="utf-8") as f:
|
with open(folder_icon_content, "r", encoding="utf-8") as f:
|
||||||
svg = f.read()
|
svg = f.read()
|
||||||
if "svg.j2" in folder_icon_content:
|
if "svg.j2" in folder_icon_content:
|
||||||
|
logger.info("foldericon in theme file is a jinja2 template")
|
||||||
colorscheme = extract_colorscheme(css_file)
|
colorscheme = extract_colorscheme(css_file)
|
||||||
for color_key, color_value in colorscheme.items():
|
for color_key, color_value in colorscheme.items():
|
||||||
svg = svg.replace(f"{{{{ {color_key} }}}}", color_value)
|
svg = svg.replace(f"{{{{ {color_key} }}}}", color_value)
|
||||||
|
logger.info("replaced colors in svg")
|
||||||
svg = urllib.parse.quote(svg)
|
svg = urllib.parse.quote(svg)
|
||||||
|
|
||||||
css_content = f'{css_head}\n.foldericon {{\n content: url("data:image/svg+xml,{svg}");\n}}\n{css_tail}'
|
css_content = f'{css_head}\n.foldericon {{\n content: url("data:image/svg+xml,{svg}");\n}}\n{css_tail}'
|
||||||
|
|
||||||
# Encode CSS content as Base64
|
# Encode CSS content as Base64
|
||||||
|
logger.info("encoding css content as base64")
|
||||||
encoded_css = base64.b64encode(css_content.encode("utf-8")).decode("utf-8")
|
encoded_css = base64.b64encode(css_content.encode("utf-8")).decode("utf-8")
|
||||||
|
|
||||||
# Inject CSS into HTML using JavaScript
|
# Inject CSS into HTML using JavaScript
|
||||||
@@ -87,24 +94,28 @@ def take_screenshot(html_file_path: str, css_file: str, output_file: str, driver
|
|||||||
style.innerHTML = atob('{encoded_css}');
|
style.innerHTML = atob('{encoded_css}');
|
||||||
document.head.appendChild(style);
|
document.head.appendChild(style);
|
||||||
"""
|
"""
|
||||||
|
logger.info("injecting CSS into HTML")
|
||||||
driver.execute_script(apply_css_script)
|
driver.execute_script(apply_css_script)
|
||||||
|
|
||||||
# Wait for a while to ensure CSS is applied
|
# Wait for a while to ensure CSS is applied
|
||||||
time.sleep(2)
|
# time.sleep(1)
|
||||||
|
|
||||||
# Move mouse to info
|
# Move mouse to info
|
||||||
|
logger.info("moving mouse to info")
|
||||||
hoverable = driver.find_element(By.CLASS_NAME, "tooltip")
|
hoverable = driver.find_element(By.CLASS_NAME, "tooltip")
|
||||||
webdriver.ActionChains(driver).move_to_element(hoverable).perform()
|
webdriver.ActionChains(driver).move_to_element(hoverable).perform()
|
||||||
|
|
||||||
# Capture screenshot
|
# Capture screenshot
|
||||||
|
logger.info("taking screenshot")
|
||||||
driver.save_screenshot(output_file)
|
driver.save_screenshot(output_file)
|
||||||
logging.info("Screenshot saved to %s", output_file)
|
logger.info("screenshot saved to %s", output_file)
|
||||||
|
|
||||||
except Exception as e:
|
except Exception as e:
|
||||||
logging.error("Failed to take screenshot for %s: %s", css_file, e)
|
logger.error("failed to take screenshot for %s: %s", css_file, e)
|
||||||
|
|
||||||
|
|
||||||
def create_preview(html_file_path: str, css_file: str, previews_folder: str):
|
def create_preview(html_file_path: str, css_file: str, previews_folder: str):
|
||||||
|
logger.info("creating preview for %s", css_file)
|
||||||
out_file = os.path.basename(css_file).removesuffix(".css") + ".html"
|
out_file = os.path.basename(css_file).removesuffix(".css") + ".html"
|
||||||
urllib.request.urlretrieve(html_file_path, os.path.join(previews_folder, out_file))
|
urllib.request.urlretrieve(html_file_path, os.path.join(previews_folder, out_file))
|
||||||
basename = os.path.basename(css_file)
|
basename = os.path.basename(css_file)
|
||||||
@@ -127,21 +138,25 @@ def create_preview(html_file_path: str, css_file: str, previews_folder: str):
|
|||||||
foldericon = foldericon.replace('"', "")
|
foldericon = foldericon.replace('"', "")
|
||||||
break
|
break
|
||||||
if "url" in foldericon:
|
if "url" in foldericon:
|
||||||
|
logger.info("foldericon in theme file, using it")
|
||||||
shutil.copyfile(css_file, os.path.join(path, "previews", basename))
|
shutil.copyfile(css_file, os.path.join(path, "previews", basename))
|
||||||
else:
|
return
|
||||||
with open(os.path.join(path, foldericon.removeprefix("themes/")), "r", encoding="utf-8") as f:
|
with open(os.path.join(path, foldericon.removeprefix("themes/")), "r", encoding="utf-8") as f:
|
||||||
|
logger.info("Reading foldericon svg")
|
||||||
svg = f.read()
|
svg = f.read()
|
||||||
if "svg.j2" in foldericon:
|
if "svg.j2" in foldericon:
|
||||||
|
logger.info("foldericon in theme file is a jinja2 template")
|
||||||
colorscheme = extract_colorscheme(css_file)
|
colorscheme = extract_colorscheme(css_file)
|
||||||
svg = svg.replace("{{ color1 }}", colorscheme["color1"])
|
for color_key, color_value in colorscheme.items():
|
||||||
svg = svg.replace("{{ color2 }}", colorscheme["color2"])
|
svg = svg.replace(f"{{{{ {color_key} }}}}", color_value)
|
||||||
svg = svg.replace("{{ color3 }}", colorscheme["color3"])
|
logger.info("replaced colors in svg")
|
||||||
svg = svg.replace("{{ color4 }}", colorscheme["color4"])
|
|
||||||
svg = urllib.parse.quote(svg)
|
svg = urllib.parse.quote(svg)
|
||||||
if os.path.exists(os.path.join(path, "previews", basename)):
|
if os.path.exists(os.path.join(path, "previews", basename)):
|
||||||
os.remove(os.path.join(path, "previews", basename))
|
os.remove(os.path.join(path, "previews", basename))
|
||||||
with open(os.path.join(path, "previews", basename), "x", encoding="utf-8") as f:
|
with open(os.path.join(path, "previews", basename), "x", encoding="utf-8") as f:
|
||||||
|
logger.info("writing theme file")
|
||||||
f.write(themehead + '\n.foldericon {\n content: url("data:image/svg+xml,' + svg + '");\n}\n' + themetail)
|
f.write(themehead + '\n.foldericon {\n content: url("data:image/svg+xml,' + svg + '");\n}\n' + themetail)
|
||||||
|
logger.info("preview created for %s", css_file)
|
||||||
|
|
||||||
|
|
||||||
def write_readme(directory_path: str, themes: List[str]) -> None:
|
def write_readme(directory_path: str, themes: List[str]) -> None:
|
||||||
@@ -155,6 +170,7 @@ def write_readme(directory_path: str, themes: List[str]) -> None:
|
|||||||
readme_path = os.path.join(directory_path, "README.md")
|
readme_path = os.path.join(directory_path, "README.md")
|
||||||
try:
|
try:
|
||||||
with open(readme_path, "r", encoding="utf-8") as f:
|
with open(readme_path, "r", encoding="utf-8") as f:
|
||||||
|
logger.info("reading README.md", extra={"file": readme_path})
|
||||||
readme = f.read()
|
readme = f.read()
|
||||||
|
|
||||||
readme_head = readme.split("## Previews of included themes")[0]
|
readme_head = readme.split("## Previews of included themes")[0]
|
||||||
@@ -162,14 +178,15 @@ def write_readme(directory_path: str, themes: List[str]) -> None:
|
|||||||
readme_head += "".join([f"\n### {theme}\n\n\n" for theme in themes])
|
readme_head += "".join([f"\n### {theme}\n\n\n" for theme in themes])
|
||||||
|
|
||||||
with open(readme_path, "w", encoding="utf-8") as f:
|
with open(readme_path, "w", encoding="utf-8") as f:
|
||||||
|
logger.info("writing README.md", extra={"file": readme_path})
|
||||||
f.write(readme_head)
|
f.write(readme_head)
|
||||||
|
|
||||||
logging.info("README.md updated with previews of included themes.")
|
logger.info("README.md updated with previews of included themes.")
|
||||||
|
|
||||||
except FileNotFoundError:
|
except FileNotFoundError:
|
||||||
logging.error("README.md not found in %s", directory_path)
|
logger.error("README.md not found in %s", directory_path)
|
||||||
except Exception as e:
|
except Exception as e:
|
||||||
logging.error("Failed to write README.md: %s", e)
|
logger.error("failed to write README.md: %s", e)
|
||||||
|
|
||||||
|
|
||||||
def write_index(directory_path: str, themes: List[str]) -> None:
|
def write_index(directory_path: str, themes: List[str]) -> None:
|
||||||
@@ -198,7 +215,7 @@ def main(directory_path: str, html_file_path: str) -> None:
|
|||||||
html_file_path (str): Path to the HTML file or URL for rendering.
|
html_file_path (str): Path to the HTML file or URL for rendering.
|
||||||
"""
|
"""
|
||||||
if not os.path.exists(directory_path):
|
if not os.path.exists(directory_path):
|
||||||
logging.error('Error: Folder path "%s" does not exist.', directory_path)
|
logger.error('Error: Folder path "%s" does not exist.', directory_path)
|
||||||
return
|
return
|
||||||
|
|
||||||
# Setup Chrome options
|
# Setup Chrome options
|
||||||
@@ -207,8 +224,9 @@ def main(directory_path: str, html_file_path: str) -> None:
|
|||||||
chrome_options.add_argument("--window-size=1920,1080") # Set window size to at least 1920x1080
|
chrome_options.add_argument("--window-size=1920,1080") # Set window size to at least 1920x1080
|
||||||
|
|
||||||
# Initialize Chrome WebDriver
|
# Initialize Chrome WebDriver
|
||||||
chromedriver_path = "/usr/bin/chromedriver" # Replace with your actual path
|
chromedriver_path = "/usr/bin/chromedriver"
|
||||||
service = Service(chromedriver_path)
|
service = Service(chromedriver_path)
|
||||||
|
logger.info("Using chromedriver at %s", chromedriver_path, extra={"chrome_options": chrome_options})
|
||||||
driver = webdriver.Chrome(service=service, options=chrome_options)
|
driver = webdriver.Chrome(service=service, options=chrome_options)
|
||||||
|
|
||||||
try:
|
try:
|
||||||
@@ -235,13 +253,16 @@ def main(directory_path: str, html_file_path: str) -> None:
|
|||||||
write_index(directory_path, themes)
|
write_index(directory_path, themes)
|
||||||
|
|
||||||
finally:
|
finally:
|
||||||
|
logger.info("closing chrome webdriver")
|
||||||
driver.quit()
|
driver.quit()
|
||||||
|
|
||||||
|
|
||||||
if __name__ == "__main__":
|
if __name__ == "__main__":
|
||||||
if len(sys.argv) != 3:
|
if len(sys.argv) != 3:
|
||||||
logging.error("Usage: python script_name.py directory_path html_file_path")
|
logger.error("Usage: python script_name.py directory_path html_file_path")
|
||||||
else:
|
else:
|
||||||
dir_path = sys.argv[1]
|
dir_path = sys.argv[1]
|
||||||
html_path = sys.argv[2]
|
html_path = sys.argv[2]
|
||||||
|
logger.info("Starting script", extra={"directory_path": dir_path, "html_file_path": html_path})
|
||||||
main(dir_path, html_path)
|
main(dir_path, html_path)
|
||||||
|
logger.info("Done!", extra={"directory_path": dir_path})
|
||||||
|
|||||||
168
help.svg
@@ -1,4 +1,4 @@
|
|||||||
<svg class="rich-terminal" viewBox="0 0 1482 806.4" xmlns="http://www.w3.org/2000/svg">
|
<svg class="rich-terminal" viewBox="0 0 1482 928.4" xmlns="http://www.w3.org/2000/svg">
|
||||||
<!-- Generated with Rich https://www.textualize.io -->
|
<!-- Generated with Rich https://www.textualize.io -->
|
||||||
<style>
|
<style>
|
||||||
|
|
||||||
@@ -19,163 +19,183 @@
|
|||||||
font-weight: 700;
|
font-weight: 700;
|
||||||
}
|
}
|
||||||
|
|
||||||
.terminal-3756623299-matrix {
|
.terminal-3378394567-matrix {
|
||||||
font-family: Fira Code, monospace;
|
font-family: Fira Code, monospace;
|
||||||
font-size: 20px;
|
font-size: 20px;
|
||||||
line-height: 24.4px;
|
line-height: 24.4px;
|
||||||
font-variant-east-asian: full-width;
|
font-variant-east-asian: full-width;
|
||||||
}
|
}
|
||||||
|
|
||||||
.terminal-3756623299-title {
|
.terminal-3378394567-title {
|
||||||
font-size: 18px;
|
font-size: 18px;
|
||||||
font-weight: bold;
|
font-weight: bold;
|
||||||
font-family: arial;
|
font-family: arial;
|
||||||
}
|
}
|
||||||
|
|
||||||
.terminal-3756623299-r1 { fill: #ff8700 }
|
.terminal-3378394567-r1 { fill: #ff8700 }
|
||||||
.terminal-3756623299-r2 { fill: #c5c8c6 }
|
.terminal-3378394567-r2 { fill: #c5c8c6 }
|
||||||
.terminal-3756623299-r3 { fill: #808080 }
|
.terminal-3378394567-r3 { fill: #808080 }
|
||||||
.terminal-3756623299-r4 { fill: #68a0b3 }
|
.terminal-3378394567-r4 { fill: #68a0b3 }
|
||||||
.terminal-3756623299-r5 { fill: #00af87 }
|
.terminal-3378394567-r5 { fill: #00af87 }
|
||||||
</style>
|
</style>
|
||||||
|
|
||||||
<defs>
|
<defs>
|
||||||
<clipPath id="terminal-3756623299-clip-terminal">
|
<clipPath id="terminal-3378394567-clip-terminal">
|
||||||
<rect x="0" y="0" width="1463.0" height="755.4" />
|
<rect x="0" y="0" width="1463.0" height="877.4" />
|
||||||
</clipPath>
|
</clipPath>
|
||||||
<clipPath id="terminal-3756623299-line-0">
|
<clipPath id="terminal-3378394567-line-0">
|
||||||
<rect x="0" y="1.5" width="1464" height="24.65"/>
|
<rect x="0" y="1.5" width="1464" height="24.65"/>
|
||||||
</clipPath>
|
</clipPath>
|
||||||
<clipPath id="terminal-3756623299-line-1">
|
<clipPath id="terminal-3378394567-line-1">
|
||||||
<rect x="0" y="25.9" width="1464" height="24.65"/>
|
<rect x="0" y="25.9" width="1464" height="24.65"/>
|
||||||
</clipPath>
|
</clipPath>
|
||||||
<clipPath id="terminal-3756623299-line-2">
|
<clipPath id="terminal-3378394567-line-2">
|
||||||
<rect x="0" y="50.3" width="1464" height="24.65"/>
|
<rect x="0" y="50.3" width="1464" height="24.65"/>
|
||||||
</clipPath>
|
</clipPath>
|
||||||
<clipPath id="terminal-3756623299-line-3">
|
<clipPath id="terminal-3378394567-line-3">
|
||||||
<rect x="0" y="74.7" width="1464" height="24.65"/>
|
<rect x="0" y="74.7" width="1464" height="24.65"/>
|
||||||
</clipPath>
|
</clipPath>
|
||||||
<clipPath id="terminal-3756623299-line-4">
|
<clipPath id="terminal-3378394567-line-4">
|
||||||
<rect x="0" y="99.1" width="1464" height="24.65"/>
|
<rect x="0" y="99.1" width="1464" height="24.65"/>
|
||||||
</clipPath>
|
</clipPath>
|
||||||
<clipPath id="terminal-3756623299-line-5">
|
<clipPath id="terminal-3378394567-line-5">
|
||||||
<rect x="0" y="123.5" width="1464" height="24.65"/>
|
<rect x="0" y="123.5" width="1464" height="24.65"/>
|
||||||
</clipPath>
|
</clipPath>
|
||||||
<clipPath id="terminal-3756623299-line-6">
|
<clipPath id="terminal-3378394567-line-6">
|
||||||
<rect x="0" y="147.9" width="1464" height="24.65"/>
|
<rect x="0" y="147.9" width="1464" height="24.65"/>
|
||||||
</clipPath>
|
</clipPath>
|
||||||
<clipPath id="terminal-3756623299-line-7">
|
<clipPath id="terminal-3378394567-line-7">
|
||||||
<rect x="0" y="172.3" width="1464" height="24.65"/>
|
<rect x="0" y="172.3" width="1464" height="24.65"/>
|
||||||
</clipPath>
|
</clipPath>
|
||||||
<clipPath id="terminal-3756623299-line-8">
|
<clipPath id="terminal-3378394567-line-8">
|
||||||
<rect x="0" y="196.7" width="1464" height="24.65"/>
|
<rect x="0" y="196.7" width="1464" height="24.65"/>
|
||||||
</clipPath>
|
</clipPath>
|
||||||
<clipPath id="terminal-3756623299-line-9">
|
<clipPath id="terminal-3378394567-line-9">
|
||||||
<rect x="0" y="221.1" width="1464" height="24.65"/>
|
<rect x="0" y="221.1" width="1464" height="24.65"/>
|
||||||
</clipPath>
|
</clipPath>
|
||||||
<clipPath id="terminal-3756623299-line-10">
|
<clipPath id="terminal-3378394567-line-10">
|
||||||
<rect x="0" y="245.5" width="1464" height="24.65"/>
|
<rect x="0" y="245.5" width="1464" height="24.65"/>
|
||||||
</clipPath>
|
</clipPath>
|
||||||
<clipPath id="terminal-3756623299-line-11">
|
<clipPath id="terminal-3378394567-line-11">
|
||||||
<rect x="0" y="269.9" width="1464" height="24.65"/>
|
<rect x="0" y="269.9" width="1464" height="24.65"/>
|
||||||
</clipPath>
|
</clipPath>
|
||||||
<clipPath id="terminal-3756623299-line-12">
|
<clipPath id="terminal-3378394567-line-12">
|
||||||
<rect x="0" y="294.3" width="1464" height="24.65"/>
|
<rect x="0" y="294.3" width="1464" height="24.65"/>
|
||||||
</clipPath>
|
</clipPath>
|
||||||
<clipPath id="terminal-3756623299-line-13">
|
<clipPath id="terminal-3378394567-line-13">
|
||||||
<rect x="0" y="318.7" width="1464" height="24.65"/>
|
<rect x="0" y="318.7" width="1464" height="24.65"/>
|
||||||
</clipPath>
|
</clipPath>
|
||||||
<clipPath id="terminal-3756623299-line-14">
|
<clipPath id="terminal-3378394567-line-14">
|
||||||
<rect x="0" y="343.1" width="1464" height="24.65"/>
|
<rect x="0" y="343.1" width="1464" height="24.65"/>
|
||||||
</clipPath>
|
</clipPath>
|
||||||
<clipPath id="terminal-3756623299-line-15">
|
<clipPath id="terminal-3378394567-line-15">
|
||||||
<rect x="0" y="367.5" width="1464" height="24.65"/>
|
<rect x="0" y="367.5" width="1464" height="24.65"/>
|
||||||
</clipPath>
|
</clipPath>
|
||||||
<clipPath id="terminal-3756623299-line-16">
|
<clipPath id="terminal-3378394567-line-16">
|
||||||
<rect x="0" y="391.9" width="1464" height="24.65"/>
|
<rect x="0" y="391.9" width="1464" height="24.65"/>
|
||||||
</clipPath>
|
</clipPath>
|
||||||
<clipPath id="terminal-3756623299-line-17">
|
<clipPath id="terminal-3378394567-line-17">
|
||||||
<rect x="0" y="416.3" width="1464" height="24.65"/>
|
<rect x="0" y="416.3" width="1464" height="24.65"/>
|
||||||
</clipPath>
|
</clipPath>
|
||||||
<clipPath id="terminal-3756623299-line-18">
|
<clipPath id="terminal-3378394567-line-18">
|
||||||
<rect x="0" y="440.7" width="1464" height="24.65"/>
|
<rect x="0" y="440.7" width="1464" height="24.65"/>
|
||||||
</clipPath>
|
</clipPath>
|
||||||
<clipPath id="terminal-3756623299-line-19">
|
<clipPath id="terminal-3378394567-line-19">
|
||||||
<rect x="0" y="465.1" width="1464" height="24.65"/>
|
<rect x="0" y="465.1" width="1464" height="24.65"/>
|
||||||
</clipPath>
|
</clipPath>
|
||||||
<clipPath id="terminal-3756623299-line-20">
|
<clipPath id="terminal-3378394567-line-20">
|
||||||
<rect x="0" y="489.5" width="1464" height="24.65"/>
|
<rect x="0" y="489.5" width="1464" height="24.65"/>
|
||||||
</clipPath>
|
</clipPath>
|
||||||
<clipPath id="terminal-3756623299-line-21">
|
<clipPath id="terminal-3378394567-line-21">
|
||||||
<rect x="0" y="513.9" width="1464" height="24.65"/>
|
<rect x="0" y="513.9" width="1464" height="24.65"/>
|
||||||
</clipPath>
|
</clipPath>
|
||||||
<clipPath id="terminal-3756623299-line-22">
|
<clipPath id="terminal-3378394567-line-22">
|
||||||
<rect x="0" y="538.3" width="1464" height="24.65"/>
|
<rect x="0" y="538.3" width="1464" height="24.65"/>
|
||||||
</clipPath>
|
</clipPath>
|
||||||
<clipPath id="terminal-3756623299-line-23">
|
<clipPath id="terminal-3378394567-line-23">
|
||||||
<rect x="0" y="562.7" width="1464" height="24.65"/>
|
<rect x="0" y="562.7" width="1464" height="24.65"/>
|
||||||
</clipPath>
|
</clipPath>
|
||||||
<clipPath id="terminal-3756623299-line-24">
|
<clipPath id="terminal-3378394567-line-24">
|
||||||
<rect x="0" y="587.1" width="1464" height="24.65"/>
|
<rect x="0" y="587.1" width="1464" height="24.65"/>
|
||||||
</clipPath>
|
</clipPath>
|
||||||
<clipPath id="terminal-3756623299-line-25">
|
<clipPath id="terminal-3378394567-line-25">
|
||||||
<rect x="0" y="611.5" width="1464" height="24.65"/>
|
<rect x="0" y="611.5" width="1464" height="24.65"/>
|
||||||
</clipPath>
|
</clipPath>
|
||||||
<clipPath id="terminal-3756623299-line-26">
|
<clipPath id="terminal-3378394567-line-26">
|
||||||
<rect x="0" y="635.9" width="1464" height="24.65"/>
|
<rect x="0" y="635.9" width="1464" height="24.65"/>
|
||||||
</clipPath>
|
</clipPath>
|
||||||
<clipPath id="terminal-3756623299-line-27">
|
<clipPath id="terminal-3378394567-line-27">
|
||||||
<rect x="0" y="660.3" width="1464" height="24.65"/>
|
<rect x="0" y="660.3" width="1464" height="24.65"/>
|
||||||
</clipPath>
|
</clipPath>
|
||||||
<clipPath id="terminal-3756623299-line-28">
|
<clipPath id="terminal-3378394567-line-28">
|
||||||
<rect x="0" y="684.7" width="1464" height="24.65"/>
|
<rect x="0" y="684.7" width="1464" height="24.65"/>
|
||||||
</clipPath>
|
</clipPath>
|
||||||
<clipPath id="terminal-3756623299-line-29">
|
<clipPath id="terminal-3378394567-line-29">
|
||||||
<rect x="0" y="709.1" width="1464" height="24.65"/>
|
<rect x="0" y="709.1" width="1464" height="24.65"/>
|
||||||
</clipPath>
|
</clipPath>
|
||||||
|
<clipPath id="terminal-3378394567-line-30">
|
||||||
|
<rect x="0" y="733.5" width="1464" height="24.65"/>
|
||||||
|
</clipPath>
|
||||||
|
<clipPath id="terminal-3378394567-line-31">
|
||||||
|
<rect x="0" y="757.9" width="1464" height="24.65"/>
|
||||||
|
</clipPath>
|
||||||
|
<clipPath id="terminal-3378394567-line-32">
|
||||||
|
<rect x="0" y="782.3" width="1464" height="24.65"/>
|
||||||
|
</clipPath>
|
||||||
|
<clipPath id="terminal-3378394567-line-33">
|
||||||
|
<rect x="0" y="806.7" width="1464" height="24.65"/>
|
||||||
|
</clipPath>
|
||||||
|
<clipPath id="terminal-3378394567-line-34">
|
||||||
|
<rect x="0" y="831.1" width="1464" height="24.65"/>
|
||||||
|
</clipPath>
|
||||||
</defs>
|
</defs>
|
||||||
|
|
||||||
<rect fill="#292929" stroke="rgba(255,255,255,0.35)" stroke-width="1" x="1" y="1" width="1480" height="804.4" rx="8"/>
|
<rect fill="#292929" stroke="rgba(255,255,255,0.35)" stroke-width="1" x="1" y="1" width="1480" height="926.4" rx="8"/>
|
||||||
<g transform="translate(26,22)">
|
<g transform="translate(26,22)">
|
||||||
<circle cx="0" cy="0" r="7" fill="#ff5f57"/>
|
<circle cx="0" cy="0" r="7" fill="#ff5f57"/>
|
||||||
<circle cx="22" cy="0" r="7" fill="#febc2e"/>
|
<circle cx="22" cy="0" r="7" fill="#febc2e"/>
|
||||||
<circle cx="44" cy="0" r="7" fill="#28c840"/>
|
<circle cx="44" cy="0" r="7" fill="#28c840"/>
|
||||||
</g>
|
</g>
|
||||||
|
|
||||||
<g transform="translate(9, 41)" clip-path="url(#terminal-3756623299-clip-terminal)">
|
<g transform="translate(9, 41)" clip-path="url(#terminal-3378394567-clip-terminal)">
|
||||||
|
|
||||||
<g class="terminal-3756623299-matrix">
|
<g class="terminal-3378394567-matrix">
|
||||||
<text class="terminal-3756623299-r1" x="0" y="20" textLength="73.2" clip-path="url(#terminal-3756623299-line-0)">Usage:</text><text class="terminal-3756623299-r3" x="85.4" y="20" textLength="122" clip-path="url(#terminal-3756623299-line-0)">builder.py</text><text class="terminal-3756623299-r2" x="207.4" y="20" textLength="24.4" clip-path="url(#terminal-3756623299-line-0)"> [</text><text class="terminal-3756623299-r4" x="231.8" y="20" textLength="24.4" clip-path="url(#terminal-3756623299-line-0)">-h</text><text class="terminal-3756623299-r2" x="256.2" y="20" textLength="24.4" clip-path="url(#terminal-3756623299-line-0)">] </text><text class="terminal-3756623299-r4" x="280.6" y="20" textLength="24.4" clip-path="url(#terminal-3756623299-line-0)">-p</text><text class="terminal-3756623299-r5" x="317.2" y="20" textLength="48.8" clip-path="url(#terminal-3756623299-line-0)">ROOT</text><text class="terminal-3756623299-r4" x="378.2" y="20" textLength="24.4" clip-path="url(#terminal-3756623299-line-0)">-w</text><text class="terminal-3756623299-r5" x="414.8" y="20" textLength="36.6" clip-path="url(#terminal-3756623299-line-0)">URL</text><text class="terminal-3756623299-r4" x="463.6" y="20" textLength="24.4" clip-path="url(#terminal-3756623299-line-0)">-t</text><text class="terminal-3756623299-r5" x="500.2" y="20" textLength="61" clip-path="url(#terminal-3756623299-line-0)">TITLE</text><text class="terminal-3756623299-r2" x="561.2" y="20" textLength="24.4" clip-path="url(#terminal-3756623299-line-0)"> [</text><text class="terminal-3756623299-r4" x="585.6" y="20" textLength="24.4" clip-path="url(#terminal-3756623299-line-0)">-r</text><text class="terminal-3756623299-r2" x="610" y="20" textLength="36.6" clip-path="url(#terminal-3756623299-line-0)">] [</text><text class="terminal-3756623299-r4" x="646.6" y="20" textLength="24.4" clip-path="url(#terminal-3756623299-line-0)">-n</text><text class="terminal-3756623299-r2" x="671" y="20" textLength="36.6" clip-path="url(#terminal-3756623299-line-0)">] [</text><text class="terminal-3756623299-r4" x="707.6" y="20" textLength="24.4" clip-path="url(#terminal-3756623299-line-0)">-l</text><text class="terminal-3756623299-r5" x="744.2" y="20" textLength="85.4" clip-path="url(#terminal-3756623299-line-0)">LICENSE</text><text class="terminal-3756623299-r2" x="829.6" y="20" textLength="36.6" clip-path="url(#terminal-3756623299-line-0)">] [</text><text class="terminal-3756623299-r4" x="866.2" y="20" textLength="24.4" clip-path="url(#terminal-3756623299-line-0)">-a</text><text class="terminal-3756623299-r5" x="902.8" y="20" textLength="73.2" clip-path="url(#terminal-3756623299-line-0)">AUTHOR</text><text class="terminal-3756623299-r2" x="976" y="20" textLength="36.6" clip-path="url(#terminal-3756623299-line-0)">] [</text><text class="terminal-3756623299-r4" x="1012.6" y="20" textLength="24.4" clip-path="url(#terminal-3756623299-line-0)">-e</text><text class="terminal-3756623299-r5" x="1049.2" y="20" textLength="109.8" clip-path="url(#terminal-3756623299-line-0)">EXTENSION</text><text class="terminal-3756623299-r2" x="1159" y="20" textLength="36.6" clip-path="url(#terminal-3756623299-line-0)">] [</text><text class="terminal-3756623299-r4" x="1195.6" y="20" textLength="24.4" clip-path="url(#terminal-3756623299-line-0)">-m</text><text class="terminal-3756623299-r2" x="1220" y="20" textLength="12.2" clip-path="url(#terminal-3756623299-line-0)">]</text><text class="terminal-3756623299-r2" x="1464" y="20" textLength="12.2" clip-path="url(#terminal-3756623299-line-0)">
|
<text class="terminal-3378394567-r1" x="0" y="20" textLength="73.2" clip-path="url(#terminal-3378394567-line-0)">Usage:</text><text class="terminal-3378394567-r3" x="85.4" y="20" textLength="122" clip-path="url(#terminal-3378394567-line-0)">builder.py</text><text class="terminal-3378394567-r2" x="207.4" y="20" textLength="24.4" clip-path="url(#terminal-3378394567-line-0)"> [</text><text class="terminal-3378394567-r4" x="231.8" y="20" textLength="24.4" clip-path="url(#terminal-3378394567-line-0)">-h</text><text class="terminal-3378394567-r2" x="256.2" y="20" textLength="36.6" clip-path="url(#terminal-3378394567-line-0)">] [</text><text class="terminal-3378394567-r4" x="292.8" y="20" textLength="24.4" clip-path="url(#terminal-3378394567-line-0)">-a</text><text class="terminal-3378394567-r5" x="329.4" y="20" textLength="73.2" clip-path="url(#terminal-3378394567-line-0)">AUTHOR</text><text class="terminal-3378394567-r2" x="402.6" y="20" textLength="36.6" clip-path="url(#terminal-3378394567-line-0)">] [</text><text class="terminal-3378394567-r4" x="439.2" y="20" textLength="24.4" clip-path="url(#terminal-3378394567-line-0)">-e</text><text class="terminal-3378394567-r5" x="475.8" y="20" textLength="109.8" clip-path="url(#terminal-3378394567-line-0)">EXTENSION</text><text class="terminal-3378394567-r2" x="585.6" y="20" textLength="36.6" clip-path="url(#terminal-3378394567-line-0)">] [</text><text class="terminal-3378394567-r4" x="622.2" y="20" textLength="24.4" clip-path="url(#terminal-3378394567-line-0)">-l</text><text class="terminal-3378394567-r5" x="658.8" y="20" textLength="85.4" clip-path="url(#terminal-3378394567-line-0)">LICENSE</text><text class="terminal-3378394567-r2" x="744.2" y="20" textLength="36.6" clip-path="url(#terminal-3378394567-line-0)">] [</text><text class="terminal-3378394567-r4" x="780.8" y="20" textLength="24.4" clip-path="url(#terminal-3378394567-line-0)">-m</text><text class="terminal-3378394567-r2" x="805.2" y="20" textLength="36.6" clip-path="url(#terminal-3378394567-line-0)">] [</text><text class="terminal-3378394567-r4" x="841.8" y="20" textLength="24.4" clip-path="url(#terminal-3378394567-line-0)">-n</text><text class="terminal-3378394567-r2" x="866.2" y="20" textLength="24.4" clip-path="url(#terminal-3378394567-line-0)">] </text><text class="terminal-3378394567-r4" x="890.6" y="20" textLength="24.4" clip-path="url(#terminal-3378394567-line-0)">-p</text><text class="terminal-3378394567-r5" x="927.2" y="20" textLength="48.8" clip-path="url(#terminal-3378394567-line-0)">ROOT</text><text class="terminal-3378394567-r4" x="988.2" y="20" textLength="24.4" clip-path="url(#terminal-3378394567-line-0)">-t</text><text class="terminal-3378394567-r5" x="1024.8" y="20" textLength="61" clip-path="url(#terminal-3378394567-line-0)">TITLE</text><text class="terminal-3378394567-r4" x="1098" y="20" textLength="24.4" clip-path="url(#terminal-3378394567-line-0)">-w</text><text class="terminal-3378394567-r5" x="1134.6" y="20" textLength="36.6" clip-path="url(#terminal-3378394567-line-0)">URL</text><text class="terminal-3378394567-r2" x="1464" y="20" textLength="12.2" clip-path="url(#terminal-3378394567-line-0)">
|
||||||
</text><text class="terminal-3756623299-r2" x="0" y="44.4" textLength="231.8" clip-path="url(#terminal-3756623299-line-1)">                  [</text><text class="terminal-3756623299-r4" x="231.8" y="44.4" textLength="146.4" clip-path="url(#terminal-3756623299-line-1)">--theme-path</text><text class="terminal-3756623299-r5" x="390.4" y="44.4" textLength="48.8" clip-path="url(#terminal-3756623299-line-1)">PATH</text><text class="terminal-3756623299-r2" x="439.2" y="44.4" textLength="36.6" clip-path="url(#terminal-3756623299-line-1)">] [</text><text class="terminal-3756623299-r4" x="475.8" y="44.4" textLength="231.8" clip-path="url(#terminal-3756623299-line-1)">--use-fancy-folders</text><text class="terminal-3756623299-r2" x="707.6" y="44.4" textLength="36.6" clip-path="url(#terminal-3756623299-line-1)">] [</text><text class="terminal-3756623299-r4" x="744.2" y="44.4" textLength="244" clip-path="url(#terminal-3756623299-line-1)">--ignore-other-files</text><text class="terminal-3756623299-r2" x="988.2" y="44.4" textLength="36.6" clip-path="url(#terminal-3756623299-line-1)">] [</text><text class="terminal-3756623299-r4" x="1024.8" y="44.4" textLength="195.2" clip-path="url(#terminal-3756623299-line-1)">--exclude-folder</text><text class="terminal-3756623299-r5" x="1232.2" y="44.4" textLength="73.2" clip-path="url(#terminal-3756623299-line-1)">FOLDER</text><text class="terminal-3756623299-r2" x="1305.4" y="44.4" textLength="12.2" clip-path="url(#terminal-3756623299-line-1)">]</text><text class="terminal-3756623299-r2" x="1464" y="44.4" textLength="12.2" clip-path="url(#terminal-3756623299-line-1)">
|
</text><text class="terminal-3378394567-r2" x="0" y="44.4" textLength="231.8" clip-path="url(#terminal-3378394567-line-1)">                  [</text><text class="terminal-3378394567-r4" x="231.8" y="44.4" textLength="195.2" clip-path="url(#terminal-3378394567-line-1)">--exclude-folder</text><text class="terminal-3378394567-r5" x="439.2" y="44.4" textLength="73.2" clip-path="url(#terminal-3378394567-line-1)">FOLDER</text><text class="terminal-3378394567-r2" x="512.4" y="44.4" textLength="36.6" clip-path="url(#terminal-3378394567-line-1)">] [</text><text class="terminal-3378394567-r4" x="549" y="44.4" textLength="219.6" clip-path="url(#terminal-3378394567-line-1)">--folderthumbnails</text><text class="terminal-3378394567-r2" x="768.6" y="44.4" textLength="36.6" clip-path="url(#terminal-3378394567-line-1)">] [</text><text class="terminal-3378394567-r4" x="805.2" y="44.4" textLength="244" clip-path="url(#terminal-3378394567-line-1)">--ignore-other-files</text><text class="terminal-3378394567-r2" x="1049.2" y="44.4" textLength="36.6" clip-path="url(#terminal-3378394567-line-1)">] [</text><text class="terminal-3378394567-r4" x="1085.8" y="44.4" textLength="280.6" clip-path="url(#terminal-3378394567-line-1)">--regenerate-thumbnails</text><text class="terminal-3378394567-r2" x="1366.4" y="44.4" textLength="12.2" clip-path="url(#terminal-3378394567-line-1)">]</text><text class="terminal-3378394567-r2" x="1464" y="44.4" textLength="12.2" clip-path="url(#terminal-3378394567-line-1)">
|
||||||
</text><text class="terminal-3756623299-r2" x="0" y="68.8" textLength="231.8" clip-path="url(#terminal-3756623299-line-2)">                  [</text><text class="terminal-3756623299-r4" x="231.8" y="68.8" textLength="109.8" clip-path="url(#terminal-3756623299-line-2)">--version</text><text class="terminal-3756623299-r2" x="341.6" y="68.8" textLength="12.2" clip-path="url(#terminal-3756623299-line-2)">]</text><text class="terminal-3756623299-r2" x="1464" y="68.8" textLength="12.2" clip-path="url(#terminal-3756623299-line-2)">
|
</text><text class="terminal-3378394567-r2" x="0" y="68.8" textLength="231.8" clip-path="url(#terminal-3378394567-line-2)">                  [</text><text class="terminal-3378394567-r4" x="231.8" y="68.8" textLength="207.4" clip-path="url(#terminal-3378394567-line-2)">--reread-metadata</text><text class="terminal-3378394567-r2" x="439.2" y="68.8" textLength="36.6" clip-path="url(#terminal-3378394567-line-2)">] [</text><text class="terminal-3378394567-r4" x="475.8" y="68.8" textLength="195.2" clip-path="url(#terminal-3378394567-line-2)">--reread-sidecar</text><text class="terminal-3378394567-r2" x="671" y="68.8" textLength="36.6" clip-path="url(#terminal-3378394567-line-2)">] [</text><text class="terminal-3378394567-r4" x="707.6" y="68.8" textLength="170.8" clip-path="url(#terminal-3378394567-line-2)">--reverse-sort</text><text class="terminal-3378394567-r2" x="878.4" y="68.8" textLength="36.6" clip-path="url(#terminal-3378394567-line-2)">] [</text><text class="terminal-3378394567-r4" x="915" y="68.8" textLength="146.4" clip-path="url(#terminal-3378394567-line-2)">--theme-path</text><text class="terminal-3378394567-r5" x="1073.6" y="68.8" textLength="48.8" clip-path="url(#terminal-3378394567-line-2)">PATH</text><text class="terminal-3378394567-r2" x="1122.4" y="68.8" textLength="36.6" clip-path="url(#terminal-3378394567-line-2)">] [</text><text class="terminal-3378394567-r4" x="1159" y="68.8" textLength="231.8" clip-path="url(#terminal-3378394567-line-2)">--use-fancy-folders</text><text class="terminal-3378394567-r2" x="1390.8" y="68.8" textLength="12.2" clip-path="url(#terminal-3378394567-line-2)">]</text><text class="terminal-3378394567-r2" x="1464" y="68.8" textLength="12.2" clip-path="url(#terminal-3378394567-line-2)">
|
||||||
</text><text class="terminal-3756623299-r2" x="1464" y="93.2" textLength="12.2" clip-path="url(#terminal-3756623299-line-3)">
|
</text><text class="terminal-3378394567-r2" x="0" y="93.2" textLength="231.8" clip-path="url(#terminal-3378394567-line-3)">                  [</text><text class="terminal-3378394567-r4" x="231.8" y="93.2" textLength="109.8" clip-path="url(#terminal-3378394567-line-3)">--version</text><text class="terminal-3378394567-r2" x="341.6" y="93.2" textLength="12.2" clip-path="url(#terminal-3378394567-line-3)">]</text><text class="terminal-3378394567-r2" x="1464" y="93.2" textLength="12.2" clip-path="url(#terminal-3378394567-line-3)">
|
||||||
</text><text class="terminal-3756623299-r2" x="0" y="117.6" textLength="671" clip-path="url(#terminal-3756623299-line-4)">Generate HTML files for a static image hosting website.</text><text class="terminal-3756623299-r2" x="1464" y="117.6" textLength="12.2" clip-path="url(#terminal-3756623299-line-4)">
|
</text><text class="terminal-3378394567-r2" x="1464" y="117.6" textLength="12.2" clip-path="url(#terminal-3378394567-line-4)">
|
||||||
</text><text class="terminal-3756623299-r2" x="1464" y="142" textLength="12.2" clip-path="url(#terminal-3756623299-line-5)">
|
</text><text class="terminal-3378394567-r2" x="0" y="142" textLength="658.8" clip-path="url(#terminal-3378394567-line-5)">generate HTML files for a static image hosting website</text><text class="terminal-3378394567-r2" x="1464" y="142" textLength="12.2" clip-path="url(#terminal-3378394567-line-5)">
|
||||||
</text><text class="terminal-3756623299-r1" x="0" y="166.4" textLength="97.6" clip-path="url(#terminal-3756623299-line-6)">Options:</text><text class="terminal-3756623299-r2" x="1464" y="166.4" textLength="12.2" clip-path="url(#terminal-3756623299-line-6)">
|
</text><text class="terminal-3378394567-r2" x="1464" y="166.4" textLength="12.2" clip-path="url(#terminal-3378394567-line-6)">
|
||||||
</text><text class="terminal-3756623299-r4" x="24.4" y="190.8" textLength="24.4" clip-path="url(#terminal-3756623299-line-7)">-h</text><text class="terminal-3756623299-r2" x="48.8" y="190.8" textLength="24.4" clip-path="url(#terminal-3756623299-line-7)">, </text><text class="terminal-3756623299-r4" x="73.2" y="190.8" textLength="73.2" clip-path="url(#terminal-3756623299-line-7)">--help</text><text class="terminal-3756623299-r2" x="292.8" y="190.8" textLength="378.2" clip-path="url(#terminal-3756623299-line-7)">show this help message and exit</text><text class="terminal-3756623299-r2" x="1464" y="190.8" textLength="12.2" clip-path="url(#terminal-3756623299-line-7)">
|
</text><text class="terminal-3378394567-r1" x="0" y="190.8" textLength="97.6" clip-path="url(#terminal-3378394567-line-7)">Options:</text><text class="terminal-3378394567-r2" x="1464" y="190.8" textLength="12.2" clip-path="url(#terminal-3378394567-line-7)">
|
||||||
</text><text class="terminal-3756623299-r4" x="24.4" y="215.2" textLength="24.4" clip-path="url(#terminal-3756623299-line-8)">-p</text><text class="terminal-3756623299-r2" x="48.8" y="215.2" textLength="24.4" clip-path="url(#terminal-3756623299-line-8)">, </text><text class="terminal-3756623299-r4" x="73.2" y="215.2" textLength="195.2" clip-path="url(#terminal-3756623299-line-8)">--root-directory</text><text class="terminal-3756623299-r5" x="280.6" y="215.2" textLength="48.8" clip-path="url(#terminal-3756623299-line-8)">ROOT</text><text class="terminal-3756623299-r2" x="1464" y="215.2" textLength="12.2" clip-path="url(#terminal-3756623299-line-8)">
|
</text><text class="terminal-3378394567-r4" x="24.4" y="215.2" textLength="24.4" clip-path="url(#terminal-3378394567-line-8)">-h</text><text class="terminal-3378394567-r2" x="48.8" y="215.2" textLength="24.4" clip-path="url(#terminal-3378394567-line-8)">, </text><text class="terminal-3378394567-r4" x="73.2" y="215.2" textLength="73.2" clip-path="url(#terminal-3378394567-line-8)">--help</text><text class="terminal-3378394567-r2" x="292.8" y="215.2" textLength="378.2" clip-path="url(#terminal-3378394567-line-8)">show this help message and exit</text><text class="terminal-3378394567-r2" x="1464" y="215.2" textLength="12.2" clip-path="url(#terminal-3378394567-line-8)">
|
||||||
</text><text class="terminal-3756623299-r2" x="292.8" y="239.6" textLength="451.4" clip-path="url(#terminal-3756623299-line-9)">Root directory containing the images.</text><text class="terminal-3756623299-r2" x="1464" y="239.6" textLength="12.2" clip-path="url(#terminal-3756623299-line-9)">
|
</text><text class="terminal-3378394567-r4" x="24.4" y="239.6" textLength="24.4" clip-path="url(#terminal-3378394567-line-9)">-a</text><text class="terminal-3378394567-r2" x="48.8" y="239.6" textLength="24.4" clip-path="url(#terminal-3378394567-line-9)">, </text><text class="terminal-3378394567-r4" x="73.2" y="239.6" textLength="158.6" clip-path="url(#terminal-3378394567-line-9)">--author-name</text><text class="terminal-3378394567-r5" x="244" y="239.6" textLength="73.2" clip-path="url(#terminal-3378394567-line-9)">AUTHOR</text><text class="terminal-3378394567-r2" x="1464" y="239.6" textLength="12.2" clip-path="url(#terminal-3378394567-line-9)">
|
||||||
</text><text class="terminal-3756623299-r4" x="24.4" y="264" textLength="24.4" clip-path="url(#terminal-3756623299-line-10)">-w</text><text class="terminal-3756623299-r2" x="48.8" y="264" textLength="24.4" clip-path="url(#terminal-3756623299-line-10)">, </text><text class="terminal-3756623299-r4" x="73.2" y="264" textLength="170.8" clip-path="url(#terminal-3756623299-line-10)">--web-root-url</text><text class="terminal-3756623299-r5" x="256.2" y="264" textLength="36.6" clip-path="url(#terminal-3756623299-line-10)">URL</text><text class="terminal-3756623299-r2" x="1464" y="264" textLength="12.2" clip-path="url(#terminal-3756623299-line-10)">
|
</text><text class="terminal-3378394567-r2" x="292.8" y="264" textLength="390.4" clip-path="url(#terminal-3378394567-line-10)">name of the author of the images</text><text class="terminal-3378394567-r2" x="1464" y="264" textLength="12.2" clip-path="url(#terminal-3378394567-line-10)">
|
||||||
</text><text class="terminal-3756623299-r2" x="292.8" y="288.4" textLength="634.4" clip-path="url(#terminal-3756623299-line-11)">Base URL of the web root for the image hosting site.</text><text class="terminal-3756623299-r2" x="1464" y="288.4" textLength="12.2" clip-path="url(#terminal-3756623299-line-11)">
|
</text><text class="terminal-3378394567-r4" x="24.4" y="288.4" textLength="24.4" clip-path="url(#terminal-3378394567-line-11)">-e</text><text class="terminal-3378394567-r2" x="48.8" y="288.4" textLength="24.4" clip-path="url(#terminal-3378394567-line-11)">, </text><text class="terminal-3378394567-r4" x="73.2" y="288.4" textLength="207.4" clip-path="url(#terminal-3378394567-line-11)">--file-extensions</text><text class="terminal-3378394567-r5" x="292.8" y="288.4" textLength="109.8" clip-path="url(#terminal-3378394567-line-11)">EXTENSION</text><text class="terminal-3378394567-r2" x="1464" y="288.4" textLength="12.2" clip-path="url(#terminal-3378394567-line-11)">
|
||||||
</text><text class="terminal-3756623299-r4" x="24.4" y="312.8" textLength="24.4" clip-path="url(#terminal-3756623299-line-12)">-t</text><text class="terminal-3756623299-r2" x="48.8" y="312.8" textLength="24.4" clip-path="url(#terminal-3756623299-line-12)">, </text><text class="terminal-3756623299-r4" x="73.2" y="312.8" textLength="146.4" clip-path="url(#terminal-3756623299-line-12)">--site-title</text><text class="terminal-3756623299-r5" x="231.8" y="312.8" textLength="61" clip-path="url(#terminal-3756623299-line-12)">TITLE</text><text class="terminal-3756623299-r2" x="1464" y="312.8" textLength="12.2" clip-path="url(#terminal-3756623299-line-12)">
|
</text><text class="terminal-3378394567-r2" x="292.8" y="312.8" textLength="732" clip-path="url(#terminal-3378394567-line-12)">file extensions to include (can be specified multiple times)</text><text class="terminal-3378394567-r2" x="1464" y="312.8" textLength="12.2" clip-path="url(#terminal-3378394567-line-12)">
|
||||||
</text><text class="terminal-3756623299-r2" x="292.8" y="337.2" textLength="390.4" clip-path="url(#terminal-3756623299-line-13)">Title of the image hosting site.</text><text class="terminal-3756623299-r2" x="1464" y="337.2" textLength="12.2" clip-path="url(#terminal-3756623299-line-13)">
|
</text><text class="terminal-3378394567-r4" x="24.4" y="337.2" textLength="24.4" clip-path="url(#terminal-3378394567-line-13)">-l</text><text class="terminal-3378394567-r2" x="48.8" y="337.2" textLength="24.4" clip-path="url(#terminal-3378394567-line-13)">, </text><text class="terminal-3378394567-r4" x="73.2" y="337.2" textLength="170.8" clip-path="url(#terminal-3378394567-line-13)">--license-type</text><text class="terminal-3378394567-r5" x="256.2" y="337.2" textLength="85.4" clip-path="url(#terminal-3378394567-line-13)">LICENSE</text><text class="terminal-3378394567-r2" x="1464" y="337.2" textLength="12.2" clip-path="url(#terminal-3378394567-line-13)">
|
||||||
</text><text class="terminal-3756623299-r4" x="24.4" y="361.6" textLength="24.4" clip-path="url(#terminal-3756623299-line-14)">-r</text><text class="terminal-3756623299-r2" x="48.8" y="361.6" textLength="24.4" clip-path="url(#terminal-3756623299-line-14)">, </text><text class="terminal-3756623299-r4" x="73.2" y="361.6" textLength="280.6" clip-path="url(#terminal-3756623299-line-14)">--regenerate-thumbnails</text><text class="terminal-3756623299-r2" x="1464" y="361.6" textLength="12.2" clip-path="url(#terminal-3756623299-line-14)">
|
</text><text class="terminal-3378394567-r2" x="292.8" y="361.6" textLength="475.8" clip-path="url(#terminal-3378394567-line-14)">specify the license type for the images</text><text class="terminal-3378394567-r2" x="1464" y="361.6" textLength="12.2" clip-path="url(#terminal-3378394567-line-14)">
|
||||||
</text><text class="terminal-3756623299-r2" x="292.8" y="386" textLength="597.8" clip-path="url(#terminal-3756623299-line-15)">Regenerate thumbnails even if they already exist.</text><text class="terminal-3756623299-r2" x="1464" y="386" textLength="12.2" clip-path="url(#terminal-3756623299-line-15)">
|
</text><text class="terminal-3378394567-r4" x="24.4" y="386" textLength="24.4" clip-path="url(#terminal-3378394567-line-15)">-m</text><text class="terminal-3378394567-r2" x="48.8" y="386" textLength="24.4" clip-path="url(#terminal-3378394567-line-15)">, </text><text class="terminal-3378394567-r4" x="73.2" y="386" textLength="170.8" clip-path="url(#terminal-3378394567-line-15)">--web-manifest</text><text class="terminal-3378394567-r2" x="292.8" y="386" textLength="341.6" clip-path="url(#terminal-3378394567-line-15)">generate a web manifest file</text><text class="terminal-3378394567-r2" x="1464" y="386" textLength="12.2" clip-path="url(#terminal-3378394567-line-15)">
|
||||||
</text><text class="terminal-3756623299-r4" x="24.4" y="410.4" textLength="24.4" clip-path="url(#terminal-3756623299-line-16)">-n</text><text class="terminal-3756623299-r2" x="48.8" y="410.4" textLength="24.4" clip-path="url(#terminal-3756623299-line-16)">, </text><text class="terminal-3756623299-r4" x="73.2" y="410.4" textLength="268.4" clip-path="url(#terminal-3756623299-line-16)">--non-interactive-mode</text><text class="terminal-3756623299-r2" x="1464" y="410.4" textLength="12.2" clip-path="url(#terminal-3756623299-line-16)">
|
</text><text class="terminal-3378394567-r4" x="24.4" y="410.4" textLength="24.4" clip-path="url(#terminal-3378394567-line-16)">-n</text><text class="terminal-3378394567-r2" x="48.8" y="410.4" textLength="24.4" clip-path="url(#terminal-3378394567-line-16)">, </text><text class="terminal-3378394567-r4" x="73.2" y="410.4" textLength="268.4" clip-path="url(#terminal-3378394567-line-16)">--non-interactive-mode</text><text class="terminal-3378394567-r2" x="1464" y="410.4" textLength="12.2" clip-path="url(#terminal-3378394567-line-16)">
|
||||||
</text><text class="terminal-3756623299-r2" x="292.8" y="434.8" textLength="646.6" clip-path="url(#terminal-3756623299-line-17)">Run in non-interactive mode, disabling progress bars.</text><text class="terminal-3756623299-r2" x="1464" y="434.8" textLength="12.2" clip-path="url(#terminal-3756623299-line-17)">
|
</text><text class="terminal-3378394567-r2" x="292.8" y="434.8" textLength="634.4" clip-path="url(#terminal-3378394567-line-17)">run in non-interactive mode, disabling progress bars</text><text class="terminal-3378394567-r2" x="1464" y="434.8" textLength="12.2" clip-path="url(#terminal-3378394567-line-17)">
|
||||||
</text><text class="terminal-3756623299-r4" x="24.4" y="459.2" textLength="24.4" clip-path="url(#terminal-3756623299-line-18)">-l</text><text class="terminal-3756623299-r2" x="48.8" y="459.2" textLength="24.4" clip-path="url(#terminal-3756623299-line-18)">, </text><text class="terminal-3756623299-r4" x="73.2" y="459.2" textLength="170.8" clip-path="url(#terminal-3756623299-line-18)">--license-type</text><text class="terminal-3756623299-r5" x="256.2" y="459.2" textLength="85.4" clip-path="url(#terminal-3756623299-line-18)">LICENSE</text><text class="terminal-3756623299-r2" x="1464" y="459.2" textLength="12.2" clip-path="url(#terminal-3756623299-line-18)">
|
</text><text class="terminal-3378394567-r4" x="24.4" y="459.2" textLength="24.4" clip-path="url(#terminal-3378394567-line-18)">-p</text><text class="terminal-3378394567-r2" x="48.8" y="459.2" textLength="24.4" clip-path="url(#terminal-3378394567-line-18)">, </text><text class="terminal-3378394567-r4" x="73.2" y="459.2" textLength="195.2" clip-path="url(#terminal-3378394567-line-18)">--root-directory</text><text class="terminal-3378394567-r5" x="280.6" y="459.2" textLength="48.8" clip-path="url(#terminal-3378394567-line-18)">ROOT</text><text class="terminal-3378394567-r2" x="1464" y="459.2" textLength="12.2" clip-path="url(#terminal-3378394567-line-18)">
|
||||||
</text><text class="terminal-3756623299-r2" x="292.8" y="483.6" textLength="488" clip-path="url(#terminal-3756623299-line-19)">Specify the license type for the images.</text><text class="terminal-3756623299-r2" x="1464" y="483.6" textLength="12.2" clip-path="url(#terminal-3756623299-line-19)">
|
</text><text class="terminal-3378394567-r2" x="292.8" y="483.6" textLength="439.2" clip-path="url(#terminal-3378394567-line-19)">root directory containing the images</text><text class="terminal-3378394567-r2" x="1464" y="483.6" textLength="12.2" clip-path="url(#terminal-3378394567-line-19)">
|
||||||
</text><text class="terminal-3756623299-r4" x="24.4" y="508" textLength="24.4" clip-path="url(#terminal-3756623299-line-20)">-a</text><text class="terminal-3756623299-r2" x="48.8" y="508" textLength="24.4" clip-path="url(#terminal-3756623299-line-20)">, </text><text class="terminal-3756623299-r4" x="73.2" y="508" textLength="158.6" clip-path="url(#terminal-3756623299-line-20)">--author-name</text><text class="terminal-3756623299-r5" x="244" y="508" textLength="73.2" clip-path="url(#terminal-3756623299-line-20)">AUTHOR</text><text class="terminal-3756623299-r2" x="1464" y="508" textLength="12.2" clip-path="url(#terminal-3756623299-line-20)">
|
</text><text class="terminal-3378394567-r4" x="24.4" y="508" textLength="24.4" clip-path="url(#terminal-3378394567-line-20)">-t</text><text class="terminal-3378394567-r2" x="48.8" y="508" textLength="24.4" clip-path="url(#terminal-3378394567-line-20)">, </text><text class="terminal-3378394567-r4" x="73.2" y="508" textLength="146.4" clip-path="url(#terminal-3378394567-line-20)">--site-title</text><text class="terminal-3378394567-r5" x="231.8" y="508" textLength="61" clip-path="url(#terminal-3378394567-line-20)">TITLE</text><text class="terminal-3378394567-r2" x="1464" y="508" textLength="12.2" clip-path="url(#terminal-3378394567-line-20)">
|
||||||
</text><text class="terminal-3756623299-r2" x="292.8" y="532.4" textLength="402.6" clip-path="url(#terminal-3756623299-line-21)">Name of the author of the images.</text><text class="terminal-3756623299-r2" x="1464" y="532.4" textLength="12.2" clip-path="url(#terminal-3756623299-line-21)">
|
</text><text class="terminal-3378394567-r2" x="292.8" y="532.4" textLength="378.2" clip-path="url(#terminal-3378394567-line-21)">title of the image hosting site</text><text class="terminal-3378394567-r2" x="1464" y="532.4" textLength="12.2" clip-path="url(#terminal-3378394567-line-21)">
|
||||||
</text><text class="terminal-3756623299-r4" x="24.4" y="556.8" textLength="24.4" clip-path="url(#terminal-3756623299-line-22)">-e</text><text class="terminal-3756623299-r2" x="48.8" y="556.8" textLength="24.4" clip-path="url(#terminal-3756623299-line-22)">, </text><text class="terminal-3756623299-r4" x="73.2" y="556.8" textLength="207.4" clip-path="url(#terminal-3756623299-line-22)">--file-extensions</text><text class="terminal-3756623299-r5" x="292.8" y="556.8" textLength="109.8" clip-path="url(#terminal-3756623299-line-22)">EXTENSION</text><text class="terminal-3756623299-r2" x="1464" y="556.8" textLength="12.2" clip-path="url(#terminal-3756623299-line-22)">
|
</text><text class="terminal-3378394567-r4" x="24.4" y="556.8" textLength="24.4" clip-path="url(#terminal-3378394567-line-22)">-w</text><text class="terminal-3378394567-r2" x="48.8" y="556.8" textLength="24.4" clip-path="url(#terminal-3378394567-line-22)">, </text><text class="terminal-3378394567-r4" x="73.2" y="556.8" textLength="170.8" clip-path="url(#terminal-3378394567-line-22)">--web-root-url</text><text class="terminal-3378394567-r5" x="256.2" y="556.8" textLength="36.6" clip-path="url(#terminal-3378394567-line-22)">URL</text><text class="terminal-3378394567-r2" x="1464" y="556.8" textLength="12.2" clip-path="url(#terminal-3378394567-line-22)">
|
||||||
</text><text class="terminal-3756623299-r2" x="292.8" y="581.2" textLength="744.2" clip-path="url(#terminal-3756623299-line-23)">File extensions to include (can be specified multiple times).</text><text class="terminal-3756623299-r2" x="1464" y="581.2" textLength="12.2" clip-path="url(#terminal-3756623299-line-23)">
|
</text><text class="terminal-3378394567-r2" x="292.8" y="581.2" textLength="622.2" clip-path="url(#terminal-3378394567-line-23)">base URL of the web root for the image hosting site</text><text class="terminal-3378394567-r2" x="1464" y="581.2" textLength="12.2" clip-path="url(#terminal-3378394567-line-23)">
|
||||||
</text><text class="terminal-3756623299-r4" x="24.4" y="605.6" textLength="24.4" clip-path="url(#terminal-3756623299-line-24)">-m</text><text class="terminal-3756623299-r2" x="48.8" y="605.6" textLength="24.4" clip-path="url(#terminal-3756623299-line-24)">, </text><text class="terminal-3756623299-r4" x="73.2" y="605.6" textLength="170.8" clip-path="url(#terminal-3756623299-line-24)">--web-manifest</text><text class="terminal-3756623299-r2" x="292.8" y="605.6" textLength="353.8" clip-path="url(#terminal-3756623299-line-24)">Generate a web manifest file.</text><text class="terminal-3756623299-r2" x="1464" y="605.6" textLength="12.2" clip-path="url(#terminal-3756623299-line-24)">
|
</text><text class="terminal-3378394567-r4" x="24.4" y="605.6" textLength="195.2" clip-path="url(#terminal-3378394567-line-24)">--exclude-folder</text><text class="terminal-3378394567-r5" x="231.8" y="605.6" textLength="73.2" clip-path="url(#terminal-3378394567-line-24)">FOLDER</text><text class="terminal-3378394567-r2" x="1464" y="605.6" textLength="12.2" clip-path="url(#terminal-3378394567-line-24)">
|
||||||
</text><text class="terminal-3756623299-r4" x="24.4" y="630" textLength="146.4" clip-path="url(#terminal-3756623299-line-25)">--theme-path</text><text class="terminal-3756623299-r5" x="183" y="630" textLength="48.8" clip-path="url(#terminal-3756623299-line-25)">PATH</text><text class="terminal-3756623299-r2" x="292.8" y="630" textLength="329.4" clip-path="url(#terminal-3756623299-line-25)">Path to the CSS theme file.</text><text class="terminal-3756623299-r2" x="1464" y="630" textLength="12.2" clip-path="url(#terminal-3756623299-line-25)">
|
</text><text class="terminal-3378394567-r2" x="292.8" y="630" textLength="1037" clip-path="url(#terminal-3378394567-line-25)">folders to exclude from processing, globs supported (can be specified multiple times)</text><text class="terminal-3378394567-r2" x="1464" y="630" textLength="12.2" clip-path="url(#terminal-3378394567-line-25)">
|
||||||
</text><text class="terminal-3756623299-r4" x="24.4" y="654.4" textLength="231.8" clip-path="url(#terminal-3756623299-line-26)">--use-fancy-folders</text><text class="terminal-3756623299-r2" x="292.8" y="654.4" textLength="890.6" clip-path="url(#terminal-3756623299-line-26)">Enable fancy folder view instead of the default Apache directory listing.</text><text class="terminal-3756623299-r2" x="1464" y="654.4" textLength="12.2" clip-path="url(#terminal-3756623299-line-26)">
|
</text><text class="terminal-3378394567-r4" x="24.4" y="654.4" textLength="219.6" clip-path="url(#terminal-3378394567-line-26)">--folderthumbnails</text><text class="terminal-3378394567-r2" x="292.8" y="654.4" textLength="817.4" clip-path="url(#terminal-3378394567-line-26)">generate subfolder thumbnails (first image in folder will be shown)</text><text class="terminal-3378394567-r2" x="1464" y="654.4" textLength="12.2" clip-path="url(#terminal-3378394567-line-26)">
|
||||||
</text><text class="terminal-3756623299-r4" x="24.4" y="678.8" textLength="244" clip-path="url(#terminal-3756623299-line-27)">--ignore-other-files</text><text class="terminal-3756623299-r2" x="292.8" y="678.8" textLength="683.2" clip-path="url(#terminal-3756623299-line-27)">Ignore files that do not match the specified extensions.</text><text class="terminal-3756623299-r2" x="1464" y="678.8" textLength="12.2" clip-path="url(#terminal-3756623299-line-27)">
|
</text><text class="terminal-3378394567-r4" x="24.4" y="678.8" textLength="244" clip-path="url(#terminal-3378394567-line-27)">--ignore-other-files</text><text class="terminal-3378394567-r2" x="292.8" y="678.8" textLength="671" clip-path="url(#terminal-3378394567-line-27)">ignore files that do not match the specified extensions</text><text class="terminal-3378394567-r2" x="1464" y="678.8" textLength="12.2" clip-path="url(#terminal-3378394567-line-27)">
|
||||||
</text><text class="terminal-3756623299-r4" x="24.4" y="703.2" textLength="195.2" clip-path="url(#terminal-3756623299-line-28)">--exclude-folder</text><text class="terminal-3756623299-r5" x="231.8" y="703.2" textLength="73.2" clip-path="url(#terminal-3756623299-line-28)">FOLDER</text><text class="terminal-3756623299-r2" x="1464" y="703.2" textLength="12.2" clip-path="url(#terminal-3756623299-line-28)">
|
</text><text class="terminal-3378394567-r4" x="24.4" y="703.2" textLength="280.6" clip-path="url(#terminal-3378394567-line-28)">--regenerate-thumbnails</text><text class="terminal-3378394567-r2" x="1464" y="703.2" textLength="12.2" clip-path="url(#terminal-3378394567-line-28)">
|
||||||
</text><text class="terminal-3756623299-r2" x="292.8" y="727.6" textLength="1049.2" clip-path="url(#terminal-3756623299-line-29)">Folders to exclude from processing, globs supported (can be specified multiple times).</text><text class="terminal-3756623299-r2" x="1464" y="727.6" textLength="12.2" clip-path="url(#terminal-3756623299-line-29)">
|
</text><text class="terminal-3378394567-r2" x="292.8" y="727.6" textLength="585.6" clip-path="url(#terminal-3378394567-line-29)">regenerate thumbnails even if they already exist</text><text class="terminal-3378394567-r2" x="1464" y="727.6" textLength="12.2" clip-path="url(#terminal-3378394567-line-29)">
|
||||||
</text><text class="terminal-3756623299-r4" x="24.4" y="752" textLength="109.8" clip-path="url(#terminal-3756623299-line-30)">--version</text><text class="terminal-3756623299-r2" x="292.8" y="752" textLength="463.6" clip-path="url(#terminal-3756623299-line-30)">show program's version number and exit</text><text class="terminal-3756623299-r2" x="1464" y="752" textLength="12.2" clip-path="url(#terminal-3756623299-line-30)">
|
</text><text class="terminal-3378394567-r4" x="24.4" y="752" textLength="207.4" clip-path="url(#terminal-3378394567-line-30)">--reread-metadata</text><text class="terminal-3378394567-r2" x="292.8" y="752" textLength="256.2" clip-path="url(#terminal-3378394567-line-30)">reread image metadata</text><text class="terminal-3378394567-r2" x="1464" y="752" textLength="12.2" clip-path="url(#terminal-3378394567-line-30)">
|
||||||
|
</text><text class="terminal-3378394567-r4" x="24.4" y="776.4" textLength="195.2" clip-path="url(#terminal-3378394567-line-31)">--reread-sidecar</text><text class="terminal-3378394567-r2" x="292.8" y="776.4" textLength="244" clip-path="url(#terminal-3378394567-line-31)">reread sidecar files</text><text class="terminal-3378394567-r2" x="1464" y="776.4" textLength="12.2" clip-path="url(#terminal-3378394567-line-31)">
|
||||||
|
</text><text class="terminal-3378394567-r4" x="24.4" y="800.8" textLength="170.8" clip-path="url(#terminal-3378394567-line-32)">--reverse-sort</text><text class="terminal-3378394567-r2" x="292.8" y="800.8" textLength="341.6" clip-path="url(#terminal-3378394567-line-32)">sort images in reverse order</text><text class="terminal-3378394567-r2" x="1464" y="800.8" textLength="12.2" clip-path="url(#terminal-3378394567-line-32)">
|
||||||
|
</text><text class="terminal-3378394567-r4" x="24.4" y="825.2" textLength="146.4" clip-path="url(#terminal-3378394567-line-33)">--theme-path</text><text class="terminal-3378394567-r5" x="183" y="825.2" textLength="48.8" clip-path="url(#terminal-3378394567-line-33)">PATH</text><text class="terminal-3378394567-r2" x="292.8" y="825.2" textLength="317.2" clip-path="url(#terminal-3378394567-line-33)">path to the CSS theme file</text><text class="terminal-3378394567-r2" x="1464" y="825.2" textLength="12.2" clip-path="url(#terminal-3378394567-line-33)">
|
||||||
|
</text><text class="terminal-3378394567-r4" x="24.4" y="849.6" textLength="231.8" clip-path="url(#terminal-3378394567-line-34)">--use-fancy-folders</text><text class="terminal-3378394567-r2" x="292.8" y="849.6" textLength="878.4" clip-path="url(#terminal-3378394567-line-34)">enable fancy folder view instead of the default Apache directory listing</text><text class="terminal-3378394567-r2" x="1464" y="849.6" textLength="12.2" clip-path="url(#terminal-3378394567-line-34)">
|
||||||
|
</text><text class="terminal-3378394567-r4" x="24.4" y="874" textLength="109.8" clip-path="url(#terminal-3378394567-line-35)">--version</text><text class="terminal-3378394567-r2" x="292.8" y="874" textLength="463.6" clip-path="url(#terminal-3378394567-line-35)">show program's version number and exit</text><text class="terminal-3378394567-r2" x="1464" y="874" textLength="12.2" clip-path="url(#terminal-3378394567-line-35)">
|
||||||
</text>
|
</text>
|
||||||
</g>
|
</g>
|
||||||
</g>
|
</g>
|
||||||
|
|||||||
|
Before Width: | Height: | Size: 24 KiB After Width: | Height: | Size: 28 KiB |
73
hl_config.yaml
Normal file
@@ -0,0 +1,73 @@
|
|||||||
|
# yaml-language-server: $schema=https://raw.githubusercontent.com/pamburus/hl/master/schema/json/config.schema.json
|
||||||
|
$schema: https://raw.githubusercontent.com/pamburus/hl/master/schema/json/config.schema.json
|
||||||
|
|
||||||
|
# Time format, see https://man7.org/linux/man-pages/man1/date.1.html for details.
|
||||||
|
time-format: "%b %d %T.%3N"
|
||||||
|
|
||||||
|
# Time zone name, see column "TZ identifier" at
|
||||||
|
# https://en.wikipedia.org/wiki/List_of_tz_database_time_zones page.
|
||||||
|
time-zone: "Europe/Vienna"
|
||||||
|
|
||||||
|
# Settings for fields processing.
|
||||||
|
fields:
|
||||||
|
# Configuration of the predefined set of fields.
|
||||||
|
predefined:
|
||||||
|
time:
|
||||||
|
show: auto
|
||||||
|
names: ["asctime"]
|
||||||
|
logger:
|
||||||
|
names: ["defaultlogger", "consolelogger"]
|
||||||
|
level:
|
||||||
|
show: auto
|
||||||
|
variants:
|
||||||
|
- names: ["levelname"]
|
||||||
|
values:
|
||||||
|
debug: ["DEBUG"]
|
||||||
|
info: ["INFO"]
|
||||||
|
warning: ["WARNING", "WARN"]
|
||||||
|
error: ["ERROR", "FATAL", "CRITICAL"]
|
||||||
|
- names: ["levelno"]
|
||||||
|
values:
|
||||||
|
debug: [10]
|
||||||
|
info: [20]
|
||||||
|
warning: [30]
|
||||||
|
error: [40, 50]
|
||||||
|
message:
|
||||||
|
names: ["message"]
|
||||||
|
caller:
|
||||||
|
names: ["funcName"]
|
||||||
|
caller-file:
|
||||||
|
names: ["filename"]
|
||||||
|
caller-line:
|
||||||
|
names: ["lineno"]
|
||||||
|
# List of wildcard field names to ignore.
|
||||||
|
ignore: ["_*"]
|
||||||
|
# List of exact field names to hide.
|
||||||
|
hide: ["pathname", "created", "levelno", "taskname", "relativeCreated", "thread", "process", "msecs"]
|
||||||
|
|
||||||
|
# Formatting settings.
|
||||||
|
formatting:
|
||||||
|
flatten: always
|
||||||
|
punctuation:
|
||||||
|
logger-name-separator: ":"
|
||||||
|
field-key-value-separator: "="
|
||||||
|
string-opening-quote: "'"
|
||||||
|
string-closing-quote: "'"
|
||||||
|
source-location-separator: "→ "
|
||||||
|
hidden-fields-indicator: " ..."
|
||||||
|
level-left-separator: "│"
|
||||||
|
level-right-separator: "│"
|
||||||
|
input-number-prefix: "#"
|
||||||
|
input-number-left-separator: ""
|
||||||
|
input-number-right-separator: " │ "
|
||||||
|
input-name-left-separator: ""
|
||||||
|
input-name-right-separator: " │ "
|
||||||
|
input-name-clipping: "··"
|
||||||
|
input-name-common-part: "··"
|
||||||
|
array-separator: " "
|
||||||
|
|
||||||
|
# Number of processing threads, configured automatically based on CPU count if not specified.
|
||||||
|
concurrency: ~
|
||||||
|
|
||||||
|
# Currently selected theme.
|
||||||
|
theme: "neutral"
|
||||||
@@ -1,8 +1,15 @@
|
|||||||
from dataclasses import dataclass
|
from dataclasses import dataclass
|
||||||
from typing import List, Optional
|
from typing import Optional
|
||||||
import os
|
import os
|
||||||
import argparse
|
import argparse
|
||||||
from rich_argparse import RichHelpFormatter, HelpPreviewAction
|
|
||||||
|
try:
|
||||||
|
from rich_argparse import RichHelpFormatter, HelpPreviewAction
|
||||||
|
|
||||||
|
RICH = True
|
||||||
|
except ModuleNotFoundError:
|
||||||
|
RICH = False
|
||||||
|
|
||||||
|
|
||||||
if __package__ is None:
|
if __package__ is None:
|
||||||
PACKAGE = ""
|
PACKAGE = ""
|
||||||
@@ -22,10 +29,12 @@ class Args:
|
|||||||
-----------
|
-----------
|
||||||
author_name : str
|
author_name : str
|
||||||
The name of the author of the images.
|
The name of the author of the images.
|
||||||
exclude_folders : List[str]
|
exclude_folders : list[str]
|
||||||
A list of folders to exclude from processing.
|
A list of folders to exclude from processing.
|
||||||
file_extensions : List[str]
|
file_extensions : list[str]
|
||||||
A list of file extensions to include.
|
A list of file extensions to include.
|
||||||
|
folder_thumbs : bool
|
||||||
|
Wether to generate subfolder thumbnails.
|
||||||
generate_webmanifest : bool
|
generate_webmanifest : bool
|
||||||
Whether to generate a web manifest file.
|
Whether to generate a web manifest file.
|
||||||
ignore_other_files : bool
|
ignore_other_files : bool
|
||||||
@@ -49,13 +58,17 @@ class Args:
|
|||||||
"""
|
"""
|
||||||
|
|
||||||
author_name: str
|
author_name: str
|
||||||
exclude_folders: List[str]
|
exclude_folders: list[str]
|
||||||
file_extensions: List[str]
|
file_extensions: list[str]
|
||||||
|
folder_thumbs: bool
|
||||||
generate_webmanifest: bool
|
generate_webmanifest: bool
|
||||||
ignore_other_files: bool
|
ignore_other_files: bool
|
||||||
license_type: Optional[str]
|
license_type: Optional[str]
|
||||||
non_interactive_mode: bool
|
non_interactive_mode: bool
|
||||||
regenerate_thumbnails: bool
|
regenerate_thumbnails: bool
|
||||||
|
reread_metadata: bool
|
||||||
|
reread_sidecar: bool
|
||||||
|
reverse_sort: bool
|
||||||
root_directory: str
|
root_directory: str
|
||||||
site_title: str
|
site_title: str
|
||||||
theme_path: str
|
theme_path: str
|
||||||
@@ -67,12 +80,16 @@ class Args:
|
|||||||
result["author_name"] = self.author_name
|
result["author_name"] = self.author_name
|
||||||
result["exclude_folders"] = self.exclude_folders
|
result["exclude_folders"] = self.exclude_folders
|
||||||
result["file_extensions"] = self.file_extensions
|
result["file_extensions"] = self.file_extensions
|
||||||
|
result["folder_thumbs"] = self.folder_thumbs
|
||||||
result["generate_webmanifest"] = self.generate_webmanifest
|
result["generate_webmanifest"] = self.generate_webmanifest
|
||||||
result["ignore_other_files"] = self.ignore_other_files
|
result["ignore_other_files"] = self.ignore_other_files
|
||||||
if self.license_type is not None:
|
if self.license_type is not None:
|
||||||
result["license_type"] = self.license_type
|
result["license_type"] = self.license_type
|
||||||
result["non_interactive_mode"] = self.non_interactive_mode
|
result["non_interactive_mode"] = self.non_interactive_mode
|
||||||
result["regenerate_thumbnails"] = self.regenerate_thumbnails
|
result["regenerate_thumbnails"] = self.regenerate_thumbnails
|
||||||
|
result["reread_metadata"] = self.reread_metadata
|
||||||
|
result["reread_sidecar"] = self.reread_sidecar
|
||||||
|
result["reverse_sort"] = self.reverse_sort
|
||||||
result["root_directory"] = self.root_directory
|
result["root_directory"] = self.root_directory
|
||||||
result["site_title"] = self.site_title
|
result["site_title"] = self.site_title
|
||||||
result["theme_path"] = self.theme_path
|
result["theme_path"] = self.theme_path
|
||||||
@@ -96,33 +113,45 @@ def parse_arguments(version: str) -> Args:
|
|||||||
An instance of the Args class containing the parsed arguments.
|
An instance of the Args class containing the parsed arguments.
|
||||||
"""
|
"""
|
||||||
# fmt: off
|
# fmt: off
|
||||||
parser = argparse.ArgumentParser(description="Generate HTML files for a static image hosting website.", formatter_class=RichHelpFormatter)
|
if RICH:
|
||||||
parser.add_argument("--exclude-folder", help="Folders to exclude from processing, globs supported (can be specified multiple times).", action="append", dest="exclude_folders", metavar="FOLDER")
|
parser = argparse.ArgumentParser(description="generate HTML files for a static image hosting website", formatter_class=RichHelpFormatter)
|
||||||
parser.add_argument("--generate-help-preview", action=HelpPreviewAction, path="help.svg")
|
else:
|
||||||
parser.add_argument("--ignore-other-files", help="Ignore files that do not match the specified extensions.", action="store_true", default=False, dest="ignore_other_files")
|
parser = argparse.ArgumentParser(description="generate HTML files for a static image hosting website")
|
||||||
parser.add_argument("--theme-path", help="Path to the CSS theme file.", default=DEFAULT_THEME_PATH, type=str, dest="theme_path", metavar="PATH")
|
parser.add_argument("-a", "--author-name", help="name of the author of the images", default=DEFAULT_AUTHOR, type=str, dest="author_name", metavar="AUTHOR")
|
||||||
parser.add_argument("--use-fancy-folders", help="Enable fancy folder view instead of the default Apache directory listing.", action="store_true", default=False, dest="use_fancy_folders")
|
parser.add_argument("-e", "--file-extensions", help="file extensions to include (can be specified multiple times)", action="append", dest="file_extensions", metavar="EXTENSION")
|
||||||
|
parser.add_argument("-l", "--license-type", help="specify the license type for the images", choices=["cc-zero", "cc-by", "cc-by-sa", "cc-by-nd", "cc-by-nc", "cc-by-nc-sa", "cc-by-nc-nd"], default=None, dest="license_type", metavar="LICENSE")
|
||||||
|
parser.add_argument("-m", "--web-manifest", help="generate a web manifest file", action="store_true", default=False, dest="generate_webmanifest")
|
||||||
|
parser.add_argument("-n", "--non-interactive-mode", help="run in non-interactive mode, disabling progress bars", action="store_true", default=False, dest="non_interactive_mode")
|
||||||
|
parser.add_argument("-p", "--root-directory", help="root directory containing the images", required=True, type=str, dest="root_directory", metavar="ROOT")
|
||||||
|
parser.add_argument("-t", "--site-title", help="title of the image hosting site", required=True, type=str, dest="site_title", metavar="TITLE")
|
||||||
|
parser.add_argument("-w", "--web-root-url", help="base URL of the web root for the image hosting site", required=True, type=str, dest="web_root_url", metavar="URL")
|
||||||
|
parser.add_argument("--exclude-folder", help="folders to exclude from processing, globs supported (can be specified multiple times)", action="append", dest="exclude_folders", metavar="FOLDER")
|
||||||
|
parser.add_argument("--folderthumbnails", help="generate subfolder thumbnails (first image in folder will be shown)", action="store_true", default=False, dest="folder_thumbs")
|
||||||
|
if RICH:
|
||||||
|
parser.add_argument("--generate-help-preview", action=HelpPreviewAction, path="help.svg", )
|
||||||
|
parser.add_argument("--ignore-other-files", help="ignore files that do not match the specified extensions", action="store_true", default=False, dest="ignore_other_files")
|
||||||
|
parser.add_argument("--regenerate-thumbnails", help="regenerate thumbnails even if they already exist", action="store_true", default=False, dest="regenerate_thumbnails")
|
||||||
|
parser.add_argument("--reread-metadata", help="reread image metadata", action="store_true", default=False, dest="reread_metadata")
|
||||||
|
parser.add_argument("--reread-sidecar", help="reread sidecar files", action="store_true", default=False, dest="reread_sidecar")
|
||||||
|
parser.add_argument("--reverse-sort", help="sort images in reverse order", action="store_true", default=False, dest="reverse_sort")
|
||||||
|
parser.add_argument("--theme-path", help="path to the CSS theme file", default=DEFAULT_THEME_PATH, type=str, dest="theme_path", metavar="PATH")
|
||||||
|
parser.add_argument("--use-fancy-folders", help="enable fancy folder view instead of the default Apache directory listing", action="store_true", default=False, dest="use_fancy_folders")
|
||||||
parser.add_argument("--version", action="version", version=f"%(prog)s {version}")
|
parser.add_argument("--version", action="version", version=f"%(prog)s {version}")
|
||||||
parser.add_argument("-a", "--author-name", help="Name of the author of the images.", default=DEFAULT_AUTHOR, type=str, dest="author_name", metavar="AUTHOR")
|
|
||||||
parser.add_argument("-e", "--file-extensions", help="File extensions to include (can be specified multiple times).", action="append", dest="file_extensions", metavar="EXTENSION")
|
|
||||||
parser.add_argument("-l", "--license-type", help="Specify the license type for the images.", choices=["cc-zero", "cc-by", "cc-by-sa", "cc-by-nd", "cc-by-nc", "cc-by-nc-sa", "cc-by-nc-nd"], default=None, dest="license_type", metavar="LICENSE")
|
|
||||||
parser.add_argument("-m", "--web-manifest", help="Generate a web manifest file.", action="store_true", default=False, dest="generate_webmanifest")
|
|
||||||
parser.add_argument("-n", "--non-interactive-mode", help="Run in non-interactive mode, disabling progress bars.", action="store_true", default=False, dest="non_interactive_mode")
|
|
||||||
parser.add_argument("-p", "--root-directory", help="Root directory containing the images.", required=True, type=str, dest="root_directory", metavar="ROOT")
|
|
||||||
parser.add_argument("-r", "--regenerate-thumbnails", help="Regenerate thumbnails even if they already exist.", action="store_true", default=False, dest="regenerate_thumbnails")
|
|
||||||
parser.add_argument("-t", "--site-title", help="Title of the image hosting site.", required=True, type=str, dest="site_title", metavar="TITLE")
|
|
||||||
parser.add_argument("-w", "--web-root-url", help="Base URL of the web root for the image hosting site.", required=True, type=str, dest="web_root_url", metavar="URL")
|
|
||||||
parsed_args = parser.parse_args()
|
parsed_args = parser.parse_args()
|
||||||
# fmt: on
|
# fmt: on
|
||||||
_args = Args(
|
_args = Args(
|
||||||
author_name=parsed_args.author_name,
|
author_name=parsed_args.author_name,
|
||||||
exclude_folders=parsed_args.exclude_folders,
|
exclude_folders=parsed_args.exclude_folders,
|
||||||
file_extensions=parsed_args.file_extensions,
|
file_extensions=parsed_args.file_extensions,
|
||||||
|
folder_thumbs=parsed_args.folder_thumbs,
|
||||||
generate_webmanifest=parsed_args.generate_webmanifest,
|
generate_webmanifest=parsed_args.generate_webmanifest,
|
||||||
ignore_other_files=parsed_args.ignore_other_files,
|
ignore_other_files=parsed_args.ignore_other_files,
|
||||||
license_type=parsed_args.license_type,
|
license_type=parsed_args.license_type,
|
||||||
non_interactive_mode=parsed_args.non_interactive_mode,
|
non_interactive_mode=parsed_args.non_interactive_mode,
|
||||||
regenerate_thumbnails=parsed_args.regenerate_thumbnails,
|
regenerate_thumbnails=parsed_args.regenerate_thumbnails,
|
||||||
|
reread_metadata=parsed_args.reread_metadata,
|
||||||
|
reread_sidecar=parsed_args.reread_sidecar,
|
||||||
|
reverse_sort=parsed_args.reverse_sort,
|
||||||
root_directory=parsed_args.root_directory,
|
root_directory=parsed_args.root_directory,
|
||||||
site_title=parsed_args.site_title,
|
site_title=parsed_args.site_title,
|
||||||
theme_path=parsed_args.theme_path,
|
theme_path=parsed_args.theme_path,
|
||||||
|
|||||||
@@ -1,9 +1,10 @@
|
|||||||
import re
|
import re
|
||||||
import colorsys
|
import colorsys
|
||||||
from typing import Dict
|
|
||||||
|
from modules.logger import logger
|
||||||
|
|
||||||
|
|
||||||
def extract_colorscheme(theme_path: str) -> Dict[str, str]:
|
def extract_colorscheme(theme_path: str) -> dict[str, str]:
|
||||||
"""
|
"""
|
||||||
Extract color scheme from a CSS theme file.
|
Extract color scheme from a CSS theme file.
|
||||||
|
|
||||||
@@ -14,9 +15,10 @@ def extract_colorscheme(theme_path: str) -> Dict[str, str]:
|
|||||||
|
|
||||||
Returns:
|
Returns:
|
||||||
--------
|
--------
|
||||||
Dict[str, str]
|
dict[str, str]
|
||||||
Dictionary containing color scheme variables and their hexadecimal values.
|
dictionary containing color scheme variables and their hexadecimal values.
|
||||||
"""
|
"""
|
||||||
|
logger.info("extracting color scheme from theme file", extra={"theme_path": theme_path})
|
||||||
pattern = r"--(color[1-4]|bcolor1):\s*(#[0-9a-fA-F]+|rgba?\([^)]*\)|hsla?\([^)]*\)|[a-zA-Z]+);"
|
pattern = r"--(color[1-4]|bcolor1):\s*(#[0-9a-fA-F]+|rgba?\([^)]*\)|hsla?\([^)]*\)|[a-zA-Z]+);"
|
||||||
colorscheme = {}
|
colorscheme = {}
|
||||||
|
|
||||||
@@ -30,6 +32,7 @@ def extract_colorscheme(theme_path: str) -> Dict[str, str]:
|
|||||||
color_value = match[1]
|
color_value = match[1]
|
||||||
hex_color_value = css_color_to_hex(color_value)
|
hex_color_value = css_color_to_hex(color_value)
|
||||||
colorscheme[variable_name] = hex_color_value
|
colorscheme[variable_name] = hex_color_value
|
||||||
|
logger.debug("extracted variable", extra={"variable": variable_name, "value": hex_color_value})
|
||||||
|
|
||||||
return colorscheme
|
return colorscheme
|
||||||
|
|
||||||
@@ -86,10 +89,12 @@ def css_color_to_hex(css_color: str) -> str:
|
|||||||
|
|
||||||
# Helper function to convert RGB tuple to hexadecimal string
|
# Helper function to convert RGB tuple to hexadecimal string
|
||||||
def rgb_to_hex(rgb: tuple[int, int, int]) -> str:
|
def rgb_to_hex(rgb: tuple[int, int, int]) -> str:
|
||||||
|
logger.debug("converting rgb tuple to hex string", extra={"rgb": rgb})
|
||||||
return "#{:02x}{:02x}{:02x}".format(*rgb)
|
return "#{:02x}{:02x}{:02x}".format(*rgb)
|
||||||
|
|
||||||
# Helper function to convert HSL tuple to RGB tuple
|
# Helper function to convert HSL tuple to RGB tuple
|
||||||
def hsl_to_rgb(hsl: tuple[int, float, float]) -> tuple[int, int, int]:
|
def hsl_to_rgb(hsl: tuple[int, float, float]) -> tuple[int, int, int]:
|
||||||
|
logger.debug("converting hsl tuple to rgb tuple", extra={"hsl": hsl})
|
||||||
return tuple(round(c * 255) for c in colorsys.hls_to_rgb(hsl[0] / 360, hsl[1] / 100, hsl[2] / 100))
|
return tuple(round(c * 255) for c in colorsys.hls_to_rgb(hsl[0] / 360, hsl[1] / 100, hsl[2] / 100))
|
||||||
|
|
||||||
# Regular expression pattern to match CSS colors
|
# Regular expression pattern to match CSS colors
|
||||||
@@ -103,6 +108,7 @@ def css_color_to_hex(css_color: str) -> str:
|
|||||||
match = color_pattern.match(css_color.strip())
|
match = color_pattern.match(css_color.strip())
|
||||||
|
|
||||||
if not match:
|
if not match:
|
||||||
|
logger.error("invalid CSS color format", extra={"css_color": css_color})
|
||||||
raise ValueError("Invalid CSS color format")
|
raise ValueError("Invalid CSS color format")
|
||||||
|
|
||||||
groups = match.groupdict()
|
groups = match.groupdict()
|
||||||
@@ -119,8 +125,10 @@ def css_color_to_hex(css_color: str) -> str:
|
|||||||
b = int(groups["b"].rstrip("%")) * 255 // 100 if "%" in groups["b"] else int(groups["b"])
|
b = int(groups["b"].rstrip("%")) * 255 // 100 if "%" in groups["b"] else int(groups["b"])
|
||||||
a = float(groups["a"]) if groups["a"] else 1.0
|
a = float(groups["a"]) if groups["a"] else 1.0
|
||||||
if a < 1.0:
|
if a < 1.0:
|
||||||
return rgb_to_hex((r, g, b)) + "{:02x}".format(round(a * 255))
|
logger.debug("converting rgba color to hex", extra={"color": css_color, "r": r, "g": g, "b": b, "a": a})
|
||||||
|
return rgb_to_hex((r, g, b)) + f"{round(a * 255):02x}"
|
||||||
else:
|
else:
|
||||||
|
logger.debug("converting rgb color to hex", extra={"color": css_color, "r": r, "g": g, "b": b})
|
||||||
return rgb_to_hex((r, g, b))
|
return rgb_to_hex((r, g, b))
|
||||||
|
|
||||||
elif groups["hsl"]:
|
elif groups["hsl"]:
|
||||||
@@ -130,8 +138,10 @@ def css_color_to_hex(css_color: str) -> str:
|
|||||||
a = float(groups["a"]) if groups["a"] else 1.0
|
a = float(groups["a"]) if groups["a"] else 1.0
|
||||||
rgb_color = hsl_to_rgb((h, s, l))
|
rgb_color = hsl_to_rgb((h, s, l))
|
||||||
if a < 1.0:
|
if a < 1.0:
|
||||||
return rgb_to_hex(rgb_color) + "{:02x}".format(round(a * 255))
|
logger.debug("converting hsla color to hex", extra={"color": css_color, "hsl": (h, s, l), "a": a})
|
||||||
|
return rgb_to_hex(rgb_color) + f"{round(a * 255):02x}"
|
||||||
else:
|
else:
|
||||||
|
logger.debug("converting hsl color to hex", extra={"color": css_color, "hsl": (h, s, l)})
|
||||||
return rgb_to_hex(rgb_color)
|
return rgb_to_hex(rgb_color)
|
||||||
|
|
||||||
# fmt: off
|
# fmt: off
|
||||||
@@ -182,7 +192,9 @@ def css_color_to_hex(css_color: str) -> str:
|
|||||||
'turquoise': '#40e0d0', 'violet': '#ee82ee', 'wheat': '#f5deb3', 'white': '#ffffff',
|
'turquoise': '#40e0d0', 'violet': '#ee82ee', 'wheat': '#f5deb3', 'white': '#ffffff',
|
||||||
'whitesmoke': '#f5f5f5', 'yellow': '#ffff00', 'yellowgreen': '#9acd32'
|
'whitesmoke': '#f5f5f5', 'yellow': '#ffff00', 'yellowgreen': '#9acd32'
|
||||||
}
|
}
|
||||||
|
logger.debug("parsing css color string", extra={"css_color": css_color})
|
||||||
return named_colors[groups['name'].lower()]
|
return named_colors[groups['name'].lower()]
|
||||||
# fmt: on
|
# fmt: on
|
||||||
|
|
||||||
|
logger.error("invalid CSS color format", extra={"css_color": css_color})
|
||||||
raise ValueError("Invalid CSS color format")
|
raise ValueError("Invalid CSS color format")
|
||||||
|
|||||||
@@ -1,15 +1,21 @@
|
|||||||
import os
|
import os
|
||||||
|
import re
|
||||||
import urllib.parse
|
import urllib.parse
|
||||||
import fnmatch
|
import fnmatch
|
||||||
import json
|
import json
|
||||||
from typing import Any, Dict, List, Tuple
|
import html
|
||||||
|
from typing import Any
|
||||||
|
from datetime import datetime
|
||||||
|
from collections import defaultdict
|
||||||
|
|
||||||
import numpy as np
|
|
||||||
from tqdm.auto import tqdm
|
from tqdm.auto import tqdm
|
||||||
from PIL import Image, ExifTags
|
from PIL import Image, ExifTags, TiffImagePlugin, UnidentifiedImageError
|
||||||
from jinja2 import Environment, FileSystemLoader
|
from jinja2 import Environment, FileSystemLoader
|
||||||
|
from defusedxml import ElementTree
|
||||||
|
from bs4 import BeautifulSoup
|
||||||
|
|
||||||
import modules.cclicense as cclicense
|
from modules.logger import logger
|
||||||
|
from modules import cclicense
|
||||||
from modules.argumentparser import Args
|
from modules.argumentparser import Args
|
||||||
|
|
||||||
# Constants for file paths and exclusions
|
# Constants for file paths and exclusions
|
||||||
@@ -22,57 +28,127 @@ FAVICON_PATH = ".static/favicon.ico"
|
|||||||
GLOBAL_CSS_PATH = ".static/global.css"
|
GLOBAL_CSS_PATH = ".static/global.css"
|
||||||
EXCLUDES = ["index.html", "manifest.json", "robots.txt"]
|
EXCLUDES = ["index.html", "manifest.json", "robots.txt"]
|
||||||
|
|
||||||
# Set the maximum image pixels to prevent decompression bomb DOS attacks
|
# Set the maximum image pixels
|
||||||
Image.MAX_IMAGE_PIXELS = 933120000
|
Image.MAX_IMAGE_PIXELS = 933120000
|
||||||
|
|
||||||
# Initialize Jinja2 environment for template rendering
|
# Initialize Jinja2 environment for template rendering
|
||||||
env = Environment(loader=FileSystemLoader(os.path.join(SCRIPTDIR, "templates")))
|
env = Environment(loader=FileSystemLoader(os.path.join(SCRIPTDIR, "templates")))
|
||||||
thumbnails: List[Tuple[str, str]] = []
|
thumbnails: list[tuple[str, str, str]] = []
|
||||||
info: Dict[str, str] = {}
|
info: dict[str, str] = {}
|
||||||
pbardict: Dict[str, tqdm] = {}
|
licens: dict[str, str] = {}
|
||||||
|
|
||||||
|
|
||||||
def initialize_sizelist(folder: str) -> Dict[str, Dict[str, int]]:
|
def getxmp(strbuffer: str) -> dict[str, Any]:
|
||||||
"""
|
"""
|
||||||
Initializes the size list JSON file if it doesn't exist.
|
Returns a dictionary containing the XMP tags.
|
||||||
|
Requires defusedxml to be installed.
|
||||||
|
|
||||||
|
:returns: XMP tags in a dictionary.
|
||||||
|
"""
|
||||||
|
|
||||||
|
def get_name(tag: str) -> str:
|
||||||
|
return re.sub("^{[^}]+}", "", tag)
|
||||||
|
|
||||||
|
def get_value(element) -> str | dict[str, Any] | None:
|
||||||
|
value: dict[str, Any] = {get_name(k): v for k, v in element.attrib.items()}
|
||||||
|
children = list(element)
|
||||||
|
if children:
|
||||||
|
for child in children:
|
||||||
|
name = get_name(child.tag)
|
||||||
|
child_value = get_value(child)
|
||||||
|
if name in value:
|
||||||
|
if not isinstance(value[name], list):
|
||||||
|
value[name] = [value[name]]
|
||||||
|
value[name].append(child_value)
|
||||||
|
else:
|
||||||
|
value[name] = child_value
|
||||||
|
elif value:
|
||||||
|
if element.text:
|
||||||
|
value["text"] = element.text
|
||||||
|
else:
|
||||||
|
return element.text
|
||||||
|
return value
|
||||||
|
|
||||||
|
root = ElementTree.fromstring(strbuffer)
|
||||||
|
return {get_name(root.tag): get_value(root)}
|
||||||
|
|
||||||
|
|
||||||
|
def initialize_metadata(folder: str) -> dict[str, dict[str, int]]:
|
||||||
|
"""
|
||||||
|
Initializes the metadata JSON file if it doesn't exist.
|
||||||
|
|
||||||
Args:
|
Args:
|
||||||
folder (str): The folder in which the size list file is located.
|
folder (str): The folder in which the metadata file is located.
|
||||||
|
|
||||||
Returns:
|
Returns:
|
||||||
Dict[str, Dict[str, int]]: The size list dictionary.
|
dict[str, dict[str, int]]: The metadata dictionary.
|
||||||
"""
|
"""
|
||||||
sizelist = {}
|
metadata = {}
|
||||||
sizelist_path = os.path.join(folder, ".sizelist.json")
|
metadata_path = os.path.join(folder, ".metadata.json")
|
||||||
if not os.path.exists(sizelist_path):
|
if not os.path.exists(metadata_path):
|
||||||
with open(sizelist_path, "x", encoding="utf-8") as sizelistfile:
|
logger.info("creating new metadata file", extra={"file": metadata_path})
|
||||||
sizelistfile.write("{}")
|
with open(metadata_path, "x", encoding="utf-8") as metadatafile:
|
||||||
with open(sizelist_path, "r+", encoding="utf-8") as sizelistfile:
|
metadatafile.write("{}")
|
||||||
|
with open(metadata_path, "r+", encoding="utf-8") as metadatafile:
|
||||||
|
logger.info("reading metadata file", extra={"file": metadata_path})
|
||||||
try:
|
try:
|
||||||
sizelist = json.loads(sizelistfile.read())
|
metadata = json.loads(metadatafile.read())
|
||||||
except json.decoder.JSONDecodeError:
|
except json.decoder.JSONDecodeError:
|
||||||
sizelist = {}
|
logger.warning("invalid JSON in metadata file", extra={"file": metadata_path})
|
||||||
return sizelist
|
metadata = {}
|
||||||
|
|
||||||
|
# remove old sizelist if it exists
|
||||||
def update_sizelist(sizelist: Dict[str, Dict[str, int]], folder: str) -> None:
|
|
||||||
"""
|
|
||||||
Updates the size list JSON file.
|
|
||||||
|
|
||||||
Args:
|
|
||||||
sizelist (Dict[str, Dict[str, int]]): The size list dictionary to be written to the file.
|
|
||||||
folder (str): The folder in which the size list file is located.
|
|
||||||
"""
|
|
||||||
sizelist_path = os.path.join(folder, ".sizelist.json")
|
sizelist_path = os.path.join(folder, ".sizelist.json")
|
||||||
if sizelist != {}:
|
|
||||||
with open(sizelist_path, "w", encoding="utf-8") as sizelistfile:
|
|
||||||
sizelistfile.write(json.dumps(sizelist, indent=4))
|
|
||||||
else:
|
|
||||||
if os.path.exists(sizelist_path):
|
if os.path.exists(sizelist_path):
|
||||||
|
with open(sizelist_path, "r") as sizelist:
|
||||||
|
metadata = json.loads(sizelist.read())
|
||||||
|
logger.warning("found old .sizelist.json, removing it...", extra={"path": sizelist_path})
|
||||||
os.remove(sizelist_path)
|
os.remove(sizelist_path)
|
||||||
|
|
||||||
|
# convert from old metadata format
|
||||||
|
if "images" not in metadata and "subfolders" not in metadata:
|
||||||
|
images = metadata.copy()
|
||||||
|
metadata = {}
|
||||||
|
metadata["images"] = images
|
||||||
|
elif "images" not in metadata:
|
||||||
|
metadata["images"] = {}
|
||||||
|
for k, v in metadata["images"].items():
|
||||||
|
if "width" in v:
|
||||||
|
metadata["images"][k]["w"] = v["width"]
|
||||||
|
del metadata["images"][k]["width"]
|
||||||
|
if "height" in v:
|
||||||
|
metadata["images"][k]["h"] = v["height"]
|
||||||
|
del metadata["images"][k]["height"]
|
||||||
|
if "tags" not in v:
|
||||||
|
metadata["images"][k]["tags"] = []
|
||||||
|
if "exifdata" not in v:
|
||||||
|
metadata["images"][k]["exifdata"] = None
|
||||||
|
if "xmp" not in v:
|
||||||
|
metadata["images"][k]["xmp"] = None
|
||||||
|
|
||||||
def get_image_info(item: str, folder: str) -> Dict[str, Any]:
|
return metadata
|
||||||
|
|
||||||
|
|
||||||
|
def update_metadata(metadata: dict[str, dict[str, Any]], folder: str) -> None:
|
||||||
|
"""
|
||||||
|
Updates the metadata JSON file.
|
||||||
|
|
||||||
|
Args:
|
||||||
|
metadata (dict[str, dict[str, int]]): The metadata dictionary to be written to the file.
|
||||||
|
folder (str): The folder in which the metadata file is located.
|
||||||
|
"""
|
||||||
|
metadata_path = os.path.join(folder, ".metadata.json")
|
||||||
|
if metadata:
|
||||||
|
with open(metadata_path, "w", encoding="utf-8") as metadatafile:
|
||||||
|
logger.info("writing metadata file", extra={"file": metadata_path})
|
||||||
|
metadatafile.write(json.dumps(metadata, indent=4))
|
||||||
|
else:
|
||||||
|
if os.path.exists(metadata_path):
|
||||||
|
logger.info("deleting empty metadata file", extra={"file": metadata_path})
|
||||||
|
os.remove(metadata_path)
|
||||||
|
|
||||||
|
|
||||||
|
def get_image_info(item: str, folder: str) -> dict[str, Any]:
|
||||||
"""
|
"""
|
||||||
Extracts image information and EXIF data.
|
Extracts image information and EXIF data.
|
||||||
|
|
||||||
@@ -81,18 +157,184 @@ def get_image_info(item: str, folder: str) -> Dict[str, Any]:
|
|||||||
folder (str): The folder containing the image.
|
folder (str): The folder containing the image.
|
||||||
|
|
||||||
Returns:
|
Returns:
|
||||||
Dict[str, Any]: A dictionary containing image width, height, and EXIF data.
|
dict[str, Any]: A dictionary containing image width, height, and EXIF data.
|
||||||
"""
|
"""
|
||||||
with Image.open(os.path.join(folder, item)) as img:
|
file = os.path.join(folder, item)
|
||||||
exif = img.getexif()
|
try:
|
||||||
|
with Image.open(file) as img:
|
||||||
|
logger.info("extracting image information", extra={"file": file})
|
||||||
width, height = img.size
|
width, height = img.size
|
||||||
exifdata = {ExifTags.TAGS.get(key, key): val for key, val in exif.items()}
|
exif = img.getexif()
|
||||||
|
xmpdata = img.getxmp()
|
||||||
|
|
||||||
|
except UnidentifiedImageError:
|
||||||
|
logger.error("cannot identify image file", extra={"file": file})
|
||||||
|
print(f"cannot identify image file: {file}")
|
||||||
|
return {"w": None, "h": None, "tags": None, "exifdata": None, "xmp": None}
|
||||||
|
if exif:
|
||||||
|
logger.info("extracting EXIF data", extra={"file": file})
|
||||||
|
ifd = exif.get_ifd(ExifTags.IFD.Exif)
|
||||||
|
exifdatas = dict(exif.items()) | ifd
|
||||||
|
exifdata = {}
|
||||||
|
for tag_id in exifdatas:
|
||||||
|
tag = ExifTags.TAGS.get(tag_id, tag_id)
|
||||||
|
content = exifdatas.get(tag_id)
|
||||||
|
if isinstance(content, bytes):
|
||||||
|
content = "0x" + content.hex()
|
||||||
|
if isinstance(content, TiffImagePlugin.IFDRational):
|
||||||
|
content = content.limit_rational(1000000)
|
||||||
|
if isinstance(content, tuple):
|
||||||
|
newtuple = ()
|
||||||
|
for i in content:
|
||||||
|
if isinstance(i, TiffImagePlugin.IFDRational):
|
||||||
|
newtuple = (*newtuple, i.limit_rational(1000000))
|
||||||
|
if newtuple:
|
||||||
|
content = newtuple
|
||||||
|
if tag in ["DateTime", "DateTimeOriginal", "DateTimeDigitized"]:
|
||||||
|
epr = r"\d{4}:\d{2}:\d{2} \d{2}:\d{2}:\d{2}"
|
||||||
|
if re.match(epr, str(content)):
|
||||||
|
try:
|
||||||
|
content = datetime.strptime(str(content), "%Y:%m:%d %H:%M:%S").strftime("%Y-%m-%d %H:%M:%S")
|
||||||
|
except ValueError:
|
||||||
|
content = None
|
||||||
|
else:
|
||||||
|
content = None
|
||||||
|
exifdata[tag] = content
|
||||||
if "Orientation" in exifdata and exifdata["Orientation"] in [6, 8]:
|
if "Orientation" in exifdata and exifdata["Orientation"] in [6, 8]:
|
||||||
|
logger.info("image is rotated", extra={"file": file})
|
||||||
width, height = height, width
|
width, height = height, width
|
||||||
return {"width": width, "height": height}
|
for key in ["PrintImageMatching", "UserComment", "MakerNote"]:
|
||||||
|
if key in exifdata:
|
||||||
|
del exifdata[key]
|
||||||
|
else:
|
||||||
|
exifdata = None
|
||||||
|
tags = []
|
||||||
|
xmp = None
|
||||||
|
if xmpdata:
|
||||||
|
logger.info("extracting XMP data", extra={"file": file})
|
||||||
|
try:
|
||||||
|
tags = xmpdata["xmpmeta"]["RDF"]["Description"]["subject"]["Bag"]["li"]
|
||||||
|
if isinstance(tags, str):
|
||||||
|
tags = [tags]
|
||||||
|
except TypeError:
|
||||||
|
pass
|
||||||
|
except KeyError:
|
||||||
|
pass
|
||||||
|
try:
|
||||||
|
tags = xmpdata["xapmeta"]["RDF"]["Description"]["subject"]["Bag"]["li"]
|
||||||
|
if isinstance(tags, str):
|
||||||
|
tags = [tags]
|
||||||
|
except TypeError:
|
||||||
|
pass
|
||||||
|
except KeyError:
|
||||||
|
pass
|
||||||
|
try:
|
||||||
|
tags = xmpdata["xmpmeta"]["RDF"]["Description"]["hierarchicalSubject"]["Bag"]["li"]
|
||||||
|
if isinstance(tags, str):
|
||||||
|
tags = [tags]
|
||||||
|
except TypeError:
|
||||||
|
pass
|
||||||
|
except KeyError:
|
||||||
|
pass
|
||||||
|
try:
|
||||||
|
tags = xmpdata["xapmeta"]["RDF"]["Description"]["hierarchicalSubject"]["Bag"]["li"]
|
||||||
|
if isinstance(tags, str):
|
||||||
|
tags = [tags]
|
||||||
|
except TypeError:
|
||||||
|
pass
|
||||||
|
except KeyError:
|
||||||
|
pass
|
||||||
|
sidecarfile = os.path.join(folder, item + ".xmp")
|
||||||
|
if os.path.exists(sidecarfile):
|
||||||
|
logger.info("xmp sidecar file found", extra={"file": sidecarfile})
|
||||||
|
try:
|
||||||
|
tags = get_tags(sidecarfile)
|
||||||
|
except Exception as e:
|
||||||
|
logger.error(e)
|
||||||
|
if None in tags:
|
||||||
|
tags.remove(None)
|
||||||
|
return {"w": width, "h": height, "tags": tags, "exifdata": exifdata, "xmp": xmp}
|
||||||
|
|
||||||
|
|
||||||
def process_image(item: str, folder: str, _args: Args, baseurl: str, sizelist: Dict[str, Dict[str, int]], raw: List[str]) -> Dict[str, Any]:
|
def nested_dict():
|
||||||
|
return defaultdict(nested_dict)
|
||||||
|
|
||||||
|
|
||||||
|
def insert_path(d, path):
|
||||||
|
for part in path[:-1]:
|
||||||
|
d = d[part]
|
||||||
|
last = path[-1]
|
||||||
|
if not isinstance(d[last], dict):
|
||||||
|
d[last] = {}
|
||||||
|
|
||||||
|
|
||||||
|
def finalize(d):
|
||||||
|
if isinstance(d, defaultdict):
|
||||||
|
return {k: finalize(d[k]) for k in sorted(d)}
|
||||||
|
return d or []
|
||||||
|
|
||||||
|
|
||||||
|
def parse_hierarchical_tags(tags, delimiter="|"):
|
||||||
|
tree = nested_dict()
|
||||||
|
for tag in tags:
|
||||||
|
parts = tag.split(delimiter)
|
||||||
|
insert_path(tree, parts)
|
||||||
|
return finalize(tree)
|
||||||
|
|
||||||
|
|
||||||
|
def get_tags(sidecarfile: str) -> list[str]:
|
||||||
|
"""
|
||||||
|
Extracts Tags from XMP sidecar file
|
||||||
|
|
||||||
|
Args:
|
||||||
|
sidecarfile (str): The path to the XMP sidecar file.
|
||||||
|
|
||||||
|
Returns:
|
||||||
|
list[str]: List containing image tags.
|
||||||
|
"""
|
||||||
|
logger.info("extracting XMP sidecar file data", extra={"file": sidecarfile})
|
||||||
|
with open(sidecarfile) as sidecar:
|
||||||
|
strbuffer = sidecar.read()
|
||||||
|
xmpdata = getxmp(strbuffer)
|
||||||
|
tags = []
|
||||||
|
try:
|
||||||
|
tags = xmpdata["xmpmeta"]["RDF"]["Description"]["subject"]["Bag"]["li"]
|
||||||
|
if isinstance(tags, str):
|
||||||
|
tags = [tags]
|
||||||
|
except TypeError:
|
||||||
|
pass
|
||||||
|
except KeyError:
|
||||||
|
pass
|
||||||
|
try:
|
||||||
|
tags = xmpdata["xapmeta"]["RDF"]["Description"]["subject"]["Bag"]["li"]
|
||||||
|
if isinstance(tags, str):
|
||||||
|
tags = [tags]
|
||||||
|
except TypeError:
|
||||||
|
pass
|
||||||
|
except KeyError:
|
||||||
|
pass
|
||||||
|
try:
|
||||||
|
tags = xmpdata["xmpmeta"]["RDF"]["Description"]["hierarchicalSubject"]["Bag"]["li"]
|
||||||
|
if isinstance(tags, str):
|
||||||
|
tags = [tags]
|
||||||
|
except TypeError:
|
||||||
|
pass
|
||||||
|
except KeyError:
|
||||||
|
pass
|
||||||
|
try:
|
||||||
|
tags = xmpdata["xapmeta"]["RDF"]["Description"]["hierarchicalSubject"]["Bag"]["li"]
|
||||||
|
if isinstance(tags, str):
|
||||||
|
tags = [tags]
|
||||||
|
except TypeError:
|
||||||
|
pass
|
||||||
|
except KeyError:
|
||||||
|
pass
|
||||||
|
if None in tags:
|
||||||
|
tags.remove(None)
|
||||||
|
return tags
|
||||||
|
|
||||||
|
|
||||||
|
def process_image(item: str, folder: str, _args: Args, baseurl: str, metadata: dict[str, dict[str, int]], raw: list[str]) -> dict[str, Any]:
|
||||||
"""
|
"""
|
||||||
Processes an image and prepares its data for the HTML template.
|
Processes an image and prepares its data for the HTML template.
|
||||||
|
|
||||||
@@ -101,22 +343,30 @@ def process_image(item: str, folder: str, _args: Args, baseurl: str, sizelist: D
|
|||||||
folder (str): The folder containing the image.
|
folder (str): The folder containing the image.
|
||||||
_args (Args): Parsed command line arguments.
|
_args (Args): Parsed command line arguments.
|
||||||
baseurl (str): Base URL for the web root.
|
baseurl (str): Base URL for the web root.
|
||||||
sizelist (Dict[str, Dict[str, int]]): Dictionary containing size information for images.
|
metadata (dict[str, dict[str, int]]): dictionary containing size information for images.
|
||||||
raw (List[str]): List of raw image file extensions.
|
raw (list[str]): list of raw image file extensions.
|
||||||
|
|
||||||
Returns:
|
Returns:
|
||||||
Dict[str, Any]: Dictionary containing image details for HTML rendering.
|
dict[str, Any]: dictionary containing image details for HTML rendering.
|
||||||
"""
|
"""
|
||||||
extsplit = os.path.splitext(item)
|
extsplit = os.path.splitext(item)
|
||||||
if item not in sizelist or _args.regenerate_thumbnails:
|
sidecarfile = os.path.join(folder, item + ".xmp")
|
||||||
sizelist[item] = get_image_info(item, folder)
|
if item not in metadata["images"] or _args.reread_metadata:
|
||||||
|
metadata["images"][item] = get_image_info(item, folder)
|
||||||
|
if _args.reread_sidecar and os.path.exists(sidecarfile):
|
||||||
|
logger.info("xmp sidecar file found", extra={"file": sidecarfile})
|
||||||
|
try:
|
||||||
|
metadata["images"][item]["tags"] = get_tags(sidecarfile)
|
||||||
|
except Exception as e:
|
||||||
|
logger.error(e)
|
||||||
|
|
||||||
image = {
|
image = {
|
||||||
"url": f"{_args.web_root_url}{baseurl}{urllib.parse.quote(item)}",
|
"src": f"{_args.web_root_url}{baseurl}{urllib.parse.quote(item)}",
|
||||||
"thumbnail": f"{_args.web_root_url}.thumbnails/{baseurl}{urllib.parse.quote(item)}.jpg",
|
"msrc": f"{_args.web_root_url}.thumbnails/{baseurl}{urllib.parse.quote(item)}.jpg",
|
||||||
"name": item,
|
"name": item,
|
||||||
"width": sizelist[item]["width"],
|
"w": metadata["images"][item]["w"],
|
||||||
"height": sizelist[item]["height"],
|
"h": metadata["images"][item]["h"],
|
||||||
|
"tags": metadata["images"][item]["tags"],
|
||||||
}
|
}
|
||||||
path = os.path.join(_args.root_directory, ".thumbnails", baseurl, item + ".jpg")
|
path = os.path.join(_args.root_directory, ".thumbnails", baseurl, item + ".jpg")
|
||||||
if not os.path.exists(path) or _args.regenerate_thumbnails:
|
if not os.path.exists(path) or _args.regenerate_thumbnails:
|
||||||
@@ -125,16 +375,22 @@ def process_image(item: str, folder: str, _args: Args, baseurl: str, sizelist: D
|
|||||||
thumbnails.append((folder, item, _args.root_directory))
|
thumbnails.append((folder, item, _args.root_directory))
|
||||||
|
|
||||||
for _raw in raw:
|
for _raw in raw:
|
||||||
if os.path.exists(os.path.join(folder, extsplit[0] + _raw)):
|
file = os.path.join(folder, extsplit[0] + _raw)
|
||||||
url = urllib.parse.quote(extsplit[0]) + _raw
|
if os.path.exists(file):
|
||||||
|
url = f"{_args.web_root_url}{baseurl}{urllib.parse.quote(extsplit[0])}{_raw}"
|
||||||
if _raw in (".tif", ".tiff"):
|
if _raw in (".tif", ".tiff"):
|
||||||
image["tiff"] = f"{_args.web_root_url}{baseurl}{url}"
|
logger.info("tiff file found", extra={"file": file})
|
||||||
|
image["tiff"] = url
|
||||||
else:
|
else:
|
||||||
image["raw"] = f"{_args.web_root_url}{baseurl}{url}"
|
logger.info("raw file found", extra={"file": file, "extension": _raw})
|
||||||
return image
|
image["raw"] = url
|
||||||
|
|
||||||
|
metadata["images"][item].update(image)
|
||||||
|
|
||||||
|
return image, metadata
|
||||||
|
|
||||||
|
|
||||||
def generate_html(folder: str, title: str, _args: Args, raw: List[str], version: str) -> None:
|
def generate_html(folder: str, title: str, _args: Args, raw: list[str], version: str, logo: str) -> list[str]:
|
||||||
"""
|
"""
|
||||||
Generates HTML content for a folder of images.
|
Generates HTML content for a folder of images.
|
||||||
|
|
||||||
@@ -142,50 +398,74 @@ def generate_html(folder: str, title: str, _args: Args, raw: List[str], version:
|
|||||||
folder (str): The folder to generate HTML for.
|
folder (str): The folder to generate HTML for.
|
||||||
title (str): The title of the HTML page.
|
title (str): The title of the HTML page.
|
||||||
_args (Args): Parsed command line arguments.
|
_args (Args): Parsed command line arguments.
|
||||||
raw (List[str]): Raw image file names.
|
raw (list[str]): Raw image file names.
|
||||||
"""
|
"""
|
||||||
sizelist = initialize_sizelist(folder)
|
logger.info("processing folder", extra={"folder": folder})
|
||||||
|
if _args.regenerate_thumbnails:
|
||||||
|
if os.path.exists(os.path.join(folder, ".metadata.json")):
|
||||||
|
logger.info("removing .metadata.json", extra={"folder": folder})
|
||||||
|
os.remove(os.path.join(folder, ".metadata.json"))
|
||||||
|
metadata = initialize_metadata(folder)
|
||||||
items = sorted(os.listdir(folder))
|
items = sorted(os.listdir(folder))
|
||||||
|
|
||||||
contains_files = False
|
contains_files = False
|
||||||
images = []
|
images = []
|
||||||
subfolders = []
|
subfolders = []
|
||||||
|
subfoldertags = set()
|
||||||
foldername = folder.removeprefix(_args.root_directory)
|
foldername = folder.removeprefix(_args.root_directory)
|
||||||
foldername = f"{foldername}/" if foldername else ""
|
foldername = f"{foldername}/" if foldername else ""
|
||||||
baseurl = urllib.parse.quote(foldername)
|
baseurl = urllib.parse.quote(foldername)
|
||||||
|
|
||||||
|
gone = [item for item in metadata["images"] if item not in items]
|
||||||
|
for gon in gone:
|
||||||
|
del metadata["images"][gon]
|
||||||
|
|
||||||
create_thumbnail_folder(foldername, _args.root_directory)
|
create_thumbnail_folder(foldername, _args.root_directory)
|
||||||
|
|
||||||
|
logger.info("processing contents", extra={"folder": folder})
|
||||||
if not _args.non_interactive_mode:
|
if not _args.non_interactive_mode:
|
||||||
pbardict[folder] = tqdm(total=len(items), desc=f"Getting image infos - {folder}", unit="files", ascii=True, dynamic_ncols=True)
|
for item in tqdm(items, total=len(items), desc=f"Getting image infos - {folder}", unit="files", ascii=True, dynamic_ncols=True):
|
||||||
|
|
||||||
for item in items:
|
|
||||||
if item not in EXCLUDES and not item.startswith("."):
|
if item not in EXCLUDES and not item.startswith("."):
|
||||||
if os.path.isdir(os.path.join(folder, item)):
|
if os.path.isdir(os.path.join(folder, item)):
|
||||||
process_subfolder(item, folder, baseurl, subfolders, _args, raw, version)
|
subfoldertags.update(process_subfolder(item, folder, baseurl, subfolders, _args, raw, version, logo))
|
||||||
else:
|
else:
|
||||||
contains_files = True
|
contains_files = True
|
||||||
if os.path.splitext(item)[1].lower() in _args.file_extensions:
|
if os.path.splitext(item)[1].lower() in _args.file_extensions:
|
||||||
images.append(process_image(item, folder, _args, baseurl, sizelist, raw))
|
img, metadata = process_image(item, folder, _args, baseurl, metadata, raw)
|
||||||
|
images.append(img)
|
||||||
if item == "info":
|
if item == "info":
|
||||||
process_info_file(folder, item)
|
process_info_file(folder, item)
|
||||||
|
if item == "LICENSE":
|
||||||
|
process_license(folder, item)
|
||||||
|
else:
|
||||||
|
for item in items:
|
||||||
|
if item not in EXCLUDES and not item.startswith("."):
|
||||||
|
if os.path.isdir(os.path.join(folder, item)):
|
||||||
|
subfoldertags.update(process_subfolder(item, folder, baseurl, subfolders, _args, raw, version, logo))
|
||||||
|
else:
|
||||||
|
contains_files = True
|
||||||
|
if os.path.splitext(item)[1].lower() in _args.file_extensions:
|
||||||
|
img, metadata = process_image(item, folder, _args, baseurl, metadata, raw)
|
||||||
|
images.append(img)
|
||||||
|
if item == "info":
|
||||||
|
process_info_file(folder, item)
|
||||||
|
if item == "LICENSE":
|
||||||
|
process_license(folder, item)
|
||||||
|
|
||||||
if not _args.non_interactive_mode:
|
metadata["subfolders"] = subfolders
|
||||||
pbardict[folder].update(1)
|
if _args.reverse_sort:
|
||||||
|
metadata["images"] = {key: metadata["images"][key] for key in sorted(metadata["images"], reverse=True)}
|
||||||
if not _args.non_interactive_mode:
|
else:
|
||||||
pbardict[folder].close()
|
metadata["images"] = {key: metadata["images"][key] for key in sorted(metadata["images"])}
|
||||||
|
update_metadata(metadata, folder)
|
||||||
update_sizelist(sizelist, folder)
|
|
||||||
|
|
||||||
if should_generate_html(images, contains_files, _args):
|
if should_generate_html(images, contains_files, _args):
|
||||||
create_html_file(folder, title, foldername, images, subfolders, _args, version)
|
subfoldertags = create_html_file(folder, title, foldername, images, subfolders, _args, version, logo, subfoldertags)
|
||||||
else:
|
else:
|
||||||
if os.path.exists(os.path.join(folder, "index.html")):
|
if os.path.exists(os.path.join(folder, "index.html")):
|
||||||
|
logger.info("removing existing index.html", extra={"folder": folder})
|
||||||
os.remove(os.path.join(folder, "index.html"))
|
os.remove(os.path.join(folder, "index.html"))
|
||||||
|
return subfoldertags
|
||||||
if not _args.non_interactive_mode:
|
|
||||||
pbardict["htmlbar"].update(1)
|
|
||||||
|
|
||||||
|
|
||||||
def create_thumbnail_folder(foldername: str, root_directory: str) -> None:
|
def create_thumbnail_folder(foldername: str, root_directory: str) -> None:
|
||||||
@@ -198,10 +478,11 @@ def create_thumbnail_folder(foldername: str, root_directory: str) -> None:
|
|||||||
"""
|
"""
|
||||||
thumbnails_path = os.path.join(root_directory, ".thumbnails", foldername)
|
thumbnails_path = os.path.join(root_directory, ".thumbnails", foldername)
|
||||||
if not os.path.exists(thumbnails_path):
|
if not os.path.exists(thumbnails_path):
|
||||||
|
logger.info("creating thumbnail folder", extra={"path": thumbnails_path})
|
||||||
os.mkdir(thumbnails_path)
|
os.mkdir(thumbnails_path)
|
||||||
|
|
||||||
|
|
||||||
def process_subfolder(item: str, folder: str, baseurl: str, subfolders: List[Dict[str, str]], _args: Args, raw: List[str], version: str) -> None:
|
def process_subfolder(item: str, folder: str, baseurl: str, subfolders: list[dict[str, str | None]], _args: Args, raw: list[str], version: str, logo: str) -> list[str]:
|
||||||
"""
|
"""
|
||||||
Processes a subfolder.
|
Processes a subfolder.
|
||||||
|
|
||||||
@@ -209,15 +490,46 @@ def process_subfolder(item: str, folder: str, baseurl: str, subfolders: List[Dic
|
|||||||
item (str): The name of the subfolder.
|
item (str): The name of the subfolder.
|
||||||
folder (str): The parent folder containing the subfolder.
|
folder (str): The parent folder containing the subfolder.
|
||||||
baseurl (str): Base URL for the web root.
|
baseurl (str): Base URL for the web root.
|
||||||
subfolders (List[Dict[str, str]]): List to store subfolder details.
|
subfolders (list[dict[str, str]]): list to store subfolder details.
|
||||||
_args (Args): Parsed command line arguments.
|
_args (Args): Parsed command line arguments.
|
||||||
raw (List[str]): Raw image file extensions.
|
raw (list[str]): Raw image file extensions.
|
||||||
"""
|
"""
|
||||||
subfolder_url = f"{_args.web_root_url}{baseurl}{urllib.parse.quote(item)}/index.html" if _args.web_root_url.startswith("file://") else f"{_args.web_root_url}{baseurl}{urllib.parse.quote(item)}"
|
subfolder_url = (
|
||||||
subfolders.append({"url": subfolder_url, "name": item})
|
f"{_args.web_root_url}{baseurl}{urllib.parse.quote(item)}/index.html"
|
||||||
|
if _args.web_root_url.startswith("file://")
|
||||||
|
else f"{_args.web_root_url}{baseurl}{urllib.parse.quote(item)}"
|
||||||
|
)
|
||||||
|
thumb = None
|
||||||
|
if _args.folder_thumbs:
|
||||||
|
thumbitems = [i for i in sorted(os.listdir(os.path.join(folder, item))) if os.path.splitext(i)[1].lower() in _args.file_extensions]
|
||||||
|
if len(thumbitems) > 0:
|
||||||
|
if _args.reverse_sort:
|
||||||
|
thumb = f"{_args.web_root_url}.thumbnails/{baseurl}{urllib.parse.quote(item)}/{urllib.parse.quote(thumbitems[-1])}.jpg"
|
||||||
|
else:
|
||||||
|
thumb = f"{_args.web_root_url}.thumbnails/{baseurl}{urllib.parse.quote(item)}/{urllib.parse.quote(thumbitems[0])}.jpg"
|
||||||
|
|
||||||
if item not in _args.exclude_folders:
|
if item not in _args.exclude_folders:
|
||||||
if not any(fnmatch.fnmatchcase(os.path.join(folder, item), exclude) for exclude in _args.exclude_folders):
|
if not any(fnmatch.fnmatchcase(os.path.join(folder, item), exclude) for exclude in _args.exclude_folders):
|
||||||
generate_html(os.path.join(folder, item), os.path.join(folder, item).removeprefix(_args.root_directory), _args, raw, version)
|
subfolders.append({"url": subfolder_url, "name": item, "thumb": thumb, "metadata": f"{_args.web_root_url}{baseurl}{urllib.parse.quote(item)}/.metadata.json"})
|
||||||
|
return generate_html(os.path.join(folder, item), os.path.join(folder, item).removeprefix(_args.root_directory), _args, raw, version, logo)
|
||||||
|
subfolders.append({"url": subfolder_url, "name": item, "thumb": thumb, "metadata": None})
|
||||||
|
return []
|
||||||
|
|
||||||
|
|
||||||
|
def process_license(folder: str, item: str) -> None:
|
||||||
|
"""
|
||||||
|
Processes a LICENSE file, preserving formatting in HTML.
|
||||||
|
|
||||||
|
Args:
|
||||||
|
folder (str): The folder containing the LICENSE file.
|
||||||
|
item (str): The LICENSE file name.
|
||||||
|
"""
|
||||||
|
path = os.path.join(folder, item)
|
||||||
|
with open(path, encoding="utf-8") as f:
|
||||||
|
logger.info("processing LICENSE", extra={"path": path})
|
||||||
|
raw_text = f.read()
|
||||||
|
escaped_text = html.escape(raw_text)
|
||||||
|
licens[urllib.parse.quote(folder)] = f"<pre>{escaped_text}</pre>"
|
||||||
|
|
||||||
|
|
||||||
def process_info_file(folder: str, item: str) -> None:
|
def process_info_file(folder: str, item: str) -> None:
|
||||||
@@ -229,15 +541,16 @@ def process_info_file(folder: str, item: str) -> None:
|
|||||||
item (str): The info file name.
|
item (str): The info file name.
|
||||||
"""
|
"""
|
||||||
with open(os.path.join(folder, item), encoding="utf-8") as f:
|
with open(os.path.join(folder, item), encoding="utf-8") as f:
|
||||||
|
logger.info("processing info file", extra={"path": os.path.join(folder, item)})
|
||||||
info[urllib.parse.quote(folder)] = f.read()
|
info[urllib.parse.quote(folder)] = f.read()
|
||||||
|
|
||||||
|
|
||||||
def should_generate_html(images: List[Dict[str, Any]], contains_files, _args: Args) -> bool:
|
def should_generate_html(images: list[dict[str, Any]], contains_files, _args: Args) -> bool:
|
||||||
"""
|
"""
|
||||||
Determines if HTML should be generated.
|
Determines if HTML should be generated.
|
||||||
|
|
||||||
Args:
|
Args:
|
||||||
images (List[Dict[str, Any]]): List of images.
|
images (list[dict[str, Any]]): list of images.
|
||||||
_args (Args): Parsed command line arguments.
|
_args (Args): Parsed command line arguments.
|
||||||
|
|
||||||
Returns:
|
Returns:
|
||||||
@@ -246,7 +559,14 @@ def should_generate_html(images: List[Dict[str, Any]], contains_files, _args: Ar
|
|||||||
return images or (_args.use_fancy_folders and not contains_files) or (_args.use_fancy_folders and _args.ignore_other_files)
|
return images or (_args.use_fancy_folders and not contains_files) or (_args.use_fancy_folders and _args.ignore_other_files)
|
||||||
|
|
||||||
|
|
||||||
def create_html_file(folder: str, title: str, foldername: str, images: List[Dict[str, Any]], subfolders: List[Dict[str, str]], _args: Args, version: str) -> None:
|
def format_html(html: str) -> str:
|
||||||
|
soup = BeautifulSoup(html, "html5lib")
|
||||||
|
return soup.prettify()
|
||||||
|
|
||||||
|
|
||||||
|
def create_html_file(
|
||||||
|
folder: str, title: str, foldername: str, images: list[dict[str, Any]], subfolders: list[dict[str, str]], _args: Args, version: str, logo: str, subfoldertags: list[str]
|
||||||
|
) -> list[str]:
|
||||||
"""
|
"""
|
||||||
Creates the HTML file using the template.
|
Creates the HTML file using the template.
|
||||||
|
|
||||||
@@ -254,11 +574,12 @@ def create_html_file(folder: str, title: str, foldername: str, images: List[Dict
|
|||||||
folder (str): The folder to create the HTML file in.
|
folder (str): The folder to create the HTML file in.
|
||||||
title (str): The title of the HTML page.
|
title (str): The title of the HTML page.
|
||||||
foldername (str): The name of the folder.
|
foldername (str): The name of the folder.
|
||||||
images (List[Dict[str, Any]]): A list of images to include in the HTML.
|
images (list[dict[str, Any]]): A list of images to include in the HTML.
|
||||||
subfolders (List[Dict[str, str]]): A list of subfolders to include in the HTML.
|
subfolders (list[dict[str, str]]): A list of subfolders to include in the HTML.
|
||||||
_args (Args): Parsed command line arguments.
|
_args (Args): Parsed command line arguments.
|
||||||
"""
|
"""
|
||||||
image_chunks = np.array_split(images, 8) if images else []
|
html_file = os.path.join(folder, "index.html")
|
||||||
|
logger.info("generating html file with jinja2", extra={"path": html_file})
|
||||||
header = os.path.basename(folder) or title
|
header = os.path.basename(folder) or title
|
||||||
parent = None if not foldername else f"{_args.web_root_url}{urllib.parse.quote(foldername.removesuffix(folder.split('/')[-1] + '/'))}"
|
parent = None if not foldername else f"{_args.web_root_url}{urllib.parse.quote(foldername.removesuffix(folder.split('/')[-1] + '/'))}"
|
||||||
if parent and _args.web_root_url.startswith("file://"):
|
if parent and _args.web_root_url.startswith("file://"):
|
||||||
@@ -276,8 +597,43 @@ def create_html_file(folder: str, title: str, foldername: str, images: List[Dict
|
|||||||
else None
|
else None
|
||||||
)
|
)
|
||||||
|
|
||||||
|
alltags = set()
|
||||||
|
for img in images:
|
||||||
|
if img["tags"]:
|
||||||
|
alltags.update(img["tags"])
|
||||||
|
|
||||||
|
alltags.update(set(subfoldertags))
|
||||||
|
|
||||||
folder_info = info.get(urllib.parse.quote(folder), "").split("\n")
|
folder_info = info.get(urllib.parse.quote(folder), "").split("\n")
|
||||||
_info = [i for i in folder_info if len(i) > 1] if folder_info else None
|
_info = [i for i in folder_info if len(i) > 1] if folder_info else None
|
||||||
|
if _args.reverse_sort:
|
||||||
|
images.sort(key=lambda i: i["name"], reverse=True)
|
||||||
|
|
||||||
|
folder_license = licens.get(urllib.parse.quote(folder), False)
|
||||||
|
|
||||||
|
license_url = ""
|
||||||
|
|
||||||
|
if folder_license:
|
||||||
|
license_html = os.path.join(folder, "license.html")
|
||||||
|
license_url = _args.web_root_url + urllib.parse.quote(foldername) + "license.html"
|
||||||
|
with open(license_html, "w+", encoding="utf-8") as f:
|
||||||
|
logger.info("writing license html file", extra={"path": license_html})
|
||||||
|
gtml = env.get_template("license.html.j2")
|
||||||
|
content = gtml.render(
|
||||||
|
title=f"{title} - LICENSE",
|
||||||
|
favicon=f"{_args.web_root_url}{FAVICON_PATH}",
|
||||||
|
stylesheet=f"{_args.web_root_url}{GLOBAL_CSS_PATH}",
|
||||||
|
theme=f"{_args.web_root_url}.static/theme.css",
|
||||||
|
root=_args.web_root_url,
|
||||||
|
parent=f"{_args.web_root_url}{urllib.parse.quote(foldername)}",
|
||||||
|
header=f"{header} - LICENSE",
|
||||||
|
license=license_info,
|
||||||
|
webmanifest=_args.generate_webmanifest,
|
||||||
|
version=version,
|
||||||
|
logo=logo,
|
||||||
|
licensefile=folder_license,
|
||||||
|
)
|
||||||
|
f.write(format_html(content))
|
||||||
|
|
||||||
html = env.get_template("index.html.j2")
|
html = env.get_template("index.html.j2")
|
||||||
content = html.render(
|
content = html.render(
|
||||||
@@ -290,32 +646,34 @@ def create_html_file(folder: str, title: str, foldername: str, images: List[Dict
|
|||||||
header=header,
|
header=header,
|
||||||
license=license_info,
|
license=license_info,
|
||||||
subdirectories=subfolders,
|
subdirectories=subfolders,
|
||||||
images=image_chunks,
|
|
||||||
info=_info,
|
info=_info,
|
||||||
allimages=images,
|
|
||||||
webmanifest=_args.generate_webmanifest,
|
webmanifest=_args.generate_webmanifest,
|
||||||
version=version,
|
version=version,
|
||||||
|
logo=logo,
|
||||||
|
licensefile=license_url,
|
||||||
|
tags=parse_hierarchical_tags(alltags),
|
||||||
)
|
)
|
||||||
|
|
||||||
with open(os.path.join(folder, "index.html"), "w", encoding="utf-8") as f:
|
with open(html_file, "w", encoding="utf-8") as f:
|
||||||
f.write(content)
|
logger.info("writing formatted html file", extra={"path": html_file})
|
||||||
|
f.write(format_html(content))
|
||||||
|
|
||||||
|
return sorted(alltags)
|
||||||
|
|
||||||
|
|
||||||
def list_folder(total: int, folder: str, title: str, _args: Args, raw: List[str], version: str) -> List[Tuple[str, str]]:
|
def list_folder(folder: str, title: str, _args: Args, raw: list[str], version: str, logo: str) -> list[tuple[str, str, str]]:
|
||||||
"""
|
"""
|
||||||
Lists and processes a folder, generating HTML files.
|
lists and processes a folder, generating HTML files.
|
||||||
|
|
||||||
Args:
|
Args:
|
||||||
total (int): Total number of folders to process.
|
total (int): Total number of folders to process.
|
||||||
folder (str): The folder to process.
|
folder (str): The folder to process.
|
||||||
title (str): The title of the HTML page.
|
title (str): The title of the HTML page.
|
||||||
_args (Args): Parsed command line arguments.
|
_args (Args): Parsed command line arguments.
|
||||||
raw (List[str]): Raw image file names.
|
raw (list[str]): Raw image file names.
|
||||||
|
|
||||||
Returns:
|
Returns:
|
||||||
List[Tuple[str, str]]: List of thumbnails generated.
|
list[tuple[str, str]]: list of thumbnails generated.
|
||||||
"""
|
"""
|
||||||
if not _args.non_interactive_mode:
|
generate_html(folder, title, _args, raw, version, logo)
|
||||||
pbardict["htmlbar"] = tqdm(total=total, desc="Generating HTML files", unit="folders", ascii=True, dynamic_ncols=True)
|
|
||||||
generate_html(folder, title, _args, raw, version)
|
|
||||||
return thumbnails
|
return thumbnails
|
||||||
|
|||||||
140
modules/logger.py
Normal file
@@ -0,0 +1,140 @@
|
|||||||
|
"""
|
||||||
|
logger.py
|
||||||
|
|
||||||
|
This module provides functionality for setting up a centralized logging system using the
|
||||||
|
`logging` library and the `python-json-logger` to output logs in JSON format. It handles
|
||||||
|
log rotation by renaming old log files and saving them based on the first timestamp entry.
|
||||||
|
|
||||||
|
Functions:
|
||||||
|
- log_format(keys): Generates the logging format string based on the list of keys.
|
||||||
|
- rotate_log_file(): Handles renaming the existing log file to a timestamp-based name.
|
||||||
|
- setup_logger(): Configures the logging system, applies a JSON format, and returns a logger instance.
|
||||||
|
- setup_consolelogger(): Configures the logging system to output logs in console format.
|
||||||
|
"""
|
||||||
|
|
||||||
|
import os
|
||||||
|
import json
|
||||||
|
import gzip
|
||||||
|
import shutil
|
||||||
|
import logging
|
||||||
|
from datetime import datetime
|
||||||
|
from pythonjsonlogger import jsonlogger
|
||||||
|
|
||||||
|
# Constants for file paths and exclusions
|
||||||
|
if __package__ is None:
|
||||||
|
PACKAGE = ""
|
||||||
|
else:
|
||||||
|
PACKAGE = __package__
|
||||||
|
SCRIPTDIR = os.path.abspath(os.path.dirname(__file__).removesuffix(PACKAGE))
|
||||||
|
LOG_DIR = os.path.join(SCRIPTDIR, "logs")
|
||||||
|
LATEST_LOG_FILE = os.path.join(LOG_DIR, "latest.jsonl")
|
||||||
|
|
||||||
|
if not os.path.exists(LOG_DIR):
|
||||||
|
os.makedirs(LOG_DIR)
|
||||||
|
|
||||||
|
|
||||||
|
def log_format(keys):
|
||||||
|
"""
|
||||||
|
Generates a list of format strings based on the given keys.
|
||||||
|
|
||||||
|
Args:
|
||||||
|
keys (list): A list of string keys that represent the log attributes (e.g., 'asctime', 'levelname').
|
||||||
|
|
||||||
|
Returns:
|
||||||
|
list: A list of formatted strings for each key, in the format "%(key)s".
|
||||||
|
"""
|
||||||
|
return [f"%({i})s" for i in keys]
|
||||||
|
|
||||||
|
|
||||||
|
def rotate_log_file(compress=False):
|
||||||
|
"""
|
||||||
|
Truncates the 'latest.jsonl' file after optionally compressing its contents to a timestamped file.
|
||||||
|
The 'latest.jsonl' file is not deleted or moved, just emptied.
|
||||||
|
|
||||||
|
Args:
|
||||||
|
compress (bool): If True, compress the old log file using gzip.
|
||||||
|
"""
|
||||||
|
if os.path.exists(LATEST_LOG_FILE):
|
||||||
|
with open(LATEST_LOG_FILE, "r+", encoding="utf-8") as f:
|
||||||
|
first_line = f.readline()
|
||||||
|
try:
|
||||||
|
first_log = json.loads(first_line)
|
||||||
|
first_timestamp = first_log.get("asctime")
|
||||||
|
first_timestamp = first_timestamp.split(",")[0]
|
||||||
|
except (json.JSONDecodeError, KeyError):
|
||||||
|
first_timestamp = datetime.now().strftime("%Y-%m-%d_%H-%M-%S")
|
||||||
|
|
||||||
|
safe_timestamp = first_timestamp.replace(":", "-").replace(" ", "_")
|
||||||
|
old_log_filename = os.path.join(LOG_DIR, f"{safe_timestamp}.jsonl")
|
||||||
|
|
||||||
|
# Write contents to the new file
|
||||||
|
with open(old_log_filename, "w", encoding="utf-8") as old_log_file:
|
||||||
|
f.seek(0) # Go back to the beginning of the file
|
||||||
|
shutil.copyfileobj(f, old_log_file)
|
||||||
|
|
||||||
|
if compress:
|
||||||
|
with open(old_log_filename, "rb") as f_in:
|
||||||
|
with gzip.open(f"{old_log_filename}.gz", "wb") as f_out:
|
||||||
|
shutil.copyfileobj(f_in, f_out)
|
||||||
|
os.remove(old_log_filename)
|
||||||
|
|
||||||
|
# Truncate the original file
|
||||||
|
f.seek(0)
|
||||||
|
f.truncate()
|
||||||
|
|
||||||
|
|
||||||
|
def setup_logger(level=logging.INFO):
|
||||||
|
"""
|
||||||
|
Configures the logging system with a custom format and outputs logs in JSON format.
|
||||||
|
|
||||||
|
The logger will write to the 'logs/latest.jsonl' file, and it will include
|
||||||
|
multiple attributes such as the time of logging, the filename, function name, log level, etc.
|
||||||
|
|
||||||
|
Returns:
|
||||||
|
logging.Logger: A configured logger instance that can be used to log messages.
|
||||||
|
"""
|
||||||
|
_logger = logging.getLogger(name="defaultlogger")
|
||||||
|
|
||||||
|
supported_keys = ["asctime", "created", "filename", "funcName", "levelname", "levelno", "lineno", "module", "msecs", "message", "process", "processName", "relativeCreated", "thread", "threadName"]
|
||||||
|
|
||||||
|
custom_format = " ".join(log_format(supported_keys))
|
||||||
|
formatter = jsonlogger.JsonFormatter(custom_format)
|
||||||
|
|
||||||
|
log_handler = logging.FileHandler(LATEST_LOG_FILE)
|
||||||
|
log_handler.setFormatter(formatter)
|
||||||
|
|
||||||
|
_logger.addHandler(log_handler)
|
||||||
|
_logger.setLevel(level=level)
|
||||||
|
|
||||||
|
return _logger
|
||||||
|
|
||||||
|
|
||||||
|
def setup_consolelogger(level=logging.INFO):
|
||||||
|
"""
|
||||||
|
Configures the logging system to output logs in console and JSON format.
|
||||||
|
|
||||||
|
The logger will write to the 'logs/latest.jsonl' file, and it will include
|
||||||
|
multiple attributes such as the time of logging, the filename, function name, log level, etc.
|
||||||
|
|
||||||
|
Returns:
|
||||||
|
logging.Logger: A configured logger instance that can be used to log messages.
|
||||||
|
"""
|
||||||
|
_logger = logging.getLogger(name="consolelogger")
|
||||||
|
|
||||||
|
supported_keys = ["asctime", "created", "filename", "funcName", "levelname", "levelno", "lineno", "module", "msecs", "message", "process", "processName", "relativeCreated", "thread", "threadName"]
|
||||||
|
|
||||||
|
custom_format = " ".join(log_format(supported_keys))
|
||||||
|
formatter = jsonlogger.JsonFormatter(custom_format)
|
||||||
|
|
||||||
|
log_handler = logging.FileHandler(LATEST_LOG_FILE)
|
||||||
|
log_handler.setFormatter(formatter)
|
||||||
|
|
||||||
|
_logger.addHandler(log_handler)
|
||||||
|
_logger.addHandler(logging.StreamHandler())
|
||||||
|
_logger.setLevel(level=level)
|
||||||
|
return _logger
|
||||||
|
|
||||||
|
|
||||||
|
rotate_log_file(compress=True)
|
||||||
|
logger = setup_logger()
|
||||||
|
consolelogger = setup_consolelogger()
|
||||||
@@ -1,6 +1,6 @@
|
|||||||
import os
|
import os
|
||||||
import shutil
|
import shutil
|
||||||
from typing import List, Dict
|
from subprocess import Popen, PIPE
|
||||||
from PIL import Image
|
from PIL import Image
|
||||||
from jinja2 import Environment, FileSystemLoader
|
from jinja2 import Environment, FileSystemLoader
|
||||||
|
|
||||||
@@ -13,6 +13,7 @@ try:
|
|||||||
except ImportError:
|
except ImportError:
|
||||||
SVGSUPPORT = False
|
SVGSUPPORT = False
|
||||||
|
|
||||||
|
from modules.logger import logger
|
||||||
from modules.argumentparser import Args
|
from modules.argumentparser import Args
|
||||||
from modules.css_color import extract_theme_color, extract_colorscheme
|
from modules.css_color import extract_theme_color, extract_colorscheme
|
||||||
|
|
||||||
@@ -36,14 +37,14 @@ class Icon:
|
|||||||
purpose: str
|
purpose: str
|
||||||
|
|
||||||
|
|
||||||
def render_svg_icon(colorscheme: Dict[str, str], iconspath: str) -> str:
|
def render_svg_icon(colorscheme: dict[str, str], iconspath: str) -> str:
|
||||||
"""
|
"""
|
||||||
Render an SVG icon using the provided color scheme.
|
Render an SVG icon using the provided color scheme.
|
||||||
|
|
||||||
Parameters:
|
Parameters:
|
||||||
-----------
|
-----------
|
||||||
colorscheme : Dict[str, str]
|
colorscheme : dict[str, str]
|
||||||
Dictionary containing color scheme variables and their values.
|
dictionary containing color scheme variables and their values.
|
||||||
iconspath : str
|
iconspath : str
|
||||||
Path to the directory where the icon will be saved.
|
Path to the directory where the icon will be saved.
|
||||||
|
|
||||||
@@ -55,6 +56,7 @@ def render_svg_icon(colorscheme: Dict[str, str], iconspath: str) -> str:
|
|||||||
svg = env.get_template("icon.svg.j2")
|
svg = env.get_template("icon.svg.j2")
|
||||||
content = svg.render(colorscheme=colorscheme)
|
content = svg.render(colorscheme=colorscheme)
|
||||||
with open(os.path.join(iconspath, "icon.svg"), "w+", encoding="utf-8") as f:
|
with open(os.path.join(iconspath, "icon.svg"), "w+", encoding="utf-8") as f:
|
||||||
|
logger.info("writing svg icon", extra={"iconspath": iconspath})
|
||||||
f.write(content)
|
f.write(content)
|
||||||
return content
|
return content
|
||||||
|
|
||||||
@@ -73,6 +75,7 @@ def save_png_icon(content: str, iconspath: str) -> None:
|
|||||||
tmpimg = BytesIO()
|
tmpimg = BytesIO()
|
||||||
cairosvg.svg2png(bytestring=content, write_to=tmpimg)
|
cairosvg.svg2png(bytestring=content, write_to=tmpimg)
|
||||||
with Image.open(tmpimg) as iconfile:
|
with Image.open(tmpimg) as iconfile:
|
||||||
|
logger.info("saving png icon", extra={"iconspath": iconspath})
|
||||||
iconfile.save(os.path.join(iconspath, "icon.png"))
|
iconfile.save(os.path.join(iconspath, "icon.png"))
|
||||||
|
|
||||||
|
|
||||||
@@ -87,10 +90,25 @@ def generate_favicon(iconspath: str, root_directory: str) -> None:
|
|||||||
root_directory : str
|
root_directory : str
|
||||||
Root directory of the project where the favicon will be saved.
|
Root directory of the project where the favicon will be saved.
|
||||||
"""
|
"""
|
||||||
command = f'magick {os.path.join(iconspath, "icon.png")} -define icon:auto-resize=16,32,48,64,72,96,144,192 {os.path.join(root_directory, ".static", "favicon.ico")}'
|
favicon = os.path.join(root_directory, ".static", "favicon.ico")
|
||||||
|
logger.info("generating favicon with imagemagick", extra={"iconspath": iconspath, "favicon": favicon})
|
||||||
|
_env = dict(os.environ)
|
||||||
|
lp_key = "LD_LIBRARY_PATH"
|
||||||
|
lp_orig = _env.get(lp_key + "_ORIG")
|
||||||
|
if lp_orig is not None:
|
||||||
|
_env[lp_key] = lp_orig
|
||||||
|
else:
|
||||||
|
_env.pop(lp_key, None)
|
||||||
if not shutil.which("magick"):
|
if not shutil.which("magick"):
|
||||||
command = f'convert {os.path.join(iconspath, "icon.png")} -define icon:auto-resize=16,32,48,64,72,96,144,192 {os.path.join(root_directory, ".static", "favicon.ico")}'
|
magick = shutil.which("convert")
|
||||||
os.system(command)
|
else:
|
||||||
|
magick = shutil.which("magick")
|
||||||
|
command = [magick, os.path.join(iconspath, "icon.png"), "-define", "icon:auto-resize=16,32,48,64,72,96,144,192", favicon]
|
||||||
|
with Popen(command, stdin=PIPE, stdout=PIPE, stderr=PIPE, env=_env, errors="ignore") as p:
|
||||||
|
out, err = p.communicate()
|
||||||
|
if p.returncode != 0:
|
||||||
|
logger.error("error generating favicon: %s", err, extra={"command": command, "out": out, "err": err})
|
||||||
|
logger.info("favicon generated successfully", extra={"command": command, "out": out, "err": err})
|
||||||
|
|
||||||
|
|
||||||
def icons(_args: Args) -> None:
|
def icons(_args: Args) -> None:
|
||||||
@@ -102,18 +120,20 @@ def icons(_args: Args) -> None:
|
|||||||
_args : Args
|
_args : Args
|
||||||
Parsed command-line arguments.
|
Parsed command-line arguments.
|
||||||
"""
|
"""
|
||||||
print("Generating icons...")
|
|
||||||
iconspath = os.path.join(_args.root_directory, ".static", "icons")
|
iconspath = os.path.join(_args.root_directory, ".static", "icons")
|
||||||
|
logger.info("generating icons", extra={"iconspath": iconspath})
|
||||||
|
print("Generating icons...")
|
||||||
colorscheme = extract_colorscheme(_args.theme_path)
|
colorscheme = extract_colorscheme(_args.theme_path)
|
||||||
content = render_svg_icon(colorscheme, iconspath)
|
content = render_svg_icon(colorscheme, iconspath)
|
||||||
if not SVGSUPPORT:
|
if not SVGSUPPORT:
|
||||||
print("Please install cairosvg to generate favicon from svg icon.")
|
print("Please install cairosvg to generate favicon from svg icon.")
|
||||||
|
logger.error("svg support not available")
|
||||||
return
|
return
|
||||||
save_png_icon(content, iconspath)
|
save_png_icon(content, iconspath)
|
||||||
generate_favicon(iconspath, _args.root_directory)
|
generate_favicon(iconspath, _args.root_directory)
|
||||||
|
|
||||||
|
|
||||||
def render_manifest_json(_args: Args, icon_list: List[Icon], colors: Dict[str, str]) -> None:
|
def render_manifest_json(_args: Args, icon_list: list[Icon], colors: dict[str, str]) -> None:
|
||||||
"""
|
"""
|
||||||
Render the manifest.json file for the web application.
|
Render the manifest.json file for the web application.
|
||||||
|
|
||||||
@@ -121,12 +141,12 @@ def render_manifest_json(_args: Args, icon_list: List[Icon], colors: Dict[str, s
|
|||||||
-----------
|
-----------
|
||||||
_args : Args
|
_args : Args
|
||||||
Parsed command-line arguments.
|
Parsed command-line arguments.
|
||||||
icon_list : List[Icon]
|
icon_list : list[Icon]
|
||||||
List of icons to be included in the manifest.
|
list of icons to be included in the manifest.
|
||||||
colors : Dict[str, str]
|
colors : dict[str, str]
|
||||||
Dictionary containing color scheme and theme color.
|
dictionary containing color scheme and theme color.
|
||||||
"""
|
"""
|
||||||
manifest = env.get_template("manifest.json.j2")
|
manifest = env.get_template("manifest.webmanifest.j2")
|
||||||
content = manifest.render(
|
content = manifest.render(
|
||||||
name=_args.web_root_url.replace("https://", "").replace("http://", "").replace("/", ""),
|
name=_args.web_root_url.replace("https://", "").replace("http://", "").replace("/", ""),
|
||||||
short_name=_args.site_title,
|
short_name=_args.site_title,
|
||||||
@@ -134,18 +154,19 @@ def render_manifest_json(_args: Args, icon_list: List[Icon], colors: Dict[str, s
|
|||||||
background_color=colors["bcolor1"],
|
background_color=colors["bcolor1"],
|
||||||
theme_color=colors["theme_color"],
|
theme_color=colors["theme_color"],
|
||||||
)
|
)
|
||||||
with open(os.path.join(_args.root_directory, ".static", "manifest.json"), "w", encoding="utf-8") as f:
|
with open(os.path.join(_args.root_directory, ".static", "manifest.webmanifest"), "w", encoding="utf-8") as f:
|
||||||
|
logger.info("rendering manifest.webmanifest", extra={"path": os.path.join(_args.root_directory, ".static", "manifest.webmanifest")})
|
||||||
f.write(content)
|
f.write(content)
|
||||||
|
|
||||||
|
|
||||||
def create_icons_from_svg(files: List[str], iconspath: str, _args: Args) -> List[Icon]:
|
def create_icons_from_svg(files: list[str], iconspath: str, _args: Args) -> list[Icon]:
|
||||||
"""
|
"""
|
||||||
Create icons from an SVG file.
|
Create icons from an SVG file.
|
||||||
|
|
||||||
Parameters:
|
Parameters:
|
||||||
-----------
|
-----------
|
||||||
files : List[str]
|
files : list[str]
|
||||||
List of files in the icons directory.
|
list of files in the icons directory.
|
||||||
iconspath : str
|
iconspath : str
|
||||||
Path to the directory where the icons will be saved.
|
Path to the directory where the icons will be saved.
|
||||||
_args : Args
|
_args : Args
|
||||||
@@ -153,10 +174,11 @@ def create_icons_from_svg(files: List[str], iconspath: str, _args: Args) -> List
|
|||||||
|
|
||||||
Returns:
|
Returns:
|
||||||
--------
|
--------
|
||||||
List[Icon]
|
list[Icon]
|
||||||
List of icons created from the SVG file.
|
list of icons created from the SVG file.
|
||||||
"""
|
"""
|
||||||
svg = [file for file in files if file.endswith(".svg")][0]
|
svg = [file for file in files if file.endswith(".svg")][0]
|
||||||
|
logger.info("creating icons for web application", extra={"iconspath": iconspath, "svg": svg})
|
||||||
icon_list = [
|
icon_list = [
|
||||||
{"src": f"{_args.web_root_url}.static/icons/{svg}", "type": "image/svg+xml", "sizes": "512x512", "purpose": "maskable"},
|
{"src": f"{_args.web_root_url}.static/icons/{svg}", "type": "image/svg+xml", "sizes": "512x512", "purpose": "maskable"},
|
||||||
{"src": f"{_args.web_root_url}.static/icons/{svg}", "type": "image/svg+xml", "sizes": "512x512", "purpose": "any"},
|
{"src": f"{_args.web_root_url}.static/icons/{svg}", "type": "image/svg+xml", "sizes": "512x512", "purpose": "any"},
|
||||||
@@ -165,6 +187,7 @@ def create_icons_from_svg(files: List[str], iconspath: str, _args: Args) -> List
|
|||||||
tmpimg = BytesIO()
|
tmpimg = BytesIO()
|
||||||
sizes = size.split("x")
|
sizes = size.split("x")
|
||||||
iconpath = os.path.join(iconspath, os.path.splitext(svg)[0] + "-" + size + ".png")
|
iconpath = os.path.join(iconspath, os.path.splitext(svg)[0] + "-" + size + ".png")
|
||||||
|
logger.info("converting svg to png", extra={"svg": svg, "size": size})
|
||||||
cairosvg.svg2png(
|
cairosvg.svg2png(
|
||||||
url=os.path.join(iconspath, svg),
|
url=os.path.join(iconspath, svg),
|
||||||
write_to=tmpimg,
|
write_to=tmpimg,
|
||||||
@@ -173,6 +196,7 @@ def create_icons_from_svg(files: List[str], iconspath: str, _args: Args) -> List
|
|||||||
scale=1,
|
scale=1,
|
||||||
)
|
)
|
||||||
with Image.open(tmpimg) as iconfile:
|
with Image.open(tmpimg) as iconfile:
|
||||||
|
logger.info("saving png file", extra={"iconpath": iconpath})
|
||||||
iconfile.save(iconpath, format="PNG")
|
iconfile.save(iconpath, format="PNG")
|
||||||
icon_list.append(
|
icon_list.append(
|
||||||
{
|
{
|
||||||
@@ -193,7 +217,7 @@ def create_icons_from_svg(files: List[str], iconspath: str, _args: Args) -> List
|
|||||||
return icon_list
|
return icon_list
|
||||||
|
|
||||||
|
|
||||||
def create_icons_from_png(iconspath: str, web_root_url: str) -> List[Icon]:
|
def create_icons_from_png(iconspath: str, web_root_url: str) -> list[Icon]:
|
||||||
"""
|
"""
|
||||||
Create icons from PNG files.
|
Create icons from PNG files.
|
||||||
|
|
||||||
@@ -206,8 +230,8 @@ def create_icons_from_png(iconspath: str, web_root_url: str) -> List[Icon]:
|
|||||||
|
|
||||||
Returns:
|
Returns:
|
||||||
--------
|
--------
|
||||||
List[Icon]
|
list[Icon]
|
||||||
List of icons created from PNG files.
|
list of icons created from PNG files.
|
||||||
"""
|
"""
|
||||||
icon_list = []
|
icon_list = []
|
||||||
for icon in os.listdir(iconspath):
|
for icon in os.listdir(iconspath):
|
||||||
@@ -215,6 +239,7 @@ def create_icons_from_png(iconspath: str, web_root_url: str) -> List[Icon]:
|
|||||||
continue
|
continue
|
||||||
with Image.open(os.path.join(iconspath, icon)) as iconfile:
|
with Image.open(os.path.join(iconspath, icon)) as iconfile:
|
||||||
iconsize = f"{iconfile.size[0]}x{iconfile.size[1]}"
|
iconsize = f"{iconfile.size[0]}x{iconfile.size[1]}"
|
||||||
|
logger.info("using icon", extra={"iconspath": iconspath, "icon": icon, "size": iconsize})
|
||||||
icon_list.append({"src": f"{web_root_url}.static/icons/{icon}", "sizes": iconsize, "type": "image/png", "purpose": "maskable"})
|
icon_list.append({"src": f"{web_root_url}.static/icons/{icon}", "sizes": iconsize, "type": "image/png", "purpose": "maskable"})
|
||||||
icon_list.append({"src": f"{web_root_url}.static/icons/{icon}", "sizes": iconsize, "type": "image/png", "purpose": "any"})
|
icon_list.append({"src": f"{web_root_url}.static/icons/{icon}", "sizes": iconsize, "type": "image/png", "purpose": "any"})
|
||||||
return icon_list
|
return icon_list
|
||||||
@@ -229,16 +254,15 @@ def webmanifest(_args: Args) -> None:
|
|||||||
_args : Args
|
_args : Args
|
||||||
Parsed command-line arguments.
|
Parsed command-line arguments.
|
||||||
"""
|
"""
|
||||||
|
logger.info("generating webmanifest")
|
||||||
|
|
||||||
iconspath = os.path.join(_args.root_directory, ".static", "icons")
|
iconspath = os.path.join(_args.root_directory, ".static", "icons")
|
||||||
files = os.listdir(iconspath)
|
files = os.listdir(iconspath)
|
||||||
icon_list = (
|
icon_list = create_icons_from_svg(files, iconspath, _args) if SVGSUPPORT and any(file.endswith(".svg") for file in files) else create_icons_from_png(iconspath, _args.web_root_url)
|
||||||
create_icons_from_svg(files, iconspath, _args)
|
|
||||||
if SVGSUPPORT and any(file.endswith(".svg") for file in files)
|
|
||||||
else create_icons_from_png(iconspath, _args.web_root_url)
|
|
||||||
)
|
|
||||||
|
|
||||||
if not icon_list:
|
if not icon_list:
|
||||||
print("No icons found in the static/icons folder!")
|
print("No icons found in the static/icons folder!")
|
||||||
|
logger.error("no icons found in the static/icons folder", extra={"iconspath": iconspath})
|
||||||
return
|
return
|
||||||
|
|
||||||
colorscheme = extract_colorscheme(os.path.join(_args.root_directory, ".static", "theme.css"))
|
colorscheme = extract_colorscheme(os.path.join(_args.root_directory, ".static", "theme.css"))
|
||||||
|
|||||||
@@ -1,8 +1,12 @@
|
|||||||
CairoSVG==2.7.1
|
beautifulsoup4~=4.13.4
|
||||||
Jinja2==3.1.4
|
CairoSVG~=2.7.1
|
||||||
numpy==2.0.0
|
defusedxml~=0.7.1
|
||||||
pillow==10.4.0
|
html5lib~=1.1
|
||||||
pyinstaller==6.9.0
|
Jinja2~=3.1.6
|
||||||
rich-argparse==1.5.2
|
jsmin~=3.0.1
|
||||||
setuptools==70.3.0
|
Pillow~=11.3.0
|
||||||
tqdm==4.66.4
|
pyinstaller~=6.11.1
|
||||||
|
python_json_logger~=2.0.7
|
||||||
|
rich_argparse~=1.7.1
|
||||||
|
selenium~=4.34.2
|
||||||
|
tqdm~=4.66.4
|
||||||
|
|||||||
@@ -52,6 +52,14 @@
|
|||||||
background-color: var(--color2);
|
background-color: var(--color2);
|
||||||
}
|
}
|
||||||
|
|
||||||
|
.tagentry label:hover {
|
||||||
|
background-color: var(--color7);
|
||||||
|
}
|
||||||
|
|
||||||
|
.tagentry .tagtoggle:hover {
|
||||||
|
background-color: var(--color4);
|
||||||
|
}
|
||||||
|
|
||||||
.column img {
|
.column img {
|
||||||
background-color: var(--color2);
|
background-color: var(--color2);
|
||||||
}
|
}
|
||||||
|
|||||||
419
templates/functionality.js
Normal file
@@ -0,0 +1,419 @@
|
|||||||
|
class PhotoGallery {
|
||||||
|
constructor() {
|
||||||
|
this.pswpElement = document.querySelector(".pswp");
|
||||||
|
this.items = [];
|
||||||
|
this.shown = [];
|
||||||
|
this.subfolders = [];
|
||||||
|
this.controllers = {};
|
||||||
|
this.tagDropdownShown = false;
|
||||||
|
|
||||||
|
this.debounce = this.debounce.bind(this);
|
||||||
|
this.openSwipe = this.openSwipe.bind(this);
|
||||||
|
this.prefetch = this.prefetch.bind(this);
|
||||||
|
this.cancel = this.cancel.bind(this);
|
||||||
|
this.reset = this.reset.bind(this);
|
||||||
|
this.recursive = this.recursive.bind(this);
|
||||||
|
this.requestMetadata = this.requestMetadata.bind(this);
|
||||||
|
this.filter = this.filter.bind(this);
|
||||||
|
this.updateImageList = this.updateImageList.bind(this);
|
||||||
|
this.setFilter = this.setFilter.bind(this);
|
||||||
|
this.toggleTag = this.toggleTag.bind(this);
|
||||||
|
this.setupDropdownToggle = this.setupDropdownToggle.bind(this);
|
||||||
|
this.setupTagHandlers = this.setupTagHandlers.bind(this);
|
||||||
|
this.setupClickHandlers = this.setupClickHandlers.bind(this);
|
||||||
|
this.scrollFunction = this.scrollFunction.bind(this);
|
||||||
|
this.topFunction = this.topFunction.bind(this);
|
||||||
|
this.onLoad = this.onLoad.bind(this);
|
||||||
|
|
||||||
|
this.init();
|
||||||
|
}
|
||||||
|
|
||||||
|
debounce(fn, delay) {
|
||||||
|
let timeoutId;
|
||||||
|
return (...args) => {
|
||||||
|
clearTimeout(timeoutId);
|
||||||
|
timeoutId = setTimeout(() => fn.apply(this, args), delay);
|
||||||
|
};
|
||||||
|
}
|
||||||
|
|
||||||
|
openSwipe(imgIndex) {
|
||||||
|
const options = { index: imgIndex };
|
||||||
|
const gallery = new PhotoSwipe(
|
||||||
|
this.pswpElement,
|
||||||
|
PhotoSwipeUI_Default,
|
||||||
|
this.shown,
|
||||||
|
options
|
||||||
|
);
|
||||||
|
gallery.init();
|
||||||
|
}
|
||||||
|
|
||||||
|
prefetch(imgIndex) {
|
||||||
|
if (this.controllers[imgIndex]) {
|
||||||
|
this.cancel(imgIndex);
|
||||||
|
}
|
||||||
|
const controller = new AbortController();
|
||||||
|
const signal = controller.signal;
|
||||||
|
this.controllers[imgIndex] = controller;
|
||||||
|
const urlToFetch = this.shown[imgIndex]?.src;
|
||||||
|
if (urlToFetch) {
|
||||||
|
fetch(urlToFetch, { method: "GET", signal }).catch(() => {});
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
cancel(imgIndex) {
|
||||||
|
if (this.controllers[imgIndex]) {
|
||||||
|
this.controllers[imgIndex].abort();
|
||||||
|
delete this.controllers[imgIndex];
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
reset() {
|
||||||
|
const content = document.documentElement.innerHTML;
|
||||||
|
const title = document.title;
|
||||||
|
const folders = document.querySelector(".folders");
|
||||||
|
let path = window.location.origin + window.location.pathname;
|
||||||
|
if (path.startsWith("null")) {
|
||||||
|
path = window.location.protocol + "//" + path.substring(4);
|
||||||
|
}
|
||||||
|
|
||||||
|
if (folders) folders.style.display = "";
|
||||||
|
document.getElementById("recursive").checked = false;
|
||||||
|
document
|
||||||
|
.querySelectorAll("#tagdropdown input.tagcheckbox:checked")
|
||||||
|
.forEach((checkbox) => (checkbox.checked = false));
|
||||||
|
window.history.replaceState({ html: content, pageTitle: title }, "", path);
|
||||||
|
this.requestMetadata();
|
||||||
|
}
|
||||||
|
|
||||||
|
showLoader() {
|
||||||
|
const imagelist = document.getElementById("imagelist");
|
||||||
|
imagelist.innerHTML = '<span class="loader"></span>';
|
||||||
|
imagelist.classList.add("centerload");
|
||||||
|
imagelist.classList.remove("row");
|
||||||
|
}
|
||||||
|
|
||||||
|
async recursive() {
|
||||||
|
this.showLoader();
|
||||||
|
const loc = new URL(window.location.href);
|
||||||
|
const content = document.documentElement.innerHTML;
|
||||||
|
const title = document.title;
|
||||||
|
const isChecked = document.getElementById("recursive")?.checked;
|
||||||
|
const folders = document.querySelector(".folders");
|
||||||
|
|
||||||
|
if (!isChecked) {
|
||||||
|
if (folders) folders.style.display = "";
|
||||||
|
loc.searchParams.delete("recursive");
|
||||||
|
window.history.replaceState({ html: content, pageTitle: title }, "", loc);
|
||||||
|
this.requestMetadata();
|
||||||
|
return;
|
||||||
|
}
|
||||||
|
|
||||||
|
if (folders) folders.style.display = "none";
|
||||||
|
loc.searchParams.delete("recursive");
|
||||||
|
loc.searchParams.append("recursive", true);
|
||||||
|
window.history.replaceState({ html: content, pageTitle: title }, "", loc);
|
||||||
|
|
||||||
|
const visited = new Set();
|
||||||
|
const existingItems = new Set();
|
||||||
|
const newItems = [];
|
||||||
|
|
||||||
|
try {
|
||||||
|
const response = await fetch(".metadata.json");
|
||||||
|
if (!response.ok) throw new Error("Failed to fetch metadata");
|
||||||
|
const data = await response.json();
|
||||||
|
|
||||||
|
this.items = [];
|
||||||
|
this.subfolders = data.subfolders || [];
|
||||||
|
|
||||||
|
for (const image of Object.values(data.images || {})) {
|
||||||
|
newItems.push(image);
|
||||||
|
existingItems.add(image.src);
|
||||||
|
}
|
||||||
|
} catch {
|
||||||
|
return;
|
||||||
|
}
|
||||||
|
|
||||||
|
const fetchFoldersRecursively = async (folderList) => {
|
||||||
|
if (!Array.isArray(folderList)) return;
|
||||||
|
const nextLevel = [];
|
||||||
|
await Promise.all(
|
||||||
|
folderList.map(async (folder) => {
|
||||||
|
if (!folder || !folder.metadata || visited.has(folder.url)) return;
|
||||||
|
visited.add(folder.url);
|
||||||
|
try {
|
||||||
|
const response = await fetch(folder.metadata);
|
||||||
|
if (!response.ok) throw new Error();
|
||||||
|
const data = await response.json();
|
||||||
|
for (const image of Object.values(data.images || {})) {
|
||||||
|
if (!existingItems.has(image.src)) {
|
||||||
|
newItems.push(image);
|
||||||
|
existingItems.add(image.src);
|
||||||
|
}
|
||||||
|
}
|
||||||
|
if (Array.isArray(data.subfolders))
|
||||||
|
nextLevel.push(...data.subfolders);
|
||||||
|
} catch {}
|
||||||
|
})
|
||||||
|
);
|
||||||
|
if (nextLevel.length > 0) await fetchFoldersRecursively(nextLevel);
|
||||||
|
};
|
||||||
|
|
||||||
|
await fetchFoldersRecursively(this.subfolders);
|
||||||
|
this.items = [...newItems];
|
||||||
|
this.filter();
|
||||||
|
}
|
||||||
|
|
||||||
|
requestMetadata() {
|
||||||
|
this.showLoader();
|
||||||
|
const hash = window.location.hash;
|
||||||
|
const searchParams = new URLSearchParams(window.location.search);
|
||||||
|
fetch(".metadata.json")
|
||||||
|
.then((response) => {
|
||||||
|
if (!response.ok) throw new Error("Failed to fetch metadata");
|
||||||
|
return response.json();
|
||||||
|
})
|
||||||
|
.then((data) => {
|
||||||
|
this.items = Object.values(data.images || {});
|
||||||
|
this.subfolders = data.subfolders || [];
|
||||||
|
|
||||||
|
if (hash != "") {
|
||||||
|
const selected = hash.replace("#", "").split(",");
|
||||||
|
this.setFilter(selected);
|
||||||
|
}
|
||||||
|
if (searchParams.get("recursive") != null) {
|
||||||
|
const recChk = document.getElementById("recursive");
|
||||||
|
if (recChk) recChk.checked = true;
|
||||||
|
this.recursive();
|
||||||
|
} else {
|
||||||
|
this.filter();
|
||||||
|
}
|
||||||
|
})
|
||||||
|
.catch(() => {});
|
||||||
|
}
|
||||||
|
|
||||||
|
filter() {
|
||||||
|
const searchParams = new URLSearchParams(window.location.search);
|
||||||
|
this.shown = [];
|
||||||
|
let path = decodeURIComponent(
|
||||||
|
window.location.origin +
|
||||||
|
window.location.pathname.replace("index.html", "")
|
||||||
|
);
|
||||||
|
if (path.startsWith("null")) {
|
||||||
|
path = window.location.protocol + "//" + path.substring(4);
|
||||||
|
}
|
||||||
|
const selectedTags = [];
|
||||||
|
|
||||||
|
document
|
||||||
|
.querySelectorAll("#tagdropdown input.tagcheckbox:checked")
|
||||||
|
.forEach((checkbox) => {
|
||||||
|
let tag = checkbox.parentElement.id.trim().substring(1);
|
||||||
|
if (checkbox.parentElement.parentElement.children.length > 1)
|
||||||
|
tag += "|";
|
||||||
|
selectedTags.push(tag);
|
||||||
|
});
|
||||||
|
|
||||||
|
const urltags = selectedTags.join(",");
|
||||||
|
|
||||||
|
let isRecursiveChecked = false;
|
||||||
|
try {
|
||||||
|
isRecursiveChecked =
|
||||||
|
document.getElementById("recursive")?.checked || false;
|
||||||
|
} catch {}
|
||||||
|
|
||||||
|
for (const item of this.items) {
|
||||||
|
const tags = item.tags || [];
|
||||||
|
const include = selectedTags.every((selected) => {
|
||||||
|
const isParent = selected.endsWith("|");
|
||||||
|
return isParent
|
||||||
|
? tags.some((t) => t.startsWith(selected))
|
||||||
|
: tags.includes(selected);
|
||||||
|
});
|
||||||
|
|
||||||
|
if (include || selectedTags.length === 0) {
|
||||||
|
if (!isRecursiveChecked) {
|
||||||
|
if (decodeURIComponent(item.src).replace(item.name, "") === path) {
|
||||||
|
this.shown.push(item);
|
||||||
|
}
|
||||||
|
} else {
|
||||||
|
this.shown.push(item);
|
||||||
|
}
|
||||||
|
}
|
||||||
|
}
|
||||||
|
this.updateImageList();
|
||||||
|
window.location.hash = urltags;
|
||||||
|
|
||||||
|
const pid = searchParams.get("pid") - 1;
|
||||||
|
if (pid != -1) {
|
||||||
|
this.openSwipe(pid);
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
updateImageList() {
|
||||||
|
const imagelist = document.getElementById("imagelist");
|
||||||
|
if (!imagelist) return;
|
||||||
|
let str = "";
|
||||||
|
this.shown.forEach((item, index) => {
|
||||||
|
str += `<div class="column"><figure><img src="${item.msrc}" data-index="${index}" /><figcaption class="caption">${item.name}`;
|
||||||
|
if (item.tiff) str += ` <a href="${item.tiff}">TIFF</a>`;
|
||||||
|
if (item.raw) str += ` <a href="${item.raw}">RAW</a>`;
|
||||||
|
str += "</figcaption></figure></div>";
|
||||||
|
});
|
||||||
|
imagelist.classList.add("row");
|
||||||
|
imagelist.classList.remove("centerload");
|
||||||
|
imagelist.innerHTML = str;
|
||||||
|
}
|
||||||
|
|
||||||
|
setFilter(selected) {
|
||||||
|
document
|
||||||
|
.querySelectorAll("#tagdropdown input.tagcheckbox")
|
||||||
|
.forEach((checkbox) => {
|
||||||
|
selected.forEach((tag) => {
|
||||||
|
if (
|
||||||
|
checkbox.parentElement.id
|
||||||
|
.trim()
|
||||||
|
.substring(1)
|
||||||
|
.replace(" ", "%20") === tag
|
||||||
|
) {
|
||||||
|
checkbox.checked = true;
|
||||||
|
}
|
||||||
|
});
|
||||||
|
});
|
||||||
|
}
|
||||||
|
|
||||||
|
toggleTag(tagid) {
|
||||||
|
const tag = document.getElementById(tagid);
|
||||||
|
const ol = tag?.closest(".tagentry")?.querySelector(".tagentryparent");
|
||||||
|
const svg = tag?.parentElement.querySelector(".tagtoggle svg");
|
||||||
|
if (!ol || !svg) return;
|
||||||
|
ol.classList.toggle("show");
|
||||||
|
svg.style.transform = ol.classList.contains("show")
|
||||||
|
? "rotate(180deg)"
|
||||||
|
: "rotate(0deg)";
|
||||||
|
}
|
||||||
|
|
||||||
|
setupDropdownToggle() {
|
||||||
|
const toggleLink = document.getElementById("tagtogglelink");
|
||||||
|
const dropdown = document.getElementById("tagdropdown");
|
||||||
|
if (!toggleLink) return;
|
||||||
|
|
||||||
|
toggleLink.addEventListener("click", (event) => {
|
||||||
|
event.stopPropagation();
|
||||||
|
const svg = toggleLink.querySelector("svg");
|
||||||
|
dropdown.classList.toggle("show");
|
||||||
|
if (svg)
|
||||||
|
svg.style.transform = dropdown.classList.contains("show")
|
||||||
|
? "rotate(180deg)"
|
||||||
|
: "rotate(0deg)";
|
||||||
|
this.tagDropdownShown = dropdown.classList.contains("show");
|
||||||
|
});
|
||||||
|
|
||||||
|
document.addEventListener("click", (event) => {
|
||||||
|
if (
|
||||||
|
!dropdown.contains(event.target) &&
|
||||||
|
!toggleLink.contains(event.target)
|
||||||
|
) {
|
||||||
|
dropdown.classList.remove("show");
|
||||||
|
this.tagDropdownShown = false;
|
||||||
|
const svg = toggleLink.querySelector("svg");
|
||||||
|
if (svg) svg.style.transform = "rotate(0deg)";
|
||||||
|
}
|
||||||
|
});
|
||||||
|
}
|
||||||
|
|
||||||
|
setupTagHandlers() {
|
||||||
|
const tagContainer = document.getElementById("tagdropdown");
|
||||||
|
if (!tagContainer) return;
|
||||||
|
|
||||||
|
const debouncedFilter = this.debounce(this.filter, 150);
|
||||||
|
tagContainer.addEventListener("change", debouncedFilter);
|
||||||
|
|
||||||
|
tagContainer.addEventListener("click", (event) => {
|
||||||
|
const toggle = event.target.closest(".tagtoggle");
|
||||||
|
if (toggle) {
|
||||||
|
event.stopPropagation();
|
||||||
|
const tagid = toggle.dataset.toggleid;
|
||||||
|
this.toggleTag(tagid);
|
||||||
|
}
|
||||||
|
});
|
||||||
|
}
|
||||||
|
|
||||||
|
setupClickHandlers() {
|
||||||
|
const resetEl = document
|
||||||
|
.getElementById("reset-filter")
|
||||||
|
?.querySelector("label");
|
||||||
|
if (resetEl) resetEl.addEventListener("click", this.reset);
|
||||||
|
|
||||||
|
const recurseEl = document.getElementById("recursive");
|
||||||
|
if (recurseEl)
|
||||||
|
recurseEl.addEventListener("change", this.debounce(this.recursive, 150));
|
||||||
|
|
||||||
|
const totop = document.getElementById("totop");
|
||||||
|
if (totop) totop.addEventListener("click", this.topFunction);
|
||||||
|
|
||||||
|
const imagelist = document.getElementById("imagelist");
|
||||||
|
if (imagelist) {
|
||||||
|
imagelist.addEventListener("click", (event) => {
|
||||||
|
const img = event.target.closest("img");
|
||||||
|
if (!img || !img.dataset.index) return;
|
||||||
|
const index = parseInt(img.dataset.index);
|
||||||
|
if (!isNaN(index)) this.openSwipe(index);
|
||||||
|
});
|
||||||
|
|
||||||
|
imagelist.addEventListener("mouseover", (event) => {
|
||||||
|
const img = event.target.closest("img");
|
||||||
|
if (!img || !img.dataset.index) return;
|
||||||
|
const index = parseInt(img.dataset.index);
|
||||||
|
if (!isNaN(index)) this.prefetch(index);
|
||||||
|
});
|
||||||
|
|
||||||
|
imagelist.addEventListener("mouseleave", (event) => {
|
||||||
|
const img = event.target.closest("img");
|
||||||
|
if (!img || !img.dataset.index) return;
|
||||||
|
const index = parseInt(img.dataset.index);
|
||||||
|
if (!isNaN(index)) this.cancel(index);
|
||||||
|
});
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
scrollFunction() {
|
||||||
|
const totopbutton = document.getElementById("totop");
|
||||||
|
if (!totopbutton) return;
|
||||||
|
if (
|
||||||
|
document.body.scrollTop > 20 ||
|
||||||
|
document.documentElement.scrollTop > 20
|
||||||
|
) {
|
||||||
|
totopbutton.style.display = "block";
|
||||||
|
} else {
|
||||||
|
totopbutton.style.display = "none";
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
topFunction() {
|
||||||
|
window.scrollTo({ top: 0, behavior: "smooth" });
|
||||||
|
}
|
||||||
|
|
||||||
|
onLoad() {
|
||||||
|
document.querySelectorAll(".tagtoggle").forEach((toggle) => {
|
||||||
|
toggle.addEventListener("mouseup", (event) => {
|
||||||
|
event.stopPropagation();
|
||||||
|
const tagid = toggle.getAttribute("data-tagid");
|
||||||
|
this.toggleTag(tagid);
|
||||||
|
});
|
||||||
|
});
|
||||||
|
|
||||||
|
this.requestMetadata();
|
||||||
|
this.setupDropdownToggle();
|
||||||
|
this.setupTagHandlers();
|
||||||
|
this.setupClickHandlers();
|
||||||
|
|
||||||
|
window.addEventListener("scroll", this.scrollFunction);
|
||||||
|
}
|
||||||
|
|
||||||
|
init() {
|
||||||
|
if (window.addEventListener) {
|
||||||
|
window.addEventListener("load", this.onLoad, false);
|
||||||
|
} else if (window.attachEvent) {
|
||||||
|
window.attachEvent("onload", this.onLoad);
|
||||||
|
}
|
||||||
|
}
|
||||||
|
}
|
||||||
@@ -1,3 +1,25 @@
|
|||||||
|
{%- macro render_tags(tag_tree, parent) -%}
|
||||||
|
{%- for key, value in tag_tree.items() %}
|
||||||
|
<li class="tagentry">
|
||||||
|
<div class="tagflex">
|
||||||
|
<label class="tag" title="{{ key }}" id="{{ parent }}|{{ key }}">
|
||||||
|
<input type="checkbox" class="tagcheckbox" />{{ key }}
|
||||||
|
</label>{% if value %} <span class="tagtoggle" data-tagid="{{ parent }}|{{ key }}">
|
||||||
|
<svg width="1em" height="1em" viewBox="0 0 129.87601 129.87624">
|
||||||
|
<g id="layer1" transform="translate(-33.816833,-52.685642)">
|
||||||
|
<path stroke="currentColor" style="fill:none;stroke-width:20;stroke-linecap:round;stroke-linejoin:round;stroke-dasharray:none;stroke-opacity:1"
|
||||||
|
d="M 54.476483,95.484647 98.754836,139.76308 143.03319,95.484647" id="path1" />
|
||||||
|
</g>
|
||||||
|
</svg></span>{% endif %}
|
||||||
|
</div>
|
||||||
|
{%- if value %}
|
||||||
|
<ol class="tagentryparent">
|
||||||
|
{{ render_tags(value, parent + '|' + key) }}
|
||||||
|
</ol>
|
||||||
|
{%- endif %}
|
||||||
|
</li><br>
|
||||||
|
{%- endfor %}
|
||||||
|
{%- endmacro -%}
|
||||||
<!DOCTYPE html>
|
<!DOCTYPE html>
|
||||||
<html lang="en">
|
<html lang="en">
|
||||||
|
|
||||||
@@ -6,46 +28,82 @@
|
|||||||
<meta name="viewport" content="width=device-width, initial-scale=1.0">
|
<meta name="viewport" content="width=device-width, initial-scale=1.0">
|
||||||
<title>{{ title }}</title>
|
<title>{{ title }}</title>
|
||||||
{%- if webmanifest %}
|
{%- if webmanifest %}
|
||||||
<link rel="manifest" href="/.static/manifest.json">
|
<link rel="manifest" href="/.static/manifest.webmanifest">
|
||||||
|
{%- endif %}
|
||||||
|
<link rel="preload" href="{{ stylesheet }}" as="style">
|
||||||
|
{%- if theme %}
|
||||||
|
<link rel="preload" href="{{ theme }}" as="style">
|
||||||
{%- endif %}
|
{%- endif %}
|
||||||
<link rel="icon" type="image/x-icon" href="{{ favicon }}">
|
<link rel="icon" type="image/x-icon" href="{{ favicon }}">
|
||||||
<link rel="stylesheet" href="{{ stylesheet }}">
|
<link rel="stylesheet" href="{{ stylesheet }}">
|
||||||
{%- if theme %}
|
{%- if theme %}
|
||||||
<link rel="stylesheet" href="{{ theme }}">
|
<link rel="stylesheet" href="{{ theme }}">
|
||||||
{%- endif %}
|
{%- endif %}
|
||||||
{%- if images %}
|
<link rel="preload" href="{{ root }}.static/pswp/photoswipe.css" as="style">
|
||||||
|
<link rel="preload" href="{{ root }}.static/pswp/default-skin/default-skin.css" as="style">
|
||||||
|
<link rel="modulepreload" href="{{ root }}.static/pswp/photoswipe.min.js">
|
||||||
|
<link rel="modulepreload" href="{{ root }}.static/pswp/photoswipe-ui-default.min.js">
|
||||||
|
<link rel="modulepreload" href="{{ root }}.static/functionality.min.js">
|
||||||
<link rel="stylesheet" href="{{ root }}.static/pswp/photoswipe.css">
|
<link rel="stylesheet" href="{{ root }}.static/pswp/photoswipe.css">
|
||||||
<link rel="stylesheet" href="{{ root }}.static/pswp/default-skin/default-skin.css">
|
<link rel="stylesheet" href="{{ root }}.static/pswp/default-skin/default-skin.css">
|
||||||
<script src="{{ root }}.static/pswp/photoswipe.min.js"></script>
|
<script src="{{ root }}.static/pswp/photoswipe.min.js"></script>
|
||||||
<script src="{{ root }}.static/pswp/photoswipe-ui-default.min.js"></script>
|
<script src="{{ root }}.static/pswp/photoswipe-ui-default.min.js"></script>
|
||||||
{%- endif %}
|
<script src="{{ root }}.static/functionality.min.js"></script>
|
||||||
</head>
|
</head>
|
||||||
|
|
||||||
<body>
|
<body>
|
||||||
<div class="header">
|
<div class="header">
|
||||||
<ul class="navbar">
|
<ol class="navbar">
|
||||||
|
<div class="navleft">
|
||||||
<li><a href="{{ root }}">Home</a></li>
|
<li><a href="{{ root }}">Home</a></li>
|
||||||
{%- if parent %}
|
{%- if parent %}
|
||||||
<li><a href="{{ parent }}">Parent Directory</a></li>
|
<li><a href="{{ parent }}">Parent Directory</a></li>
|
||||||
{%- endif %}
|
{%- endif %}
|
||||||
{%- if info %}
|
{%- if info %}
|
||||||
<li class="tooltip"><a>Info</a><span class="tooltiptext">
|
<li class="tooltip"><a>Info</a><span class="tooltiptext infotext">
|
||||||
{%- for infoline in info -%}
|
{%- for infoline in info -%}
|
||||||
{{ infoline }}<br />
|
{{ infoline }}<br />
|
||||||
{%- endfor -%}
|
{%- endfor -%}
|
||||||
</span></li>
|
</span></li>
|
||||||
{%- endif %}
|
{%- endif %}
|
||||||
|
</div>
|
||||||
|
<div class="navcenter">
|
||||||
<li class="title"><span class="header">{{ header }}</span></li>
|
<li class="title"><span class="header">{{ header }}</span></li>
|
||||||
{%- if license %}
|
</div>
|
||||||
<li class="license"><a href="{{ license.url }}" rel="license noopener noreferrer" target="_blank">License</a></li>
|
<div class="navright">
|
||||||
|
{% if tags %}
|
||||||
|
<li class="tooltip">
|
||||||
|
<a id="tagtogglelink">Filter by Tags <svg width="0.8em" height="0.8em" viewBox="0 0 129.87601 129.87624">
|
||||||
|
<g id="layer1" transform="translate(-33.816833,-52.685642)">
|
||||||
|
<path stroke="currentColor" style="fill:none;stroke-width:20;stroke-linecap:round;stroke-linejoin:round;stroke-dasharray:none;stroke-opacity:1"
|
||||||
|
d="M 54.476483,95.484647 98.754836,139.76308 143.03319,95.484647" id="path1" />
|
||||||
|
</g>
|
||||||
|
</svg></a>
|
||||||
|
<ol class="tooltiptext tagdropdown" id="tagdropdown">
|
||||||
|
<span class="tagentry" id="reset-filter"><label>reset filter</label></span>
|
||||||
|
<span class="tagentry">
|
||||||
|
<label>
|
||||||
|
<input type="checkbox" id="recursive" />recursive filter
|
||||||
|
</label>
|
||||||
|
</span>
|
||||||
|
{{ render_tags(tags, '') }}
|
||||||
|
</ol>
|
||||||
|
</li>
|
||||||
|
{% endif %}
|
||||||
|
{%- if licensefile %}
|
||||||
|
<li class="license"><a href="{{ licensefile }}">License</a></li>
|
||||||
{%- endif %}
|
{%- endif %}
|
||||||
</ul>
|
</div>
|
||||||
|
</ol>
|
||||||
{% if subdirectories %}
|
{% if subdirectories %}
|
||||||
<div class="folders">
|
<div class="folders">
|
||||||
{%- for subdirectory in subdirectories %}
|
{%- for subdirectory in subdirectories %}
|
||||||
<a href="{{ subdirectory.url }}">
|
<a href="{{ subdirectory.url }}">
|
||||||
<figure>
|
<figure>
|
||||||
<img class="foldericon" />
|
<img class="foldericon" />
|
||||||
|
{%- if subdirectory.thumb %}
|
||||||
|
<img class="folderthumb" src="{{ subdirectory.thumb }}" />
|
||||||
|
{%- endif %}
|
||||||
<figcaption>{{ subdirectory.name }}</figcaption>
|
<figcaption>{{ subdirectory.name }}</figcaption>
|
||||||
</figure>
|
</figure>
|
||||||
</a>
|
</a>
|
||||||
@@ -53,29 +111,8 @@
|
|||||||
</div>
|
</div>
|
||||||
{%- endif %}
|
{%- endif %}
|
||||||
</div>
|
</div>
|
||||||
{% if images %}
|
<div class="row" id="imagelist">
|
||||||
{%- set ns = namespace(count = 0) -%}
|
|
||||||
<div class="row">
|
|
||||||
{%- for imageblock in images %}
|
|
||||||
<div class="column">
|
|
||||||
{%- for image in imageblock %}
|
|
||||||
<figure>
|
|
||||||
<img src="{{ image.thumbnail }}" alt="{{ image.name }}" onclick="openSwipe({{ ns.count }})" />
|
|
||||||
{%- set ns.count = ns.count + 1 %}
|
|
||||||
<figcaption class="caption">{{ image.name }}
|
|
||||||
{%- if image.tiff %}
|
|
||||||
<a href="{{ image.tiff }}">TIFF</a>
|
|
||||||
{%- endif %}
|
|
||||||
{%- if image.raw %}
|
|
||||||
<a href="{{ image.raw }}">RAW</a>
|
|
||||||
{%- endif %}
|
|
||||||
</figcaption>
|
|
||||||
</figure>
|
|
||||||
{%- endfor %}
|
|
||||||
</div>
|
</div>
|
||||||
{%- endfor %}
|
|
||||||
</div>
|
|
||||||
{%- endif %}
|
|
||||||
{% if license %}
|
{% if license %}
|
||||||
{%- if 'CC' in license.type %}
|
{%- if 'CC' in license.type %}
|
||||||
<div class="footer" xmlns:cc="http://creativecommons.org/ns#" xmlns:dct="http://purl.org/dc/terms/">
|
<div class="footer" xmlns:cc="http://creativecommons.org/ns#" xmlns:dct="http://purl.org/dc/terms/">
|
||||||
@@ -95,18 +132,17 @@
|
|||||||
</a>
|
</a>
|
||||||
{%- endif %}
|
{%- endif %}
|
||||||
<span class="attribution">Made with <a href="https://github.com/greflm13/StaticGalleryBuilder" target="_blank" rel="noopener noreferrer">StaticGalleryBuilder {{ version }}</a> by <a
|
<span class="attribution">Made with <a href="https://github.com/greflm13/StaticGalleryBuilder" target="_blank" rel="noopener noreferrer">StaticGalleryBuilder {{ version }}</a> by <a
|
||||||
href="https://github.com/greflm13" target="_blank" rel="noopener noreferrer">Flo Greistorfer</a>.</span>
|
href="https://github.com/greflm13" target="_blank" rel="noopener noreferrer">{{ logo }}</a>.</span>
|
||||||
<button onclick="topFunction()" id="totop" title="Back to Top">Back to Top</button>
|
<button type="button" id="totop" title="Back to Top">Back to Top</button>
|
||||||
</div>
|
</div>
|
||||||
{%- endif %}
|
{%- endif %}
|
||||||
{%- else %}
|
{%- else %}
|
||||||
<div class="footer">
|
<div class="footer">
|
||||||
<span class="attribution">Made with <a href="https://github.com/greflm13/StaticGalleryBuilder" target="_blank" rel="noopener noreferrer">StaticGalleryBuilder {{ version }}</a> by <a
|
<span class="attribution">Made with <a href="https://github.com/greflm13/StaticGalleryBuilder" target="_blank" rel="noopener noreferrer">StaticGalleryBuilder {{ version }}</a> by <a
|
||||||
href="https://github.com/greflm13" target="_blank" rel="noopener noreferrer">Flo Greistorfer</a>.</span>
|
href="https://github.com/greflm13" target="_blank" rel="noopener noreferrer">{{ logo }}</a>.</span>
|
||||||
<button onclick="topFunction()" id="totop" title="Back to Top">Back to Top</button>
|
<button type="button" id="totop" title="Back to Top">Back to Top</button>
|
||||||
</div>
|
</div>
|
||||||
{%- endif %}
|
{%- endif %}
|
||||||
{% if images %}
|
|
||||||
<div class="pswp" tabindex="-1" role="dialog" aria-hidden="true">
|
<div class="pswp" tabindex="-1" role="dialog" aria-hidden="true">
|
||||||
<div class="pswp__bg"></div>
|
<div class="pswp__bg"></div>
|
||||||
<div class="pswp__scroll-wrap">
|
<div class="pswp__scroll-wrap">
|
||||||
@@ -143,46 +179,9 @@
|
|||||||
</div>
|
</div>
|
||||||
</div>
|
</div>
|
||||||
</div>
|
</div>
|
||||||
<script>
|
|
||||||
var pswpElement = document.querySelectorAll('.pswp')[0];
|
|
||||||
var items = [
|
|
||||||
{%- for image in allimages %}
|
|
||||||
{ src: "{{ image.url }}", w: {{ image.width }}, h: {{ image.height }}, msrc: "{{ image.thumbnail }}" },
|
|
||||||
{%- endfor %}
|
|
||||||
];
|
|
||||||
var re = /pid=(\d+)/;
|
|
||||||
|
|
||||||
function openSwipe(img) {
|
|
||||||
var options = {
|
|
||||||
index: img
|
|
||||||
};
|
|
||||||
var gallery = new PhotoSwipe(pswpElement, PhotoSwipeUI_Default, items, options);
|
|
||||||
gallery.init();
|
|
||||||
}
|
|
||||||
|
|
||||||
if (re.test(window.location.href)) {
|
|
||||||
var pid = window.location.href.match(re)[1];
|
|
||||||
openSwipe(parseInt(pid));
|
|
||||||
}
|
|
||||||
|
|
||||||
let totopbutton = document.getElementById("totop");
|
|
||||||
|
|
||||||
window.onscroll = function () { scrollFunction() };
|
|
||||||
|
|
||||||
function scrollFunction() {
|
|
||||||
if (document.body.scrollTop > 20 || document.documentElement.scrollTop > 20) {
|
|
||||||
totopbutton.style.display = "block";
|
|
||||||
} else {
|
|
||||||
totopbutton.style.display = "none";
|
|
||||||
}
|
|
||||||
}
|
|
||||||
|
|
||||||
function topFunction() {
|
|
||||||
document.body.scrollTop = 0;
|
|
||||||
document.documentElement.scrollTop = 0;
|
|
||||||
}
|
|
||||||
</script>
|
|
||||||
{%- endif %}
|
|
||||||
</body>
|
</body>
|
||||||
|
<script>
|
||||||
|
new PhotoGallery();
|
||||||
|
</script>
|
||||||
|
|
||||||
</html>
|
</html>
|
||||||
70
templates/license.html.j2
Normal file
@@ -0,0 +1,70 @@
|
|||||||
|
<!DOCTYPE html>
|
||||||
|
<html lang="en">
|
||||||
|
|
||||||
|
<head>
|
||||||
|
<meta charset="UTF-8">
|
||||||
|
<meta name="viewport" content="width=device-width, initial-scale=1.0">
|
||||||
|
<title>{{ title }}</title>
|
||||||
|
{%- if webmanifest %}
|
||||||
|
<link rel="manifest" href="/.static/manifest.json">
|
||||||
|
{%- endif %}
|
||||||
|
<link rel="preload" href="{{ stylesheet }}" as="style">
|
||||||
|
{%- if theme %}
|
||||||
|
<link rel="preload" href="{{ theme }}" as="style">
|
||||||
|
{%- endif %}
|
||||||
|
<link rel="icon" type="image/x-icon" href="{{ favicon }}">
|
||||||
|
<link rel="stylesheet" href="{{ stylesheet }}">
|
||||||
|
{%- if theme %}
|
||||||
|
<link rel="stylesheet" href="{{ theme }}">
|
||||||
|
{%- endif %}
|
||||||
|
</head>
|
||||||
|
|
||||||
|
<body>
|
||||||
|
<div class="header">
|
||||||
|
<ol class="navbar">
|
||||||
|
<div class="navleft">
|
||||||
|
<li><a href="{{ root }}">Home</a></li>
|
||||||
|
{%- if parent %}
|
||||||
|
<li><a href="{{ parent }}">Parent Directory</a></li>
|
||||||
|
{%- endif %}
|
||||||
|
</div>
|
||||||
|
<div class="navcenter">
|
||||||
|
<li class="title"><span class="header">{{ header }}</span></li>
|
||||||
|
</div>
|
||||||
|
</div>
|
||||||
|
{%- if licensefile %}
|
||||||
|
<div class="licensefile">
|
||||||
|
{{ licensefile }}
|
||||||
|
</div>
|
||||||
|
{%- endif %}
|
||||||
|
{% if license %}
|
||||||
|
{%- if 'CC' in license.type %}
|
||||||
|
<div class="footer" xmlns:cc="http://creativecommons.org/ns#" xmlns:dct="http://purl.org/dc/terms/">
|
||||||
|
{%- if license.type == 'CC0 1.0' %}
|
||||||
|
<a property="dct:title" rel="cc:attributionURL" href="{{ root }}">{{ license.project }}</a> by <span property="cc:attributionName">{{ license.author }}</span> is marked with
|
||||||
|
<a href="{{ license.url }}" target="_blank" rel="license noopener noreferrer" style="display: inline-block">CC0 1.0
|
||||||
|
{%- for pic in license.pics %}
|
||||||
|
<img style="height: 22px !important; margin-left: 3px; vertical-align: text-bottom" src="{{ pic }}" alt="" />
|
||||||
|
{%- endfor %}
|
||||||
|
</a>
|
||||||
|
{%- else %}
|
||||||
|
<a property="dct:title" rel="cc:attributionURL" href="{{ root }}">{{ license.project }}</a> by <span property="cc:attributionName">{{ license.author }}</span> is licensed under
|
||||||
|
<a href="{{ license.url }}" target="_blank" rel="license noopener noreferrer">{{ license.type }}
|
||||||
|
{%- for pic in license.pics %}
|
||||||
|
<img src="{{ pic }}" alt="" />
|
||||||
|
{%- endfor %}
|
||||||
|
</a>
|
||||||
|
{%- endif %}
|
||||||
|
<span class="attribution">Made with <a href="https://github.com/greflm13/StaticGalleryBuilder" target="_blank" rel="noopener noreferrer">StaticGalleryBuilder {{ version }}</a> by <a
|
||||||
|
href="https://github.com/greflm13" target="_blank" rel="noopener noreferrer">{{ logo }}</a>.</span>
|
||||||
|
<button type="button" onclick="topFunction()" id="totop" title="Back to Top">Back to Top</button>
|
||||||
|
</div>
|
||||||
|
{%- endif %}
|
||||||
|
{%- else %}
|
||||||
|
<div class="footer">
|
||||||
|
<span class="attribution">Made with <a href="https://github.com/greflm13/StaticGalleryBuilder" target="_blank" rel="noopener noreferrer">StaticGalleryBuilder {{ version }}</a> by <a
|
||||||
|
href="https://github.com/greflm13" target="_blank" rel="noopener noreferrer">{{ logo }}</a>.</span>
|
||||||
|
<button type="button" onclick="topFunction()" id="totop" title="Back to Top">Back to Top</button>
|
||||||
|
</div>
|
||||||
|
{%- endif %}
|
||||||
|
</body>
|
||||||
BIN
test/example/DSC00009.jpg
Normal file
|
After Width: | Height: | Size: 28 MiB |
7
test/example/DSC00009.jpg.xmp
Normal file
@@ -0,0 +1,7 @@
|
|||||||
|
<?xml version="1.0" encoding="utf-8"?>
|
||||||
|
<x:xmpmeta x:xmptk="XMP Core 4.4.0-Exiv2" xmlns:x="adobe:ns:meta/">
|
||||||
|
<rdf:RDF xmlns:rdf="http://www.w3.org/1999/02/22-rdf-syntax-ns#">
|
||||||
|
<rdf:Description rdf:about="" xmlns:dc="http://purl.org/dc/elements/1.1/" xmlns:exif="http://ns.adobe.com/exif/1.0/" xmlns:lr="http://ns.adobe.com/lightroom/1.0/" xmlns:xmp="http://ns.adobe.com/xap/1.0/" xmlns:xmpMM="http://ns.adobe.com/xap/1.0/mm/" xmpMM:DerivedFrom="DSC00009.jpg">
|
||||||
|
<dc:subject><rdf:Bag><rdf:li>st</rdf:li><rdf:li>aqueduct</rdf:li><rdf:li>arch</rdf:li><rdf:li>arch bridge</rdf:li><rdf:li>bridge</rdf:li><rdf:li>hillside</rdf:li><rdf:li>passenger train</rdf:li><rdf:li>railroad</rdf:li><rdf:li>railroad bridge</rdf:li><rdf:li>span</rdf:li><rdf:li>train track</rdf:li><rdf:li>tree</rdf:li><rdf:li>viaduct</rdf:li></rdf:Bag></dc:subject><lr:hierarchicalSubject><rdf:Bag><rdf:li>st|aqueduct</rdf:li><rdf:li>st|arch</rdf:li><rdf:li>st|arch bridge</rdf:li><rdf:li>st|bridge</rdf:li><rdf:li>st|hillside</rdf:li><rdf:li>st|passenger train</rdf:li><rdf:li>st|railroad</rdf:li><rdf:li>st|railroad bridge</rdf:li><rdf:li>st|span</rdf:li><rdf:li>st|train track</rdf:li><rdf:li>st|tree</rdf:li><rdf:li>st|viaduct</rdf:li></rdf:Bag></lr:hierarchicalSubject></rdf:Description>
|
||||||
|
</rdf:RDF>
|
||||||
|
</x:xmpmeta>
|
||||||
BIN
test/example/DSC03470.JPG
Executable file
|
After Width: | Height: | Size: 20 MiB |
39
test/example/DSC03470.JPG.xmp
Normal file
@@ -0,0 +1,39 @@
|
|||||||
|
<?xml version="1.0" encoding="utf-8"?>
|
||||||
|
<x:xmpmeta x:xmptk="XMP Core 4.4.0-Exiv2" xmlns:x="adobe:ns:meta/">
|
||||||
|
<rdf:RDF xmlns:rdf="http://www.w3.org/1999/02/22-rdf-syntax-ns#">
|
||||||
|
<rdf:Description rdf:about="" xmlns:dc="http://purl.org/dc/elements/1.1/"
|
||||||
|
xmlns:exif="http://ns.adobe.com/exif/1.0/" xmlns:lr="http://ns.adobe.com/lightroom/1.0/"
|
||||||
|
xmlns:xmp="http://ns.adobe.com/xap/1.0/" xmlns:xmpMM="http://ns.adobe.com/xap/1.0/mm/"
|
||||||
|
xmpMM:DerivedFrom="DSC03470.JPG">
|
||||||
|
<dc:subject>
|
||||||
|
<rdf:Bag>
|
||||||
|
<rdf:li>st</rdf:li>
|
||||||
|
<rdf:li>bus</rdf:li>
|
||||||
|
<rdf:li>illuminate</rdf:li>
|
||||||
|
<rdf:li>neon</rdf:li>
|
||||||
|
<rdf:li>neon light</rdf:li>
|
||||||
|
<rdf:li>night</rdf:li>
|
||||||
|
<rdf:li>sign</rdf:li>
|
||||||
|
<rdf:li>train car</rdf:li>
|
||||||
|
<rdf:li>trolley</rdf:li>
|
||||||
|
<rdf:li>window</rdf:li>
|
||||||
|
<rdf:li>train</rdf:li>
|
||||||
|
</rdf:Bag>
|
||||||
|
</dc:subject>
|
||||||
|
<lr:hierarchicalSubject>
|
||||||
|
<rdf:Bag>
|
||||||
|
<rdf:li>st|bus</rdf:li>
|
||||||
|
<rdf:li>st|illuminate</rdf:li>
|
||||||
|
<rdf:li>st|neon</rdf:li>
|
||||||
|
<rdf:li>st|neon light</rdf:li>
|
||||||
|
<rdf:li>st|night</rdf:li>
|
||||||
|
<rdf:li>st|sign</rdf:li>
|
||||||
|
<rdf:li>st|train car</rdf:li>
|
||||||
|
<rdf:li>st|trolley</rdf:li>
|
||||||
|
<rdf:li>st|window</rdf:li>
|
||||||
|
<rdf:li>st|train</rdf:li>
|
||||||
|
</rdf:Bag>
|
||||||
|
</lr:hierarchicalSubject>
|
||||||
|
</rdf:Description>
|
||||||
|
</rdf:RDF>
|
||||||
|
</x:xmpmeta>
|
||||||
BIN
test/example/DSC03508.ARW
Executable file
7
test/example/DSC03508.ARW.xmp
Normal file
@@ -0,0 +1,7 @@
|
|||||||
|
<?xml version="1.0" encoding="utf-8"?>
|
||||||
|
<x:xmpmeta x:xmptk="XMP Core 4.4.0-Exiv2" xmlns:x="adobe:ns:meta/">
|
||||||
|
<rdf:RDF xmlns:rdf="http://www.w3.org/1999/02/22-rdf-syntax-ns#">
|
||||||
|
<rdf:Description rdf:about="" xmlns:dc="http://purl.org/dc/elements/1.1/" xmlns:exif="http://ns.adobe.com/exif/1.0/" xmlns:lr="http://ns.adobe.com/lightroom/1.0/" xmlns:xmp="http://ns.adobe.com/xap/1.0/" xmlns:xmpMM="http://ns.adobe.com/xap/1.0/mm/" xmpMM:DerivedFrom="DSC03508.ARW">
|
||||||
|
<dc:subject><rdf:Bag><rdf:li>st</rdf:li><rdf:li>building</rdf:li><rdf:li>ceiling</rdf:li><rdf:li>pillar</rdf:li><rdf:li>display</rdf:li><rdf:li>rail</rdf:li><rdf:li>steam engine</rdf:li><rdf:li>steam locomotive</rdf:li><rdf:li>train</rdf:li><rdf:li>train car</rdf:li><rdf:li>train track</rdf:li></rdf:Bag></dc:subject><lr:hierarchicalSubject><rdf:Bag><rdf:li>st|building</rdf:li><rdf:li>st|ceiling</rdf:li><rdf:li>st|pillar</rdf:li><rdf:li>st|display</rdf:li><rdf:li>st|rail</rdf:li><rdf:li>st|steam engine</rdf:li><rdf:li>st|steam locomotive</rdf:li><rdf:li>st|train</rdf:li><rdf:li>st|train car</rdf:li><rdf:li>st|train track</rdf:li></rdf:Bag></lr:hierarchicalSubject></rdf:Description>
|
||||||
|
</rdf:RDF>
|
||||||
|
</x:xmpmeta>
|
||||||
BIN
test/example/DSC03508.JPG
Executable file
|
After Width: | Height: | Size: 20 MiB |
49
test/example/DSC03508.JPG.xmp
Normal file
@@ -0,0 +1,49 @@
|
|||||||
|
<?xml version="1.0" encoding="utf-8"?>
|
||||||
|
<x:xmpmeta x:xmptk="XMP Core 4.4.0-Exiv2" xmlns:x="adobe:ns:meta/">
|
||||||
|
<rdf:RDF xmlns:rdf="http://www.w3.org/1999/02/22-rdf-syntax-ns#">
|
||||||
|
<rdf:Description rdf:about="" xmlns:dc="http://purl.org/dc/elements/1.1/"
|
||||||
|
xmlns:exif="http://ns.adobe.com/exif/1.0/" xmlns:lr="http://ns.adobe.com/lightroom/1.0/"
|
||||||
|
xmlns:xmp="http://ns.adobe.com/xap/1.0/" xmlns:xmpMM="http://ns.adobe.com/xap/1.0/mm/"
|
||||||
|
xmpMM:DerivedFrom="DSC03508.JPG">
|
||||||
|
<dc:subject>
|
||||||
|
<rdf:Bag>
|
||||||
|
<rdf:li>st</rdf:li>
|
||||||
|
<rdf:li>attach</rdf:li>
|
||||||
|
<rdf:li>basement</rdf:li>
|
||||||
|
<rdf:li>beam</rdf:li>
|
||||||
|
<rdf:li>building</rdf:li>
|
||||||
|
<rdf:li>ceiling</rdf:li>
|
||||||
|
<rdf:li>equipment</rdf:li>
|
||||||
|
<rdf:li>floor</rdf:li>
|
||||||
|
<rdf:li>pipe</rdf:li>
|
||||||
|
<rdf:li>red</rdf:li>
|
||||||
|
<rdf:li>room</rdf:li>
|
||||||
|
<rdf:li>scaffold</rdf:li>
|
||||||
|
<rdf:li>tube</rdf:li>
|
||||||
|
<rdf:li>warehouse</rdf:li>
|
||||||
|
<rdf:li>water pipe</rdf:li>
|
||||||
|
<rdf:li>train</rdf:li>
|
||||||
|
</rdf:Bag>
|
||||||
|
</dc:subject>
|
||||||
|
<lr:hierarchicalSubject>
|
||||||
|
<rdf:Bag>
|
||||||
|
<rdf:li>st|attach</rdf:li>
|
||||||
|
<rdf:li>st|basement</rdf:li>
|
||||||
|
<rdf:li>st|beam</rdf:li>
|
||||||
|
<rdf:li>st|building</rdf:li>
|
||||||
|
<rdf:li>st|ceiling</rdf:li>
|
||||||
|
<rdf:li>st|equipment</rdf:li>
|
||||||
|
<rdf:li>st|floor</rdf:li>
|
||||||
|
<rdf:li>st|pipe</rdf:li>
|
||||||
|
<rdf:li>st|red</rdf:li>
|
||||||
|
<rdf:li>st|room</rdf:li>
|
||||||
|
<rdf:li>st|scaffold</rdf:li>
|
||||||
|
<rdf:li>st|tube</rdf:li>
|
||||||
|
<rdf:li>st|warehouse</rdf:li>
|
||||||
|
<rdf:li>st|water pipe</rdf:li>
|
||||||
|
<rdf:li>st|train</rdf:li>
|
||||||
|
</rdf:Bag>
|
||||||
|
</lr:hierarchicalSubject>
|
||||||
|
</rdf:Description>
|
||||||
|
</rdf:RDF>
|
||||||
|
</x:xmpmeta>
|
||||||
427
test/example/LICENSE
Normal file
@@ -0,0 +1,427 @@
|
|||||||
|
Attribution-ShareAlike 4.0 International
|
||||||
|
|
||||||
|
=======================================================================
|
||||||
|
|
||||||
|
Creative Commons Corporation ("Creative Commons") is not a law firm and
|
||||||
|
does not provide legal services or legal advice. Distribution of
|
||||||
|
Creative Commons public licenses does not create a lawyer-client or
|
||||||
|
other relationship. Creative Commons makes its licenses and related
|
||||||
|
information available on an "as-is" basis. Creative Commons gives no
|
||||||
|
warranties regarding its licenses, any material licensed under their
|
||||||
|
terms and conditions, or any related information. Creative Commons
|
||||||
|
disclaims all liability for damages resulting from their use to the
|
||||||
|
fullest extent possible.
|
||||||
|
|
||||||
|
Using Creative Commons Public Licenses
|
||||||
|
|
||||||
|
Creative Commons public licenses provide a standard set of terms and
|
||||||
|
conditions that creators and other rights holders may use to share
|
||||||
|
original works of authorship and other material subject to copyright
|
||||||
|
and certain other rights specified in the public license below. The
|
||||||
|
following considerations are for informational purposes only, are not
|
||||||
|
exhaustive, and do not form part of our licenses.
|
||||||
|
|
||||||
|
Considerations for licensors: Our public licenses are
|
||||||
|
intended for use by those authorized to give the public
|
||||||
|
permission to use material in ways otherwise restricted by
|
||||||
|
copyright and certain other rights. Our licenses are
|
||||||
|
irrevocable. Licensors should read and understand the terms
|
||||||
|
and conditions of the license they choose before applying it.
|
||||||
|
Licensors should also secure all rights necessary before
|
||||||
|
applying our licenses so that the public can reuse the
|
||||||
|
material as expected. Licensors should clearly mark any
|
||||||
|
material not subject to the license. This includes other CC-
|
||||||
|
licensed material, or material used under an exception or
|
||||||
|
limitation to copyright. More considerations for licensors:
|
||||||
|
wiki.creativecommons.org/Considerations_for_licensors
|
||||||
|
|
||||||
|
Considerations for the public: By using one of our public
|
||||||
|
licenses, a licensor grants the public permission to use the
|
||||||
|
licensed material under specified terms and conditions. If
|
||||||
|
the licensor's permission is not necessary for any reason--for
|
||||||
|
example, because of any applicable exception or limitation to
|
||||||
|
copyright--then that use is not regulated by the license. Our
|
||||||
|
licenses grant only permissions under copyright and certain
|
||||||
|
other rights that a licensor has authority to grant. Use of
|
||||||
|
the licensed material may still be restricted for other
|
||||||
|
reasons, including because others have copyright or other
|
||||||
|
rights in the material. A licensor may make special requests,
|
||||||
|
such as asking that all changes be marked or described.
|
||||||
|
Although not required by our licenses, you are encouraged to
|
||||||
|
respect those requests where reasonable. More considerations
|
||||||
|
for the public:
|
||||||
|
wiki.creativecommons.org/Considerations_for_licensees
|
||||||
|
|
||||||
|
=======================================================================
|
||||||
|
|
||||||
|
Creative Commons Attribution-ShareAlike 4.0 International Public
|
||||||
|
License
|
||||||
|
|
||||||
|
By exercising the Licensed Rights (defined below), You accept and agree
|
||||||
|
to be bound by the terms and conditions of this Creative Commons
|
||||||
|
Attribution-ShareAlike 4.0 International Public License ("Public
|
||||||
|
License"). To the extent this Public License may be interpreted as a
|
||||||
|
contract, You are granted the Licensed Rights in consideration of Your
|
||||||
|
acceptance of these terms and conditions, and the Licensor grants You
|
||||||
|
such rights in consideration of benefits the Licensor receives from
|
||||||
|
making the Licensed Material available under these terms and
|
||||||
|
conditions.
|
||||||
|
|
||||||
|
|
||||||
|
Section 1 -- Definitions.
|
||||||
|
|
||||||
|
a. Adapted Material means material subject to Copyright and Similar
|
||||||
|
Rights that is derived from or based upon the Licensed Material
|
||||||
|
and in which the Licensed Material is translated, altered,
|
||||||
|
arranged, transformed, or otherwise modified in a manner requiring
|
||||||
|
permission under the Copyright and Similar Rights held by the
|
||||||
|
Licensor. For purposes of this Public License, where the Licensed
|
||||||
|
Material is a musical work, performance, or sound recording,
|
||||||
|
Adapted Material is always produced where the Licensed Material is
|
||||||
|
synched in timed relation with a moving image.
|
||||||
|
|
||||||
|
b. Adapter's License means the license You apply to Your Copyright
|
||||||
|
and Similar Rights in Your contributions to Adapted Material in
|
||||||
|
accordance with the terms and conditions of this Public License.
|
||||||
|
|
||||||
|
c. BY-SA Compatible License means a license listed at
|
||||||
|
creativecommons.org/compatiblelicenses, approved by Creative
|
||||||
|
Commons as essentially the equivalent of this Public License.
|
||||||
|
|
||||||
|
d. Copyright and Similar Rights means copyright and/or similar rights
|
||||||
|
closely related to copyright including, without limitation,
|
||||||
|
performance, broadcast, sound recording, and Sui Generis Database
|
||||||
|
Rights, without regard to how the rights are labeled or
|
||||||
|
categorized. For purposes of this Public License, the rights
|
||||||
|
specified in Section 2(b)(1)-(2) are not Copyright and Similar
|
||||||
|
Rights.
|
||||||
|
|
||||||
|
e. Effective Technological Measures means those measures that, in the
|
||||||
|
absence of proper authority, may not be circumvented under laws
|
||||||
|
fulfilling obligations under Article 11 of the WIPO Copyright
|
||||||
|
Treaty adopted on December 20, 1996, and/or similar international
|
||||||
|
agreements.
|
||||||
|
|
||||||
|
f. Exceptions and Limitations means fair use, fair dealing, and/or
|
||||||
|
any other exception or limitation to Copyright and Similar Rights
|
||||||
|
that applies to Your use of the Licensed Material.
|
||||||
|
|
||||||
|
g. License Elements means the license attributes listed in the name
|
||||||
|
of a Creative Commons Public License. The License Elements of this
|
||||||
|
Public License are Attribution and ShareAlike.
|
||||||
|
|
||||||
|
h. Licensed Material means the artistic or literary work, database,
|
||||||
|
or other material to which the Licensor applied this Public
|
||||||
|
License.
|
||||||
|
|
||||||
|
i. Licensed Rights means the rights granted to You subject to the
|
||||||
|
terms and conditions of this Public License, which are limited to
|
||||||
|
all Copyright and Similar Rights that apply to Your use of the
|
||||||
|
Licensed Material and that the Licensor has authority to license.
|
||||||
|
|
||||||
|
j. Licensor means the individual(s) or entity(ies) granting rights
|
||||||
|
under this Public License.
|
||||||
|
|
||||||
|
k. Share means to provide material to the public by any means or
|
||||||
|
process that requires permission under the Licensed Rights, such
|
||||||
|
as reproduction, public display, public performance, distribution,
|
||||||
|
dissemination, communication, or importation, and to make material
|
||||||
|
available to the public including in ways that members of the
|
||||||
|
public may access the material from a place and at a time
|
||||||
|
individually chosen by them.
|
||||||
|
|
||||||
|
l. Sui Generis Database Rights means rights other than copyright
|
||||||
|
resulting from Directive 96/9/EC of the European Parliament and of
|
||||||
|
the Council of 11 March 1996 on the legal protection of databases,
|
||||||
|
as amended and/or succeeded, as well as other essentially
|
||||||
|
equivalent rights anywhere in the world.
|
||||||
|
|
||||||
|
m. You means the individual or entity exercising the Licensed Rights
|
||||||
|
under this Public License. Your has a corresponding meaning.
|
||||||
|
|
||||||
|
|
||||||
|
Section 2 -- Scope.
|
||||||
|
|
||||||
|
a. License grant.
|
||||||
|
|
||||||
|
1. Subject to the terms and conditions of this Public License,
|
||||||
|
the Licensor hereby grants You a worldwide, royalty-free,
|
||||||
|
non-sublicensable, non-exclusive, irrevocable license to
|
||||||
|
exercise the Licensed Rights in the Licensed Material to:
|
||||||
|
|
||||||
|
a. reproduce and Share the Licensed Material, in whole or
|
||||||
|
in part; and
|
||||||
|
|
||||||
|
b. produce, reproduce, and Share Adapted Material.
|
||||||
|
|
||||||
|
2. Exceptions and Limitations. For the avoidance of doubt, where
|
||||||
|
Exceptions and Limitations apply to Your use, this Public
|
||||||
|
License does not apply, and You do not need to comply with
|
||||||
|
its terms and conditions.
|
||||||
|
|
||||||
|
3. Term. The term of this Public License is specified in Section
|
||||||
|
6(a).
|
||||||
|
|
||||||
|
4. Media and formats; technical modifications allowed. The
|
||||||
|
Licensor authorizes You to exercise the Licensed Rights in
|
||||||
|
all media and formats whether now known or hereafter created,
|
||||||
|
and to make technical modifications necessary to do so. The
|
||||||
|
Licensor waives and/or agrees not to assert any right or
|
||||||
|
authority to forbid You from making technical modifications
|
||||||
|
necessary to exercise the Licensed Rights, including
|
||||||
|
technical modifications necessary to circumvent Effective
|
||||||
|
Technological Measures. For purposes of this Public License,
|
||||||
|
simply making modifications authorized by this Section 2(a)
|
||||||
|
(4) never produces Adapted Material.
|
||||||
|
|
||||||
|
5. Downstream recipients.
|
||||||
|
|
||||||
|
a. Offer from the Licensor -- Licensed Material. Every
|
||||||
|
recipient of the Licensed Material automatically
|
||||||
|
receives an offer from the Licensor to exercise the
|
||||||
|
Licensed Rights under the terms and conditions of this
|
||||||
|
Public License.
|
||||||
|
|
||||||
|
b. Additional offer from the Licensor -- Adapted Material.
|
||||||
|
Every recipient of Adapted Material from You
|
||||||
|
automatically receives an offer from the Licensor to
|
||||||
|
exercise the Licensed Rights in the Adapted Material
|
||||||
|
under the conditions of the Adapter's License You apply.
|
||||||
|
|
||||||
|
c. No downstream restrictions. You may not offer or impose
|
||||||
|
any additional or different terms or conditions on, or
|
||||||
|
apply any Effective Technological Measures to, the
|
||||||
|
Licensed Material if doing so restricts exercise of the
|
||||||
|
Licensed Rights by any recipient of the Licensed
|
||||||
|
Material.
|
||||||
|
|
||||||
|
6. No endorsement. Nothing in this Public License constitutes or
|
||||||
|
may be construed as permission to assert or imply that You
|
||||||
|
are, or that Your use of the Licensed Material is, connected
|
||||||
|
with, or sponsored, endorsed, or granted official status by,
|
||||||
|
the Licensor or others designated to receive attribution as
|
||||||
|
provided in Section 3(a)(1)(A)(i).
|
||||||
|
|
||||||
|
b. Other rights.
|
||||||
|
|
||||||
|
1. Moral rights, such as the right of integrity, are not
|
||||||
|
licensed under this Public License, nor are publicity,
|
||||||
|
privacy, and/or other similar personality rights; however, to
|
||||||
|
the extent possible, the Licensor waives and/or agrees not to
|
||||||
|
assert any such rights held by the Licensor to the limited
|
||||||
|
extent necessary to allow You to exercise the Licensed
|
||||||
|
Rights, but not otherwise.
|
||||||
|
|
||||||
|
2. Patent and trademark rights are not licensed under this
|
||||||
|
Public License.
|
||||||
|
|
||||||
|
3. To the extent possible, the Licensor waives any right to
|
||||||
|
collect royalties from You for the exercise of the Licensed
|
||||||
|
Rights, whether directly or through a collecting society
|
||||||
|
under any voluntary or waivable statutory or compulsory
|
||||||
|
licensing scheme. In all other cases the Licensor expressly
|
||||||
|
reserves any right to collect such royalties.
|
||||||
|
|
||||||
|
|
||||||
|
Section 3 -- License Conditions.
|
||||||
|
|
||||||
|
Your exercise of the Licensed Rights is expressly made subject to the
|
||||||
|
following conditions.
|
||||||
|
|
||||||
|
a. Attribution.
|
||||||
|
|
||||||
|
1. If You Share the Licensed Material (including in modified
|
||||||
|
form), You must:
|
||||||
|
|
||||||
|
a. retain the following if it is supplied by the Licensor
|
||||||
|
with the Licensed Material:
|
||||||
|
|
||||||
|
i. identification of the creator(s) of the Licensed
|
||||||
|
Material and any others designated to receive
|
||||||
|
attribution, in any reasonable manner requested by
|
||||||
|
the Licensor (including by pseudonym if
|
||||||
|
designated);
|
||||||
|
|
||||||
|
ii. a copyright notice;
|
||||||
|
|
||||||
|
iii. a notice that refers to this Public License;
|
||||||
|
|
||||||
|
iv. a notice that refers to the disclaimer of
|
||||||
|
warranties;
|
||||||
|
|
||||||
|
v. a URI or hyperlink to the Licensed Material to the
|
||||||
|
extent reasonably practicable;
|
||||||
|
|
||||||
|
b. indicate if You modified the Licensed Material and
|
||||||
|
retain an indication of any previous modifications; and
|
||||||
|
|
||||||
|
c. indicate the Licensed Material is licensed under this
|
||||||
|
Public License, and include the text of, or the URI or
|
||||||
|
hyperlink to, this Public License.
|
||||||
|
|
||||||
|
2. You may satisfy the conditions in Section 3(a)(1) in any
|
||||||
|
reasonable manner based on the medium, means, and context in
|
||||||
|
which You Share the Licensed Material. For example, it may be
|
||||||
|
reasonable to satisfy the conditions by providing a URI or
|
||||||
|
hyperlink to a resource that includes the required
|
||||||
|
information.
|
||||||
|
|
||||||
|
3. If requested by the Licensor, You must remove any of the
|
||||||
|
information required by Section 3(a)(1)(A) to the extent
|
||||||
|
reasonably practicable.
|
||||||
|
|
||||||
|
b. ShareAlike.
|
||||||
|
|
||||||
|
In addition to the conditions in Section 3(a), if You Share
|
||||||
|
Adapted Material You produce, the following conditions also apply.
|
||||||
|
|
||||||
|
1. The Adapter's License You apply must be a Creative Commons
|
||||||
|
license with the same License Elements, this version or
|
||||||
|
later, or a BY-SA Compatible License.
|
||||||
|
|
||||||
|
2. You must include the text of, or the URI or hyperlink to, the
|
||||||
|
Adapter's License You apply. You may satisfy this condition
|
||||||
|
in any reasonable manner based on the medium, means, and
|
||||||
|
context in which You Share Adapted Material.
|
||||||
|
|
||||||
|
3. You may not offer or impose any additional or different terms
|
||||||
|
or conditions on, or apply any Effective Technological
|
||||||
|
Measures to, Adapted Material that restrict exercise of the
|
||||||
|
rights granted under the Adapter's License You apply.
|
||||||
|
|
||||||
|
|
||||||
|
Section 4 -- Sui Generis Database Rights.
|
||||||
|
|
||||||
|
Where the Licensed Rights include Sui Generis Database Rights that
|
||||||
|
apply to Your use of the Licensed Material:
|
||||||
|
|
||||||
|
a. for the avoidance of doubt, Section 2(a)(1) grants You the right
|
||||||
|
to extract, reuse, reproduce, and Share all or a substantial
|
||||||
|
portion of the contents of the database;
|
||||||
|
|
||||||
|
b. if You include all or a substantial portion of the database
|
||||||
|
contents in a database in which You have Sui Generis Database
|
||||||
|
Rights, then the database in which You have Sui Generis Database
|
||||||
|
Rights (but not its individual contents) is Adapted Material,
|
||||||
|
including for purposes of Section 3(b); and
|
||||||
|
|
||||||
|
c. You must comply with the conditions in Section 3(a) if You Share
|
||||||
|
all or a substantial portion of the contents of the database.
|
||||||
|
|
||||||
|
For the avoidance of doubt, this Section 4 supplements and does not
|
||||||
|
replace Your obligations under this Public License where the Licensed
|
||||||
|
Rights include other Copyright and Similar Rights.
|
||||||
|
|
||||||
|
|
||||||
|
Section 5 -- Disclaimer of Warranties and Limitation of Liability.
|
||||||
|
|
||||||
|
a. UNLESS OTHERWISE SEPARATELY UNDERTAKEN BY THE LICENSOR, TO THE
|
||||||
|
EXTENT POSSIBLE, THE LICENSOR OFFERS THE LICENSED MATERIAL AS-IS
|
||||||
|
AND AS-AVAILABLE, AND MAKES NO REPRESENTATIONS OR WARRANTIES OF
|
||||||
|
ANY KIND CONCERNING THE LICENSED MATERIAL, WHETHER EXPRESS,
|
||||||
|
IMPLIED, STATUTORY, OR OTHER. THIS INCLUDES, WITHOUT LIMITATION,
|
||||||
|
WARRANTIES OF TITLE, MERCHANTABILITY, FITNESS FOR A PARTICULAR
|
||||||
|
PURPOSE, NON-INFRINGEMENT, ABSENCE OF LATENT OR OTHER DEFECTS,
|
||||||
|
ACCURACY, OR THE PRESENCE OR ABSENCE OF ERRORS, WHETHER OR NOT
|
||||||
|
KNOWN OR DISCOVERABLE. WHERE DISCLAIMERS OF WARRANTIES ARE NOT
|
||||||
|
ALLOWED IN FULL OR IN PART, THIS DISCLAIMER MAY NOT APPLY TO YOU.
|
||||||
|
|
||||||
|
b. TO THE EXTENT POSSIBLE, IN NO EVENT WILL THE LICENSOR BE LIABLE
|
||||||
|
TO YOU ON ANY LEGAL THEORY (INCLUDING, WITHOUT LIMITATION,
|
||||||
|
NEGLIGENCE) OR OTHERWISE FOR ANY DIRECT, SPECIAL, INDIRECT,
|
||||||
|
INCIDENTAL, CONSEQUENTIAL, PUNITIVE, EXEMPLARY, OR OTHER LOSSES,
|
||||||
|
COSTS, EXPENSES, OR DAMAGES ARISING OUT OF THIS PUBLIC LICENSE OR
|
||||||
|
USE OF THE LICENSED MATERIAL, EVEN IF THE LICENSOR HAS BEEN
|
||||||
|
ADVISED OF THE POSSIBILITY OF SUCH LOSSES, COSTS, EXPENSES, OR
|
||||||
|
DAMAGES. WHERE A LIMITATION OF LIABILITY IS NOT ALLOWED IN FULL OR
|
||||||
|
IN PART, THIS LIMITATION MAY NOT APPLY TO YOU.
|
||||||
|
|
||||||
|
c. The disclaimer of warranties and limitation of liability provided
|
||||||
|
above shall be interpreted in a manner that, to the extent
|
||||||
|
possible, most closely approximates an absolute disclaimer and
|
||||||
|
waiver of all liability.
|
||||||
|
|
||||||
|
|
||||||
|
Section 6 -- Term and Termination.
|
||||||
|
|
||||||
|
a. This Public License applies for the term of the Copyright and
|
||||||
|
Similar Rights licensed here. However, if You fail to comply with
|
||||||
|
this Public License, then Your rights under this Public License
|
||||||
|
terminate automatically.
|
||||||
|
|
||||||
|
b. Where Your right to use the Licensed Material has terminated under
|
||||||
|
Section 6(a), it reinstates:
|
||||||
|
|
||||||
|
1. automatically as of the date the violation is cured, provided
|
||||||
|
it is cured within 30 days of Your discovery of the
|
||||||
|
violation; or
|
||||||
|
|
||||||
|
2. upon express reinstatement by the Licensor.
|
||||||
|
|
||||||
|
For the avoidance of doubt, this Section 6(b) does not affect any
|
||||||
|
right the Licensor may have to seek remedies for Your violations
|
||||||
|
of this Public License.
|
||||||
|
|
||||||
|
c. For the avoidance of doubt, the Licensor may also offer the
|
||||||
|
Licensed Material under separate terms or conditions or stop
|
||||||
|
distributing the Licensed Material at any time; however, doing so
|
||||||
|
will not terminate this Public License.
|
||||||
|
|
||||||
|
d. Sections 1, 5, 6, 7, and 8 survive termination of this Public
|
||||||
|
License.
|
||||||
|
|
||||||
|
|
||||||
|
Section 7 -- Other Terms and Conditions.
|
||||||
|
|
||||||
|
a. The Licensor shall not be bound by any additional or different
|
||||||
|
terms or conditions communicated by You unless expressly agreed.
|
||||||
|
|
||||||
|
b. Any arrangements, understandings, or agreements regarding the
|
||||||
|
Licensed Material not stated herein are separate from and
|
||||||
|
independent of the terms and conditions of this Public License.
|
||||||
|
|
||||||
|
|
||||||
|
Section 8 -- Interpretation.
|
||||||
|
|
||||||
|
a. For the avoidance of doubt, this Public License does not, and
|
||||||
|
shall not be interpreted to, reduce, limit, restrict, or impose
|
||||||
|
conditions on any use of the Licensed Material that could lawfully
|
||||||
|
be made without permission under this Public License.
|
||||||
|
|
||||||
|
b. To the extent possible, if any provision of this Public License is
|
||||||
|
deemed unenforceable, it shall be automatically reformed to the
|
||||||
|
minimum extent necessary to make it enforceable. If the provision
|
||||||
|
cannot be reformed, it shall be severed from this Public License
|
||||||
|
without affecting the enforceability of the remaining terms and
|
||||||
|
conditions.
|
||||||
|
|
||||||
|
c. No term or condition of this Public License will be waived and no
|
||||||
|
failure to comply consented to unless expressly agreed to by the
|
||||||
|
Licensor.
|
||||||
|
|
||||||
|
d. Nothing in this Public License constitutes or may be interpreted
|
||||||
|
as a limitation upon, or waiver of, any privileges and immunities
|
||||||
|
that apply to the Licensor or You, including from the legal
|
||||||
|
processes of any jurisdiction or authority.
|
||||||
|
|
||||||
|
|
||||||
|
=======================================================================
|
||||||
|
|
||||||
|
Creative Commons is not a party to its public
|
||||||
|
licenses. Notwithstanding, Creative Commons may elect to apply one of
|
||||||
|
its public licenses to material it publishes and in those instances
|
||||||
|
will be considered the “Licensor.” The text of the Creative Commons
|
||||||
|
public licenses is dedicated to the public domain under the CC0 Public
|
||||||
|
Domain Dedication. Except for the limited purpose of indicating that
|
||||||
|
material is shared under a Creative Commons public license or as
|
||||||
|
otherwise permitted by the Creative Commons policies published at
|
||||||
|
creativecommons.org/policies, Creative Commons does not authorize the
|
||||||
|
use of the trademark "Creative Commons" or any other trademark or logo
|
||||||
|
of Creative Commons without its prior written consent including,
|
||||||
|
without limitation, in connection with any unauthorized modifications
|
||||||
|
to any of its public licenses or any other arrangements,
|
||||||
|
understandings, or agreements concerning use of licensed material. For
|
||||||
|
the avoidance of doubt, this paragraph does not form part of the
|
||||||
|
public licenses.
|
||||||
|
|
||||||
|
Creative Commons may be contacted at creativecommons.org.
|
||||||
35
test/example/example.jpg.xmp
Normal file
@@ -0,0 +1,35 @@
|
|||||||
|
<?xml version="1.0" encoding="utf-8"?>
|
||||||
|
<x:xmpmeta x:xmptk="XMP Core 4.4.0-Exiv2" xmlns:x="adobe:ns:meta/">
|
||||||
|
<rdf:RDF xmlns:rdf="http://www.w3.org/1999/02/22-rdf-syntax-ns#">
|
||||||
|
<rdf:Description rdf:about="" xmlns:dc="http://purl.org/dc/elements/1.1/"
|
||||||
|
xmlns:exif="http://ns.adobe.com/exif/1.0/" xmlns:lr="http://ns.adobe.com/lightroom/1.0/"
|
||||||
|
xmlns:xmp="http://ns.adobe.com/xap/1.0/" xmlns:xmpMM="http://ns.adobe.com/xap/1.0/mm/"
|
||||||
|
xmpMM:DerivedFrom="example.jpg">
|
||||||
|
<dc:subject>
|
||||||
|
<rdf:Bag>
|
||||||
|
<rdf:li>st</rdf:li>
|
||||||
|
<rdf:li>cloudy</rdf:li>
|
||||||
|
<rdf:li>evening sky</rdf:li>
|
||||||
|
<rdf:li>sea</rdf:li>
|
||||||
|
<rdf:li>sky</rdf:li>
|
||||||
|
<rdf:li>storm cloud</rdf:li>
|
||||||
|
<rdf:li>stormy</rdf:li>
|
||||||
|
<rdf:li>sun</rdf:li>
|
||||||
|
<rdf:li>sunset</rdf:li>
|
||||||
|
</rdf:Bag>
|
||||||
|
</dc:subject>
|
||||||
|
<lr:hierarchicalSubject>
|
||||||
|
<rdf:Bag>
|
||||||
|
<rdf:li>st|cloudy</rdf:li>
|
||||||
|
<rdf:li>st|evening sky</rdf:li>
|
||||||
|
<rdf:li>st|sea</rdf:li>
|
||||||
|
<rdf:li>st|sky</rdf:li>
|
||||||
|
<rdf:li>st|storm cloud</rdf:li>
|
||||||
|
<rdf:li>st|stormy</rdf:li>
|
||||||
|
<rdf:li>st|sun</rdf:li>
|
||||||
|
<rdf:li>st|sunset</rdf:li>
|
||||||
|
</rdf:Bag>
|
||||||
|
</lr:hierarchicalSubject>
|
||||||
|
</rdf:Description>
|
||||||
|
</rdf:RDF>
|
||||||
|
</x:xmpmeta>
|
||||||
7
test/example/example.tif.xmp
Normal file
@@ -0,0 +1,7 @@
|
|||||||
|
<?xml version="1.0" encoding="utf-8"?>
|
||||||
|
<x:xmpmeta x:xmptk="XMP Core 4.4.0-Exiv2" xmlns:x="adobe:ns:meta/">
|
||||||
|
<rdf:RDF xmlns:rdf="http://www.w3.org/1999/02/22-rdf-syntax-ns#">
|
||||||
|
<rdf:Description rdf:about="" xmlns:dc="http://purl.org/dc/elements/1.1/" xmlns:exif="http://ns.adobe.com/exif/1.0/" xmlns:lr="http://ns.adobe.com/lightroom/1.0/" xmlns:xmp="http://ns.adobe.com/xap/1.0/" xmlns:xmpMM="http://ns.adobe.com/xap/1.0/mm/" xmpMM:DerivedFrom="example.tif">
|
||||||
|
<dc:subject><rdf:Bag><rdf:li>st</rdf:li><rdf:li>cloud</rdf:li><rdf:li>cloudy</rdf:li><rdf:li>evening sky</rdf:li><rdf:li>sea</rdf:li><rdf:li>sky</rdf:li><rdf:li>storm cloud</rdf:li><rdf:li>stormy</rdf:li><rdf:li>sun</rdf:li><rdf:li>sunset</rdf:li></rdf:Bag></dc:subject><lr:hierarchicalSubject><rdf:Bag><rdf:li>st|cloud</rdf:li><rdf:li>st|cloudy</rdf:li><rdf:li>st|evening sky</rdf:li><rdf:li>st|sea</rdf:li><rdf:li>st|sky</rdf:li><rdf:li>st|storm cloud</rdf:li><rdf:li>st|stormy</rdf:li><rdf:li>st|sun</rdf:li><rdf:li>st|sunset</rdf:li></rdf:Bag></lr:hierarchicalSubject></rdf:Description>
|
||||||
|
</rdf:RDF>
|
||||||
|
</x:xmpmeta>
|
||||||
BIN
test/example/subfolder/DSC01106.jpg
Normal file
|
After Width: | Height: | Size: 3.5 MiB |
30
test/example/subfolder/DSC01106.jpg.xmp
Normal file
@@ -0,0 +1,30 @@
|
|||||||
|
<?xml version="1.0" encoding="utf-8"?>
|
||||||
|
<x:xmpmeta x:xmptk="XMP Core 4.4.0-Exiv2" xmlns:x="adobe:ns:meta/">
|
||||||
|
<rdf:RDF xmlns:rdf="http://www.w3.org/1999/02/22-rdf-syntax-ns#">
|
||||||
|
<rdf:Description rdf:about="" xmlns:dc="http://purl.org/dc/elements/1.1/"
|
||||||
|
xmlns:exif="http://ns.adobe.com/exif/1.0/" xmlns:lr="http://ns.adobe.com/lightroom/1.0/"
|
||||||
|
xmlns:xmp="http://ns.adobe.com/xap/1.0/" xmlns:xmpMM="http://ns.adobe.com/xap/1.0/mm/"
|
||||||
|
xmpMM:DerivedFrom="DSC01106.jpg">
|
||||||
|
<dc:subject>
|
||||||
|
<rdf:Bag>
|
||||||
|
<rdf:li>st</rdf:li>
|
||||||
|
<rdf:li>clear</rdf:li>
|
||||||
|
<rdf:li>dark</rdf:li>
|
||||||
|
<rdf:li>moon</rdf:li>
|
||||||
|
<rdf:li>night</rdf:li>
|
||||||
|
<rdf:li>night sky</rdf:li>
|
||||||
|
<rdf:li>sky</rdf:li>
|
||||||
|
</rdf:Bag>
|
||||||
|
</dc:subject>
|
||||||
|
<lr:hierarchicalSubject>
|
||||||
|
<rdf:Bag>
|
||||||
|
<rdf:li>st|clear</rdf:li>
|
||||||
|
<rdf:li>st|dark</rdf:li>
|
||||||
|
<rdf:li>st|moon</rdf:li>
|
||||||
|
<rdf:li>st|night</rdf:li>
|
||||||
|
<rdf:li>st|sky|night sky</rdf:li>
|
||||||
|
</rdf:Bag>
|
||||||
|
</lr:hierarchicalSubject>
|
||||||
|
</rdf:Description>
|
||||||
|
</rdf:RDF>
|
||||||
|
</x:xmpmeta>
|
||||||
|
After Width: | Height: | Size: 7.3 MiB |
@@ -0,0 +1,25 @@
|
|||||||
|
<?xml version="1.0" encoding="utf-8"?>
|
||||||
|
<x:xmpmeta x:xmptk="XMP Core 4.4.0-Exiv2" xmlns:x="adobe:ns:meta/">
|
||||||
|
<rdf:RDF xmlns:rdf="http://www.w3.org/1999/02/22-rdf-syntax-ns#">
|
||||||
|
<rdf:Description rdf:about="" xmlns:dc="http://purl.org/dc/elements/1.1/"
|
||||||
|
xmlns:exif="http://ns.adobe.com/exif/1.0/" xmlns:lr="http://ns.adobe.com/lightroom/1.0/"
|
||||||
|
xmlns:xmp="http://ns.adobe.com/xap/1.0/" xmlns:xmpMM="http://ns.adobe.com/xap/1.0/mm/"
|
||||||
|
xmpMM:DerivedFrom="000041900001.jpg">
|
||||||
|
<dc:subject>
|
||||||
|
<rdf:Bag>
|
||||||
|
<rdf:li>st</rdf:li>
|
||||||
|
<rdf:li>cloud</rdf:li>
|
||||||
|
<rdf:li>fly</rdf:li>
|
||||||
|
<rdf:li>sky</rdf:li>
|
||||||
|
</rdf:Bag>
|
||||||
|
</dc:subject>
|
||||||
|
<lr:hierarchicalSubject>
|
||||||
|
<rdf:Bag>
|
||||||
|
<rdf:li>st|cloud</rdf:li>
|
||||||
|
<rdf:li>st|fly</rdf:li>
|
||||||
|
<rdf:li>st|sky</rdf:li>
|
||||||
|
</rdf:Bag>
|
||||||
|
</lr:hierarchicalSubject>
|
||||||
|
</rdf:Description>
|
||||||
|
</rdf:RDF>
|
||||||
|
</x:xmpmeta>
|
||||||
@@ -0,0 +1,7 @@
|
|||||||
|
<?xml version="1.0" encoding="utf-8"?>
|
||||||
|
<x:xmpmeta x:xmptk="XMP Core 4.4.0-Exiv2" xmlns:x="adobe:ns:meta/">
|
||||||
|
<rdf:RDF xmlns:rdf="http://www.w3.org/1999/02/22-rdf-syntax-ns#">
|
||||||
|
<rdf:Description rdf:about="" xmlns:dc="http://purl.org/dc/elements/1.1/" xmlns:exif="http://ns.adobe.com/exif/1.0/" xmlns:lr="http://ns.adobe.com/lightroom/1.0/" xmlns:xmp="http://ns.adobe.com/xap/1.0/" xmlns:xmpMM="http://ns.adobe.com/xap/1.0/mm/" xmpMM:DerivedFrom="000041900001.tif">
|
||||||
|
<dc:subject><rdf:Bag><rdf:li>st</rdf:li><rdf:li>cloud</rdf:li><rdf:li>cloudy</rdf:li><rdf:li>fly</rdf:li><rdf:li>sky</rdf:li></rdf:Bag></dc:subject><lr:hierarchicalSubject><rdf:Bag><rdf:li>st|cloud</rdf:li><rdf:li>st|cloudy</rdf:li><rdf:li>st|fly</rdf:li><rdf:li>st|sky</rdf:li></rdf:Bag></lr:hierarchicalSubject></rdf:Description>
|
||||||
|
</rdf:RDF>
|
||||||
|
</x:xmpmeta>
|
||||||
@@ -97,6 +97,14 @@ body {
|
|||||||
background-color: var(--color6);
|
background-color: var(--color6);
|
||||||
}
|
}
|
||||||
|
|
||||||
|
.tagentry label:hover {
|
||||||
|
background-color: var(--color3);
|
||||||
|
}
|
||||||
|
|
||||||
|
.tagentry .tagtoggle:hover {
|
||||||
|
background-color: var(--color3);
|
||||||
|
}
|
||||||
|
|
||||||
.column img {
|
.column img {
|
||||||
background-color: var(--bcolor2);
|
background-color: var(--bcolor2);
|
||||||
}
|
}
|
||||||
@@ -110,3 +118,23 @@ body {
|
|||||||
color: var(--bcolor1);
|
color: var(--bcolor1);
|
||||||
font-weight: 600;
|
font-weight: 600;
|
||||||
}
|
}
|
||||||
|
|
||||||
|
.loader {
|
||||||
|
width: 48px;
|
||||||
|
height: 48px;
|
||||||
|
border-radius: 50%;
|
||||||
|
display: inline-block;
|
||||||
|
border-top: 3px solid var(--bcolor1);
|
||||||
|
border-right: 3px solid transparent;
|
||||||
|
box-sizing: border-box;
|
||||||
|
animation: rotation 1s linear infinite;
|
||||||
|
}
|
||||||
|
|
||||||
|
@keyframes rotation {
|
||||||
|
0% {
|
||||||
|
transform: rotate(0deg);
|
||||||
|
}
|
||||||
|
100% {
|
||||||
|
transform: rotate(360deg);
|
||||||
|
}
|
||||||
|
}
|
||||||
@@ -96,6 +96,14 @@ body {
|
|||||||
background-color: var(--color3);
|
background-color: var(--color3);
|
||||||
}
|
}
|
||||||
|
|
||||||
|
.tagentry label:hover {
|
||||||
|
background-color: var(--color7);
|
||||||
|
}
|
||||||
|
|
||||||
|
.tagentry .tagtoggle:hover {
|
||||||
|
background-color: var(--color7);
|
||||||
|
}
|
||||||
|
|
||||||
.column img {
|
.column img {
|
||||||
background-color: var(--bcolor1);
|
background-color: var(--bcolor1);
|
||||||
}
|
}
|
||||||
@@ -109,3 +117,23 @@ body {
|
|||||||
color: var(--bcolor2);
|
color: var(--bcolor2);
|
||||||
font-weight: 600;
|
font-weight: 600;
|
||||||
}
|
}
|
||||||
|
|
||||||
|
.loader {
|
||||||
|
width: 48px;
|
||||||
|
height: 48px;
|
||||||
|
border-radius: 50%;
|
||||||
|
display: inline-block;
|
||||||
|
border-top: 3px solid var(--bcolor2);
|
||||||
|
border-right: 3px solid transparent;
|
||||||
|
box-sizing: border-box;
|
||||||
|
animation: rotation 1s linear infinite;
|
||||||
|
}
|
||||||
|
|
||||||
|
@keyframes rotation {
|
||||||
|
0% {
|
||||||
|
transform: rotate(0deg);
|
||||||
|
}
|
||||||
|
100% {
|
||||||
|
transform: rotate(360deg);
|
||||||
|
}
|
||||||
|
}
|
||||||
@@ -74,6 +74,14 @@
|
|||||||
background-color: var(--bcolor1);
|
background-color: var(--bcolor1);
|
||||||
}
|
}
|
||||||
|
|
||||||
|
.tagentry label:hover {
|
||||||
|
background-color: var(--bcolor3);
|
||||||
|
}
|
||||||
|
|
||||||
|
.tagentry .tagtoggle:hover {
|
||||||
|
background-color: var(--bcolor3);
|
||||||
|
}
|
||||||
|
|
||||||
.column img {
|
.column img {
|
||||||
background-color: var(--bcolor1);
|
background-color: var(--bcolor1);
|
||||||
}
|
}
|
||||||
@@ -88,6 +96,52 @@
|
|||||||
font-weight: 600;
|
font-weight: 600;
|
||||||
}
|
}
|
||||||
|
|
||||||
|
.loader {
|
||||||
|
width: 48px;
|
||||||
|
height: 48px;
|
||||||
|
border-radius: 50%;
|
||||||
|
position: relative;
|
||||||
|
animation: rotate 1s linear infinite
|
||||||
|
}
|
||||||
|
|
||||||
|
.loader::before {
|
||||||
|
content: "";
|
||||||
|
box-sizing: border-box;
|
||||||
|
position: absolute;
|
||||||
|
inset: 0px;
|
||||||
|
border-radius: 50%;
|
||||||
|
border: 5px solid var(--bcolor2);
|
||||||
|
animation: prixClipFix 2s linear infinite;
|
||||||
|
}
|
||||||
|
|
||||||
|
@keyframes rotate {
|
||||||
|
100% {
|
||||||
|
transform: rotate(360deg)
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
@keyframes prixClipFix {
|
||||||
|
0% {
|
||||||
|
clip-path: polygon(50% 50%, 0 0, 0 0, 0 0, 0 0, 0 0)
|
||||||
|
}
|
||||||
|
|
||||||
|
25% {
|
||||||
|
clip-path: polygon(50% 50%, 0 0, 100% 0, 100% 0, 100% 0, 100% 0)
|
||||||
|
}
|
||||||
|
|
||||||
|
50% {
|
||||||
|
clip-path: polygon(50% 50%, 0 0, 100% 0, 100% 100%, 100% 100%, 100% 100%)
|
||||||
|
}
|
||||||
|
|
||||||
|
75% {
|
||||||
|
clip-path: polygon(50% 50%, 0 0, 100% 0, 100% 100%, 0 100%, 0 100%)
|
||||||
|
}
|
||||||
|
|
||||||
|
100% {
|
||||||
|
clip-path: polygon(50% 50%, 0 0, 100% 0, 100% 100%, 0 100%, 0 0)
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
body {
|
body {
|
||||||
color: var(--bcolor2);
|
color: var(--bcolor2);
|
||||||
background-color: var(--bcolor3);
|
background-color: var(--bcolor3);
|
||||||
|
|||||||
@@ -74,6 +74,14 @@
|
|||||||
background-color: var(--bcolor1);
|
background-color: var(--bcolor1);
|
||||||
}
|
}
|
||||||
|
|
||||||
|
.tagentry label:hover {
|
||||||
|
background-color: var(--bcolor3);
|
||||||
|
}
|
||||||
|
|
||||||
|
.tagentry .tagtoggle:hover {
|
||||||
|
background-color: var(--bcolor3);
|
||||||
|
}
|
||||||
|
|
||||||
.column img {
|
.column img {
|
||||||
background-color: var(--bcolor1);
|
background-color: var(--bcolor1);
|
||||||
}
|
}
|
||||||
@@ -88,6 +96,52 @@
|
|||||||
font-weight: 600;
|
font-weight: 600;
|
||||||
}
|
}
|
||||||
|
|
||||||
|
.loader {
|
||||||
|
width: 48px;
|
||||||
|
height: 48px;
|
||||||
|
border-radius: 50%;
|
||||||
|
position: relative;
|
||||||
|
animation: rotate 1s linear infinite
|
||||||
|
}
|
||||||
|
|
||||||
|
.loader::before {
|
||||||
|
content: "";
|
||||||
|
box-sizing: border-box;
|
||||||
|
position: absolute;
|
||||||
|
inset: 0px;
|
||||||
|
border-radius: 50%;
|
||||||
|
border: 5px solid var(--bcolor2);
|
||||||
|
animation: prixClipFix 2s linear infinite;
|
||||||
|
}
|
||||||
|
|
||||||
|
@keyframes rotate {
|
||||||
|
100% {
|
||||||
|
transform: rotate(360deg)
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
@keyframes prixClipFix {
|
||||||
|
0% {
|
||||||
|
clip-path: polygon(50% 50%, 0 0, 0 0, 0 0, 0 0, 0 0)
|
||||||
|
}
|
||||||
|
|
||||||
|
25% {
|
||||||
|
clip-path: polygon(50% 50%, 0 0, 100% 0, 100% 0, 100% 0, 100% 0)
|
||||||
|
}
|
||||||
|
|
||||||
|
50% {
|
||||||
|
clip-path: polygon(50% 50%, 0 0, 100% 0, 100% 100%, 100% 100%, 100% 100%)
|
||||||
|
}
|
||||||
|
|
||||||
|
75% {
|
||||||
|
clip-path: polygon(50% 50%, 0 0, 100% 0, 100% 100%, 0 100%, 0 100%)
|
||||||
|
}
|
||||||
|
|
||||||
|
100% {
|
||||||
|
clip-path: polygon(50% 50%, 0 0, 100% 0, 100% 100%, 0 100%, 0 0)
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
body {
|
body {
|
||||||
color: var(--bcolor2);
|
color: var(--bcolor2);
|
||||||
background-color: var(--bcolor3);
|
background-color: var(--bcolor3);
|
||||||
|
|||||||
@@ -1,14 +1,22 @@
|
|||||||
@import url("https://fonts.googleapis.com/css2?family=Playfair+Display:wght@400;500;600;700;800;900&display=swap");
|
@import url("https://fonts.googleapis.com/css2?family=Playfair+Display:wght@400;500;600;700;800;900&display=swap");
|
||||||
|
|
||||||
* {
|
* {
|
||||||
--color1: #FF7F50; /* Coral */
|
--color1: #FF7F50;
|
||||||
--color2: #D2691E; /* Chocolate */
|
/* Coral */
|
||||||
--color3: #8B4513; /* SaddleBrown */
|
--color2: #D2691E;
|
||||||
--color4: #FFA07A; /* LightSalmon */
|
/* Chocolate */
|
||||||
--bcolor1: #FFF8DC; /* Cornsilk */
|
--color3: #8B4513;
|
||||||
--bcolor2: #2F4F4F; /* DarkSlateGray */
|
/* SaddleBrown */
|
||||||
--bcolor3: #3E2723; /* Darker brown */
|
--color4: #FFA07A;
|
||||||
--bcolor4: #4E342E; /* Slightly lighter brown */
|
/* LightSalmon */
|
||||||
|
--bcolor1: #FFF8DC;
|
||||||
|
/* Cornsilk */
|
||||||
|
--bcolor2: #2F4F4F;
|
||||||
|
/* DarkSlateGray */
|
||||||
|
--bcolor3: #3E2723;
|
||||||
|
/* Darker brown */
|
||||||
|
--bcolor4: #4E342E;
|
||||||
|
/* Slightly lighter brown */
|
||||||
}
|
}
|
||||||
|
|
||||||
.navbar {
|
.navbar {
|
||||||
@@ -79,6 +87,14 @@
|
|||||||
font-family: "Playfair Display", serif;
|
font-family: "Playfair Display", serif;
|
||||||
}
|
}
|
||||||
|
|
||||||
|
.tagentry label:hover {
|
||||||
|
background-color: var(--color3);
|
||||||
|
}
|
||||||
|
|
||||||
|
.tagentry .tagtoggle:hover {
|
||||||
|
background-color: var(--color3);
|
||||||
|
}
|
||||||
|
|
||||||
.column img {
|
.column img {
|
||||||
background-color: var(--bcolor4);
|
background-color: var(--bcolor4);
|
||||||
}
|
}
|
||||||
@@ -94,6 +110,54 @@
|
|||||||
font-family: "Playfair Display", serif;
|
font-family: "Playfair Display", serif;
|
||||||
}
|
}
|
||||||
|
|
||||||
|
.loader,
|
||||||
|
.loader:before,
|
||||||
|
.loader:after {
|
||||||
|
border-radius: 50%;
|
||||||
|
width: 2.5em;
|
||||||
|
height: 2.5em;
|
||||||
|
animation-fill-mode: both;
|
||||||
|
animation: bblFadInOut 1.8s infinite ease-in-out;
|
||||||
|
}
|
||||||
|
|
||||||
|
.loader {
|
||||||
|
color: var(--bcolor2);
|
||||||
|
font-size: 7px;
|
||||||
|
position: relative;
|
||||||
|
text-indent: -9999em;
|
||||||
|
transform: translateZ(0);
|
||||||
|
animation-delay: -0.16s;
|
||||||
|
}
|
||||||
|
|
||||||
|
.loader:before,
|
||||||
|
.loader:after {
|
||||||
|
content: '';
|
||||||
|
position: absolute;
|
||||||
|
top: 0;
|
||||||
|
}
|
||||||
|
|
||||||
|
.loader:before {
|
||||||
|
left: -3.5em;
|
||||||
|
animation-delay: -0.32s;
|
||||||
|
}
|
||||||
|
|
||||||
|
.loader:after {
|
||||||
|
left: 3.5em;
|
||||||
|
}
|
||||||
|
|
||||||
|
@keyframes bblFadInOut {
|
||||||
|
|
||||||
|
0%,
|
||||||
|
80%,
|
||||||
|
100% {
|
||||||
|
box-shadow: 0 2.5em 0 -1.3em
|
||||||
|
}
|
||||||
|
|
||||||
|
40% {
|
||||||
|
box-shadow: 0 2.5em 0 0
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
body {
|
body {
|
||||||
color: var(--bcolor1);
|
color: var(--bcolor1);
|
||||||
background-color: var(--bcolor3);
|
background-color: var(--bcolor3);
|
||||||
|
|||||||
@@ -5,22 +5,22 @@
|
|||||||
--color2: #b22222;
|
--color2: #b22222;
|
||||||
--color3: #ff4500;
|
--color3: #ff4500;
|
||||||
--color4: #6e0000;
|
--color4: #6e0000;
|
||||||
--bcolor1: #ebebeb;
|
--bcolor1: #171717;
|
||||||
--bcolor2: #191919;
|
--bcolor2: #191919;
|
||||||
--bcolor3: #171717;
|
--bcolor3: #ebebeb;
|
||||||
--bcolor4: #0a0a0a;
|
--bcolor4: #0a0a0a;
|
||||||
}
|
}
|
||||||
|
|
||||||
.navbar {
|
.navbar {
|
||||||
font-weight: bold;
|
font-weight: bold;
|
||||||
color: var(--bcolor1);
|
color: var(--bcolor3);
|
||||||
background-color: var(--color1);
|
background-color: var(--color1);
|
||||||
font-weight: 900;
|
font-weight: 900;
|
||||||
}
|
}
|
||||||
|
|
||||||
.navbar li a {
|
.navbar li a {
|
||||||
font-weight: 800;
|
font-weight: 800;
|
||||||
color: var(--bcolor1);
|
color: var(--bcolor3);
|
||||||
}
|
}
|
||||||
|
|
||||||
/* Change the link color on hover */
|
/* Change the link color on hover */
|
||||||
@@ -30,7 +30,7 @@
|
|||||||
}
|
}
|
||||||
|
|
||||||
.footer {
|
.footer {
|
||||||
color: var(--bcolor1);
|
color: var(--bcolor3);
|
||||||
background-color: var(--color3);
|
background-color: var(--color3);
|
||||||
font-weight: 700;
|
font-weight: 700;
|
||||||
}
|
}
|
||||||
@@ -73,6 +73,14 @@
|
|||||||
background-color: var(--bcolor2);
|
background-color: var(--bcolor2);
|
||||||
}
|
}
|
||||||
|
|
||||||
|
.tagentry label:hover {
|
||||||
|
background-color: var(--bcolor4);
|
||||||
|
}
|
||||||
|
|
||||||
|
.tagentry .tagtoggle:hover {
|
||||||
|
background-color: var(--bcolor4);
|
||||||
|
}
|
||||||
|
|
||||||
.column img {
|
.column img {
|
||||||
background-color: var(--bcolor4);
|
background-color: var(--bcolor4);
|
||||||
}
|
}
|
||||||
@@ -87,6 +95,36 @@
|
|||||||
font-weight: 800;
|
font-weight: 800;
|
||||||
}
|
}
|
||||||
|
|
||||||
|
.loader {
|
||||||
|
width: 12px;
|
||||||
|
height: 12px;
|
||||||
|
border-radius: 50%;
|
||||||
|
display: block;
|
||||||
|
margin: 15px auto;
|
||||||
|
position: relative;
|
||||||
|
color: var(--color1);
|
||||||
|
box-sizing: border-box;
|
||||||
|
animation: animloader 1s linear infinite alternate;
|
||||||
|
}
|
||||||
|
|
||||||
|
@keyframes animloader {
|
||||||
|
0% {
|
||||||
|
box-shadow: -38px -12px, -14px 0, 14px 0, 38px 0;
|
||||||
|
}
|
||||||
|
|
||||||
|
33% {
|
||||||
|
box-shadow: -38px 0px, -14px -12px, 14px 0, 38px 0;
|
||||||
|
}
|
||||||
|
|
||||||
|
66% {
|
||||||
|
box-shadow: -38px 0px, -14px 0, 14px -12px, 38px 0;
|
||||||
|
}
|
||||||
|
|
||||||
|
100% {
|
||||||
|
box-shadow: -38px 0, -14px 0, 14px 0, 38px -12px;
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
body {
|
body {
|
||||||
color: var(--bcolor1);
|
color: var(--bcolor1);
|
||||||
background-color: var(--bcolor3);
|
background-color: var(--bcolor3);
|
||||||
|
|||||||
@@ -79,6 +79,14 @@
|
|||||||
font-family: "Nunito", sans-serif;
|
font-family: "Nunito", sans-serif;
|
||||||
}
|
}
|
||||||
|
|
||||||
|
.tagentry label:hover {
|
||||||
|
background-color: var(--color4);
|
||||||
|
}
|
||||||
|
|
||||||
|
.tagentry .tagtoggle:hover {
|
||||||
|
background-color: var(--color4);
|
||||||
|
}
|
||||||
|
|
||||||
.column img {
|
.column img {
|
||||||
background-color: var(--bcolor4);
|
background-color: var(--bcolor4);
|
||||||
}
|
}
|
||||||
@@ -95,6 +103,91 @@
|
|||||||
font-family: "Nunito", sans-serif;
|
font-family: "Nunito", sans-serif;
|
||||||
}
|
}
|
||||||
|
|
||||||
|
.loader {
|
||||||
|
transform: rotateZ(45deg);
|
||||||
|
perspective: 1000px;
|
||||||
|
border-radius: 50%;
|
||||||
|
width: 48px;
|
||||||
|
height: 48px;
|
||||||
|
color: var(--color1);
|
||||||
|
}
|
||||||
|
|
||||||
|
.loader:before,
|
||||||
|
.loader:after {
|
||||||
|
content: '';
|
||||||
|
display: block;
|
||||||
|
position: absolute;
|
||||||
|
top: 0;
|
||||||
|
left: 0;
|
||||||
|
width: inherit;
|
||||||
|
height: inherit;
|
||||||
|
border-radius: 50%;
|
||||||
|
transform: rotateX(70deg);
|
||||||
|
animation: 1s spin linear infinite;
|
||||||
|
}
|
||||||
|
|
||||||
|
.loader:after {
|
||||||
|
color: var(--bcolor4);
|
||||||
|
transform: rotateY(70deg);
|
||||||
|
animation-delay: .4s;
|
||||||
|
}
|
||||||
|
|
||||||
|
@keyframes rotate {
|
||||||
|
0% {
|
||||||
|
transform: translate(-50%, -50%) rotateZ(0deg);
|
||||||
|
}
|
||||||
|
|
||||||
|
100% {
|
||||||
|
transform: translate(-50%, -50%) rotateZ(360deg);
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
@keyframes rotateccw {
|
||||||
|
0% {
|
||||||
|
transform: translate(-50%, -50%) rotate(0deg);
|
||||||
|
}
|
||||||
|
|
||||||
|
100% {
|
||||||
|
transform: translate(-50%, -50%) rotate(-360deg);
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
@keyframes spin {
|
||||||
|
|
||||||
|
0%,
|
||||||
|
100% {
|
||||||
|
box-shadow: .2em 0px 0 0px currentcolor;
|
||||||
|
}
|
||||||
|
|
||||||
|
12% {
|
||||||
|
box-shadow: .2em .2em 0 0 currentcolor;
|
||||||
|
}
|
||||||
|
|
||||||
|
25% {
|
||||||
|
box-shadow: 0 .2em 0 0px currentcolor;
|
||||||
|
}
|
||||||
|
|
||||||
|
37% {
|
||||||
|
box-shadow: -.2em .2em 0 0 currentcolor;
|
||||||
|
}
|
||||||
|
|
||||||
|
50% {
|
||||||
|
box-shadow: -.2em 0 0 0 currentcolor;
|
||||||
|
}
|
||||||
|
|
||||||
|
62% {
|
||||||
|
box-shadow: -.2em -.2em 0 0 currentcolor;
|
||||||
|
}
|
||||||
|
|
||||||
|
75% {
|
||||||
|
box-shadow: 0px -.2em 0 0 currentcolor;
|
||||||
|
}
|
||||||
|
|
||||||
|
87% {
|
||||||
|
box-shadow: .2em -.2em 0 0 currentcolor;
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
body {
|
body {
|
||||||
color: var(--bcolor2);
|
color: var(--bcolor2);
|
||||||
background-color: var(--bcolor3);
|
background-color: var(--bcolor3);
|
||||||
|
|||||||
@@ -5,22 +5,22 @@
|
|||||||
--color2: #1346a4;
|
--color2: #1346a4;
|
||||||
--color3: #0e3377;
|
--color3: #0e3377;
|
||||||
--color4: #3674e7;
|
--color4: #3674e7;
|
||||||
--bcolor1: #ebebeb;
|
--bcolor1: #171717;
|
||||||
--bcolor2: #191919;
|
--bcolor2: #191919;
|
||||||
--bcolor3: #171717;
|
--bcolor3: #ebebeb;
|
||||||
--bcolor4: #0a0a0a;
|
--bcolor4: #0a0a0a;
|
||||||
}
|
}
|
||||||
|
|
||||||
.navbar {
|
.navbar {
|
||||||
font-weight: bold;
|
font-weight: bold;
|
||||||
color: var(--bcolor1);
|
color: var(--bcolor3);
|
||||||
background-color: var(--color1);
|
background-color: var(--color1);
|
||||||
font-weight: 900;
|
font-weight: 900;
|
||||||
}
|
}
|
||||||
|
|
||||||
.navbar li a {
|
.navbar li a {
|
||||||
font-weight: 800;
|
font-weight: 800;
|
||||||
color: var(--bcolor1);
|
color: var(--bcolor3);
|
||||||
}
|
}
|
||||||
|
|
||||||
/* Change the link color on hover */
|
/* Change the link color on hover */
|
||||||
@@ -30,7 +30,7 @@
|
|||||||
}
|
}
|
||||||
|
|
||||||
.footer {
|
.footer {
|
||||||
color: var(--bcolor1);
|
color: var(--bcolor3);
|
||||||
background-color: var(--color3);
|
background-color: var(--color3);
|
||||||
font-weight: 700;
|
font-weight: 700;
|
||||||
}
|
}
|
||||||
@@ -73,6 +73,14 @@
|
|||||||
background-color: var(--bcolor2);
|
background-color: var(--bcolor2);
|
||||||
}
|
}
|
||||||
|
|
||||||
|
.tagentry label:hover {
|
||||||
|
background-color: var(--bcolor4);
|
||||||
|
}
|
||||||
|
|
||||||
|
.tagentry .tagtoggle:hover {
|
||||||
|
background-color: var(--bcolor4);
|
||||||
|
}
|
||||||
|
|
||||||
.column img {
|
.column img {
|
||||||
background-color: var(--bcolor4);
|
background-color: var(--bcolor4);
|
||||||
}
|
}
|
||||||
@@ -87,6 +95,36 @@
|
|||||||
font-weight: 800;
|
font-weight: 800;
|
||||||
}
|
}
|
||||||
|
|
||||||
|
.loader {
|
||||||
|
width: 12px;
|
||||||
|
height: 12px;
|
||||||
|
border-radius: 50%;
|
||||||
|
display: block;
|
||||||
|
margin: 15px auto;
|
||||||
|
position: relative;
|
||||||
|
color: var(--color1);
|
||||||
|
box-sizing: border-box;
|
||||||
|
animation: animloader 1s linear infinite alternate;
|
||||||
|
}
|
||||||
|
|
||||||
|
@keyframes animloader {
|
||||||
|
0% {
|
||||||
|
box-shadow: -38px -12px, -14px 0, 14px 0, 38px 0;
|
||||||
|
}
|
||||||
|
|
||||||
|
33% {
|
||||||
|
box-shadow: -38px 0px, -14px -12px, 14px 0, 38px 0;
|
||||||
|
}
|
||||||
|
|
||||||
|
66% {
|
||||||
|
box-shadow: -38px 0px, -14px 0, 14px -12px, 38px 0;
|
||||||
|
}
|
||||||
|
|
||||||
|
100% {
|
||||||
|
box-shadow: -38px 0, -14px 0, 14px 0, 38px -12px;
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
body {
|
body {
|
||||||
color: var(--bcolor1);
|
color: var(--bcolor1);
|
||||||
background-color: var(--bcolor3);
|
background-color: var(--bcolor3);
|
||||||
|
|||||||
@@ -52,6 +52,14 @@
|
|||||||
background-color: var(--color3);
|
background-color: var(--color3);
|
||||||
}
|
}
|
||||||
|
|
||||||
|
.tagentry label:hover {
|
||||||
|
background-color: var(--color4);
|
||||||
|
}
|
||||||
|
|
||||||
|
.tagentry .tagtoggle:hover {
|
||||||
|
background-color: var(--color4);
|
||||||
|
}
|
||||||
|
|
||||||
.column img {
|
.column img {
|
||||||
background-color: var(--bcolor2);
|
background-color: var(--bcolor2);
|
||||||
}
|
}
|
||||||
@@ -66,6 +74,27 @@
|
|||||||
font-weight: 800;
|
font-weight: 800;
|
||||||
}
|
}
|
||||||
|
|
||||||
|
.loader {
|
||||||
|
width: 48px;
|
||||||
|
height: 48px;
|
||||||
|
border-radius: 50%;
|
||||||
|
display: inline-block;
|
||||||
|
border-top: 3px solid var(--bcolor1);
|
||||||
|
border-right: 3px solid transparent;
|
||||||
|
box-sizing: border-box;
|
||||||
|
animation: rotation 1s linear infinite;
|
||||||
|
}
|
||||||
|
|
||||||
|
@keyframes rotation {
|
||||||
|
0% {
|
||||||
|
transform: rotate(0deg);
|
||||||
|
}
|
||||||
|
|
||||||
|
100% {
|
||||||
|
transform: rotate(360deg);
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
body {
|
body {
|
||||||
color: var(--bcolor1);
|
color: var(--bcolor1);
|
||||||
background-color: var(--color4);
|
background-color: var(--color4);
|
||||||
|
|||||||
@@ -52,6 +52,14 @@
|
|||||||
background-color: var(--color2);
|
background-color: var(--color2);
|
||||||
}
|
}
|
||||||
|
|
||||||
|
.tagentry label:hover {
|
||||||
|
background-color: var(--color4);
|
||||||
|
}
|
||||||
|
|
||||||
|
.tagentry .tagtoggle:hover {
|
||||||
|
background-color: var(--color4);
|
||||||
|
}
|
||||||
|
|
||||||
.column img {
|
.column img {
|
||||||
background-color: var(--color2);
|
background-color: var(--color2);
|
||||||
}
|
}
|
||||||
@@ -66,6 +74,27 @@
|
|||||||
font-weight: 800;
|
font-weight: 800;
|
||||||
}
|
}
|
||||||
|
|
||||||
|
.loader {
|
||||||
|
width: 48px;
|
||||||
|
height: 48px;
|
||||||
|
border-radius: 50%;
|
||||||
|
display: inline-block;
|
||||||
|
border-top: 3px solid var(--bcolor2);
|
||||||
|
border-right: 3px solid transparent;
|
||||||
|
box-sizing: border-box;
|
||||||
|
animation: rotation 1s linear infinite;
|
||||||
|
}
|
||||||
|
|
||||||
|
@keyframes rotation {
|
||||||
|
0% {
|
||||||
|
transform: rotate(0deg);
|
||||||
|
}
|
||||||
|
|
||||||
|
100% {
|
||||||
|
transform: rotate(360deg);
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
body {
|
body {
|
||||||
color: var(--bcolor2);
|
color: var(--bcolor2);
|
||||||
background-color: var(--bcolor1);
|
background-color: var(--bcolor1);
|
||||||
|
|||||||
@@ -1,6 +1,6 @@
|
|||||||
<svg width='64' height='64' viewBox='0 0 24 24' fill='none' xmlns='http://www.w3.org/2000/svg'>
|
<svg width='64' height='64' viewBox='0 0 24 24' fill='none' xmlns='http://www.w3.org/2000/svg'>
|
||||||
<path d='M20 6H10L8 4H4C2.89543 4 2 4.89543 2 6V18C2 19.1046 2.89543 20 4 20H20C21.1046 20 22 19.1046 22 18V8C22 6.89543 21.1046 6 20 6Z' fill='{{ color1 }}' />
|
<path d='M20 6H10L8 4H4C2.89543 4 2 4.89543 2 6V18C2 19.1046 2.89543 20 4 20H20C21.1046 20 22 19.1046 22 18V8C22 6.89543 21.1046 6 20 6Z' fill='{{ color1 }}' />
|
||||||
<path d='M4 4H8L10 6H20C21.1046 6 22 6.89543 22 8H2C2 6.89543 2.89543 6 4 6V4Z' fill='{{ color4 }}' />
|
<path d='M4 4H8L10 6H20C21.1046 6 22 6.89543 22 8H2C2 6.89543 2.89543 6 4 6V4Z' fill='{{ color4 }}' />
|
||||||
<path d='M10 6H14L12 4H8L10 6Z' fill='{{ color2 }}' />
|
<path d='M10 6H14L12 4H8L10 6Z' fill='{{ color3 }}' />
|
||||||
<path d='M14 6H18L16 4H12L14 6Z' fill='{{ color3 }}' />
|
<path d='M14 6H18L16 4H12L14 6Z' fill='{{ color2 }}' />
|
||||||
</svg>
|
</svg>
|
||||||
|
Before Width: | Height: | Size: 493 B After Width: | Height: | Size: 493 B |
@@ -5,22 +5,22 @@
|
|||||||
--color2: #008000;
|
--color2: #008000;
|
||||||
--color3: #32cd32;
|
--color3: #32cd32;
|
||||||
--color4: #004300;
|
--color4: #004300;
|
||||||
--bcolor1: #ebebeb;
|
--bcolor1: #171717;
|
||||||
--bcolor2: #191919;
|
--bcolor2: #191919;
|
||||||
--bcolor3: #171717;
|
--bcolor3: #ebebeb;
|
||||||
--bcolor4: #0a0a0a;
|
--bcolor4: #0a0a0a;
|
||||||
}
|
}
|
||||||
|
|
||||||
.navbar {
|
.navbar {
|
||||||
font-weight: bold;
|
font-weight: bold;
|
||||||
color: var(--bcolor1);
|
color: var(--bcolor3);
|
||||||
background-color: var(--color1);
|
background-color: var(--color1);
|
||||||
font-weight: 900;
|
font-weight: 900;
|
||||||
}
|
}
|
||||||
|
|
||||||
.navbar li a {
|
.navbar li a {
|
||||||
font-weight: 800;
|
font-weight: 800;
|
||||||
color: var(--bcolor1);
|
color: var(--bcolor3);
|
||||||
}
|
}
|
||||||
|
|
||||||
/* Change the link color on hover */
|
/* Change the link color on hover */
|
||||||
@@ -30,7 +30,7 @@
|
|||||||
}
|
}
|
||||||
|
|
||||||
.footer {
|
.footer {
|
||||||
color: var(--bcolor1);
|
color: var(--bcolor3);
|
||||||
background-color: var(--color3);
|
background-color: var(--color3);
|
||||||
font-weight: 700;
|
font-weight: 700;
|
||||||
}
|
}
|
||||||
@@ -73,6 +73,14 @@
|
|||||||
background-color: var(--bcolor2);
|
background-color: var(--bcolor2);
|
||||||
}
|
}
|
||||||
|
|
||||||
|
.tagentry label:hover {
|
||||||
|
background-color: var(--bcolor4);
|
||||||
|
}
|
||||||
|
|
||||||
|
.tagentry .tagtoggle:hover {
|
||||||
|
background-color: var(--bcolor4);
|
||||||
|
}
|
||||||
|
|
||||||
.column img {
|
.column img {
|
||||||
background-color: var(--bcolor4);
|
background-color: var(--bcolor4);
|
||||||
}
|
}
|
||||||
@@ -85,6 +93,36 @@
|
|||||||
background-color: var(--color1);
|
background-color: var(--color1);
|
||||||
}
|
}
|
||||||
|
|
||||||
|
.loader {
|
||||||
|
width: 12px;
|
||||||
|
height: 12px;
|
||||||
|
border-radius: 50%;
|
||||||
|
display: block;
|
||||||
|
margin: 15px auto;
|
||||||
|
position: relative;
|
||||||
|
color: var(--color1);
|
||||||
|
box-sizing: border-box;
|
||||||
|
animation: animloader 1s linear infinite alternate;
|
||||||
|
}
|
||||||
|
|
||||||
|
@keyframes animloader {
|
||||||
|
0% {
|
||||||
|
box-shadow: -38px -12px, -14px 0, 14px 0, 38px 0;
|
||||||
|
}
|
||||||
|
|
||||||
|
33% {
|
||||||
|
box-shadow: -38px 0px, -14px -12px, 14px 0, 38px 0;
|
||||||
|
}
|
||||||
|
|
||||||
|
66% {
|
||||||
|
box-shadow: -38px 0px, -14px 0, 14px -12px, 38px 0;
|
||||||
|
}
|
||||||
|
|
||||||
|
100% {
|
||||||
|
box-shadow: -38px 0, -14px 0, 14px 0, 38px -12px;
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
body {
|
body {
|
||||||
color: var(--bcolor1);
|
color: var(--bcolor1);
|
||||||
background-color: var(--bcolor3);
|
background-color: var(--bcolor3);
|
||||||
|
|||||||
@@ -72,6 +72,14 @@
|
|||||||
background-color: var(--color3);
|
background-color: var(--color3);
|
||||||
}
|
}
|
||||||
|
|
||||||
|
.tagentry label:hover {
|
||||||
|
background-color: var(--color2);
|
||||||
|
}
|
||||||
|
|
||||||
|
.tagentry .tagtoggle:hover {
|
||||||
|
background-color: var(--color2);
|
||||||
|
}
|
||||||
|
|
||||||
.column img {
|
.column img {
|
||||||
background-color: var(--bcolor3);
|
background-color: var(--bcolor3);
|
||||||
}
|
}
|
||||||
@@ -86,6 +94,36 @@
|
|||||||
font-weight: 800;
|
font-weight: 800;
|
||||||
}
|
}
|
||||||
|
|
||||||
|
.loader {
|
||||||
|
width: 12px;
|
||||||
|
height: 12px;
|
||||||
|
border-radius: 50%;
|
||||||
|
display: block;
|
||||||
|
margin: 15px auto;
|
||||||
|
position: relative;
|
||||||
|
color: var(--bcolor1);
|
||||||
|
box-sizing: border-box;
|
||||||
|
animation: animloader 1s linear infinite alternate;
|
||||||
|
}
|
||||||
|
|
||||||
|
@keyframes animloader {
|
||||||
|
0% {
|
||||||
|
box-shadow: -38px -12px, -14px 0, 14px 0, 38px 0;
|
||||||
|
}
|
||||||
|
|
||||||
|
33% {
|
||||||
|
box-shadow: -38px 0px, -14px -12px, 14px 0, 38px 0;
|
||||||
|
}
|
||||||
|
|
||||||
|
66% {
|
||||||
|
box-shadow: -38px 0px, -14px 0, 14px -12px, 38px 0;
|
||||||
|
}
|
||||||
|
|
||||||
|
100% {
|
||||||
|
box-shadow: -38px 0, -14px 0, 14px 0, 38px -12px;
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
body {
|
body {
|
||||||
color: var(--bcolor1);
|
color: var(--bcolor1);
|
||||||
background-color: var(--bcolor2);
|
background-color: var(--bcolor2);
|
||||||
|
|||||||
@@ -55,6 +55,14 @@
|
|||||||
background-color: var(--bcolor2);
|
background-color: var(--bcolor2);
|
||||||
}
|
}
|
||||||
|
|
||||||
|
.tagentry label:hover {
|
||||||
|
background-color: var(--bcolor3);
|
||||||
|
}
|
||||||
|
|
||||||
|
.tagentry .tagtoggle:hover {
|
||||||
|
background-color: var(--bcolor3);
|
||||||
|
}
|
||||||
|
|
||||||
.column img {
|
.column img {
|
||||||
background-color: var(--bcolor5);
|
background-color: var(--bcolor5);
|
||||||
}
|
}
|
||||||
@@ -69,6 +77,91 @@
|
|||||||
font-weight: 800;
|
font-weight: 800;
|
||||||
}
|
}
|
||||||
|
|
||||||
|
.loader {
|
||||||
|
transform: rotateZ(45deg);
|
||||||
|
perspective: 1000px;
|
||||||
|
border-radius: 50%;
|
||||||
|
width: 48px;
|
||||||
|
height: 48px;
|
||||||
|
color: var(--color2);
|
||||||
|
}
|
||||||
|
|
||||||
|
.loader:before,
|
||||||
|
.loader:after {
|
||||||
|
content: '';
|
||||||
|
display: block;
|
||||||
|
position: absolute;
|
||||||
|
top: 0;
|
||||||
|
left: 0;
|
||||||
|
width: inherit;
|
||||||
|
height: inherit;
|
||||||
|
border-radius: 50%;
|
||||||
|
transform: rotateX(70deg);
|
||||||
|
animation: 1s spin linear infinite;
|
||||||
|
}
|
||||||
|
|
||||||
|
.loader:after {
|
||||||
|
color: var(--color4);
|
||||||
|
transform: rotateY(70deg);
|
||||||
|
animation-delay: .4s;
|
||||||
|
}
|
||||||
|
|
||||||
|
@keyframes rotate {
|
||||||
|
0% {
|
||||||
|
transform: translate(-50%, -50%) rotateZ(0deg);
|
||||||
|
}
|
||||||
|
|
||||||
|
100% {
|
||||||
|
transform: translate(-50%, -50%) rotateZ(360deg);
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
@keyframes rotateccw {
|
||||||
|
0% {
|
||||||
|
transform: translate(-50%, -50%) rotate(0deg);
|
||||||
|
}
|
||||||
|
|
||||||
|
100% {
|
||||||
|
transform: translate(-50%, -50%) rotate(-360deg);
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
@keyframes spin {
|
||||||
|
|
||||||
|
0%,
|
||||||
|
100% {
|
||||||
|
box-shadow: .2em 0px 0 0px currentcolor;
|
||||||
|
}
|
||||||
|
|
||||||
|
12% {
|
||||||
|
box-shadow: .2em .2em 0 0 currentcolor;
|
||||||
|
}
|
||||||
|
|
||||||
|
25% {
|
||||||
|
box-shadow: 0 .2em 0 0px currentcolor;
|
||||||
|
}
|
||||||
|
|
||||||
|
37% {
|
||||||
|
box-shadow: -.2em .2em 0 0 currentcolor;
|
||||||
|
}
|
||||||
|
|
||||||
|
50% {
|
||||||
|
box-shadow: -.2em 0 0 0 currentcolor;
|
||||||
|
}
|
||||||
|
|
||||||
|
62% {
|
||||||
|
box-shadow: -.2em -.2em 0 0 currentcolor;
|
||||||
|
}
|
||||||
|
|
||||||
|
75% {
|
||||||
|
box-shadow: 0px -.2em 0 0 currentcolor;
|
||||||
|
}
|
||||||
|
|
||||||
|
87% {
|
||||||
|
box-shadow: .2em -.2em 0 0 currentcolor;
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
body {
|
body {
|
||||||
color: var(--bcolor1);
|
color: var(--bcolor1);
|
||||||
background-color: var(--bcolor4);
|
background-color: var(--bcolor4);
|
||||||
|
|||||||
@@ -76,6 +76,14 @@
|
|||||||
background-color: var(--bcolor2);
|
background-color: var(--bcolor2);
|
||||||
}
|
}
|
||||||
|
|
||||||
|
.tagentry label:hover {
|
||||||
|
background-color: var(--bcolor4);
|
||||||
|
}
|
||||||
|
|
||||||
|
.tagentry .tagtoggle:hover {
|
||||||
|
background-color: var(--bcolor4);
|
||||||
|
}
|
||||||
|
|
||||||
.column img {
|
.column img {
|
||||||
background-color: var(--bcolor5);
|
background-color: var(--bcolor5);
|
||||||
}
|
}
|
||||||
@@ -90,6 +98,36 @@
|
|||||||
font-weight: 800;
|
font-weight: 800;
|
||||||
}
|
}
|
||||||
|
|
||||||
|
.loader {
|
||||||
|
width: 12px;
|
||||||
|
height: 12px;
|
||||||
|
border-radius: 50%;
|
||||||
|
display: block;
|
||||||
|
margin: 15px auto;
|
||||||
|
position: relative;
|
||||||
|
color: var(--color4);
|
||||||
|
box-sizing: border-box;
|
||||||
|
animation: animloader 1s linear infinite alternate;
|
||||||
|
}
|
||||||
|
|
||||||
|
@keyframes animloader {
|
||||||
|
0% {
|
||||||
|
box-shadow: -38px -12px, -14px 0, 14px 0, 38px 0;
|
||||||
|
}
|
||||||
|
|
||||||
|
33% {
|
||||||
|
box-shadow: -38px 0px, -14px -12px, 14px 0, 38px 0;
|
||||||
|
}
|
||||||
|
|
||||||
|
66% {
|
||||||
|
box-shadow: -38px 0px, -14px 0, 14px -12px, 38px 0;
|
||||||
|
}
|
||||||
|
|
||||||
|
100% {
|
||||||
|
box-shadow: -38px 0, -14px 0, 14px 0, 38px -12px;
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
body {
|
body {
|
||||||
color: var(--bcolor3);
|
color: var(--bcolor3);
|
||||||
background-color: var(--bcolor1);
|
background-color: var(--bcolor1);
|
||||||
|
|||||||
|
Before Width: | Height: | Size: 1.5 MiB After Width: | Height: | Size: 1.1 MiB |
|
Before Width: | Height: | Size: 1.5 MiB After Width: | Height: | Size: 1.1 MiB |
|
Before Width: | Height: | Size: 1.5 MiB After Width: | Height: | Size: 1.0 MiB |
|
Before Width: | Height: | Size: 1.5 MiB After Width: | Height: | Size: 1.1 MiB |
|
Before Width: | Height: | Size: 1.5 MiB After Width: | Height: | Size: 1.1 MiB |
|
Before Width: | Height: | Size: 1.5 MiB After Width: | Height: | Size: 1.1 MiB |
|
Before Width: | Height: | Size: 1.5 MiB After Width: | Height: | Size: 1.1 MiB |
|
Before Width: | Height: | Size: 1.5 MiB After Width: | Height: | Size: 1.1 MiB |
|
Before Width: | Height: | Size: 1.5 MiB After Width: | Height: | Size: 1.1 MiB |
|
Before Width: | Height: | Size: 1.5 MiB After Width: | Height: | Size: 1.1 MiB |
|
Before Width: | Height: | Size: 1.5 MiB After Width: | Height: | Size: 1.1 MiB |
|
Before Width: | Height: | Size: 1.5 MiB After Width: | Height: | Size: 1.1 MiB |
|
Before Width: | Height: | Size: 1.5 MiB After Width: | Height: | Size: 1.1 MiB |
|
Before Width: | Height: | Size: 1.5 MiB After Width: | Height: | Size: 1.1 MiB |
|
Before Width: | Height: | Size: 1.5 MiB After Width: | Height: | Size: 1.1 MiB |
|
Before Width: | Height: | Size: 1.5 MiB After Width: | Height: | Size: 1.0 MiB |
|
Before Width: | Height: | Size: 1.5 MiB After Width: | Height: | Size: 1.1 MiB |
|
Before Width: | Height: | Size: 1.5 MiB After Width: | Height: | Size: 1.1 MiB |
|
Before Width: | Height: | Size: 1.5 MiB After Width: | Height: | Size: 1.1 MiB |
@@ -1,14 +1,22 @@
|
|||||||
@import url("https://fonts.googleapis.com/css2?family=Lora:wght@300;400;500;600;700&display=swap");
|
@import url("https://fonts.googleapis.com/css2?family=Lora:wght@300;400;500;600;700&display=swap");
|
||||||
|
|
||||||
* {
|
* {
|
||||||
--color1: #FFB6C1; /* LightPink */
|
--color1: #FFB6C1;
|
||||||
--color2: #98FB98; /* PaleGreen */
|
/* LightPink */
|
||||||
--color3: #FFD700; /* Gold */
|
--color2: #98FB98;
|
||||||
--color4: #87CEFA; /* LightSkyBlue */
|
/* PaleGreen */
|
||||||
--bcolor1: #FFFFFF; /* White */
|
--color3: #FFD700;
|
||||||
--bcolor2: #2F4F4F; /* DarkSlateGray */
|
/* Gold */
|
||||||
--bcolor3: #FAF0E6; /* Linen */
|
--color4: #87CEFA;
|
||||||
--bcolor4: #E6E6FA; /* Lavender */
|
/* LightSkyBlue */
|
||||||
|
--bcolor1: #FFFFFF;
|
||||||
|
/* White */
|
||||||
|
--bcolor2: #2F4F4F;
|
||||||
|
/* DarkSlateGray */
|
||||||
|
--bcolor3: #FAF0E6;
|
||||||
|
/* Linen */
|
||||||
|
--bcolor4: #E6E6FA;
|
||||||
|
/* Lavender */
|
||||||
}
|
}
|
||||||
|
|
||||||
.navbar {
|
.navbar {
|
||||||
@@ -79,6 +87,14 @@
|
|||||||
font-family: "Lora", serif;
|
font-family: "Lora", serif;
|
||||||
}
|
}
|
||||||
|
|
||||||
|
.tagentry label:hover {
|
||||||
|
background-color: var(--bcolor3);
|
||||||
|
}
|
||||||
|
|
||||||
|
.tagentry .tagtoggle:hover {
|
||||||
|
background-color: var(--bcolor3);
|
||||||
|
}
|
||||||
|
|
||||||
.column img {
|
.column img {
|
||||||
background-color: var(--bcolor4);
|
background-color: var(--bcolor4);
|
||||||
}
|
}
|
||||||
@@ -94,6 +110,54 @@
|
|||||||
font-family: "Lora", serif;
|
font-family: "Lora", serif;
|
||||||
}
|
}
|
||||||
|
|
||||||
|
.loader,
|
||||||
|
.loader:before,
|
||||||
|
.loader:after {
|
||||||
|
border-radius: 50%;
|
||||||
|
width: 2.5em;
|
||||||
|
height: 2.5em;
|
||||||
|
animation-fill-mode: both;
|
||||||
|
animation: bblFadInOut 1.8s infinite ease-in-out;
|
||||||
|
}
|
||||||
|
|
||||||
|
.loader {
|
||||||
|
color: var(--bcolor2);
|
||||||
|
font-size: 7px;
|
||||||
|
position: relative;
|
||||||
|
text-indent: -9999em;
|
||||||
|
transform: translateZ(0);
|
||||||
|
animation-delay: -0.16s;
|
||||||
|
}
|
||||||
|
|
||||||
|
.loader:before,
|
||||||
|
.loader:after {
|
||||||
|
content: '';
|
||||||
|
position: absolute;
|
||||||
|
top: 0;
|
||||||
|
}
|
||||||
|
|
||||||
|
.loader:before {
|
||||||
|
left: -3.5em;
|
||||||
|
animation-delay: -0.32s;
|
||||||
|
}
|
||||||
|
|
||||||
|
.loader:after {
|
||||||
|
left: 3.5em;
|
||||||
|
}
|
||||||
|
|
||||||
|
@keyframes bblFadInOut {
|
||||||
|
|
||||||
|
0%,
|
||||||
|
80%,
|
||||||
|
100% {
|
||||||
|
box-shadow: 0 2.5em 0 -1.3em
|
||||||
|
}
|
||||||
|
|
||||||
|
40% {
|
||||||
|
box-shadow: 0 2.5em 0 0
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
body {
|
body {
|
||||||
color: var(--bcolor2);
|
color: var(--bcolor2);
|
||||||
background-color: var(--bcolor3);
|
background-color: var(--bcolor3);
|
||||||
|
|||||||
@@ -80,6 +80,14 @@
|
|||||||
background-color: var(--color4);
|
background-color: var(--color4);
|
||||||
}
|
}
|
||||||
|
|
||||||
|
.tagentry label:hover {
|
||||||
|
background-color: var(--color3);
|
||||||
|
}
|
||||||
|
|
||||||
|
.tagentry .tagtoggle:hover {
|
||||||
|
background-color: var(--color3);
|
||||||
|
}
|
||||||
|
|
||||||
.column img {
|
.column img {
|
||||||
background-color: var(--color4);
|
background-color: var(--color4);
|
||||||
}
|
}
|
||||||
@@ -94,6 +102,52 @@
|
|||||||
font-weight: 700;
|
font-weight: 700;
|
||||||
}
|
}
|
||||||
|
|
||||||
|
.loader {
|
||||||
|
width: 48px;
|
||||||
|
height: 48px;
|
||||||
|
border-radius: 50%;
|
||||||
|
position: relative;
|
||||||
|
animation: rotate 1s linear infinite
|
||||||
|
}
|
||||||
|
|
||||||
|
.loader::before {
|
||||||
|
content: "";
|
||||||
|
box-sizing: border-box;
|
||||||
|
position: absolute;
|
||||||
|
inset: 0px;
|
||||||
|
border-radius: 50%;
|
||||||
|
border: 5px solid var(--color2);
|
||||||
|
animation: prixClipFix 2s linear infinite;
|
||||||
|
}
|
||||||
|
|
||||||
|
@keyframes rotate {
|
||||||
|
100% {
|
||||||
|
transform: rotate(360deg)
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
@keyframes prixClipFix {
|
||||||
|
0% {
|
||||||
|
clip-path: polygon(50% 50%, 0 0, 0 0, 0 0, 0 0, 0 0)
|
||||||
|
}
|
||||||
|
|
||||||
|
25% {
|
||||||
|
clip-path: polygon(50% 50%, 0 0, 100% 0, 100% 0, 100% 0, 100% 0)
|
||||||
|
}
|
||||||
|
|
||||||
|
50% {
|
||||||
|
clip-path: polygon(50% 50%, 0 0, 100% 0, 100% 100%, 100% 100%, 100% 100%)
|
||||||
|
}
|
||||||
|
|
||||||
|
75% {
|
||||||
|
clip-path: polygon(50% 50%, 0 0, 100% 0, 100% 100%, 0 100%, 0 100%)
|
||||||
|
}
|
||||||
|
|
||||||
|
100% {
|
||||||
|
clip-path: polygon(50% 50%, 0 0, 100% 0, 100% 100%, 0 100%, 0 0)
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
body {
|
body {
|
||||||
color: var(--bcolor1);
|
color: var(--bcolor1);
|
||||||
background-color: var(--bcolor2);
|
background-color: var(--bcolor2);
|
||||||
|
|||||||
@@ -80,6 +80,15 @@
|
|||||||
font-family: "Roboto", sans-serif;
|
font-family: "Roboto", sans-serif;
|
||||||
}
|
}
|
||||||
|
|
||||||
|
.tagentry label:hover {
|
||||||
|
background-color: var(--bcolor3);
|
||||||
|
}
|
||||||
|
|
||||||
|
.tagentry .tagtoggle:hover {
|
||||||
|
background-color: var(--bcolor3);
|
||||||
|
color: var(--bcolor2);
|
||||||
|
}
|
||||||
|
|
||||||
.column img {
|
.column img {
|
||||||
background-color: var(--bcolor4);
|
background-color: var(--bcolor4);
|
||||||
}
|
}
|
||||||
@@ -95,6 +104,54 @@
|
|||||||
font-family: "Roboto", sans-serif;
|
font-family: "Roboto", sans-serif;
|
||||||
}
|
}
|
||||||
|
|
||||||
|
.loader,
|
||||||
|
.loader:before,
|
||||||
|
.loader:after {
|
||||||
|
border-radius: 50%;
|
||||||
|
width: 2.5em;
|
||||||
|
height: 2.5em;
|
||||||
|
animation-fill-mode: both;
|
||||||
|
animation: bblFadInOut 1.8s infinite ease-in-out;
|
||||||
|
}
|
||||||
|
|
||||||
|
.loader {
|
||||||
|
color: var(--bcolor2);
|
||||||
|
font-size: 7px;
|
||||||
|
position: relative;
|
||||||
|
text-indent: -9999em;
|
||||||
|
transform: translateZ(0);
|
||||||
|
animation-delay: -0.16s;
|
||||||
|
}
|
||||||
|
|
||||||
|
.loader:before,
|
||||||
|
.loader:after {
|
||||||
|
content: '';
|
||||||
|
position: absolute;
|
||||||
|
top: 0;
|
||||||
|
}
|
||||||
|
|
||||||
|
.loader:before {
|
||||||
|
left: -3.5em;
|
||||||
|
animation-delay: -0.32s;
|
||||||
|
}
|
||||||
|
|
||||||
|
.loader:after {
|
||||||
|
left: 3.5em;
|
||||||
|
}
|
||||||
|
|
||||||
|
@keyframes bblFadInOut {
|
||||||
|
|
||||||
|
0%,
|
||||||
|
80%,
|
||||||
|
100% {
|
||||||
|
box-shadow: 0 2.5em 0 -1.3em
|
||||||
|
}
|
||||||
|
|
||||||
|
40% {
|
||||||
|
box-shadow: 0 2.5em 0 0
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
body {
|
body {
|
||||||
color: var(--bcolor2);
|
color: var(--bcolor2);
|
||||||
background-color: var(--bcolor3);
|
background-color: var(--bcolor3);
|
||||||
|
|||||||
@@ -5,9 +5,9 @@
|
|||||||
--color2: #ffd700;
|
--color2: #ffd700;
|
||||||
--color3: #ffe135;
|
--color3: #ffe135;
|
||||||
--color4: #ce8c00;
|
--color4: #ce8c00;
|
||||||
--bcolor1: #ebebeb;
|
--bcolor1: #171717;
|
||||||
--bcolor2: #191919;
|
--bcolor2: #191919;
|
||||||
--bcolor3: #171717;
|
--bcolor3: #ebebeb;
|
||||||
--bcolor4: #0a0a0a;
|
--bcolor4: #0a0a0a;
|
||||||
}
|
}
|
||||||
|
|
||||||
@@ -60,7 +60,7 @@
|
|||||||
|
|
||||||
.row a {
|
.row a {
|
||||||
font-weight: 800;
|
font-weight: 800;
|
||||||
color: var(--color2);
|
color: var(--color1);
|
||||||
text-decoration: none;
|
text-decoration: none;
|
||||||
}
|
}
|
||||||
|
|
||||||
@@ -70,10 +70,18 @@
|
|||||||
|
|
||||||
.tooltiptext {
|
.tooltiptext {
|
||||||
font-weight: 600;
|
font-weight: 600;
|
||||||
color: var(--bcolor1);
|
color: var(--bcolor3);
|
||||||
background-color: var(--bcolor2);
|
background-color: var(--bcolor2);
|
||||||
}
|
}
|
||||||
|
|
||||||
|
.tagentry label:hover {
|
||||||
|
background-color: var(--bcolor4);
|
||||||
|
}
|
||||||
|
|
||||||
|
.tagentry .tagtoggle:hover {
|
||||||
|
background-color: var(--bcolor4);
|
||||||
|
}
|
||||||
|
|
||||||
.column img {
|
.column img {
|
||||||
background-color: var(--bcolor4);
|
background-color: var(--bcolor4);
|
||||||
}
|
}
|
||||||
@@ -88,6 +96,36 @@
|
|||||||
font-weight: 800;
|
font-weight: 800;
|
||||||
}
|
}
|
||||||
|
|
||||||
|
.loader {
|
||||||
|
width: 12px;
|
||||||
|
height: 12px;
|
||||||
|
border-radius: 50%;
|
||||||
|
display: block;
|
||||||
|
margin: 15px auto;
|
||||||
|
position: relative;
|
||||||
|
color: var(--color1);
|
||||||
|
box-sizing: border-box;
|
||||||
|
animation: animloader 1s linear infinite alternate;
|
||||||
|
}
|
||||||
|
|
||||||
|
@keyframes animloader {
|
||||||
|
0% {
|
||||||
|
box-shadow: -38px -12px, -14px 0, 14px 0, 38px 0;
|
||||||
|
}
|
||||||
|
|
||||||
|
33% {
|
||||||
|
box-shadow: -38px 0px, -14px -12px, 14px 0, 38px 0;
|
||||||
|
}
|
||||||
|
|
||||||
|
66% {
|
||||||
|
box-shadow: -38px 0px, -14px 0, 14px -12px, 38px 0;
|
||||||
|
}
|
||||||
|
|
||||||
|
100% {
|
||||||
|
box-shadow: -38px 0, -14px 0, 14px 0, 38px -12px;
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
body {
|
body {
|
||||||
color: var(--bcolor1);
|
color: var(--bcolor1);
|
||||||
background-color: var(--bcolor3);
|
background-color: var(--bcolor3);
|
||||||
|
|||||||
@@ -88,6 +88,14 @@
|
|||||||
font-family: "Montserrat", sans-serif;
|
font-family: "Montserrat", sans-serif;
|
||||||
}
|
}
|
||||||
|
|
||||||
|
.tagentry label:hover {
|
||||||
|
background-color: var(--color2);
|
||||||
|
}
|
||||||
|
|
||||||
|
.tagentry .tagtoggle:hover {
|
||||||
|
background-color: var(--color2);
|
||||||
|
}
|
||||||
|
|
||||||
.column img {
|
.column img {
|
||||||
background-color: var(--bcolor4);
|
background-color: var(--bcolor4);
|
||||||
}
|
}
|
||||||
@@ -103,6 +111,54 @@
|
|||||||
font-family: "Montserrat", sans-serif;
|
font-family: "Montserrat", sans-serif;
|
||||||
}
|
}
|
||||||
|
|
||||||
|
.loader,
|
||||||
|
.loader:before,
|
||||||
|
.loader:after {
|
||||||
|
border-radius: 50%;
|
||||||
|
width: 2.5em;
|
||||||
|
height: 2.5em;
|
||||||
|
animation-fill-mode: both;
|
||||||
|
animation: bblFadInOut 1.8s infinite ease-in-out;
|
||||||
|
}
|
||||||
|
|
||||||
|
.loader {
|
||||||
|
color: var(--bcolor2);
|
||||||
|
font-size: 7px;
|
||||||
|
position: relative;
|
||||||
|
text-indent: -9999em;
|
||||||
|
transform: translateZ(0);
|
||||||
|
animation-delay: -0.16s;
|
||||||
|
}
|
||||||
|
|
||||||
|
.loader:before,
|
||||||
|
.loader:after {
|
||||||
|
content: '';
|
||||||
|
position: absolute;
|
||||||
|
top: 0;
|
||||||
|
}
|
||||||
|
|
||||||
|
.loader:before {
|
||||||
|
left: -3.5em;
|
||||||
|
animation-delay: -0.32s;
|
||||||
|
}
|
||||||
|
|
||||||
|
.loader:after {
|
||||||
|
left: 3.5em;
|
||||||
|
}
|
||||||
|
|
||||||
|
@keyframes bblFadInOut {
|
||||||
|
|
||||||
|
0%,
|
||||||
|
80%,
|
||||||
|
100% {
|
||||||
|
box-shadow: 0 2.5em 0 -1.3em
|
||||||
|
}
|
||||||
|
|
||||||
|
40% {
|
||||||
|
box-shadow: 0 2.5em 0 0
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
body {
|
body {
|
||||||
color: var(--bcolor2);
|
color: var(--bcolor2);
|
||||||
background-color: var(--bcolor3);
|
background-color: var(--bcolor3);
|
||||||
|
|||||||
4
view-latest-log.sh
Executable file
@@ -0,0 +1,4 @@
|
|||||||
|
#!/usr/bin/env bash
|
||||||
|
|
||||||
|
LESS=-SR hl logs/latest.jsonl --config hl_config.yaml
|
||||||
|
|
||||||
3
view-logs.sh
Executable file
@@ -0,0 +1,3 @@
|
|||||||
|
#!/usr/bin/env bash
|
||||||
|
|
||||||
|
LESS=-SR hl $(ls -tr logs/*.{jsonl,jsonl.gz}) --config hl_config.yaml
|
||||||