Compare commits
21 Commits
release/1.3.0
...
main
| Author | SHA1 | Date | |
|---|---|---|---|
| 6680d464c3 | |||
| b6a5460a04 | |||
| 84c408d149 | |||
| 53d366f6b6 | |||
| 6beca83fa3 | |||
| cbd83360a6 | |||
| 767fb8cfc6 | |||
| 1b626c1e89 | |||
| 44a08ad715 | |||
| 2639c87576 | |||
| 85783efb81 | |||
| 6127e132e4 | |||
| 89b83e3e43 | |||
| 39cf54624a | |||
| 722548118a | |||
| a5b24dc9bf | |||
| 99867aa03e | |||
| 65f4dd1ac5 | |||
| 748f52241f | |||
| 21e1469b51 | |||
| 649102d1aa |
+17
-5
@@ -1,8 +1,20 @@
|
|||||||
[*.{js,jsx,mjs,cjs,ts,tsx,mts,cts,vue,css,scss,sass,less,styl}]
|
# EditorConfig is awesome: https://EditorConfig.org
|
||||||
charset = utf-8
|
|
||||||
indent_size = 2
|
# top-most EditorConfig file
|
||||||
|
root = true
|
||||||
|
|
||||||
|
[*]
|
||||||
indent_style = space
|
indent_style = space
|
||||||
insert_final_newline = true
|
indent_size = 4
|
||||||
trim_trailing_whitespace = true
|
|
||||||
end_of_line = lf
|
end_of_line = lf
|
||||||
|
charset = utf-8
|
||||||
|
trim_trailing_whitespace = true
|
||||||
|
insert_final_newline = true
|
||||||
|
|
||||||
|
[Makefile]
|
||||||
|
indent_style = tab
|
||||||
|
indent_size = 2
|
||||||
|
|
||||||
|
[*.{js,jsx,mjs,cjs,ts,tsx,mts,cts,vue,css,scss,sass,less,styl}]
|
||||||
|
indent_size = 2
|
||||||
max_line_length = 100
|
max_line_length = 100
|
||||||
|
|||||||
@@ -1,37 +1,65 @@
|
|||||||
name: Deploy
|
name: Deploy
|
||||||
|
|
||||||
|
concurrency:
|
||||||
|
group: deploy-${{ github.ref }}
|
||||||
|
cancel-in-progress: true
|
||||||
|
|
||||||
on:
|
on:
|
||||||
schedule:
|
|
||||||
- cron: "*/30 * * * *"
|
|
||||||
push:
|
push:
|
||||||
branches:
|
branches:
|
||||||
- "main"
|
- "main"
|
||||||
|
paths:
|
||||||
|
- '.github/workflows/deploy.yml'
|
||||||
|
- 'package.json'
|
||||||
|
- 'bun.lock'
|
||||||
|
- 'src/**'
|
||||||
|
- 'index.html'
|
||||||
|
- 'vite.config.ts'
|
||||||
|
- 'post-build.ts'
|
||||||
|
- 'tsconfig.*'
|
||||||
|
- 'env.d.ts'
|
||||||
|
- 'vite.d.ts'
|
||||||
|
|
||||||
jobs:
|
jobs:
|
||||||
build:
|
build:
|
||||||
name: Build
|
name: "Build"
|
||||||
runs-on: ubuntu-latest
|
runs-on: ubuntu-latest
|
||||||
steps:
|
steps:
|
||||||
- uses: actions/checkout@v6
|
- name: Set up Bun
|
||||||
- uses: oven-sh/setup-bun@v2
|
uses: actions/setup-bun@v2
|
||||||
- run: bun install
|
- name: Checkout repository
|
||||||
- name: Setup SSH Key
|
uses: actions/checkout@v6
|
||||||
uses: webfactory/ssh-agent@v0.9.1
|
- name: Install dependencies
|
||||||
|
run: bun ci
|
||||||
|
- name: Checkout articles repository
|
||||||
|
uses: actions/checkout@v6
|
||||||
with:
|
with:
|
||||||
ssh-private-key: ${{ secrets.SSH_PRIVATE_KEY }}
|
repository: ${{ vars.ARTICLES_REPOSITORY }}
|
||||||
- run: git clone ${{ vars.ARTICLES_REPOSITORY }} articles
|
ref: main
|
||||||
- run: bun run build
|
token: ${{ secrets.PRIVATE_CLONE_TOKEN }}
|
||||||
- uses: actions/upload-artifact@v7
|
path: ./articles
|
||||||
|
- name: Build project
|
||||||
|
run: bun run build
|
||||||
|
- name: Upload production files
|
||||||
|
uses: actions/upload-artifact@v3
|
||||||
with:
|
with:
|
||||||
name: production-files
|
name: production-files
|
||||||
path: ./dist
|
path: dist/
|
||||||
deploy:
|
|
||||||
name: Deploy
|
stapler:
|
||||||
|
name: "Deploy to Stapler"
|
||||||
|
runs-on: ubuntu-latest
|
||||||
needs: build
|
needs: build
|
||||||
runs-on: ubuntu-latest
|
|
||||||
steps:
|
steps:
|
||||||
- uses: actions/download-artifact@v8
|
- name: Download production files
|
||||||
|
uses: actions/download-artifact@v3
|
||||||
with:
|
with:
|
||||||
name: production-files
|
name: production-files
|
||||||
path: ./dist
|
path: ./dist
|
||||||
- run: tar -czC dist -f dist.tar.gz .
|
- name: Upload to Stapler server
|
||||||
- run: |
|
uses: actions/stapler-deploy@v1
|
||||||
curl -v -X PUT -H 'X-Token: ${{ secrets.STAPLER_TOKEN }}' -H 'X-Host-Only: ${{ vars.TARGET_HOST }}' -H 'X-SPA: index.html' --data-binary "@dist.tar.gz" ${{ vars.STAPLER_TARGET }}
|
with:
|
||||||
|
path: dist
|
||||||
|
token: ${{ secrets.STAPLER_TOKEN }}
|
||||||
|
target: ${{ github.repository }}
|
||||||
|
extra_curl_args: ${{ vars.STAPLER_CURL_ARGS }}
|
||||||
|
|||||||
@@ -0,0 +1,66 @@
|
|||||||
|
name: Lint
|
||||||
|
|
||||||
|
concurrency:
|
||||||
|
group: lint-${{ github.ref }}
|
||||||
|
cancel-in-progress: true
|
||||||
|
|
||||||
|
on:
|
||||||
|
workflow_dispatch:
|
||||||
|
push:
|
||||||
|
paths:
|
||||||
|
- '.github/workflows/lint.yml'
|
||||||
|
- 'package.json'
|
||||||
|
- 'bun.lock'
|
||||||
|
- 'src/**'
|
||||||
|
- 'articles.example/**'
|
||||||
|
- '.oxlintrc.json'
|
||||||
|
- 'eslint.config.ts'
|
||||||
|
- 'tsconfig.*'
|
||||||
|
- 'env.d.ts'
|
||||||
|
- 'vite.d.ts'
|
||||||
|
|
||||||
|
jobs:
|
||||||
|
lint-eslint:
|
||||||
|
name: 'ESLint'
|
||||||
|
runs-on: ubuntu-latest
|
||||||
|
steps:
|
||||||
|
- name: Set up Bun
|
||||||
|
uses: actions/setup-bun@v2
|
||||||
|
- name: Checkout repository
|
||||||
|
uses: actions/checkout@v6
|
||||||
|
- name: Install dependencies
|
||||||
|
run: bun ci
|
||||||
|
- name: Create fake articles
|
||||||
|
run: mv articles.example articles
|
||||||
|
- name: Run ESLint
|
||||||
|
run: bun run lint:eslint
|
||||||
|
|
||||||
|
lint-oxlint:
|
||||||
|
name: 'Oxlint'
|
||||||
|
runs-on: ubuntu-latest
|
||||||
|
steps:
|
||||||
|
- name: Set up Bun
|
||||||
|
uses: actions/setup-bun@v2
|
||||||
|
- name: Checkout repository
|
||||||
|
uses: actions/checkout@v6
|
||||||
|
- name: Install dependencies
|
||||||
|
run: bun ci
|
||||||
|
- name: Create fake articles
|
||||||
|
run: mv articles.example articles
|
||||||
|
- name: Run Oxlint
|
||||||
|
run: bun run lint:oxlint
|
||||||
|
|
||||||
|
tsc:
|
||||||
|
name: 'TypeScript'
|
||||||
|
runs-on: ubuntu-latest
|
||||||
|
steps:
|
||||||
|
- name: Set up Bun
|
||||||
|
uses: actions/setup-bun@v2
|
||||||
|
- name: Checkout repository
|
||||||
|
uses: actions/checkout@v6
|
||||||
|
- name: Install dependencies
|
||||||
|
run: bun ci
|
||||||
|
- name: Create fake articles
|
||||||
|
run: mv articles.example articles
|
||||||
|
- name: Run type check
|
||||||
|
run: bun run type-check
|
||||||
@@ -1,6 +0,0 @@
|
|||||||
{
|
|
||||||
"$schema": "https://json.schemastore.org/prettierrc",
|
|
||||||
"semi": false,
|
|
||||||
"singleQuote": true,
|
|
||||||
"printWidth": 100
|
|
||||||
}
|
|
||||||
Vendored
-9
@@ -1,9 +0,0 @@
|
|||||||
{
|
|
||||||
"recommendations": [
|
|
||||||
"Vue.volar",
|
|
||||||
"dbaeumer.vscode-eslint",
|
|
||||||
"EditorConfig.EditorConfig",
|
|
||||||
"oxc.oxc-vscode",
|
|
||||||
"esbenp.prettier-vscode"
|
|
||||||
]
|
|
||||||
}
|
|
||||||
@@ -0,0 +1,61 @@
|
|||||||
|
# ENV
|
||||||
|
|
||||||
|
ifeq (,$(shell which bun))
|
||||||
|
NPM ?= npm
|
||||||
|
endif
|
||||||
|
|
||||||
|
NPM ?= bun
|
||||||
|
GIT ?= git
|
||||||
|
|
||||||
|
.PHONY: help
|
||||||
|
help: ## show this message
|
||||||
|
@echo "Usage: $(MAKE) [target1] [target2] ..."
|
||||||
|
@echo ""
|
||||||
|
@echo "Commands/Targets:"
|
||||||
|
@cat $(MAKEFILE_LIST) | grep -E '(^[a-zA-Z0-9_%-]+:.*?##.*$$)|(^##)' | awk 'BEGIN {FS = ":.*?## "}{printf "\033[32m%-20s\033[0m %s\n", $$1, $$2}' | sed -e 's/\[32m##/[33m/'
|
||||||
|
@echo ""
|
||||||
|
@echo "Environment:"
|
||||||
|
@cat $(MAKEFILE_LIST) | grep -E '^[a-zA-Z0-9_-]+\s*\??=.*$$' | grep -Eo '^[a-zA-Z0-9_-]+' | xargs -I {} $(MAKE) -s print-{} 2> /dev/null
|
||||||
|
|
||||||
|
.PHONY: print-%
|
||||||
|
print-%:
|
||||||
|
@echo -e '\033[32m$*\033[0m = $($*)'
|
||||||
|
|
||||||
|
# FILES
|
||||||
|
|
||||||
|
node_modules: bun.lock
|
||||||
|
@$(MAKE) -s npm-install
|
||||||
|
|
||||||
|
# ACTIONS
|
||||||
|
|
||||||
|
.PHONY: install
|
||||||
|
install: npm-install ## install project
|
||||||
|
|
||||||
|
.PHONY: update
|
||||||
|
update: npm-update ## update project
|
||||||
|
|
||||||
|
.PHONY: build
|
||||||
|
build: npm-run-build ## build static site in "dist"
|
||||||
|
|
||||||
|
.PHONY: dev
|
||||||
|
dev: npm-run-dev ## run dev server
|
||||||
|
|
||||||
|
.PHONY: lint
|
||||||
|
lint: npm-run-lint npm-run-type-check ## lint code
|
||||||
|
|
||||||
|
.PHONY: format
|
||||||
|
format: npm-run-lint-fix ## fix and reformat code
|
||||||
|
|
||||||
|
# TOOLS
|
||||||
|
|
||||||
|
.PHONY: npm-install
|
||||||
|
npm-install: ## npm install
|
||||||
|
$(NPM) install
|
||||||
|
|
||||||
|
.PHONY: npm-update
|
||||||
|
npm-update: ## npm update
|
||||||
|
$(NPM) update
|
||||||
|
|
||||||
|
.PHONY: npm-run-%
|
||||||
|
npm-run-%: node_modules ## npm run (script)
|
||||||
|
$(NPM) run $*
|
||||||
@@ -1,3 +1,5 @@
|
|||||||
|
[](https://git.klemek.fr/klemek/md-blog/actions?workflow=lint.yml) [](https://git.klemek.fr/klemek/md-blog/actions?workflow=deploy.yml)
|
||||||
|
|
||||||
# md-blog
|
# md-blog
|
||||||
|
|
||||||
## Minimal setup
|
## Minimal setup
|
||||||
@@ -20,10 +22,9 @@ bun run build
|
|||||||
- [x] build with github actions
|
- [x] build with github actions
|
||||||
- [x] config in sub repo
|
- [x] config in sub repo
|
||||||
- [x] copyright
|
- [x] copyright
|
||||||
- [ ] nav bar on top
|
- [x] nav bar on top
|
||||||
- [ ] date updated
|
- [x] date updated
|
||||||
- [ ] archive page
|
- [ ] archive page
|
||||||
- [ ] about page
|
- [x] about page
|
||||||
- [ ] contact/links
|
|
||||||
- [ ] link to previous/next article
|
- [ ] link to previous/next article
|
||||||
- [ ] proper docs
|
- [ ] proper docs
|
||||||
Binary file not shown.
|
After Width: | Height: | Size: 806 B |
@@ -0,0 +1,40 @@
|
|||||||
|
---
|
||||||
|
title: Demo
|
||||||
|
author: kleπek
|
||||||
|
draft: false
|
||||||
|
thumbnail: ./thumbnail.jpg
|
||||||
|
date: 2026-01-01
|
||||||
|
---
|
||||||
|
|
||||||
|
## Images
|
||||||
|
|
||||||
|
PNG: 
|
||||||
|
|
||||||
|
## highlight.js
|
||||||
|
|
||||||
|
```typescript
|
||||||
|
function test(arg1: string, arg2: number): string {
|
||||||
|
return `${arg1}: ${arg2}`
|
||||||
|
}
|
||||||
|
```
|
||||||
|
|
||||||
|
```text
|
||||||
|
text/plain
|
||||||
|
```
|
||||||
|
|
||||||
|
## Mermaid
|
||||||
|
|
||||||
|
```mermaid
|
||||||
|
flowchart LR
|
||||||
|
A --> B
|
||||||
|
```
|
||||||
|
|
||||||
|
## LaTex
|
||||||
|
|
||||||
|
$$\Large\color{Red}{\frac{\Delta K}{2}=\frac{T_{2}}{x_{2}(x_{2}-x_{1})}-\frac{T_{1}}{x_{1}(x_{2}-x_{1})}}$$
|
||||||
|
|
||||||
|
Inline LaTex $$\frac{\Delta K}{2}$$ in text
|
||||||
|
|
||||||
|
## Lucide Icons
|
||||||
|
|
||||||
|
Icon inside <i icon="atom"></i> text
|
||||||
@@ -0,0 +1,15 @@
|
|||||||
|
{
|
||||||
|
"base_url": "http://localhost/",
|
||||||
|
"title": "<i icon=notepad-text></i> My Blog",
|
||||||
|
"signature": "By <b>Me</b>",
|
||||||
|
"lang": "en",
|
||||||
|
"custom_head": "",
|
||||||
|
"home_count": 5,
|
||||||
|
"rss_link": "<i icon=rss></i> RSS",
|
||||||
|
"about_link": "<i icon=info></i> ABOUT",
|
||||||
|
"back_link": "<i icon=undo-2></i> Back to home",
|
||||||
|
"published_on": "Published on",
|
||||||
|
"updated_on": "Updated on",
|
||||||
|
"authored": "By",
|
||||||
|
"copyright": "<a style=\"text-decoration:none;color:inherit;\" target=_blank href=\"https://creativecommons.org/licenses/by-nc/4.0/\">CC BY-NC</a>"
|
||||||
|
}
|
||||||
@@ -0,0 +1,187 @@
|
|||||||
|
/*
|
||||||
|
=================================================
|
||||||
|
https://www.joshwcomeau.com/css/custom-css-reset/
|
||||||
|
=================================================
|
||||||
|
*/
|
||||||
|
|
||||||
|
/*
|
||||||
|
1. Use a more-intuitive box-sizing model.
|
||||||
|
*/
|
||||||
|
*,
|
||||||
|
*::before,
|
||||||
|
*::after {
|
||||||
|
box-sizing: border-box;
|
||||||
|
}
|
||||||
|
/*
|
||||||
|
2. Remove default margin
|
||||||
|
*/
|
||||||
|
* {
|
||||||
|
margin: 0;
|
||||||
|
}
|
||||||
|
/*
|
||||||
|
3. Allow percentage-based heights in the application
|
||||||
|
*/
|
||||||
|
html,
|
||||||
|
body,
|
||||||
|
div#app {
|
||||||
|
height: 100%;
|
||||||
|
}
|
||||||
|
/*
|
||||||
|
Typographic tweaks!
|
||||||
|
4. Add accessible line-height
|
||||||
|
5. Improve text rendering
|
||||||
|
*/
|
||||||
|
body {
|
||||||
|
line-height: 1.5;
|
||||||
|
-webkit-font-smoothing: antialiased;
|
||||||
|
}
|
||||||
|
/*
|
||||||
|
6. Improve media defaults
|
||||||
|
*/
|
||||||
|
img,
|
||||||
|
picture,
|
||||||
|
video,
|
||||||
|
canvas,
|
||||||
|
svg {
|
||||||
|
display: block;
|
||||||
|
max-width: 100%;
|
||||||
|
}
|
||||||
|
/*
|
||||||
|
7. Remove built-in form typography styles
|
||||||
|
*/
|
||||||
|
input,
|
||||||
|
button,
|
||||||
|
textarea,
|
||||||
|
select {
|
||||||
|
font: inherit;
|
||||||
|
}
|
||||||
|
/*
|
||||||
|
8. Avoid text overflows
|
||||||
|
*/
|
||||||
|
p,
|
||||||
|
h1,
|
||||||
|
h2,
|
||||||
|
h3,
|
||||||
|
h4,
|
||||||
|
h5,
|
||||||
|
h6 {
|
||||||
|
overflow-wrap: break-word;
|
||||||
|
}
|
||||||
|
/*
|
||||||
|
9. Create a root stacking context
|
||||||
|
*/
|
||||||
|
#root,
|
||||||
|
#__next {
|
||||||
|
isolation: isolate;
|
||||||
|
}
|
||||||
|
|
||||||
|
/*
|
||||||
|
=================================================
|
||||||
|
https://blog.koley.in/2019/339-bytes-of-responsive-css
|
||||||
|
https://www.swyx.io/css-100-bytes
|
||||||
|
https://gist.github.com/JoeyBurzynski/617fb6201335779f8424ad9528b72c41
|
||||||
|
=================================================
|
||||||
|
*/
|
||||||
|
|
||||||
|
html,
|
||||||
|
body {
|
||||||
|
padding: 0;
|
||||||
|
max-width: 100%;
|
||||||
|
color: #222;
|
||||||
|
font-family: Verdana, serif;
|
||||||
|
}
|
||||||
|
|
||||||
|
body {
|
||||||
|
background-color: #eee;
|
||||||
|
}
|
||||||
|
|
||||||
|
main {
|
||||||
|
padding: 1.5rem;
|
||||||
|
margin: auto;
|
||||||
|
background-color: #ccc;
|
||||||
|
min-height: 100%;
|
||||||
|
}
|
||||||
|
|
||||||
|
table {
|
||||||
|
border-collapse: collapse;
|
||||||
|
width: 100%;
|
||||||
|
font-size: 0.9em;
|
||||||
|
}
|
||||||
|
|
||||||
|
h1,
|
||||||
|
h2,
|
||||||
|
h3,
|
||||||
|
h4,
|
||||||
|
h5,
|
||||||
|
h6 {
|
||||||
|
margin: 1em 0 0.5em;
|
||||||
|
}
|
||||||
|
|
||||||
|
p,
|
||||||
|
ul,
|
||||||
|
ol,
|
||||||
|
pre,
|
||||||
|
img,
|
||||||
|
blockquote,
|
||||||
|
details {
|
||||||
|
margin-bottom: 1em;
|
||||||
|
color: #444;
|
||||||
|
}
|
||||||
|
|
||||||
|
hr {
|
||||||
|
opacity: 25%;
|
||||||
|
border-bottom: 0;
|
||||||
|
margin-bottom: 0.5em;
|
||||||
|
}
|
||||||
|
|
||||||
|
textarea,
|
||||||
|
input,
|
||||||
|
select,
|
||||||
|
pre,
|
||||||
|
.mono {
|
||||||
|
font-family: monospace;
|
||||||
|
}
|
||||||
|
|
||||||
|
@media only screen and (min-width: 768px) {
|
||||||
|
main {
|
||||||
|
max-width: 42rem;
|
||||||
|
}
|
||||||
|
table {
|
||||||
|
font-size: inherit;
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
/*
|
||||||
|
* MD BLOG
|
||||||
|
*/
|
||||||
|
|
||||||
|
nav {
|
||||||
|
display: flex;
|
||||||
|
align-items: flex-end;
|
||||||
|
}
|
||||||
|
|
||||||
|
.home nav {
|
||||||
|
margin-bottom: 1em;
|
||||||
|
vertical-align: text-bottom;
|
||||||
|
}
|
||||||
|
|
||||||
|
.home .nav-title {
|
||||||
|
font-size: xx-large;
|
||||||
|
}
|
||||||
|
|
||||||
|
nav .nav-title {
|
||||||
|
font-size: larger;
|
||||||
|
font-weight: bold;
|
||||||
|
color: inherit;
|
||||||
|
display: inline-block;
|
||||||
|
flex-grow: 1;
|
||||||
|
}
|
||||||
|
|
||||||
|
nav .nav-items {
|
||||||
|
display: flex;
|
||||||
|
gap: 0.5em;
|
||||||
|
}
|
||||||
|
|
||||||
|
.article-info {
|
||||||
|
font-style: italic;
|
||||||
|
}
|
||||||
@@ -5,9 +5,10 @@
|
|||||||
"": {
|
"": {
|
||||||
"name": "md-blog",
|
"name": "md-blog",
|
||||||
"dependencies": {
|
"dependencies": {
|
||||||
|
"@keithclark/shaderview": "https://github.com/keithclark/shaderview/archive/refs/tags/1.2.0.tar.gz",
|
||||||
"highlight.js": "^11.11.1",
|
"highlight.js": "^11.11.1",
|
||||||
"katex": "^0.16.45",
|
"katex": "^0.16.45",
|
||||||
"lucide": "^1.11.0",
|
"lucide": "^1.14.0",
|
||||||
"mermaid": "^11.14.0",
|
"mermaid": "^11.14.0",
|
||||||
"vue": "^3.5.33",
|
"vue": "^3.5.33",
|
||||||
"vue-router": "^5.0.6",
|
"vue-router": "^5.0.6",
|
||||||
@@ -16,10 +17,10 @@
|
|||||||
"@tsconfig/node24": "^24.0.4",
|
"@tsconfig/node24": "^24.0.4",
|
||||||
"@types/node": "^24.12.2",
|
"@types/node": "^24.12.2",
|
||||||
"@vitejs/plugin-vue": "^6.0.6",
|
"@vitejs/plugin-vue": "^6.0.6",
|
||||||
|
"@vue/eslint-config-prettier": "^10.2.0",
|
||||||
"@vue/eslint-config-typescript": "^14.7.0",
|
"@vue/eslint-config-typescript": "^14.7.0",
|
||||||
"@vue/tsconfig": "^0.9.1",
|
"@vue/tsconfig": "^0.9.1",
|
||||||
"eslint": "^10.2.1",
|
"eslint": "^10.3.0",
|
||||||
"eslint-config-prettier": "^10.1.8",
|
|
||||||
"eslint-plugin-oxlint": "~1.60.0",
|
"eslint-plugin-oxlint": "~1.60.0",
|
||||||
"eslint-plugin-vue": "~10.8.0",
|
"eslint-plugin-vue": "~10.8.0",
|
||||||
"feed": "^5.2.1",
|
"feed": "^5.2.1",
|
||||||
@@ -210,6 +211,8 @@
|
|||||||
|
|
||||||
"@jridgewell/trace-mapping": ["@jridgewell/trace-mapping@0.3.31", "", { "dependencies": { "@jridgewell/resolve-uri": "^3.1.0", "@jridgewell/sourcemap-codec": "^1.4.14" } }, "sha512-zzNR+SdQSDJzc8joaeP8QQoCQr8NuYx2dIIytl1QeBEZHJ9uW6hebsrYgbz8hJwUQao3TWCMtmfV8Nu1twOLAw=="],
|
"@jridgewell/trace-mapping": ["@jridgewell/trace-mapping@0.3.31", "", { "dependencies": { "@jridgewell/resolve-uri": "^3.1.0", "@jridgewell/sourcemap-codec": "^1.4.14" } }, "sha512-zzNR+SdQSDJzc8joaeP8QQoCQr8NuYx2dIIytl1QeBEZHJ9uW6hebsrYgbz8hJwUQao3TWCMtmfV8Nu1twOLAw=="],
|
||||||
|
|
||||||
|
"@keithclark/shaderview": ["@keithclark/shaderview@https://github.com/keithclark/shaderview/archive/refs/tags/1.2.0.tar.gz", {}],
|
||||||
|
|
||||||
"@mermaid-js/parser": ["@mermaid-js/parser@1.1.0", "", { "dependencies": { "langium": "^4.0.0" } }, "sha512-gxK9ZX2+Fex5zu8LhRQoMeMPEHbc73UKZ0FQ54YrQtUxE1VVhMwzeNtKRPAu5aXks4FasbMe4xB4bWrmq6Jlxw=="],
|
"@mermaid-js/parser": ["@mermaid-js/parser@1.1.0", "", { "dependencies": { "langium": "^4.0.0" } }, "sha512-gxK9ZX2+Fex5zu8LhRQoMeMPEHbc73UKZ0FQ54YrQtUxE1VVhMwzeNtKRPAu5aXks4FasbMe4xB4bWrmq6Jlxw=="],
|
||||||
|
|
||||||
"@napi-rs/wasm-runtime": ["@napi-rs/wasm-runtime@1.1.4", "", { "dependencies": { "@tybys/wasm-util": "^0.10.1" }, "peerDependencies": { "@emnapi/core": "^1.7.1", "@emnapi/runtime": "^1.7.1" } }, "sha512-3NQNNgA1YSlJb/kMH1ildASP9HW7/7kYnRI2szWJaofaS1hWmbGI4H+d3+22aGzXXN9IJ+n+GiFVcGipJP18ow=="],
|
"@napi-rs/wasm-runtime": ["@napi-rs/wasm-runtime@1.1.4", "", { "dependencies": { "@tybys/wasm-util": "^0.10.1" }, "peerDependencies": { "@emnapi/core": "^1.7.1", "@emnapi/runtime": "^1.7.1" } }, "sha512-3NQNNgA1YSlJb/kMH1ildASP9HW7/7kYnRI2szWJaofaS1hWmbGI4H+d3+22aGzXXN9IJ+n+GiFVcGipJP18ow=="],
|
||||||
@@ -288,6 +291,8 @@
|
|||||||
|
|
||||||
"@parcel/watcher-win32-x64": ["@parcel/watcher-win32-x64@2.5.6", "", { "os": "win32", "cpu": "x64" }, "sha512-hbQlYcCq5dlAX9Qx+kFb0FHue6vbjlf0FrNzSKdYK2APUf7tGfGxQCk2ihEREmbR6ZMc0MVAD5RIX/41gpUzTw=="],
|
"@parcel/watcher-win32-x64": ["@parcel/watcher-win32-x64@2.5.6", "", { "os": "win32", "cpu": "x64" }, "sha512-hbQlYcCq5dlAX9Qx+kFb0FHue6vbjlf0FrNzSKdYK2APUf7tGfGxQCk2ihEREmbR6ZMc0MVAD5RIX/41gpUzTw=="],
|
||||||
|
|
||||||
|
"@pkgr/core": ["@pkgr/core@0.2.9", "", {}, "sha512-QNqXyfVS2wm9hweSYD2O7F0G06uurj9kZ96TRQE5Y9hU7+tgdZwIkbAKc5Ocy1HxEY2kuDQa6cQ1WRs/O5LFKA=="],
|
||||||
|
|
||||||
"@polka/url": ["@polka/url@1.0.0-next.29", "", {}, "sha512-wwQAWhWSuHaag8c4q/KN/vCoeOJYshAIvMQwD4GpSb3OiZklFfvAgmj0VCBBImRpuF/aFgIRzllXlVX93Jevww=="],
|
"@polka/url": ["@polka/url@1.0.0-next.29", "", {}, "sha512-wwQAWhWSuHaag8c4q/KN/vCoeOJYshAIvMQwD4GpSb3OiZklFfvAgmj0VCBBImRpuF/aFgIRzllXlVX93Jevww=="],
|
||||||
|
|
||||||
"@rolldown/binding-android-arm64": ["@rolldown/binding-android-arm64@1.0.0-rc.17", "", { "os": "android", "cpu": "arm64" }, "sha512-s70pVGhw4zqGeFnXWvAzJDlvxhlRollagdCCKRgOsgUOH3N1l0LIxf83AtGzmb5SiVM4Hjl5HyarMRfdfj3DaQ=="],
|
"@rolldown/binding-android-arm64": ["@rolldown/binding-android-arm64@1.0.0-rc.17", "", { "os": "android", "cpu": "arm64" }, "sha512-s70pVGhw4zqGeFnXWvAzJDlvxhlRollagdCCKRgOsgUOH3N1l0LIxf83AtGzmb5SiVM4Hjl5HyarMRfdfj3DaQ=="],
|
||||||
@@ -454,6 +459,8 @@
|
|||||||
|
|
||||||
"@vue/devtools-shared": ["@vue/devtools-shared@8.1.1", "", {}, "sha512-+h4ttmJYl/txpxHKaoZcaKpC+pvckgLzIDiSQlaQ7kKthKh8KuwoLW2D8hPJEnqKzXOvu15UHEoGyngAXCz0EQ=="],
|
"@vue/devtools-shared": ["@vue/devtools-shared@8.1.1", "", {}, "sha512-+h4ttmJYl/txpxHKaoZcaKpC+pvckgLzIDiSQlaQ7kKthKh8KuwoLW2D8hPJEnqKzXOvu15UHEoGyngAXCz0EQ=="],
|
||||||
|
|
||||||
|
"@vue/eslint-config-prettier": ["@vue/eslint-config-prettier@10.2.0", "", { "dependencies": { "eslint-config-prettier": "^10.0.1", "eslint-plugin-prettier": "^5.2.2" }, "peerDependencies": { "eslint": ">= 8.21.0", "prettier": ">= 3.0.0" } }, "sha512-GL3YBLwv/+b86yHcNNfPJxOTtVFJ4Mbc9UU3zR+KVoG7SwGTjPT+32fXamscNumElhcpXW3mT0DgzS9w32S7Bw=="],
|
||||||
|
|
||||||
"@vue/eslint-config-typescript": ["@vue/eslint-config-typescript@14.7.0", "", { "dependencies": { "@typescript-eslint/utils": "^8.56.0", "fast-glob": "^3.3.3", "typescript-eslint": "^8.56.0", "vue-eslint-parser": "^10.4.0" }, "peerDependencies": { "eslint": "^9.10.0 || ^10.0.0", "eslint-plugin-vue": "^9.28.0 || ^10.0.0", "typescript": ">=4.8.4" }, "optionalPeers": ["typescript"] }, "sha512-iegbMINVc+seZ/QxtzWiOBozctrHiF2WvGedruu2EbLujg9VuU0FQiNcN2z1ycuaoKKpF4m2qzB5HDEMKbxtIg=="],
|
"@vue/eslint-config-typescript": ["@vue/eslint-config-typescript@14.7.0", "", { "dependencies": { "@typescript-eslint/utils": "^8.56.0", "fast-glob": "^3.3.3", "typescript-eslint": "^8.56.0", "vue-eslint-parser": "^10.4.0" }, "peerDependencies": { "eslint": "^9.10.0 || ^10.0.0", "eslint-plugin-vue": "^9.28.0 || ^10.0.0", "typescript": ">=4.8.4" }, "optionalPeers": ["typescript"] }, "sha512-iegbMINVc+seZ/QxtzWiOBozctrHiF2WvGedruu2EbLujg9VuU0FQiNcN2z1ycuaoKKpF4m2qzB5HDEMKbxtIg=="],
|
||||||
|
|
||||||
"@vue/language-core": ["@vue/language-core@3.2.7", "", { "dependencies": { "@volar/language-core": "2.4.28", "@vue/compiler-dom": "^3.5.0", "@vue/shared": "^3.5.0", "alien-signals": "^3.1.2", "muggle-string": "^0.4.1", "path-browserify": "^1.0.1", "picomatch": "^4.0.4" } }, "sha512-Gn4q/tRxbpVGLEuARQ43p3YELlNAFgRUVCgW9U5Cr+5q4vfD2bWDWpl3ABbJMXUt5xlE1dF8dkigg2aUq7JYYw=="],
|
"@vue/language-core": ["@vue/language-core@3.2.7", "", { "dependencies": { "@volar/language-core": "2.4.28", "@vue/compiler-dom": "^3.5.0", "@vue/shared": "^3.5.0", "alien-signals": "^3.1.2", "muggle-string": "^0.4.1", "path-browserify": "^1.0.1", "picomatch": "^4.0.4" } }, "sha512-Gn4q/tRxbpVGLEuARQ43p3YELlNAFgRUVCgW9U5Cr+5q4vfD2bWDWpl3ABbJMXUt5xlE1dF8dkigg2aUq7JYYw=="],
|
||||||
@@ -638,12 +645,14 @@
|
|||||||
|
|
||||||
"escape-string-regexp": ["escape-string-regexp@4.0.0", "", {}, "sha512-TtpcNJ3XAzx3Gq8sWRzJaVajRs0uVxA2YAkdb1jm2YkPz4G6egUFAyA3n5vtEIZefPk5Wa4UXbKuS5fKkJWdgA=="],
|
"escape-string-regexp": ["escape-string-regexp@4.0.0", "", {}, "sha512-TtpcNJ3XAzx3Gq8sWRzJaVajRs0uVxA2YAkdb1jm2YkPz4G6egUFAyA3n5vtEIZefPk5Wa4UXbKuS5fKkJWdgA=="],
|
||||||
|
|
||||||
"eslint": ["eslint@10.2.1", "", { "dependencies": { "@eslint-community/eslint-utils": "^4.8.0", "@eslint-community/regexpp": "^4.12.2", "@eslint/config-array": "^0.23.5", "@eslint/config-helpers": "^0.5.5", "@eslint/core": "^1.2.1", "@eslint/plugin-kit": "^0.7.1", "@humanfs/node": "^0.16.6", "@humanwhocodes/module-importer": "^1.0.1", "@humanwhocodes/retry": "^0.4.2", "@types/estree": "^1.0.6", "ajv": "^6.14.0", "cross-spawn": "^7.0.6", "debug": "^4.3.2", "escape-string-regexp": "^4.0.0", "eslint-scope": "^9.1.2", "eslint-visitor-keys": "^5.0.1", "espree": "^11.2.0", "esquery": "^1.7.0", "esutils": "^2.0.2", "fast-deep-equal": "^3.1.3", "file-entry-cache": "^8.0.0", "find-up": "^5.0.0", "glob-parent": "^6.0.2", "ignore": "^5.2.0", "imurmurhash": "^0.1.4", "is-glob": "^4.0.0", "json-stable-stringify-without-jsonify": "^1.0.1", "minimatch": "^10.2.4", "natural-compare": "^1.4.0", "optionator": "^0.9.3" }, "peerDependencies": { "jiti": "*" }, "optionalPeers": ["jiti"], "bin": { "eslint": "bin/eslint.js" } }, "sha512-wiyGaKsDgqXvF40P8mDwiUp/KQjE1FdrIEJsM8PZ3XCiniTMXS3OHWWUe5FI5agoCnr8x4xPrTDZuxsBlNHl+Q=="],
|
"eslint": ["eslint@10.3.0", "", { "dependencies": { "@eslint-community/eslint-utils": "^4.8.0", "@eslint-community/regexpp": "^4.12.2", "@eslint/config-array": "^0.23.5", "@eslint/config-helpers": "^0.5.5", "@eslint/core": "^1.2.1", "@eslint/plugin-kit": "^0.7.1", "@humanfs/node": "^0.16.6", "@humanwhocodes/module-importer": "^1.0.1", "@humanwhocodes/retry": "^0.4.2", "@types/estree": "^1.0.6", "ajv": "^6.14.0", "cross-spawn": "^7.0.6", "debug": "^4.3.2", "escape-string-regexp": "^4.0.0", "eslint-scope": "^9.1.2", "eslint-visitor-keys": "^5.0.1", "espree": "^11.2.0", "esquery": "^1.7.0", "esutils": "^2.0.2", "fast-deep-equal": "^3.1.3", "file-entry-cache": "^8.0.0", "find-up": "^5.0.0", "glob-parent": "^6.0.2", "ignore": "^5.2.0", "imurmurhash": "^0.1.4", "is-glob": "^4.0.0", "json-stable-stringify-without-jsonify": "^1.0.1", "minimatch": "^10.2.4", "natural-compare": "^1.4.0", "optionator": "^0.9.3" }, "peerDependencies": { "jiti": "*" }, "optionalPeers": ["jiti"], "bin": { "eslint": "bin/eslint.js" } }, "sha512-XbEXaRva5cF0ZQB8w6MluHA0kZZfV2DuCMJ3ozyEOHLwDpZX2Lmm/7Pp0xdJmI0GL1W05VH5VwIFHEm1Vcw2gw=="],
|
||||||
|
|
||||||
"eslint-config-prettier": ["eslint-config-prettier@10.1.8", "", { "peerDependencies": { "eslint": ">=7.0.0" }, "bin": { "eslint-config-prettier": "bin/cli.js" } }, "sha512-82GZUjRS0p/jganf6q1rEO25VSoHH0hKPCTrgillPjdI/3bgBhAE1QzHrHTizjpRvy6pGAvKjDJtk2pF9NDq8w=="],
|
"eslint-config-prettier": ["eslint-config-prettier@10.1.8", "", { "peerDependencies": { "eslint": ">=7.0.0" }, "bin": { "eslint-config-prettier": "bin/cli.js" } }, "sha512-82GZUjRS0p/jganf6q1rEO25VSoHH0hKPCTrgillPjdI/3bgBhAE1QzHrHTizjpRvy6pGAvKjDJtk2pF9NDq8w=="],
|
||||||
|
|
||||||
"eslint-plugin-oxlint": ["eslint-plugin-oxlint@1.60.0", "", { "dependencies": { "jsonc-parser": "^3.3.1" }, "peerDependencies": { "oxlint": "~1.60.0" } }, "sha512-9RUD23k7ablez1qg7JWnyPYPOlbucDDqaDr+qNUi0TbIQCPqIPCLzfllgqKF9lOxlg+l17H8hISErmarvm2J1w=="],
|
"eslint-plugin-oxlint": ["eslint-plugin-oxlint@1.60.0", "", { "dependencies": { "jsonc-parser": "^3.3.1" }, "peerDependencies": { "oxlint": "~1.60.0" } }, "sha512-9RUD23k7ablez1qg7JWnyPYPOlbucDDqaDr+qNUi0TbIQCPqIPCLzfllgqKF9lOxlg+l17H8hISErmarvm2J1w=="],
|
||||||
|
|
||||||
|
"eslint-plugin-prettier": ["eslint-plugin-prettier@5.5.5", "", { "dependencies": { "prettier-linter-helpers": "^1.0.1", "synckit": "^0.11.12" }, "peerDependencies": { "@types/eslint": ">=8.0.0", "eslint": ">=8.0.0", "eslint-config-prettier": ">= 7.0.0 <10.0.0 || >=10.1.0", "prettier": ">=3.0.0" }, "optionalPeers": ["@types/eslint", "eslint-config-prettier"] }, "sha512-hscXkbqUZ2sPithAuLm5MXL+Wph+U7wHngPBv9OMWwlP8iaflyxpjTYZkmdgB4/vPIhemRlBEoLrH7UC1n7aUw=="],
|
||||||
|
|
||||||
"eslint-plugin-vue": ["eslint-plugin-vue@10.8.0", "", { "dependencies": { "@eslint-community/eslint-utils": "^4.4.0", "natural-compare": "^1.4.0", "nth-check": "^2.1.1", "postcss-selector-parser": "^7.1.0", "semver": "^7.6.3", "xml-name-validator": "^4.0.0" }, "peerDependencies": { "@stylistic/eslint-plugin": "^2.0.0 || ^3.0.0 || ^4.0.0 || ^5.0.0", "@typescript-eslint/parser": "^7.0.0 || ^8.0.0", "eslint": "^8.57.0 || ^9.0.0 || ^10.0.0", "vue-eslint-parser": "^10.0.0" }, "optionalPeers": ["@stylistic/eslint-plugin", "@typescript-eslint/parser"] }, "sha512-f1J/tcbnrpgC8suPN5AtdJ5MQjuXbSU9pGRSSYAuF3SHoiYCOdEX6O22pLaRyLHXvDcOe+O5ENgc1owQ587agA=="],
|
"eslint-plugin-vue": ["eslint-plugin-vue@10.8.0", "", { "dependencies": { "@eslint-community/eslint-utils": "^4.4.0", "natural-compare": "^1.4.0", "nth-check": "^2.1.1", "postcss-selector-parser": "^7.1.0", "semver": "^7.6.3", "xml-name-validator": "^4.0.0" }, "peerDependencies": { "@stylistic/eslint-plugin": "^2.0.0 || ^3.0.0 || ^4.0.0 || ^5.0.0", "@typescript-eslint/parser": "^7.0.0 || ^8.0.0", "eslint": "^8.57.0 || ^9.0.0 || ^10.0.0", "vue-eslint-parser": "^10.0.0" }, "optionalPeers": ["@stylistic/eslint-plugin", "@typescript-eslint/parser"] }, "sha512-f1J/tcbnrpgC8suPN5AtdJ5MQjuXbSU9pGRSSYAuF3SHoiYCOdEX6O22pLaRyLHXvDcOe+O5ENgc1owQ587agA=="],
|
||||||
|
|
||||||
"eslint-scope": ["eslint-scope@9.1.2", "", { "dependencies": { "@types/esrecurse": "^4.3.1", "@types/estree": "^1.0.8", "esrecurse": "^4.3.0", "estraverse": "^5.2.0" } }, "sha512-xS90H51cKw0jltxmvmHy2Iai1LIqrfbw57b79w/J7MfvDfkIkFZ+kj6zC3BjtUwh150HsSSdxXZcsuv72miDFQ=="],
|
"eslint-scope": ["eslint-scope@9.1.2", "", { "dependencies": { "@types/esrecurse": "^4.3.1", "@types/estree": "^1.0.8", "esrecurse": "^4.3.0", "estraverse": "^5.2.0" } }, "sha512-xS90H51cKw0jltxmvmHy2Iai1LIqrfbw57b79w/J7MfvDfkIkFZ+kj6zC3BjtUwh150HsSSdxXZcsuv72miDFQ=="],
|
||||||
@@ -668,6 +677,8 @@
|
|||||||
|
|
||||||
"fast-deep-equal": ["fast-deep-equal@3.1.3", "", {}, "sha512-f3qQ9oQy9j2AhBe/H9VC91wLmKBCCU/gDOnKNAYG5hswO7BLKj09Hc5HYNz9cGI++xlpDCIgDaitVs03ATR84Q=="],
|
"fast-deep-equal": ["fast-deep-equal@3.1.3", "", {}, "sha512-f3qQ9oQy9j2AhBe/H9VC91wLmKBCCU/gDOnKNAYG5hswO7BLKj09Hc5HYNz9cGI++xlpDCIgDaitVs03ATR84Q=="],
|
||||||
|
|
||||||
|
"fast-diff": ["fast-diff@1.3.0", "", {}, "sha512-VxPP4NqbUjj6MaAOafWeUn2cXWLcCtljklUtZf0Ind4XQ+QPtmA0b18zZy0jIQx+ExRVCR/ZQpBmik5lXshNsw=="],
|
||||||
|
|
||||||
"fast-glob": ["fast-glob@3.3.3", "", { "dependencies": { "@nodelib/fs.stat": "^2.0.2", "@nodelib/fs.walk": "^1.2.3", "glob-parent": "^5.1.2", "merge2": "^1.3.0", "micromatch": "^4.0.8" } }, "sha512-7MptL8U0cqcFdzIzwOTHoilX9x5BrNqye7Z/LuC7kCMRio1EMSyqRK3BEAUD7sXRq4iT4AzTVuZdhgQ2TCvYLg=="],
|
"fast-glob": ["fast-glob@3.3.3", "", { "dependencies": { "@nodelib/fs.stat": "^2.0.2", "@nodelib/fs.walk": "^1.2.3", "glob-parent": "^5.1.2", "merge2": "^1.3.0", "micromatch": "^4.0.8" } }, "sha512-7MptL8U0cqcFdzIzwOTHoilX9x5BrNqye7Z/LuC7kCMRio1EMSyqRK3BEAUD7sXRq4iT4AzTVuZdhgQ2TCvYLg=="],
|
||||||
|
|
||||||
"fast-json-stable-stringify": ["fast-json-stable-stringify@2.1.0", "", {}, "sha512-lhd/wF+Lk98HZoTCtlVraHtfh5XYijIjalXck7saUtuanSDyLMxnHhSXEDJqHxD7msR8D0uCmqlkwjCV8xvwHw=="],
|
"fast-json-stable-stringify": ["fast-json-stable-stringify@2.1.0", "", {}, "sha512-lhd/wF+Lk98HZoTCtlVraHtfh5XYijIjalXck7saUtuanSDyLMxnHhSXEDJqHxD7msR8D0uCmqlkwjCV8xvwHw=="],
|
||||||
@@ -802,7 +813,7 @@
|
|||||||
|
|
||||||
"lru-cache": ["lru-cache@5.1.1", "", { "dependencies": { "yallist": "^3.0.2" } }, "sha512-KpNARQA3Iwv+jTA0utUVVbrh+Jlrr1Fv0e56GGzAFOXN7dk/FviaDW8LHmK52DlcH4WP2n6gI8vN1aesBFgo9w=="],
|
"lru-cache": ["lru-cache@5.1.1", "", { "dependencies": { "yallist": "^3.0.2" } }, "sha512-KpNARQA3Iwv+jTA0utUVVbrh+Jlrr1Fv0e56GGzAFOXN7dk/FviaDW8LHmK52DlcH4WP2n6gI8vN1aesBFgo9w=="],
|
||||||
|
|
||||||
"lucide": ["lucide@1.11.0", "", {}, "sha512-2uvbGlcztUY+z0Ef++YCaxD6mtzrPsUJ1qWbIfqZrZGRxZBCL3icE6g2nzRTtJ6YywOoXC5blL/NmkLI66en5g=="],
|
"lucide": ["lucide@1.14.0", "", {}, "sha512-IoRC3lHwemJWvsXKcHK90hkgY4h1HGztBL63w2XwFtIu8gFDPp4/kiuqVtlN3vaM9bxsLQ4ZUBJfGsbKFaB2IA=="],
|
||||||
|
|
||||||
"magic-string": ["magic-string@0.30.21", "", { "dependencies": { "@jridgewell/sourcemap-codec": "^1.5.5" } }, "sha512-vd2F4YUyEXKGcLHoq+TEyCjxueSeHnFxyyjNp80yg0XV4vUhnDer/lvvlqM/arB5bXQN5K2/3oinyCRyx8T2CQ=="],
|
"magic-string": ["magic-string@0.30.21", "", { "dependencies": { "@jridgewell/sourcemap-codec": "^1.5.5" } }, "sha512-vd2F4YUyEXKGcLHoq+TEyCjxueSeHnFxyyjNp80yg0XV4vUhnDer/lvvlqM/arB5bXQN5K2/3oinyCRyx8T2CQ=="],
|
||||||
|
|
||||||
@@ -892,6 +903,8 @@
|
|||||||
|
|
||||||
"prettier": ["prettier@3.8.3", "", { "bin": { "prettier": "bin/prettier.cjs" } }, "sha512-7igPTM53cGHMW8xWuVTydi2KO233VFiTNyF5hLJqpilHfmn8C8gPf+PS7dUT64YcXFbiMGZxS9pCSxL/Dxm/Jw=="],
|
"prettier": ["prettier@3.8.3", "", { "bin": { "prettier": "bin/prettier.cjs" } }, "sha512-7igPTM53cGHMW8xWuVTydi2KO233VFiTNyF5hLJqpilHfmn8C8gPf+PS7dUT64YcXFbiMGZxS9pCSxL/Dxm/Jw=="],
|
||||||
|
|
||||||
|
"prettier-linter-helpers": ["prettier-linter-helpers@1.0.1", "", { "dependencies": { "fast-diff": "^1.1.2" } }, "sha512-SxToR7P8Y2lWmv/kTzVLC1t/GDI2WGjMwNhLLE9qtH8Q13C+aEmuRlzDst4Up4s0Wc8sF2M+J57iB3cMLqftfg=="],
|
||||||
|
|
||||||
"punycode": ["punycode@2.3.1", "", {}, "sha512-vYt7UD1U9Wg6138shLtLOvdAu+8DsC/ilFtEVHcH+wydcSpNE20AfSOduf6MkRFahL5FY7X1oU7nKVZFtfq8Fg=="],
|
"punycode": ["punycode@2.3.1", "", {}, "sha512-vYt7UD1U9Wg6138shLtLOvdAu+8DsC/ilFtEVHcH+wydcSpNE20AfSOduf6MkRFahL5FY7X1oU7nKVZFtfq8Fg=="],
|
||||||
|
|
||||||
"quansync": ["quansync@0.2.11", "", {}, "sha512-AifT7QEbW9Nri4tAwR5M/uzpBuqfZf+zwaEM/QkzEjj7NBuFD2rBuy0K3dE+8wltbezDV7JMA0WfnCPYRSYbXA=="],
|
"quansync": ["quansync@0.2.11", "", {}, "sha512-AifT7QEbW9Nri4tAwR5M/uzpBuqfZf+zwaEM/QkzEjj7NBuFD2rBuy0K3dE+8wltbezDV7JMA0WfnCPYRSYbXA=="],
|
||||||
@@ -988,6 +1001,8 @@
|
|||||||
|
|
||||||
"sync-message-port": ["sync-message-port@1.2.0", "", {}, "sha512-gAQ9qrUN/UCypHtGFbbe7Rc/f9bzO88IwrG8TDo/aMKAApKyD6E3W4Cm0EfhfBb6Z6SKt59tTCTfD+n1xmAvMg=="],
|
"sync-message-port": ["sync-message-port@1.2.0", "", {}, "sha512-gAQ9qrUN/UCypHtGFbbe7Rc/f9bzO88IwrG8TDo/aMKAApKyD6E3W4Cm0EfhfBb6Z6SKt59tTCTfD+n1xmAvMg=="],
|
||||||
|
|
||||||
|
"synckit": ["synckit@0.11.12", "", { "dependencies": { "@pkgr/core": "^0.2.9" } }, "sha512-Bh7QjT8/SuKUIfObSXNHNSK6WHo6J1tHCqJsuaFDP7gP0fkzSfTxI8y85JrppZ0h8l0maIgc2tfuZQ6/t3GtnQ=="],
|
||||||
|
|
||||||
"tinyexec": ["tinyexec@1.1.1", "", {}, "sha512-VKS/ZaQhhkKFMANmAOhhXVoIfBXblQxGX1myCQ2faQrfmobMftXeJPcZGp0gS07ocvGJWDLZGyOZDadDBqYIJg=="],
|
"tinyexec": ["tinyexec@1.1.1", "", {}, "sha512-VKS/ZaQhhkKFMANmAOhhXVoIfBXblQxGX1myCQ2faQrfmobMftXeJPcZGp0gS07ocvGJWDLZGyOZDadDBqYIJg=="],
|
||||||
|
|
||||||
"tinyglobby": ["tinyglobby@0.2.16", "", { "dependencies": { "fdir": "^6.5.0", "picomatch": "^4.0.4" } }, "sha512-pn99VhoACYR8nFHhxqix+uvsbXineAasWm5ojXoN8xEwK5Kd3/TrhNn1wByuD52UxWRLy8pu+kRMniEi6Eq9Zg=="],
|
"tinyglobby": ["tinyglobby@0.2.16", "", { "dependencies": { "fdir": "^6.5.0", "picomatch": "^4.0.4" } }, "sha512-pn99VhoACYR8nFHhxqix+uvsbXineAasWm5ojXoN8xEwK5Kd3/TrhNn1wByuD52UxWRLy8pu+kRMniEi6Eq9Zg=="],
|
||||||
|
|||||||
+24
-18
@@ -1,26 +1,32 @@
|
|||||||
import { globalIgnores } from 'eslint/config'
|
import { globalIgnores } from "eslint/config";
|
||||||
import { defineConfigWithVueTs, vueTsConfigs } from '@vue/eslint-config-typescript'
|
import {
|
||||||
import pluginVue from 'eslint-plugin-vue'
|
defineConfigWithVueTs,
|
||||||
import pluginOxlint from 'eslint-plugin-oxlint'
|
vueTsConfigs,
|
||||||
import skipFormatting from 'eslint-config-prettier/flat'
|
} from "@vue/eslint-config-typescript";
|
||||||
|
import pluginVue from "eslint-plugin-vue";
|
||||||
|
import skipFormatting from "@vue/eslint-config-prettier/skip-formatting";
|
||||||
|
import { configureVueProject } from "@vue/eslint-config-typescript";
|
||||||
|
import pluginOxlint from "eslint-plugin-oxlint";
|
||||||
|
|
||||||
// To allow more languages other than `ts` in `.vue` files, uncomment the following lines:
|
configureVueProject({ scriptLangs: ["ts", "tsx"] });
|
||||||
// import { configureVueProject } from '@vue/eslint-config-typescript'
|
|
||||||
// configureVueProject({ scriptLangs: ['ts', 'tsx'] })
|
|
||||||
// More info at https://github.com/vuejs/eslint-config-typescript/#advanced-setup
|
|
||||||
|
|
||||||
export default defineConfigWithVueTs(
|
export default defineConfigWithVueTs(
|
||||||
{
|
{
|
||||||
name: 'app/files-to-lint',
|
name: "app/files-to-lint",
|
||||||
files: ['**/*.{vue,ts,mts,tsx}'],
|
files: ["**/*.{ts,mts,tsx,vue}"],
|
||||||
},
|
},
|
||||||
|
|
||||||
globalIgnores(['**/dist/**', '**/dist-ssr/**', '**/coverage/**']),
|
globalIgnores(["**/dist/**"]),
|
||||||
|
|
||||||
...pluginVue.configs['flat/essential'],
|
|
||||||
vueTsConfigs.recommended,
|
|
||||||
|
|
||||||
...pluginOxlint.buildFromOxlintConfigFile('.oxlintrc.json'),
|
|
||||||
|
|
||||||
|
pluginVue.configs["flat/recommended"],
|
||||||
|
vueTsConfigs.strictTypeChecked,
|
||||||
|
vueTsConfigs.stylisticTypeChecked,
|
||||||
|
...pluginOxlint.buildFromOxlintConfigFile(".oxlintrc.json"),
|
||||||
skipFormatting,
|
skipFormatting,
|
||||||
)
|
|
||||||
|
{
|
||||||
|
rules: {
|
||||||
|
"vue/no-v-html": "off",
|
||||||
|
},
|
||||||
|
},
|
||||||
|
);
|
||||||
|
|||||||
+11
-8
@@ -1,9 +1,9 @@
|
|||||||
{
|
{
|
||||||
"name": "md-blog",
|
"name": "md-blog",
|
||||||
"version": "1.3.0",
|
"version": "1.9.2",
|
||||||
"private": true,
|
"private": true,
|
||||||
"type": "module",
|
"type": "module",
|
||||||
"repository": "https://github.com/klemek/md-blog",
|
"repository": "https://git.klemek.fr/klemek/md-blog",
|
||||||
"scripts": {
|
"scripts": {
|
||||||
"dev": "vite",
|
"dev": "vite",
|
||||||
"build": "run-p type-check \"build-only {@}\" -- && run-p post-build",
|
"build": "run-p type-check \"build-only {@}\" -- && run-p post-build",
|
||||||
@@ -12,14 +12,17 @@
|
|||||||
"build-only": "vite build",
|
"build-only": "vite build",
|
||||||
"type-check": "vue-tsc --build",
|
"type-check": "vue-tsc --build",
|
||||||
"lint": "run-s lint:*",
|
"lint": "run-s lint:*",
|
||||||
"lint:oxlint": "oxlint . --fix",
|
"lint-fix": "run-s lint-fix:*",
|
||||||
"lint:eslint": "eslint . --fix --cache",
|
"lint:eslint": "eslint . --cache",
|
||||||
"format": "prettier --write src/ *.ts *.json"
|
"lint:oxlint": "oxlint .",
|
||||||
|
"lint-fix:eslint": "eslint . --fix --cache",
|
||||||
|
"lint-fix:oxlint": "oxlint . --fix"
|
||||||
},
|
},
|
||||||
"dependencies": {
|
"dependencies": {
|
||||||
|
"@keithclark/shaderview": "https://github.com/keithclark/shaderview/archive/refs/tags/1.2.0.tar.gz",
|
||||||
"highlight.js": "^11.11.1",
|
"highlight.js": "^11.11.1",
|
||||||
"katex": "^0.16.45",
|
"katex": "^0.16.45",
|
||||||
"lucide": "^1.11.0",
|
"lucide": "^1.14.0",
|
||||||
"mermaid": "^11.14.0",
|
"mermaid": "^11.14.0",
|
||||||
"vue": "^3.5.33",
|
"vue": "^3.5.33",
|
||||||
"vue-router": "^5.0.6"
|
"vue-router": "^5.0.6"
|
||||||
@@ -28,10 +31,10 @@
|
|||||||
"@tsconfig/node24": "^24.0.4",
|
"@tsconfig/node24": "^24.0.4",
|
||||||
"@types/node": "^24.12.2",
|
"@types/node": "^24.12.2",
|
||||||
"@vitejs/plugin-vue": "^6.0.6",
|
"@vitejs/plugin-vue": "^6.0.6",
|
||||||
|
"@vue/eslint-config-prettier": "^10.2.0",
|
||||||
"@vue/eslint-config-typescript": "^14.7.0",
|
"@vue/eslint-config-typescript": "^14.7.0",
|
||||||
"@vue/tsconfig": "^0.9.1",
|
"@vue/tsconfig": "^0.9.1",
|
||||||
"eslint": "^10.2.1",
|
"eslint": "^10.3.0",
|
||||||
"eslint-config-prettier": "^10.1.8",
|
|
||||||
"eslint-plugin-oxlint": "~1.60.0",
|
"eslint-plugin-oxlint": "~1.60.0",
|
||||||
"eslint-plugin-vue": "~10.8.0",
|
"eslint-plugin-vue": "~10.8.0",
|
||||||
"feed": "^5.2.1",
|
"feed": "^5.2.1",
|
||||||
|
|||||||
+83
-71
@@ -1,129 +1,141 @@
|
|||||||
import fs from 'fs'
|
import fs from "fs";
|
||||||
import process from 'process'
|
import process from "process";
|
||||||
import { Feed } from 'feed'
|
import { Feed } from "feed";
|
||||||
import articlesConfig from './articles/config.json'
|
import articlesConfig from "./articles/config.json";
|
||||||
|
|
||||||
function getFiles(dir: string): string[] {
|
function getFiles(dir: string): string[] {
|
||||||
return fs.readdirSync(dir).flatMap((name) => {
|
return fs.readdirSync(dir).flatMap((name) => {
|
||||||
const path = `${dir}/${name}`
|
const path = `${dir}/${name}`;
|
||||||
if (fs.statSync(path).isDirectory()) {
|
if (fs.statSync(path).isDirectory()) {
|
||||||
return getFiles(path)
|
return getFiles(path);
|
||||||
} else {
|
} else {
|
||||||
return [path]
|
return [path];
|
||||||
}
|
}
|
||||||
})
|
});
|
||||||
}
|
}
|
||||||
|
|
||||||
const METADATA_BLOCK_REGEX = /^---\n([\s\S]*?)---\n/m
|
const METADATA_BLOCK_REGEX = /^---\n([\s\S]*?)---\n/m;
|
||||||
const METADATA_REGEX = /^(\w+):(.*)$/gm
|
const METADATA_REGEX = /^(\w+):(.*)$/gm;
|
||||||
|
|
||||||
function readArticleMetadata(path: string): Record<string, string> | null {
|
function readArticleMetadata(path: string): Record<string, string> | null {
|
||||||
const content = fs.readFileSync(path, { encoding: 'utf8' })
|
const content = fs.readFileSync(path, { encoding: "utf8" });
|
||||||
const match: RegExpExecArray | null = METADATA_BLOCK_REGEX.exec(content)
|
const match: RegExpExecArray | null = METADATA_BLOCK_REGEX.exec(content);
|
||||||
if (!match || !match[1]) {
|
if (!match?.[1]) {
|
||||||
console.warn(`No metadata for: ${path}`)
|
console.warn(`No metadata for: ${path}`);
|
||||||
return null
|
return null;
|
||||||
}
|
}
|
||||||
let subMatch: RegExpExecArray | null = null
|
let subMatch: RegExpExecArray | null = null;
|
||||||
const metadata: Record<string, string> = {
|
const metadata: Record<string, string> = {
|
||||||
path: path.replaceAll('/index.md', ''),
|
path: path.replaceAll("/index.md", ""),
|
||||||
}
|
};
|
||||||
do {
|
do {
|
||||||
subMatch = METADATA_REGEX.exec(match[1])
|
subMatch = METADATA_REGEX.exec(match[1]);
|
||||||
if (subMatch && subMatch[1] && subMatch[2]) {
|
if (subMatch?.[1] && subMatch[2]) {
|
||||||
metadata[subMatch[1]] = subMatch[2].trim()
|
metadata[subMatch[1]] = subMatch[2].trim();
|
||||||
}
|
}
|
||||||
} while (subMatch)
|
} while (subMatch);
|
||||||
return metadata
|
return metadata;
|
||||||
}
|
}
|
||||||
|
|
||||||
function formatArticlePage(metadata: Record<string, string>, baseHtml: string): string {
|
function formatArticlePage(
|
||||||
let outHtml = baseHtml
|
metadata: Record<string, string>,
|
||||||
outHtml = outHtml.replace(/<.*?property="og:url".*?>/gm, '')
|
baseHtml: string,
|
||||||
|
): string {
|
||||||
|
let outHtml = baseHtml;
|
||||||
|
outHtml = outHtml.replace(/<.*?property="og:url".*?>/gm, "");
|
||||||
outHtml = outHtml.replace(
|
outHtml = outHtml.replace(
|
||||||
/<\/head>/gm,
|
/<\/head>/gm,
|
||||||
`<meta property="og:url" content="${articlesConfig['base_url']}${metadata.path}/">\n</head>`,
|
`<meta property="og:url" content="${articlesConfig.base_url}${metadata.path}/">\n</head>`,
|
||||||
)
|
);
|
||||||
const blog_title = articlesConfig['title']?.replace(/(<([^>]+)>)/gi, '').trim()
|
const blog_title = articlesConfig.title.replace(/(<([^>]+)>)/gi, "").trim();
|
||||||
if (metadata.title) {
|
if (metadata.title) {
|
||||||
const title = metadata.title.replace(/(<([^>]+)>)/gi, '').trim()
|
const title = metadata.title.replace(/(<([^>]+)>)/gi, "").trim();
|
||||||
outHtml = outHtml.replace(/<title>.*?<\/title>/gm, `<title>${blog_title} — ${title}</title>`)
|
outHtml = outHtml.replace(
|
||||||
outHtml = outHtml.replace(/<.*?property="og:title".*?>/gm, '')
|
/<title>.*?<\/title>/gm,
|
||||||
|
`<title>${blog_title} — ${title}</title>`,
|
||||||
|
);
|
||||||
|
outHtml = outHtml.replace(/<.*?property="og:title".*?>/gm, "");
|
||||||
outHtml = outHtml.replace(
|
outHtml = outHtml.replace(
|
||||||
/<\/head>/gm,
|
/<\/head>/gm,
|
||||||
`<meta property="og:title" content="${title}">\n</head>`,
|
`<meta property="og:title" content="${title}">\n</head>`,
|
||||||
)
|
);
|
||||||
outHtml = outHtml.replace(/<.*?property="og:description".*?>/gm, '')
|
outHtml = outHtml.replace(/<.*?property="og:description".*?>/gm, "");
|
||||||
outHtml = outHtml.replace(
|
outHtml = outHtml.replace(
|
||||||
/<\/head>/gm,
|
/<\/head>/gm,
|
||||||
`<meta property="og:description" content="${blog_title}">\n</head>`,
|
`<meta property="og:description" content="${blog_title}">\n</head>`,
|
||||||
)
|
);
|
||||||
}
|
}
|
||||||
if (metadata.thumbnail) {
|
if (metadata.thumbnail) {
|
||||||
outHtml = outHtml.replace(/<.*?property="og:image".*?>/gm, '')
|
outHtml = outHtml.replace(/<.*?property="og:image".*?>/gm, "");
|
||||||
outHtml = outHtml.replace(
|
outHtml = outHtml.replace(
|
||||||
/<\/head>/gm,
|
/<\/head>/gm,
|
||||||
`<meta property="og:image" content="${metadata.thumbnail.replace('./', articlesConfig['base_url'] + metadata.path + '/')}">\n</head>`,
|
`<meta property="og:image" content="${metadata.thumbnail.replace("./", articlesConfig.base_url + metadata.path + "/")}">\n</head>`,
|
||||||
)
|
);
|
||||||
}
|
}
|
||||||
return outHtml
|
return outHtml;
|
||||||
}
|
}
|
||||||
|
|
||||||
function addFeedArticle(metadata: Record<string, string>, feed: Feed) {
|
function addFeedArticle(metadata: Record<string, string>, feed: Feed) {
|
||||||
if (metadata.draft !== 'true') {
|
if (metadata.draft !== "true") {
|
||||||
feed.addItem({
|
feed.addItem({
|
||||||
title: metadata.title.replace(/(<([^>]+)>)/gi, '').trim(),
|
title: metadata.title.replace(/(<([^>]+)>)/gi, "").trim(),
|
||||||
id: metadata.path,
|
id: metadata.path,
|
||||||
link: `${articlesConfig['base_url']}${metadata.path}/`,
|
link: `${articlesConfig.base_url}${metadata.path}/`,
|
||||||
date: new Date(Date.parse(metadata.date)),
|
date: new Date(Date.parse(metadata.date)),
|
||||||
image: metadata.thumbnail.replace('./', articlesConfig['base_url'] + metadata.path + '/'),
|
image: metadata.thumbnail.replace(
|
||||||
})
|
"./",
|
||||||
|
articlesConfig.base_url + metadata.path + "/",
|
||||||
|
),
|
||||||
|
});
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
|
||||||
const indexContent = fs.readFileSync('dist/index.html', { encoding: 'utf8' })
|
const indexContent = fs.readFileSync("dist/index.html", { encoding: "utf8" });
|
||||||
|
|
||||||
if (!indexContent) {
|
if (!indexContent) {
|
||||||
console.error('Could not read dist/index.html')
|
console.error("Could not read dist/index.html");
|
||||||
process.exit(1)
|
process.exit(1);
|
||||||
}
|
}
|
||||||
|
|
||||||
const metadatas = getFiles('articles')
|
const metadatas = getFiles("articles")
|
||||||
.filter((path) => path.match(/\/index.md$/))
|
.filter((path) => /\/index.md$/.exec(path))
|
||||||
.map((path) => readArticleMetadata(path))
|
.map((path) => readArticleMetadata(path))
|
||||||
.filter((metadata) => !!metadata)
|
.filter((metadata) => !!metadata);
|
||||||
|
|
||||||
const feed = new Feed({
|
const feed = new Feed({
|
||||||
title: articlesConfig['title']?.replace(/(<([^>]+)>)/gi, '').trim() ?? '',
|
title: articlesConfig.title.replace(/(<([^>]+)>)/gi, "").trim(),
|
||||||
id: articlesConfig['base_url'],
|
id: articlesConfig.base_url,
|
||||||
link: articlesConfig['base_url'],
|
link: articlesConfig.base_url,
|
||||||
language: articlesConfig['lang'],
|
language: articlesConfig.lang,
|
||||||
favicon: articlesConfig['base_url'] + 'articles/favicon.ico',
|
favicon: articlesConfig.base_url + "articles/favicon.ico",
|
||||||
generator: 'md-blog',
|
generator: "md-blog",
|
||||||
feedLinks: {
|
feedLinks: {
|
||||||
json: articlesConfig['base_url'] + 'feed.json',
|
json: articlesConfig.base_url + "feed.json",
|
||||||
atom: articlesConfig['base_url'] + 'atom.xml',
|
atom: articlesConfig.base_url + "atom.xml",
|
||||||
rss: articlesConfig['base_url'] + 'rss',
|
rss: articlesConfig.base_url + "rss",
|
||||||
},
|
},
|
||||||
updated: new Date(
|
updated: new Date(
|
||||||
Math.max(
|
Math.max(
|
||||||
0,
|
0,
|
||||||
...metadatas
|
...metadatas
|
||||||
.filter((metadata) => metadata.draft !== 'true')
|
.filter((metadata) => metadata.draft !== "true")
|
||||||
.map((metadata) => Date.parse(metadata.date)),
|
.map((metadata) => Date.parse(metadata.date)),
|
||||||
),
|
),
|
||||||
),
|
),
|
||||||
})
|
});
|
||||||
|
|
||||||
metadatas.forEach((metadata) => {
|
metadatas.forEach((metadata) => {
|
||||||
fs.writeFileSync(`dist/${metadata.path}/index.html`, formatArticlePage(metadata, indexContent))
|
fs.writeFileSync(
|
||||||
console.info(`Wrote dist/${metadata.path}/index.html`)
|
`dist/${metadata.path}/index.html`,
|
||||||
addFeedArticle(metadata, feed)
|
formatArticlePage(metadata, indexContent),
|
||||||
})
|
);
|
||||||
|
console.info(`Wrote dist/${metadata.path}/index.html`);
|
||||||
|
addFeedArticle(metadata, feed);
|
||||||
|
});
|
||||||
|
|
||||||
fs.writeFileSync('dist/feed.json', feed.json1())
|
fs.writeFileSync("dist/feed.json", feed.json1());
|
||||||
console.info(`Wrote dist/feed.json`)
|
console.info(`Wrote dist/feed.json`);
|
||||||
fs.writeFileSync('dist/atom.xml', feed.atom1())
|
fs.writeFileSync("dist/atom.xml", feed.atom1());
|
||||||
console.info(`Wrote dist/atom.xml`)
|
console.info(`Wrote dist/atom.xml`);
|
||||||
fs.writeFileSync('dist/rss.xml', feed.rss2())
|
fs.writeFileSync("dist/rss.xml", feed.rss2());
|
||||||
console.info(`Wrote dist/rss.xml`)
|
console.info(`Wrote dist/rss.xml`);
|
||||||
|
|||||||
+7
-24
@@ -1,29 +1,12 @@
|
|||||||
<script setup lang="ts">
|
<script setup lang="ts">
|
||||||
import hljs from 'highlight.js'
|
import PageFooter from '@components/PageFooter.vue'
|
||||||
import { createIcons, icons } from 'lucide'
|
import NavBar from '@components/NavBar.vue'
|
||||||
import { onMounted, onUpdated, nextTick } from 'vue'
|
|
||||||
import mermaid from 'mermaid'
|
|
||||||
|
|
||||||
async function update() {
|
|
||||||
setTimeout(async () => {
|
|
||||||
await nextTick()
|
|
||||||
hljs.highlightAll()
|
|
||||||
createIcons({
|
|
||||||
icons,
|
|
||||||
nameAttr: 'icon',
|
|
||||||
attrs: {
|
|
||||||
width: '1.1em',
|
|
||||||
height: '1.1em',
|
|
||||||
},
|
|
||||||
})
|
|
||||||
mermaid.run()
|
|
||||||
}, 100)
|
|
||||||
}
|
|
||||||
|
|
||||||
onMounted(update)
|
|
||||||
onUpdated(update)
|
|
||||||
</script>
|
</script>
|
||||||
|
|
||||||
<template>
|
<template>
|
||||||
<RouterView @vue:mounted="update" @vue:updated="update" />
|
<main>
|
||||||
|
<NavBar />
|
||||||
|
<RouterView />
|
||||||
|
<PageFooter />
|
||||||
|
</main>
|
||||||
</template>
|
</template>
|
||||||
|
|||||||
@@ -0,0 +1,9 @@
|
|||||||
|
<script setup lang="ts">
|
||||||
|
import { BACK_LINK } from '@lib/config'
|
||||||
|
</script>
|
||||||
|
|
||||||
|
<template>
|
||||||
|
<template v-if="$route.fullPath != '/'">
|
||||||
|
<RouterLink class="link-back" to="/"><span v-html="BACK_LINK"></span></RouterLink>
|
||||||
|
</template>
|
||||||
|
</template>
|
||||||
@@ -0,0 +1,13 @@
|
|||||||
|
<script setup lang="ts">
|
||||||
|
import { BASE_URL, TITLE, RSS_LINK, ABOUT_LINK } from '@lib/config'
|
||||||
|
</script>
|
||||||
|
|
||||||
|
<template>
|
||||||
|
<nav>
|
||||||
|
<RouterLink to="/" class="nav-title"><span v-html="TITLE"></span></RouterLink>
|
||||||
|
<span class="nav-items">
|
||||||
|
<RouterLink to="/about/"><span v-html="ABOUT_LINK"></span></RouterLink>
|
||||||
|
<a :href="BASE_URL + 'atom.xml'" v-html="RSS_LINK"></a>
|
||||||
|
</span>
|
||||||
|
</nav>
|
||||||
|
</template>
|
||||||
@@ -1,16 +1,12 @@
|
|||||||
<script setup lang="ts">
|
<script setup lang="ts">
|
||||||
import { REPOSITORY, NAME, VERSION, BASE_URL, TITLE, COPYRIGHT } from '@/lib/meta'
|
import { REPOSITORY, NAME, VERSION, TITLE, COPYRIGHT } from '@lib/config'
|
||||||
import { stripHTML } from '@/lib/strings';
|
import { stripHTML } from '@lib/strings'
|
||||||
</script>
|
</script>
|
||||||
|
|
||||||
<template>
|
<template>
|
||||||
<template v-if="$route.fullPath != '/'">
|
|
||||||
<RouterLink to="/"><i icon="undo-2"></i> Back to home</RouterLink>
|
|
||||||
</template>
|
|
||||||
<hr />
|
|
||||||
<footer>
|
<footer>
|
||||||
{{ stripHTML(TITLE) }} © {{ new Date().getFullYear() }}, <span v-html="COPYRIGHT"></span> | Made with
|
{{ stripHTML(TITLE) }} © {{ new Date().getFullYear() }}, <span v-html="COPYRIGHT"></span> |
|
||||||
<a :href="REPOSITORY">{{ NAME }} {{ VERSION }}</a> |
|
Made with
|
||||||
<a :href="BASE_URL + 'atom.xml'"><i icon="rss"></i> RSS</a>
|
<a :href="REPOSITORY">{{ NAME }} {{ VERSION }}</a>
|
||||||
</footer>
|
</footer>
|
||||||
</template>
|
</template>
|
||||||
|
|||||||
+2
-1
@@ -7,7 +7,8 @@ export interface ArticleMetadata {
|
|||||||
path: string
|
path: string
|
||||||
title: string
|
title: string
|
||||||
date: Date
|
date: Date
|
||||||
author: string
|
updated?: Date
|
||||||
|
author?: string
|
||||||
thumbnail?: string
|
thumbnail?: string
|
||||||
draft?: boolean
|
draft?: boolean
|
||||||
}
|
}
|
||||||
|
|||||||
+92
-51
@@ -1,105 +1,146 @@
|
|||||||
import type { MarkdownData, Article, ArticleMetadata } from '@interfaces'
|
import type { MarkdownData, Article, ArticleMetadata } from "@interfaces";
|
||||||
import katex from 'katex'
|
import katex from "katex";
|
||||||
|
import { nextTick } from "vue";
|
||||||
|
import hljs from "highlight.js";
|
||||||
|
import mermaid from "mermaid";
|
||||||
|
import { createIcons, icons } from "lucide";
|
||||||
|
|
||||||
|
export async function updateDynamicContent() {
|
||||||
|
await nextTick();
|
||||||
|
hljs.highlightAll();
|
||||||
|
createIcons({
|
||||||
|
icons,
|
||||||
|
nameAttr: "icon",
|
||||||
|
attrs: {
|
||||||
|
width: "1.1em",
|
||||||
|
height: "1.1em",
|
||||||
|
},
|
||||||
|
});
|
||||||
|
await mermaid.run();
|
||||||
|
}
|
||||||
|
|
||||||
function parseMetadata(
|
function parseMetadata(
|
||||||
srcAttributes: Record<string, unknown>,
|
srcAttributes: Record<string, unknown>,
|
||||||
pathPrefix: string,
|
pathPrefix: string,
|
||||||
): ArticleMetadata {
|
): ArticleMetadata {
|
||||||
|
const draft = !!srcAttributes.draft;
|
||||||
return {
|
return {
|
||||||
path: pathPrefix,
|
path: pathPrefix,
|
||||||
title: decodeURIComponent((srcAttributes.title as string) ?? 'Untitled'),
|
title:
|
||||||
date: new Date(Date.parse((srcAttributes.date as string) ?? '')),
|
(draft ? "[DRAFT] " : "") +
|
||||||
author: decodeURIComponent((srcAttributes.author as string) ?? ''),
|
decodeURIComponent(
|
||||||
thumbnail: (srcAttributes.thumbnail as string) ?? '',
|
(srcAttributes.title as string | undefined) ?? "Untitled",
|
||||||
draft: !!srcAttributes.draft,
|
),
|
||||||
}
|
date: (srcAttributes.date as string | undefined)
|
||||||
|
? new Date(Date.parse(srcAttributes.date as string))
|
||||||
|
: new Date(),
|
||||||
|
updated: (srcAttributes.updated as string | undefined)
|
||||||
|
? new Date(Date.parse(srcAttributes.updated as string))
|
||||||
|
: undefined,
|
||||||
|
author: decodeURIComponent(
|
||||||
|
(srcAttributes.author as string | undefined) ?? "",
|
||||||
|
),
|
||||||
|
thumbnail: (srcAttributes.thumbnail as string | undefined) ?? "",
|
||||||
|
draft: draft,
|
||||||
|
};
|
||||||
}
|
}
|
||||||
|
|
||||||
const LATEX_REGEX = /\$\$((?:(?!\$\$)[\s\S])*)\$\$/m
|
const LATEX_REGEX = /\$\$((?:(?!\$\$)[\s\S])*)\$\$/m;
|
||||||
const MERMAID_REGEX = /<pre>\s*<code class="language-mermaid">([\s\S]*)<\/code>\s*<\/pre>/m
|
const MERMAID_REGEX =
|
||||||
|
/<pre>\s*<code class="language-mermaid">([\s\S]*)<\/code>\s*<\/pre>/m;
|
||||||
|
|
||||||
function transformLatexBlocks(srcHtml: string): string {
|
function transformLatexBlocks(srcHtml: string): string {
|
||||||
let outHtml = srcHtml
|
let outHtml = srcHtml;
|
||||||
let match: RegExpExecArray | null = null
|
let match: RegExpExecArray | null = null;
|
||||||
do {
|
do {
|
||||||
match = LATEX_REGEX.exec(outHtml)
|
match = LATEX_REGEX.exec(outHtml);
|
||||||
if (match && match[1]) {
|
if (match?.[1]) {
|
||||||
try {
|
try {
|
||||||
outHtml = outHtml.replace(match[0], katex.renderToString(match[1]))
|
outHtml = outHtml.replace(match[0], katex.renderToString(match[1]));
|
||||||
} catch (ex) {
|
} catch (ex) {
|
||||||
outHtml = outHtml.replace(match[0], `<i>katex error: ${ex}</i>`)
|
outHtml = outHtml.replace(
|
||||||
|
match[0],
|
||||||
|
`<i>katex error: ${ex as Error}</i>`,
|
||||||
|
);
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
} while (match && match[1])
|
} while (match?.[1]);
|
||||||
return outHtml
|
return outHtml;
|
||||||
}
|
}
|
||||||
|
|
||||||
function transformMermaidBlocks(srcHtml: string): string {
|
function transformMermaidBlocks(srcHtml: string): string {
|
||||||
let outHtml = srcHtml
|
let outHtml = srcHtml;
|
||||||
let match: RegExpExecArray | null = null
|
let match: RegExpExecArray | null = null;
|
||||||
do {
|
do {
|
||||||
match = MERMAID_REGEX.exec(outHtml)
|
match = MERMAID_REGEX.exec(outHtml);
|
||||||
if (match && match[1]) {
|
if (match?.[1]) {
|
||||||
outHtml = outHtml.replace(match[0], `<pre class="mermaid">\n${match[1]}\n</pre>`)
|
outHtml = outHtml.replace(
|
||||||
|
match[0],
|
||||||
|
`<pre class="mermaid">\n${match[1]}\n</pre>`,
|
||||||
|
);
|
||||||
}
|
}
|
||||||
} while (match && match[1])
|
} while (match?.[1]);
|
||||||
return outHtml
|
return outHtml;
|
||||||
}
|
}
|
||||||
|
|
||||||
function transformHtml(srcHtml: string): string {
|
function transformHtml(srcHtml: string): string {
|
||||||
let outHtml: string = srcHtml
|
let outHtml: string = srcHtml;
|
||||||
outHtml = transformLatexBlocks(outHtml)
|
outHtml = transformLatexBlocks(outHtml);
|
||||||
outHtml = transformMermaidBlocks(outHtml)
|
outHtml = transformMermaidBlocks(outHtml);
|
||||||
return outHtml
|
return outHtml;
|
||||||
}
|
}
|
||||||
|
|
||||||
/**
|
/**
|
||||||
* @deprecated
|
* @deprecated
|
||||||
*/
|
*/
|
||||||
export async function loadArticleOld(date: Date): Promise<Article | null> {
|
export async function loadArticleOld(date: Date): Promise<Article | null> {
|
||||||
const year = date.getFullYear().toString()
|
const year = date.getFullYear().toString();
|
||||||
const month = (date.getMonth() + 1).toString().padStart(2, '0')
|
const month = (date.getMonth() + 1).toString().padStart(2, "0");
|
||||||
const day = date.getDate().toString().padStart(2, '0')
|
const day = date.getDate().toString().padStart(2, "0");
|
||||||
const path = `./articles/${year}/${month}/${day}/`
|
const path = `./articles/${year}/${month}/${day}/`;
|
||||||
try {
|
try {
|
||||||
const data = (await import(`@articles/${year}/${month}/${day}/index.md`)) as MarkdownData
|
const data = (await import(
|
||||||
|
`@articles/${year}/${month}/${day}/index.md`
|
||||||
|
)) as MarkdownData;
|
||||||
return {
|
return {
|
||||||
metadata: parseMetadata(data.attributes, path),
|
metadata: parseMetadata(data.attributes, path),
|
||||||
html: transformHtml(data.html),
|
html: transformHtml(data.html),
|
||||||
}
|
};
|
||||||
} catch (ex) {
|
} catch (ex) {
|
||||||
console.error(ex)
|
console.error(ex);
|
||||||
return null
|
return null;
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
|
||||||
export async function loadArticle(path: string): Promise<Article | null> {
|
export async function loadArticle(path: string): Promise<Article | null> {
|
||||||
const raw_articles = import.meta.glob('@articles/**/index.md')
|
const raw_articles = import.meta.glob("@articles/**/index.md");
|
||||||
const key = `/articles/${path}index.md`
|
const key = `/articles/${path}index.md`;
|
||||||
if (!raw_articles[key]) return null
|
if (!raw_articles[key]) return null;
|
||||||
try {
|
try {
|
||||||
const data = (await raw_articles[key]()) as MarkdownData
|
const data = (await raw_articles[key]()) as MarkdownData;
|
||||||
return {
|
return {
|
||||||
metadata: parseMetadata(data.attributes, path),
|
metadata: parseMetadata(data.attributes, path),
|
||||||
html: transformHtml(data.html),
|
html: transformHtml(data.html),
|
||||||
}
|
};
|
||||||
} catch (ex) {
|
} catch (ex) {
|
||||||
console.error(ex)
|
console.error(ex);
|
||||||
return null
|
return null;
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
|
||||||
export async function listArticles(): Promise<ArticleMetadata[]> {
|
export async function listArticles(): Promise<ArticleMetadata[]> {
|
||||||
const raw_articles = import.meta.glob('@articles/**/index.md')
|
const raw_articles = import.meta.glob("@articles/**/index.md");
|
||||||
const articles: ArticleMetadata[] = (
|
const articles: ArticleMetadata[] = (
|
||||||
await Promise.all(
|
await Promise.all(
|
||||||
Object.keys(raw_articles).map(async (key) => {
|
Object.keys(raw_articles).map(async (key) => {
|
||||||
if (!raw_articles[key]) return null
|
if (!raw_articles[key]) return null;
|
||||||
const data = (await raw_articles[key]()) as MarkdownData
|
const data = (await raw_articles[key]()) as MarkdownData;
|
||||||
return parseMetadata(data.attributes, key.replace('index.md', ''))
|
return parseMetadata(data.attributes, key.replace("index.md", ""));
|
||||||
}),
|
}),
|
||||||
)
|
)
|
||||||
).filter((item) => item !== null)
|
).filter((item) => item !== null);
|
||||||
articles.sort((article1, article2) => article2.date.valueOf() - article1.date.valueOf())
|
articles.sort(
|
||||||
return articles
|
(article1, article2) => article2.date.valueOf() - article1.date.valueOf(),
|
||||||
|
);
|
||||||
|
return articles;
|
||||||
}
|
}
|
||||||
|
|||||||
@@ -0,0 +1,18 @@
|
|||||||
|
import packageJson from '@/../package.json'
|
||||||
|
import articlesConfig from '@articles/config.json'
|
||||||
|
|
||||||
|
export const NAME: string = packageJson.name
|
||||||
|
export const VERSION: string = packageJson.version
|
||||||
|
export const REPOSITORY: string = packageJson.repository
|
||||||
|
export const TITLE: string = articlesConfig.title
|
||||||
|
export const SIGNATURE: string = articlesConfig.signature
|
||||||
|
export const COPYRIGHT: string = articlesConfig.copyright
|
||||||
|
export const RSS_LINK: string = articlesConfig.rss_link
|
||||||
|
export const BACK_LINK: string = articlesConfig.back_link
|
||||||
|
export const ABOUT_LINK: string = articlesConfig.about_link
|
||||||
|
export const HOME_COUNT: number = articlesConfig.home_count
|
||||||
|
export const PUBLISHED_ON: string = articlesConfig.published_on
|
||||||
|
export const UPDATED_ON: string = articlesConfig.updated_on
|
||||||
|
export const AUTHORED: string = articlesConfig.authored
|
||||||
|
export const BASE_URL: string = import.meta.env.BASE_URL
|
||||||
|
export const PROD: boolean = import.meta.env.PROD
|
||||||
+6
-6
@@ -1,9 +1,9 @@
|
|||||||
export function simpleDateFormat(date: Date): string {
|
export function simpleDateFormat(date: Date): string {
|
||||||
return (
|
return (
|
||||||
date.getFullYear() +
|
date.getFullYear().toString() +
|
||||||
'-' +
|
"-" +
|
||||||
(date.getMonth() + 1).toString().padStart(2, '0') +
|
(date.getMonth() + 1).toString().padStart(2, "0") +
|
||||||
'-' +
|
"-" +
|
||||||
date.getDate().toString().padStart(2, '0')
|
date.getDate().toString().padStart(2, "0")
|
||||||
)
|
);
|
||||||
}
|
}
|
||||||
|
|||||||
@@ -1,9 +0,0 @@
|
|||||||
import packageJson from '@/../package.json'
|
|
||||||
|
|
||||||
export const NAME = packageJson.name
|
|
||||||
export const VERSION = packageJson.version
|
|
||||||
export const REPOSITORY = packageJson.repository
|
|
||||||
export const TITLE = import.meta.env.VITE_APP_TITLE
|
|
||||||
export const SIGNATURE = import.meta.env.VITE_APP_SIGNATURE
|
|
||||||
export const COPYRIGHT = import.meta.env.VITE_APP_COPYRIGHT
|
|
||||||
export const BASE_URL = import.meta.env.BASE_URL
|
|
||||||
@@ -2,6 +2,9 @@ import { createApp } from 'vue'
|
|||||||
import App from './App.vue'
|
import App from './App.vue'
|
||||||
import router from './router'
|
import router from './router'
|
||||||
|
|
||||||
|
import ShaderviewElement from '@keithclark/shaderview'
|
||||||
|
customElements.define('kc-shaderview', ShaderviewElement)
|
||||||
|
|
||||||
const app = createApp(App)
|
const app = createApp(App)
|
||||||
|
|
||||||
app.use(router)
|
app.use(router)
|
||||||
|
|||||||
+5
-3
@@ -1,12 +1,14 @@
|
|||||||
import ArticleView from '@views/ArticleView.vue'
|
|
||||||
import HomeView from '@views/HomeView.vue'
|
|
||||||
import NotFoundView from '@views/NotFoundView.vue'
|
|
||||||
import { createRouter, createWebHistory } from 'vue-router'
|
import { createRouter, createWebHistory } from 'vue-router'
|
||||||
|
import HomeView from '@views/HomeView.vue'
|
||||||
|
import AboutView from '@views/AboutView.vue'
|
||||||
|
import ArticleView from '@views/ArticleView.vue'
|
||||||
|
import NotFoundView from '@views/NotFoundView.vue'
|
||||||
|
|
||||||
const router = createRouter({
|
const router = createRouter({
|
||||||
history: createWebHistory(),
|
history: createWebHistory(),
|
||||||
routes: [
|
routes: [
|
||||||
{ path: '/', component: HomeView },
|
{ path: '/', component: HomeView },
|
||||||
|
{ path: '/about/', component: AboutView },
|
||||||
{ path: '/articles/:pathMatch(.*)/', component: ArticleView },
|
{ path: '/articles/:pathMatch(.*)/', component: ArticleView },
|
||||||
{ path: '/:pathMatch(.*)', component: NotFoundView },
|
{ path: '/:pathMatch(.*)', component: NotFoundView },
|
||||||
],
|
],
|
||||||
|
|||||||
@@ -0,0 +1,15 @@
|
|||||||
|
<script setup lang="ts">
|
||||||
|
import { onMounted, onUpdated } from 'vue'
|
||||||
|
import { updateDynamicContent } from '@lib/articles'
|
||||||
|
import { html } from '@articles/about.md'
|
||||||
|
|
||||||
|
onMounted(updateDynamicContent)
|
||||||
|
onUpdated(updateDynamicContent)
|
||||||
|
</script>
|
||||||
|
|
||||||
|
<template>
|
||||||
|
<div class="article">
|
||||||
|
<div v-html="html"></div>
|
||||||
|
<BackHomeButton />
|
||||||
|
</div>
|
||||||
|
</template>
|
||||||
+23
-19
@@ -1,13 +1,13 @@
|
|||||||
<script setup lang="ts">
|
<script setup lang="ts">
|
||||||
import type { Article } from '@interfaces'
|
import type { Article } from '@interfaces'
|
||||||
import { ref, onBeforeMount } from 'vue'
|
import { ref, onBeforeMount, onUpdated, onMounted } from 'vue'
|
||||||
import { loadArticle } from '@lib/articles'
|
import { loadArticle, updateDynamicContent } from '@lib/articles'
|
||||||
import { useRoute, onBeforeRouteUpdate, type RouteLocation } from 'vue-router'
|
import { useRoute, onBeforeRouteUpdate, type RouteLocation } from 'vue-router'
|
||||||
import NotFoundView from './NotFoundView.vue'
|
|
||||||
import { simpleDateFormat } from '@lib/dates'
|
import { simpleDateFormat } from '@lib/dates'
|
||||||
import { SIGNATURE, TITLE } from '@lib/meta'
|
import { AUTHORED, PUBLISHED_ON, SIGNATURE, TITLE, UPDATED_ON } from '@lib/config'
|
||||||
import PageFooter from '@components/PageFooter.vue'
|
import { stripHTML } from '@lib/strings'
|
||||||
import { stripHTML } from '@/lib/strings'
|
import BackHomeButton from '@components/BackHomeButton.vue'
|
||||||
|
import NotFoundView from '@views/NotFoundView.vue'
|
||||||
|
|
||||||
const article = ref<Article | null>(null)
|
const article = ref<Article | null>(null)
|
||||||
const loading = ref<boolean>(true)
|
const loading = ref<boolean>(true)
|
||||||
@@ -19,27 +19,32 @@ async function loadPage(target: RouteLocation) {
|
|||||||
window.document.title =
|
window.document.title =
|
||||||
stripHTML(TITLE) + ' — ' + stripHTML(article.value?.metadata.title ?? 'Not Found')
|
stripHTML(TITLE) + ' — ' + stripHTML(article.value?.metadata.title ?? 'Not Found')
|
||||||
loading.value = false
|
loading.value = false
|
||||||
|
await updateDynamicContent()
|
||||||
}
|
}
|
||||||
|
|
||||||
onBeforeMount(() => loadPage(route))
|
onBeforeMount(() => loadPage(route))
|
||||||
onBeforeRouteUpdate(loadPage)
|
onBeforeRouteUpdate(loadPage)
|
||||||
|
onMounted(updateDynamicContent)
|
||||||
|
onUpdated(updateDynamicContent)
|
||||||
</script>
|
</script>
|
||||||
|
|
||||||
<template>
|
<template>
|
||||||
<template v-if="loading">
|
<template v-if="!loading && !article">
|
||||||
<main></main>
|
|
||||||
</template>
|
|
||||||
<template v-else-if="!article">
|
|
||||||
<NotFoundView />
|
<NotFoundView />
|
||||||
</template>
|
</template>
|
||||||
<template v-else>
|
<div v-if="!loading" class="article">
|
||||||
<main class="article">
|
<template v-if="!loading && article">
|
||||||
<div class="article-header">
|
<div class="article-header">
|
||||||
<RouterLink class="link-home" to="/"><i icon="undo-2">↑</i></RouterLink>
|
|
||||||
<h1 class="article-title" v-html="article.metadata.title"></h1>
|
<h1 class="article-title" v-html="article.metadata.title"></h1>
|
||||||
<div class="article-published">
|
<div class="article-info">
|
||||||
{{ article.metadata.draft ? 'Drafted on' : 'Published on' }}
|
<span v-if="article.metadata.author"
|
||||||
{{ simpleDateFormat(article.metadata.date) }}
|
><span v-html="AUTHORED"></span> <span v-html="article.metadata.author"></span> —
|
||||||
|
</span>
|
||||||
|
<span v-html="PUBLISHED_ON"></span> {{ simpleDateFormat(article.metadata.date) }}
|
||||||
|
<span v-if="article.metadata.updated"
|
||||||
|
>— <span v-html="UPDATED_ON"></span>
|
||||||
|
{{ simpleDateFormat(article.metadata.updated) }}</span
|
||||||
|
>
|
||||||
</div>
|
</div>
|
||||||
<img
|
<img
|
||||||
v-if="article.metadata.thumbnail"
|
v-if="article.metadata.thumbnail"
|
||||||
@@ -50,8 +55,7 @@ onBeforeRouteUpdate(loadPage)
|
|||||||
</div>
|
</div>
|
||||||
<div class="article-text" v-html="article.html"></div>
|
<div class="article-text" v-html="article.html"></div>
|
||||||
<div class="article-signature" v-html="SIGNATURE"></div>
|
<div class="article-signature" v-html="SIGNATURE"></div>
|
||||||
<br />
|
<BackHomeButton />
|
||||||
<PageFooter />
|
|
||||||
</main>
|
|
||||||
</template>
|
</template>
|
||||||
|
</div>
|
||||||
</template>
|
</template>
|
||||||
|
|||||||
+19
-13
@@ -1,29 +1,36 @@
|
|||||||
<script setup lang="ts">
|
<script setup lang="ts">
|
||||||
import { ref, onBeforeMount } from 'vue'
|
import { ref, onBeforeMount, onUpdated, onMounted } from 'vue'
|
||||||
import { listArticles } from '@lib/articles'
|
import { listArticles, updateDynamicContent } from '@lib/articles'
|
||||||
import type { ArticleMetadata } from '@interfaces'
|
import type { ArticleMetadata } from '@interfaces'
|
||||||
import { simpleDateFormat } from '@lib/dates'
|
import { simpleDateFormat } from '@lib/dates'
|
||||||
import { TITLE } from '@lib/meta'
|
import { HOME_COUNT, PROD, PUBLISHED_ON, TITLE } from '@lib/config'
|
||||||
import PageFooter from '@components/PageFooter.vue'
|
import { stripHTML } from '@lib/strings'
|
||||||
import { stripHTML } from '@/lib/strings'
|
|
||||||
|
|
||||||
const articles = ref<ArticleMetadata[]>([])
|
const articles = ref<ArticleMetadata[]>([])
|
||||||
|
const loading = ref<boolean>(true)
|
||||||
|
|
||||||
onBeforeMount(async () => {
|
onBeforeMount(async () => {
|
||||||
const newArticles = await listArticles()
|
const newArticles = (await listArticles())
|
||||||
|
.filter((metadata) => (!metadata.draft || !PROD) && metadata.path)
|
||||||
|
.slice(0, HOME_COUNT)
|
||||||
articles.value.splice(0, articles.value.length, ...newArticles)
|
articles.value.splice(0, articles.value.length, ...newArticles)
|
||||||
window.document.title = stripHTML(TITLE) + ' — Home'
|
window.document.title = stripHTML(TITLE) + ' — Home'
|
||||||
|
loading.value = false
|
||||||
|
await updateDynamicContent()
|
||||||
})
|
})
|
||||||
|
onMounted(updateDynamicContent)
|
||||||
|
onUpdated(updateDynamicContent)
|
||||||
</script>
|
</script>
|
||||||
|
|
||||||
<template>
|
<template>
|
||||||
<main>
|
<div v-if="!loading" class="home">
|
||||||
<h1 class="title" v-html="TITLE"></h1>
|
<template v-for="(metadata, index) in articles" :key="index">
|
||||||
<template v-for="(metadata, index) in articles" v-bind:key="index">
|
<div v-if="(!metadata.draft || !PROD) && metadata.path" class="article-item">
|
||||||
<div v-if="!metadata.draft && metadata.path" class="article-item">
|
|
||||||
<RouterLink :to="metadata.path">
|
<RouterLink :to="metadata.path">
|
||||||
<h2 v-html="metadata.title"></h2>
|
<h2 v-html="metadata.title"></h2>
|
||||||
<span class="article-published">Published on {{ simpleDateFormat(metadata.date) }}</span>
|
<span class="article-info"
|
||||||
|
><span v-html="PUBLISHED_ON"></span> {{ simpleDateFormat(metadata.date) }}</span
|
||||||
|
>
|
||||||
<img
|
<img
|
||||||
v-if="metadata.thumbnail"
|
v-if="metadata.thumbnail"
|
||||||
alt="thumbnail"
|
alt="thumbnail"
|
||||||
@@ -32,6 +39,5 @@ onBeforeMount(async () => {
|
|||||||
</RouterLink>
|
</RouterLink>
|
||||||
</div>
|
</div>
|
||||||
</template>
|
</template>
|
||||||
<PageFooter />
|
</div>
|
||||||
</main>
|
|
||||||
</template>
|
</template>
|
||||||
|
|||||||
@@ -1,17 +1,12 @@
|
|||||||
<script setup lang="ts">
|
<script setup lang="ts">
|
||||||
import { TITLE } from '@/lib/meta'
|
import { onMounted, onUpdated } from 'vue'
|
||||||
import { stripHTML } from '@/lib/strings'
|
import { updateDynamicContent } from '@lib/articles'
|
||||||
import PageFooter from '@components/PageFooter.vue'
|
import { html } from '@articles/not_found.md'
|
||||||
import { onBeforeMount } from 'vue'
|
|
||||||
|
|
||||||
onBeforeMount(() => {
|
onMounted(updateDynamicContent)
|
||||||
window.document.title = stripHTML(TITLE) + ' — Not Found'
|
onUpdated(updateDynamicContent)
|
||||||
})
|
|
||||||
</script>
|
</script>
|
||||||
|
|
||||||
<template>
|
<template>
|
||||||
<main>
|
<div class="article" v-html="html"></div>
|
||||||
<h1>Page not found</h1>
|
|
||||||
<PageFooter />
|
|
||||||
</main>
|
|
||||||
</template>
|
</template>
|
||||||
|
|||||||
+5
-7
@@ -10,13 +10,11 @@ import articlesConfig from './articles/config.json'
|
|||||||
export default ({ mode }: { mode: string }) => {
|
export default ({ mode }: { mode: string }) => {
|
||||||
process.env = { ...process.env, ...loadEnv(mode, process.cwd()) }
|
process.env = { ...process.env, ...loadEnv(mode, process.cwd()) }
|
||||||
|
|
||||||
process.env.VITE_BASE_URL = articlesConfig['base_url']
|
process.env.VITE_BASE_URL = articlesConfig.base_url
|
||||||
process.env.VITE_APP_TITLE = articlesConfig['title']
|
process.env.VITE_APP_TITLE = articlesConfig.title
|
||||||
process.env.VITE_APP_TITLE_NO_HTML = articlesConfig['title'].replace(/(<([^>]+)>)/gi, '').trim()
|
process.env.VITE_APP_TITLE_NO_HTML = articlesConfig.title.replace(/(<([^>]+)>)/gi, '').trim()
|
||||||
process.env.VITE_APP_LANG = articlesConfig['lang']
|
process.env.VITE_APP_LANG = articlesConfig.lang
|
||||||
process.env.VITE_APP_SIGNATURE = articlesConfig['signature']
|
process.env.VITE_CUSTOM_HEAD = articlesConfig.custom_head
|
||||||
process.env.VITE_CUSTOM_HEAD = articlesConfig['custom_head']
|
|
||||||
process.env.VITE_APP_COPYRIGHT = articlesConfig['copyright']
|
|
||||||
|
|
||||||
return defineConfig({
|
return defineConfig({
|
||||||
plugins: [vue(), vueDevTools(), mdPlugin({ mode: [Mode.HTML] })],
|
plugins: [vue(), vueDevTools(), mdPlugin({ mode: [Mode.HTML] })],
|
||||||
|
|||||||
@@ -23,3 +23,5 @@ declare module '*.md' {
|
|||||||
// Modify below per your usage
|
// Modify below per your usage
|
||||||
export { attributes, toc, html, ReactComponent, VueComponent, VueComponentWith }
|
export { attributes, toc, html, ReactComponent, VueComponent, VueComponentWith }
|
||||||
}
|
}
|
||||||
|
|
||||||
|
declare module '*.vue'
|
||||||
|
|||||||
Reference in New Issue
Block a user