From 0126b7172201d7075fa92657ad818cfd178cb9b2 Mon Sep 17 00:00:00 2001 From: Yufan Sheng Date: Wed, 19 Jun 2024 03:34:36 +0800 Subject: [PATCH] feat: add upyun support. move images to upyun on build. (#45) * style: better reply button. * feat: add astro badge. * chore: drop useless styles. * feat: use new options.ts for global configuration. * fix: the invalid import.meta.env.PROD in astro build. * feat: move og to new assert prefix. * chore: add better og description. * chore: make sure all the images link could be correctly replaced. * feat: add upyun uploader. --- .vscode/settings.json | 27 +- Dockerfile | 4 +- astro.config.ts | 23 +- options.ts | 177 +++++ package-lock.json | 227 +++++- package.json | 4 +- {remark-plugins => plugins}/images.ts | 2 +- plugins/upyun.ts | 89 +++ src/assets/styles/globals.css | 662 ++---------------- src/assets/styles/reset.css | 5 +- src/components/comment/CommentItem.astro | 2 +- src/components/comment/artalk.ts | 4 +- src/components/footer/Footer.astro | 48 +- src/components/footer/GoTop.astro | 9 - src/components/header/Header.astro | 2 +- src/components/header/Logo.astro | 4 +- src/components/header/LogoLarge.astro | 4 +- src/components/like/Share.astro | 3 +- src/components/meta/PageMeta.astro | 10 +- src/components/meta/PostMeta.astro | 12 +- .../page/category/PinnedCategories.astro | 3 +- src/components/page/post/FeaturePosts.astro | 3 +- src/components/page/post/PostPagination.astro | 3 +- src/components/search/SearchBar.astro | 2 +- src/components/sidebar/RandomPosts.astro | 3 +- src/components/sidebar/RandomTags.astro | 3 +- src/components/sidebar/RecentComments.astro | 2 +- src/content/config.ts | 62 +- src/content/options/index.yml | 92 --- src/helpers/db/query.ts | 2 +- src/helpers/formatter.ts | 3 +- src/helpers/images.ts | 6 +- src/helpers/og.ts | 9 +- src/helpers/schema.ts | 17 +- src/helpers/seo.ts | 213 +++--- src/layouts/BaseLayout.astro | 13 +- src/layouts/PageLayout.astro | 5 +- src/layouts/PostLayout.astro | 7 +- src/layouts/posts/CategoryLayout.astro | 3 +- src/layouts/posts/SearchLayout.astro | 1 - src/layouts/posts/TagLayout.astro | 3 +- src/pages/404.astro | 1 - src/pages/cats/[slug]/page/[num].astro | 3 +- src/pages/feed.ts | 5 +- src/pages/og/[slug].png.ts | 4 +- src/pages/search/index.astro | 3 +- src/pages/sitemap.xml.ts | 4 +- src/pages/tags/[slug]/page/[num].astro | 3 +- tsconfig.json | 4 +- 49 files changed, 782 insertions(+), 1018 deletions(-) create mode 100644 options.ts rename {remark-plugins => plugins}/images.ts (96%) create mode 100644 plugins/upyun.ts delete mode 100644 src/components/footer/GoTop.astro delete mode 100644 src/content/options/index.yml diff --git a/.vscode/settings.json b/.vscode/settings.json index 3472512..b101389 100644 --- a/.vscode/settings.json +++ b/.vscode/settings.json @@ -12,6 +12,10 @@ }, "cSpell.words": [ "alexinea", + "alignfull", + "alignleft", + "alignright", + "alignwide", "ameho", "amehochan", "aplayer", @@ -21,6 +25,7 @@ "batang", "bigserial", "biomejs", + "blogroll", "blogster", "blurhash", "captainofphb", @@ -45,12 +50,13 @@ "fong", "forencrypt", "giscus", + "gogogo", "gotop", "gungseo", "hefei", "heiti", - "HONORBKK", - "HUAWEIYAL", + "honorbkk", + "huaweiyal", "ianvs", "iconfont", "iroha", @@ -61,7 +67,7 @@ "jing", "jungshik", "khalil", - "KHTML", + "khtml", "koanughi", "koaunghi", "lantinghei", @@ -73,15 +79,17 @@ "luxon", "mboker", "minagi", - "Miui", - "MMWEBID", - "MMWEBSDK", + "miui", + "mmwebid", + "mmwebsdk", "mochi", "napi", "netease", "nextval", "nian", + "nocolor", "nofollow", + "nopd", "noto", "oppo", "opposans", @@ -95,7 +103,7 @@ "qrcode", "quan", "recma", - "Redmi", + "redmi", "regclass", "sauvignon", "sheng", @@ -104,6 +112,8 @@ "shmily", "skrs", "syhily", + "tabindex", + "tagcloud", "taza", "teruteru", "timestamptz", @@ -111,6 +121,7 @@ "tsconfigs", "ultrahtml", "unpic", + "upyun", "urlset", "varchar", "velite", @@ -120,7 +131,7 @@ "weibo", "xiao", "xinsenz", - "XWEB", + "xweb", "yefengs", "yetgul", "ying", diff --git a/Dockerfile b/Dockerfile index ca267f5..1bf529a 100644 --- a/Dockerfile +++ b/Dockerfile @@ -8,7 +8,7 @@ COPY . . ENV ASTRO_TELEMETRY_DISABLED=1 RUN NODE_ENV=development npm install RUN npm i patch-package && npm exec patch-package -RUN npm run build +RUN NODE_ENV=production npm run build FROM base AS runtime RUN npm install --omit=dev @@ -16,4 +16,4 @@ COPY --from=build /app/dist ./dist ENV HOST=0.0.0.0 ENV PORT=4321 EXPOSE 4321 -CMD node ./dist/server/entry.mjs +CMD NODE_ENV=production node ./dist/server/entry.mjs diff --git a/astro.config.ts b/astro.config.ts index 59d561f..c5f64cf 100644 --- a/astro.config.ts +++ b/astro.config.ts @@ -1,15 +1,14 @@ import mdx from '@astrojs/mdx'; import node from '@astrojs/node'; import { defineConfig, envField } from 'astro/config'; -import { astroImage } from './remark-plugins/images'; - -// Dynamic switch the site. This is hard coded. -const port = 4321; -const site = import.meta.env.PROD ? 'https://yufan.me' : `http://localhost:${port}`; +import options from './options'; +import { astroImage } from './plugins/images'; +import { upyun } from './plugins/upyun'; // https://astro.build/config export default defineConfig({ - site: site, + // This will override the import.meta.env.SITE. No need to introduce method. + site: options.isProd() ? options.website : options.local.website, output: 'server', security: { checkOrigin: true, @@ -17,11 +16,13 @@ export default defineConfig({ experimental: { env: { schema: { + // Postgres Database POSTGRES_HOST: envField.string({ context: 'server', access: 'secret' }), POSTGRES_PORT: envField.number({ context: 'server', access: 'secret' }), POSTGRES_USERNAME: envField.string({ context: 'server', access: 'secret' }), POSTGRES_PASSWORD: envField.string({ context: 'server', access: 'secret' }), POSTGRES_DATABASE: envField.string({ context: 'server', access: 'secret' }), + // Artalk Comment ARTALK_HOST: envField.string({ context: 'server', access: 'secret' }), }, }, @@ -30,6 +31,9 @@ export default defineConfig({ mdx({ remarkPlugins: [astroImage], }), + upyun({ + path: ['images', 'og', 'cats'], + }), ], adapter: node({ mode: 'standalone', @@ -42,8 +46,7 @@ export default defineConfig({ }, }, server: { - host: true, - port: port, + port: options.local.port, }, devToolbar: { // I don't need such toolbar. @@ -53,4 +56,8 @@ export default defineConfig({ // Add this for avoiding the needless import optimize in Vite. optimizeDeps: { exclude: ['@napi-rs/canvas'] }, }, + build: { + assets: 'cats', + assetsPrefix: options.assetsPrefix(), + }, }); diff --git a/options.ts b/options.ts new file mode 100644 index 0000000..a4f5814 --- /dev/null +++ b/options.ts @@ -0,0 +1,177 @@ +import { z } from 'astro/zod'; + +// The type of the options, use zod for better validation. +const Options = z + .object({ + local: z + .object({ + port: z.number(), + }) + .transform((local) => ({ ...local, website: `http://localhost:${local.port}` })), + title: z.string().max(40), + website: z + .string() + .url() + .refine((u) => !u.endsWith('/')) + .readonly(), + description: z.string().max(100), + keywords: z.array(z.string()), + author: z.object({ name: z.string(), email: z.string().email(), url: z.string().url() }), + navigation: z.array(z.object({ text: z.string(), link: z.string(), target: z.string().optional() })), + socials: z.array( + z.object({ + name: z.string(), + icon: z.string(), + type: z.enum(['link', 'qrcode']), + title: z.string().optional(), + link: z.string().url(), + }), + ), + settings: z.object({ + initialYear: z.number().max(2024), + icpNo: z.string().optional(), + locale: z.string().optional().default('zh-CN'), + timeZone: z.string().optional().default('Asia/Shanghai'), + timeFormat: z.string().optional().default('yyyy-MM-dd HH:mm:ss'), + twitter: z.string(), + assetPrefix: z + .string() + .url() + .refine((u) => !u.endsWith('/')) + .readonly(), + post: z.object({ + sort: z.enum(['asc', 'desc']), + feature: z.array(z.string()).optional(), + category: z.array(z.string()).optional(), + }), + pagination: z.object({ + posts: z.number().optional().default(5), + category: z.number().optional().default(7), + tags: z.number().optional().default(7), + search: z.number().optional().default(7), + }), + feed: z.object({ + full: z.boolean().optional().default(true), + size: z.number().optional().default(20), + }), + sidebar: z.object({ + search: z.boolean().default(false), + post: z.number().default(6), + comment: z.number().default(0), + tag: z.number().default(20), + }), + comments: z.object({ + server: z.string().url().readonly(), + admins: z.array(z.number()), + }), + }), + }) + .transform((opts) => { + const isProd = (): boolean => import.meta.env.MODE === 'production' || process.env.NODE_ENV === 'production'; + const assetsPrefix = (): string => (isProd() ? opts.settings.assetPrefix : opts.local.website); + return { + ...opts, + // Monkey patch for the issue https://github.com/withastro/astro/issues/11282 + // No need to fallback to the import.meta.env.PROD I think. + isProd, + // Given that the import.meta.env.ASSETS_PREFIX has two types. + // I have to use this uniform method instead. + assetsPrefix, + defaultOpenGraph: (): string => `${assetsPrefix()}/images/open-graph.png`, + }; + }); + +const options = { + local: { + port: 4321, + }, + title: '且听书吟', + website: 'https://yufan.me', + description: '诗与梦想的远方', + keywords: ['雨帆', '且听书吟', 'syhily', 'amehochan', 'yufan'], + author: { + name: '雨帆', + email: 'syhily@gmail.com', + url: 'https://yufan.me', + }, + navigation: [ + { + text: '首页', + link: '/', + }, + { + text: '关于', + link: '/about', + }, + { + text: '留言', + link: '/guestbook', + }, + { + text: '友链', + link: '/links', + }, + { + text: '笔记', + link: 'https://note.yufan.me', + target: '_blank', + }, + ], + socials: [ + { + name: 'GitHub', + icon: 'icon-github-fill', + type: 'link', + link: 'https://github.com/syhily', + }, + { + name: 'Twitter', + icon: 'icon-twitter', + type: 'link', + link: 'https://twitter.com/amehochan', + }, + { + name: 'Wechat', + icon: 'icon-wechat', + type: 'qrcode', + title: '扫码加我微信好友', + link: 'https://u.wechat.com/EBpmuKmrVz4YVFnoCJdnruA', + }, + ], + settings: { + initialYear: 2011, + icpNo: '皖ICP备2021002315号-2', + locale: 'zh-CN', + timeZone: 'Asia/Shanghai', + timeFormat: 'yyyy-MM-dd', + twitter: 'amehochan', + assetPrefix: 'https://cat.yufan.me', + post: { + sort: 'desc', + feature: ['secret-of-boys-mind', 'my-darling', 'happiness-caprice'], + category: ['article', 'think', 'gossip', 'coding'], + }, + pagination: { + posts: 5, + category: 7, + tags: 7, + search: 7, + }, + feed: { + full: true, + size: 10, + }, + sidebar: { + search: true, + post: 6, + comment: 6, + tag: 20, + }, + comments: { + server: 'https://comment.yufan.me', + admins: [3], + }, + }, +}; + +export default Options.parse(options); diff --git a/package-lock.json b/package-lock.json index a8acc90..acdfcff 100644 --- a/package-lock.json +++ b/package-lock.json @@ -32,6 +32,7 @@ "@types/pg": "^8.11.6", "@types/qrcode-svg": "^1.1.4", "@types/unist": "^3.0.2", + "@types/upyun": "^3.4.3", "aplayer": "^1.10.1", "bootstrap": "^5.3.3", "prettier": "^3.3.2", @@ -42,7 +43,8 @@ "rimraf": "^5.0.7", "sharp": "^0.33.4", "typescript": "^5.4.5", - "unist-util-visit": "^5.0.0" + "unist-util-visit": "^5.0.0", + "upyun": "^3.4.6" } }, "node_modules/@ampproject/remapping": { @@ -171,6 +173,29 @@ "integrity": "sha512-IfYcSBWE3hLpBg8+X2SEa8LVkJdJEkT2Ese2aaLs3ptGdVtABxndrMaxuFlQ1qdFf9Q5rDvDpxI3WwgvKFAsQA==", "license": "MIT" }, + "node_modules/@astrojs/markdown-remark/node_modules/is-buffer": { + "version": "2.0.5", + "resolved": "https://registry.npmjs.org/is-buffer/-/is-buffer-2.0.5.tgz", + "integrity": "sha512-i2R6zNFDwgEHJyQUtJEk0XFi1i0dPFn/oqjK3/vPCcDeJvW5NQ83V8QbicfF1SupOaB0h8ntgBC2YiE7dfyctQ==", + "funding": [ + { + "type": "github", + "url": "https://github.com/sponsors/feross" + }, + { + "type": "patreon", + "url": "https://www.patreon.com/feross" + }, + { + "type": "consulting", + "url": "https://feross.org/support" + } + ], + "license": "MIT", + "engines": { + "node": ">=4" + } + }, "node_modules/@astrojs/markdown-remark/node_modules/nlcst-to-string": { "version": "3.1.1", "resolved": "https://registry.npmjs.org/nlcst-to-string/-/nlcst-to-string-3.1.1.tgz", @@ -2774,6 +2799,16 @@ "integrity": "sha512-dqId9J8K/vGi5Zr7oo212BGii5m3q5Hxlkwy3WpYuKPklmBEvsbMYYyLxAQpSffdLl/gdW0XUpKWFvYmyoWCoQ==", "license": "MIT" }, + "node_modules/@types/upyun": { + "version": "3.4.3", + "resolved": "https://registry.npmjs.org/@types/upyun/-/upyun-3.4.3.tgz", + "integrity": "sha512-iTZTvDxt1h4dmZeJnWsB7YeVRwTwEr8GE22LSKM3F464+4VnaUoIYtcwqkWnxEDPV97ZN3XA0BzSgCx84QR8jA==", + "dev": true, + "license": "MIT", + "dependencies": { + "@types/node": "*" + } + }, "node_modules/@ungap/structured-clone": { "version": "1.2.0", "resolved": "https://registry.npmjs.org/@ungap/structured-clone/-/structured-clone-1.2.0.tgz", @@ -3136,6 +3171,23 @@ "sharp": "^0.33.3" } }, + "node_modules/asynckit": { + "version": "0.4.0", + "resolved": "https://registry.npmjs.org/asynckit/-/asynckit-0.4.0.tgz", + "integrity": "sha512-Oei9OH4tRh0YqU3GxhX79dM/mwVgvbZJaSNaRk+bshkj0S5cfHcgYakreBjrHwatXKbz+IoIdYLxrKim2MjW0Q==", + "dev": true, + "license": "MIT" + }, + "node_modules/axios": { + "version": "0.26.1", + "resolved": "https://registry.npmjs.org/axios/-/axios-0.26.1.tgz", + "integrity": "sha512-fPwcX4EvnSHuInCMItEhAGnaSEXRBjtzh9fOtsE6E1G6p7vl7edEeZe11QHf18+6+9gR5PbKV/sGKNaD8YaMeA==", + "dev": true, + "license": "MIT", + "dependencies": { + "follow-redirects": "^1.14.8" + } + }, "node_modules/axobject-query": { "version": "4.0.0", "resolved": "https://registry.npmjs.org/axobject-query/-/axobject-query-4.0.0.tgz", @@ -3414,6 +3466,16 @@ "url": "https://github.com/sponsors/wooorm" } }, + "node_modules/charenc": { + "version": "0.0.2", + "resolved": "https://registry.npmjs.org/charenc/-/charenc-0.0.2.tgz", + "integrity": "sha512-yrLQ/yVUFXkzg7EDQsPieE/53+0RlaWTs+wBrvW36cyilJ2SaDWfl4Yj7MtLTXleV9uEKefbAGUPv2/iWSooRA==", + "dev": true, + "license": "BSD-3-Clause", + "engines": { + "node": "*" + } + }, "node_modules/chokidar": { "version": "3.6.0", "resolved": "https://registry.npmjs.org/chokidar/-/chokidar-3.6.0.tgz", @@ -3685,6 +3747,19 @@ "devOptional": true, "license": "MIT" }, + "node_modules/combined-stream": { + "version": "1.0.8", + "resolved": "https://registry.npmjs.org/combined-stream/-/combined-stream-1.0.8.tgz", + "integrity": "sha512-FQN4MRfuJeHf7cBbBMJFXhKSDq+2kAArBlmRBvcvFE5BB1HZKXtSFASDhdlz9zOYwxh8lDdnvmMOe/+5cdoEdg==", + "dev": true, + "license": "MIT", + "dependencies": { + "delayed-stream": "~1.0.0" + }, + "engines": { + "node": ">= 0.8" + } + }, "node_modules/comma-separated-tokens": { "version": "2.0.3", "resolved": "https://registry.npmjs.org/comma-separated-tokens/-/comma-separated-tokens-2.0.3.tgz", @@ -3730,6 +3805,16 @@ "node": ">= 8" } }, + "node_modules/crypt": { + "version": "0.0.2", + "resolved": "https://registry.npmjs.org/crypt/-/crypt-0.0.2.tgz", + "integrity": "sha512-mCxBlsHFYh9C+HVpiEacem8FEBnMXgU9gy4zmNC+SXAZNB/1idgp/aulFJ4FgCi7GPEVbfyng092GqL2k2rmow==", + "dev": true, + "license": "BSD-3-Clause", + "engines": { + "node": "*" + } + }, "node_modules/cssesc": { "version": "3.0.0", "resolved": "https://registry.npmjs.org/cssesc/-/cssesc-3.0.0.tgz", @@ -3772,6 +3857,16 @@ "url": "https://github.com/sponsors/wooorm" } }, + "node_modules/delayed-stream": { + "version": "1.0.0", + "resolved": "https://registry.npmjs.org/delayed-stream/-/delayed-stream-1.0.0.tgz", + "integrity": "sha512-ZySD7Nf91aLB0RxL4KGrKHBXl7Eds1DAmEdcoVawXnLD7SDhpNgtuII2aAkg7a7QS41jxPSZ17p4VdGnMHk3MQ==", + "dev": true, + "license": "MIT", + "engines": { + "node": ">=0.4.0" + } + }, "node_modules/depd": { "version": "2.0.0", "resolved": "https://registry.npmjs.org/depd/-/depd-2.0.0.tgz", @@ -4349,6 +4444,27 @@ "node": ">=8" } }, + "node_modules/follow-redirects": { + "version": "1.15.6", + "resolved": "https://registry.npmjs.org/follow-redirects/-/follow-redirects-1.15.6.tgz", + "integrity": "sha512-wWN62YITEaOpSK584EZXJafH1AGpO8RVgElfkuXbTOrPX4fIfOyEpW/CsiNd8JdYrAoOvafRTOEnvsO++qCqFA==", + "dev": true, + "funding": [ + { + "type": "individual", + "url": "https://github.com/sponsors/RubenVerborgh" + } + ], + "license": "MIT", + "engines": { + "node": ">=4.0" + }, + "peerDependenciesMeta": { + "debug": { + "optional": true + } + } + }, "node_modules/foreground-child": { "version": "3.2.1", "resolved": "https://registry.npmjs.org/foreground-child/-/foreground-child-3.2.1.tgz", @@ -4366,6 +4482,21 @@ "url": "https://github.com/sponsors/isaacs" } }, + "node_modules/form-data": { + "version": "4.0.0", + "resolved": "https://registry.npmjs.org/form-data/-/form-data-4.0.0.tgz", + "integrity": "sha512-ETEklSGi5t0QMZuiXoA/Q6vcnxcLQP5vdugSpuAyi6SVGi2clPPp+xgEhuMaHC+zGgn31Kd235W35f7Hykkaww==", + "dev": true, + "license": "MIT", + "dependencies": { + "asynckit": "^0.4.0", + "combined-stream": "^1.0.8", + "mime-types": "^2.1.12" + }, + "engines": { + "node": ">= 6" + } + }, "node_modules/fresh": { "version": "0.5.2", "resolved": "https://registry.npmjs.org/fresh/-/fresh-0.5.2.tgz", @@ -4812,6 +4943,13 @@ "url": "https://opencollective.com/unified" } }, + "node_modules/hmacsha1": { + "version": "1.0.0", + "resolved": "https://registry.npmjs.org/hmacsha1/-/hmacsha1-1.0.0.tgz", + "integrity": "sha512-4FP6J0oI8jqb6gLLl9tSwVdosWJ/AKSGJ+HwYf6Ixe4MUcEkst4uWzpVQrNOCin0fzTRQbXV8ePheU8WiiDYBw==", + "dev": true, + "license": "BSD" + }, "node_modules/html-escaper": { "version": "3.0.3", "resolved": "https://registry.npmjs.org/html-escaper/-/html-escaper-3.0.3.tgz", @@ -4925,27 +5063,11 @@ } }, "node_modules/is-buffer": { - "version": "2.0.5", - "resolved": "https://registry.npmjs.org/is-buffer/-/is-buffer-2.0.5.tgz", - "integrity": "sha512-i2R6zNFDwgEHJyQUtJEk0XFi1i0dPFn/oqjK3/vPCcDeJvW5NQ83V8QbicfF1SupOaB0h8ntgBC2YiE7dfyctQ==", - "funding": [ - { - "type": "github", - "url": "https://github.com/sponsors/feross" - }, - { - "type": "patreon", - "url": "https://www.patreon.com/feross" - }, - { - "type": "consulting", - "url": "https://feross.org/support" - } - ], - "license": "MIT", - "engines": { - "node": ">=4" - } + "version": "1.1.6", + "resolved": "https://registry.npmjs.org/is-buffer/-/is-buffer-1.1.6.tgz", + "integrity": "sha512-NcdALwpXkTm5Zvvbk7owOUSvVvBKDgKP5/ewfXEznmQFfs4ZRmanOeKBTjRVjka3QFoN6XJ+9F3USqfHqTaU5w==", + "dev": true, + "license": "MIT" }, "node_modules/is-core-module": { "version": "2.13.1", @@ -5084,6 +5206,13 @@ "url": "https://github.com/sponsors/sindresorhus" } }, + "node_modules/is-promise": { + "version": "4.0.0", + "resolved": "https://registry.npmjs.org/is-promise/-/is-promise-4.0.0.tgz", + "integrity": "sha512-hvpoI6korhJMnej285dSg6nu1+e6uxs7zG3BYAm5byqDsgJNWwxzM6z6iZiAgQR4TJ30JmBTOwqZUw3WlyH3AQ==", + "dev": true, + "license": "MIT" + }, "node_modules/is-reference": { "version": "3.0.2", "resolved": "https://registry.npmjs.org/is-reference/-/is-reference-3.0.2.tgz", @@ -5393,6 +5522,18 @@ "node": ">= 18" } }, + "node_modules/md5": { + "version": "2.3.0", + "resolved": "https://registry.npmjs.org/md5/-/md5-2.3.0.tgz", + "integrity": "sha512-T1GITYmFaKuO91vxyoQMFETst+O71VUPEU3ze5GNzDm0OWdP8v1ziTaAEPUr/3kLsY3Sftgz242A1SetQiDL7g==", + "dev": true, + "license": "BSD-3-Clause", + "dependencies": { + "charenc": "0.0.2", + "crypt": "0.0.2", + "is-buffer": "~1.1.6" + } + }, "node_modules/mdast-util-definitions": { "version": "6.0.0", "resolved": "https://registry.npmjs.org/mdast-util-definitions/-/mdast-util-definitions-6.0.0.tgz", @@ -6464,6 +6605,29 @@ "node": ">=4" } }, + "node_modules/mime-db": { + "version": "1.52.0", + "resolved": "https://registry.npmjs.org/mime-db/-/mime-db-1.52.0.tgz", + "integrity": "sha512-sPU4uV7dYlvtWJxwwxHD0PuihVNiE7TyAbQ5SWxDCB9mUYvOgroQOwYQQOKPJ8CIbE+1ETVlOoK1UC2nU3gYvg==", + "dev": true, + "license": "MIT", + "engines": { + "node": ">= 0.6" + } + }, + "node_modules/mime-types": { + "version": "2.1.35", + "resolved": "https://registry.npmjs.org/mime-types/-/mime-types-2.1.35.tgz", + "integrity": "sha512-ZDY+bPm5zTTF+YpCrAU9nK0UgICYPT0QtT1NZWFv4s++TNkcgVaT0g6+4R2uI4MjQjzysHB1zxuWL50hzaeXiw==", + "dev": true, + "license": "MIT", + "dependencies": { + "mime-db": "1.52.0" + }, + "engines": { + "node": ">= 0.6" + } + }, "node_modules/mimic-fn": { "version": "4.0.0", "resolved": "https://registry.npmjs.org/mimic-fn/-/mimic-fn-4.0.0.tgz", @@ -8609,6 +8773,25 @@ "browserslist": ">= 4.21.0" } }, + "node_modules/upyun": { + "version": "3.4.6", + "resolved": "https://registry.npmjs.org/upyun/-/upyun-3.4.6.tgz", + "integrity": "sha512-ThAI7woGkVE2lsOq8MFYb0Oeg8avOQQbY3XmXmaq1aZVjzcglcMuI/RImBrq+KJw7nX39iNKCJKYs65xiAF53Q==", + "dev": true, + "license": "MIT", + "dependencies": { + "axios": "^0.26.1", + "base-64": "^1.0.0", + "form-data": "^4.0.0", + "hmacsha1": "^1.0.0", + "is-promise": "^4.0.0", + "md5": "^2.3.0", + "mime-types": "^2.1.15" + }, + "engines": { + "node": ">=8.0.0" + } + }, "node_modules/vfile": { "version": "6.0.1", "resolved": "https://registry.npmjs.org/vfile/-/vfile-6.0.1.tgz", diff --git a/package.json b/package.json index 080a773..8677b99 100644 --- a/package.json +++ b/package.json @@ -64,6 +64,7 @@ "@types/pg": "^8.11.6", "@types/qrcode-svg": "^1.1.4", "@types/unist": "^3.0.2", + "@types/upyun": "^3.4.3", "aplayer": "^1.10.1", "bootstrap": "^5.3.3", "prettier": "^3.3.2", @@ -74,6 +75,7 @@ "rimraf": "^5.0.7", "sharp": "^0.33.4", "typescript": "^5.4.5", - "unist-util-visit": "^5.0.0" + "unist-util-visit": "^5.0.0", + "upyun": "^3.4.6" } } diff --git a/remark-plugins/images.ts b/plugins/images.ts similarity index 96% rename from remark-plugins/images.ts rename to plugins/images.ts index ba0ff11..72e42f2 100644 --- a/remark-plugins/images.ts +++ b/plugins/images.ts @@ -43,7 +43,7 @@ const transformAstroImage = async (imageNode: ImageNode) => { imageNode.name = 'Image'; imageNode.attributes = [ { type: 'mdxJsxAttribute', name: 'alt', value: imageNode.alt }, - { type: 'mdxJsxAttribute', name: 'src', value: imageNode.url }, + { type: 'mdxJsxAttribute', name: 'src', value: metadata.src }, { type: 'mdxJsxAttribute', name: 'width', value: imageNode.width ?? metadata.width }, { type: 'mdxJsxAttribute', name: 'height', value: imageNode.height ?? metadata.height }, { type: 'mdxJsxAttribute', name: 'blurDataURL', value: metadata.blurDataURL }, diff --git a/plugins/upyun.ts b/plugins/upyun.ts new file mode 100644 index 0000000..731fa4e --- /dev/null +++ b/plugins/upyun.ts @@ -0,0 +1,89 @@ +import type { AstroIntegration, AstroIntegrationLogger, RouteData } from 'astro'; +import fs from 'node:fs'; +import path from 'node:path'; +import up from 'upyun'; + +export type UpyunOption = { + path: string[]; + bucket?: string; + operator?: string; + password?: string; +}; + +const defaultOption: UpyunOption = { + path: ['images'], + bucket: process.env.UPYUN_BUCKET, + operator: process.env.UPYUN_OPERATOR, + password: process.env.UPYUN_PASSWORD, +}; + +export const upyun = (opt: UpyunOption): AstroIntegration => ({ + name: 'upyun', + hooks: { + 'astro:build:done': async ({ dir, logger }: { dir: URL; routes: RouteData[]; logger: AstroIntegrationLogger }) => { + const option: UpyunOption = { ...defaultOption, ...opt }; + if (typeof option.bucket === 'undefined' || opt.bucket === null) { + logger.error('No "bucket" found on your configuration, skip deploying.'); + return; + } + if (typeof option.operator === 'undefined' || opt.operator === null) { + logger.error('No "operator" found on your configuration, skip deploying.'); + return; + } + if (typeof option.password === 'undefined' || opt.password === null) { + logger.error('No "password" found on your configuration, skip deploying.'); + return; + } + if (option.path.length === 0) { + logger.warn('No files need to be upload to upyun. Skip.'); + return; + } + + // Create UPYUN Client + const service = new up.Service(option.bucket, option.operator, option.password); + const client = new up.Client(service); + + // Upload one by one + const staticRootPath = dir.pathname; + for (const dir of option.path) { + logger.info(`Start to upload the ${dir} to upyun`); + await uploadFile(logger, client, staticRootPath, dir); + } + }, + }, +}); + +const normalizePath = (p: string): string => { + return p.includes(path.win32.sep) ? p.split(path.win32.sep).join(path.posix.sep) : p; +}; + +const uploadFile = async (logger: AstroIntegrationLogger, client: up.Client, root: string, current: string) => { + const fullPath = path.join(root, current); + const isDir = fs.statSync(fullPath).isDirectory(); + + // Visit file. + if (!isDir) { + const filePath = normalizePath(current); + const res1 = await client.headFile(filePath); + + if (res1 === false) { + // This file need to be uploaded to upyun. + // Try Create directory first. + const newDir = filePath.substring(0, filePath.lastIndexOf(path.posix.sep)); + const res2 = await client.headFile(newDir); + if (res2 === false) { + logger.info(`Try to create ${newDir} on upyun`); + await client.makeDir(newDir); + } + // Upload file. + logger.info(`Try to upload file ${filePath} to upyun`); + await client.putFile(filePath, fs.readFileSync(fullPath)); + } + + return; + } + + for (const item of fs.readdirSync(fullPath)) { + await uploadFile(logger, client, root, path.join(current, item)); + } +}; diff --git a/src/assets/styles/globals.css b/src/assets/styles/globals.css index 025456f..efedce7 100644 --- a/src/assets/styles/globals.css +++ b/src/assets/styles/globals.css @@ -44,7 +44,7 @@ } /*-------------------------------------------------------------- -Buttons + Buttons --------------------------------------------------------------*/ .btn { @@ -197,7 +197,7 @@ Buttons } /*-------------------------------------------------------------- - btn link style + btn link style --------------------------------------------------------------*/ .btn-link { @@ -221,7 +221,7 @@ Buttons } /*-------------------------------------------------------------- - btn link style + btn link style --------------------------------------------------------------*/ .btn-text { @@ -244,7 +244,7 @@ Buttons } /*-------------------------------------------------------------- - btn link style + btn link style --------------------------------------------------------------*/ .btn-success { @@ -269,7 +269,7 @@ Buttons } /*-------------------------------------------------------------- - btn outline style + btn outline style --------------------------------------------------------------*/ .btn-outline-primary { @@ -293,7 +293,7 @@ Buttons } /*-------------------------------------------------------------- - btn outline secondary style + btn outline secondary style --------------------------------------------------------------*/ .btn-outline-secondary { @@ -316,7 +316,7 @@ Buttons } /*-------------------------------------------------------------- - btn outline light style + btn outline light style --------------------------------------------------------------*/ .btn-outline-light { @@ -338,7 +338,7 @@ Buttons } /*-------------------------------------------------------------- - btn outline secondary style + btn outline secondary style --------------------------------------------------------------*/ .btn-outline-dark { @@ -421,7 +421,7 @@ Buttons } /*-------------------------------------------------------------- - btn radius style + btn radius style --------------------------------------------------------------*/ .btn-circle { @@ -472,7 +472,7 @@ Buttons } /*-------------------------------------------------------------- - btn size style + btn size style --------------------------------------------------------------*/ .btn-xs { @@ -520,7 +520,7 @@ Buttons } /*-------------------------------------------------------------- - btn icon style + btn icon style --------------------------------------------------------------*/ .btn-icon { @@ -641,7 +641,7 @@ Buttons } /*-------------------------------------------------------------- - btn width style + btn width style --------------------------------------------------------------*/ .btn-w-xs { @@ -852,7 +852,7 @@ Buttons } /*-------------------------------------------------------------- -badge + badge --------------------------------------------------------------*/ .badge { @@ -936,7 +936,7 @@ badge } /*-------------------------------------------------------------- -border + border --------------------------------------------------------------*/ .border-secondary { @@ -1016,7 +1016,6 @@ border .form-control[readonly] { background-color: var(--bg-light); opacity: 1; - color: color; } textarea.form-control { @@ -1399,6 +1398,14 @@ textarea.form-control { color: inherit; } +.list-favicon { + background-size: cover; + display: inline-block; + margin-left: 8px; + width: 12px; + height: 12px; +} + @media (min-width: 1200px) and (max-width: 1399.98px) { .list-desc .h-2x { -webkit-line-clamp: 1; @@ -1512,7 +1519,6 @@ textarea.form-control { #tag_cloud a:hover, #recent-comments a:hover { color: var(--color-primary); - text-shadow: var(--color-muted) 0.05em 0 0.05em; } a.list-title:hover { @@ -1577,60 +1583,6 @@ a.list-title:hover { } } -/*-------------------------------------------------------------- - custom style ---------------------------------------------------------------*/ - -.custom-hover { - position: relative; - overflow: hidden; -} - -.custom-hover img, -.custom-hover-img { - max-width: 100%; - -webkit-transition: all 444ms ease-in-out; - -o-transition: all 444ms ease-in-out; - transition: all 444ms ease-in-out; -} - -.custom-hover:hover img, -.custom-hover:hover .custom-hover-img { - -webkit-transform: scale(1.05); - -ms-transform: scale(1.05); - transform: scale(1.05); -} - -.custom-hover .custom-hover-icon { - position: absolute; - left: 0; - right: 0; - text-align: center; - top: 50%; - opacity: 0; - -webkit-transform: translateY(-50%); - -ms-transform: translateY(-50%); - transform: translateY(-50%); - visibility: visible; - z-index: 9; - -webkit-transition: 0.3s; - -o-transition: 0.3s; - transition: 0.3s; -} - -.custom-hover .custom-hover-icon span { - width: 40px; - height: 40px; - line-height: 40px; - border-radius: 40px; - display: inline-block; - background-color: rgba(255, 255, 255, 0.68); -} - -.custom-hover:hover .custom-hover-icon { - opacity: 1; -} - /*-------------------------------------------------------------- overlay style --------------------------------------------------------------*/ @@ -1709,7 +1661,7 @@ a:hover .overlay { } /*-------------------------------------------------------------- - headere style + header style --------------------------------------------------------------*/ .site-layout { @@ -1846,7 +1798,7 @@ a:hover .overlay { } /*-------------------------------------------------------------- - header style + header style --------------------------------------------------------------*/ .navbar-brand { @@ -1934,7 +1886,6 @@ a:hover .overlay { flex: 1 1 auto; overflow-x: hidden; overflow-y: auto; - -webkit-overflow-scrolling: touch; } .site-menu > ul { @@ -2009,8 +1960,8 @@ a:hover .overlay { .site-menu li.menu-item-has-children.in .menu-icon .iconfont { -ms-transform: rotate(90deg); - transform: rotate(90deg); -webkit-transform: rotate(90deg); + transform: rotate(90deg); } /*-------------------------------------------------------------- @@ -2053,7 +2004,7 @@ a:hover .overlay { } .site-menu .sub-menu li.current-menu-item a, -.site-menu .sub-menu > li .sub-memu > li:hover > a, +.site-menu .sub-menu > li .sub-menu > li:hover > a, .site-menu .sub-menu > li:hover > a, .site-menu > li > ul > li > ul > li:hover > a, .site-menu > li > ul > li:hover > a, @@ -2425,22 +2376,6 @@ a:hover .overlay { } } -/*-------------------------------------------------------------- - pushes style02 ---------------------------------------------------------------*/ - -.list-pushes-style02 .list-item { - margin-bottom: 0; -} - -.list-pushes-style02 .list-item.block .list-content { - padding: 0.75rem; -} - -.list-pushes-style02 .card-body { - /* margin-top: -5rem; */ -} - /* Entry Content ----------------------------- */ .post-content { @@ -2653,15 +2588,6 @@ a:hover .overlay { justify-content: center; } -/*-------------------------------------------------------------- - top thumbnail style ---------------------------------------------------------------*/ - -.post-thumbnail img { - width: 100%; - height: auto; -} - /*-------------------------------------------------------------- post like style --------------------------------------------------------------*/ @@ -2705,34 +2631,32 @@ a:hover .overlay { } } -/*-------------------------------------------------------------- - tag style ---------------------------------------------------------------*/ - -.post-tags a { - background-color: #f7f8fa; - padding: 0.25rem 0.5rem; - font-size: 0.75rem; - margin: 0 0.5rem 0.5rem 0; - border-radius: 0.15rem; - display: inline-block; -} - -/*-------------------------------------------------------------- - comment style ---------------------------------------------------------------*/ - /*-------------------------------------------------------------- footer style --------------------------------------------------------------*/ -.footer, .footer a { - color: #e8e9ea; + color: var(--color-dark); } .footer a:hover { - color: #fff; + color: var(--color-primary); +} + +.footer .line { + margin-top: 3px; + justify-content: center; + display: flex; + flex-direction: row; + gap: 0.5em; + flex-wrap: wrap; +} + +.astro-badge:hover stop:first-child { + stop-color: #3245ff; +} +.astro-badge:hover stop:last-child { + stop-color: #bc52ee; } /*-------------------------------------------------------------- @@ -2851,99 +2775,6 @@ a:hover .overlay { margin-right: 5px; } -/*-------------------------------------------------------------- - widget_categories style ---------------------------------------------------------------*/ - -.blogroll li, -.widget_meta li, -.widget_archive li, -.widget_categories li { - position: relative; - display: -ms-flexbox; - display: -webkit-box; - display: flex; - -ms-flex-direction: row; - -webkit-box-orient: horizontal; - -webkit-box-direction: normal; - flex-direction: row; - -ms-flex-align: center; - -webkit-box-align: center; - align-items: center; - text-align: justify; - -ms-flex-pack: justify; - -webkit-box-pack: justify; - justify-content: space-between; - font-size: inherit; - color: var(--color-muted); - padding: 0.75rem 0 0.5rem 0; - text-transform: uppercase; - border-top: 1px solid var(--border-light); -} - -.blogroll li { - -ms-flex-direction: column; - -webkit-box-orient: vertical; - -webkit-box-direction: normal; - flex-direction: column; - -ms-flex-pack: normal; - -webkit-box-pack: normal; - justify-content: normal; - -ms-flex-align: normal; - -webkit-box-align: normal; - align-items: normal; -} - -.widget_meta li a, -.blogroll li a, -.widget_archive li a, -.widget_categories li a { - font-size: 0.875rem; -} - -.blogroll li a { - display: -ms-flexbox; - display: -webkit-box; - display: flex; - -ms-flex-direction: row; - -webkit-box-orient: horizontal; - -webkit-box-direction: normal; - flex-direction: row; - -ms-flex-align: center; - -webkit-box-align: center; - align-items: center; -} - -.blogroll li a img { - width: 26px; - height: 26px; - border-radius: 100%; - margin-right: 10px; -} - -.blogroll li:last-child, -.widget_archive li:last-child, -.widget_categories li:last-child { - padding: 0.75rem 0 0; -} - -.widget_meta li:first-child, -.blogroll li:first-child, -.widget_archive li:first-child, -.widget_categories li:first-child { - border: none; - padding: 0 0 0.75rem 0; -} - -.widget_categories select, -.widget_archive select { - width: 100%; - font-size: inherit; - border: 1px solid var(--border-light); - border-radius: 0; - height: 35px; -} - /*-------------------------------------------------------------- tagcloud style --------------------------------------------------------------*/ @@ -2976,50 +2807,6 @@ a:hover .overlay { margin-right: 5px; } -/*-------------------------------------------------------------- - searchform style ---------------------------------------------------------------*/ - -.searchform { - position: relative; -} - -.searchform label { - display: none; -} - -.searchform input[type='submit'] { - position: absolute; - bottom: 0; - right: 0; - top: 0; - border: 0; - border-radius: 0; - padding: 0 20px; - color: #fff; - background-color: var(--bg-dark); - -webkit-transition: all 0.3s ease; - -o-transition: all 0.3s ease; - transition: all 0.3s ease; -} - -.searchform input[type='submit']:hover { - background-color: var(--bg-dark); -} - -.searchform input[type='text'] { - width: 100%; -} - -/*-------------------------------------------------------------- - widget_media_image style ---------------------------------------------------------------*/ - -.widget_media_image img { - max-width: 100%; - height: auto; -} - /*-------------------------------------------------------------- widget_recent_comments style --------------------------------------------------------------*/ @@ -3040,118 +2827,7 @@ a:hover .overlay { } /*-------------------------------------------------------------- - widget_calendar style ---------------------------------------------------------------*/ - -.widget_calendar table { - border-collapse: collapse; - margin: 0 0 0.5rem; - width: 100%; - caption-side: top; -} - -.widget_calendar caption { - border: 0; - border-bottom: 0; - padding: 10px 0; -} - -.widget_calendar caption { - padding: 0.25rem 0; -} - -.widget_calendar th, -.widget_calendar td { - text-align: center; - padding: 0.25rem 0.5rem; -} - -@media (min-width: 992px) and (max-width: 1399.98px) { - .widget_calendar th, - .widget_calendar td { - text-align: center; - padding: 0.125rem 0.125rem; - } -} - -.widget_calendar tbody td a { - position: relative; -} - -.widget_calendar tbody td a::after { - content: ''; - position: absolute; - width: 6px; - height: 6px; - background: var(--bg-dark); - border-radius: var(--radius-lg); - bottom: -6px; - left: 50%; - margin-left: -3px; -} - -.widget_calendar td#today { - font-weight: 600; - color: var(--color-primary); -} - -.widget_calendar tfoot { - border: 1px solid var(--border-light); - border-top: 0; -} - -.widget_calendar tfoot td { - border: 0; -} - -.widget_calendar .wp-calendar-nav { - font-size: 0.75rem; -} - -/*-------------------------------------------------------------- - widget_rss style ---------------------------------------------------------------*/ - -.widget_rss .widget-title .rsswidget:first-child { - float: right; -} - -.widget_rss ul li { - text-transform: uppercase; - margin-bottom: 10px; - padding-top: 12px; - border-top: 1px solid var(--border-light); -} - -.widget_rss ul li:first-child { - border: none; - padding-top: 0; -} - -.widget_rss ul li .rsswidget { - font-size: inherit; - display: block; -} - -.widget_rss ul li .rss-date { - font-size: inherit; - color: var(--color-muted); -} - -.widget_rss ul li .rssSummary { - font-size: inherit; - margin-top: 5px; - color: var(--color-secondary); -} - -.widget_rss ul li cite { - font-size: inherit; - font-style: normal; - color: var(--color-muted); -} - -/*-------------------------------------------------------------- -## Pagination (WordPress CSS classes) + Pagination --------------------------------------------------------------*/ .screen-reader-text { @@ -3227,116 +2903,6 @@ a:hover .overlay { } } -/*-------------------------------------------------------------- - dialog ---------------------------------------------------------------*/ - -#nice-back-to-top { - position: fixed; - width: 48px; - height: 48px; - right: 25px; - bottom: 25px; - z-index: 10000; - opacity: 0; - visibility: hidden; -} - -#nice-back-to-top.active { - opacity: 1; - visibility: visible; -} - -#nice-back-to-top .icon-stack { - display: -webkit-box; - display: -ms-flexbox; - display: flex; - -webkit-box-orient: vertical; - -webkit-box-direction: normal; - -ms-flex-direction: column; - flex-direction: column; - -webkit-box-pack: center; - -ms-flex-pack: center; - justify-content: center; - width: 100%; - height: 100%; - font-size: 1rem; - text-align: center; - color: #fff; - background-color: #252c41; - overflow: hidden; - -webkit-transition: - color 0.2s ease-in-out, - background-color 0.2s ease-in-out, - border-color 0.2s ease-in-out; - -o-transition: - color 0.2s ease-in-out, - background-color 0.2s ease-in-out, - border-color 0.2s ease-in-out; - transition: - color 0.2s ease-in-out, - background-color 0.2s ease-in-out, - border-color 0.2s ease-in-out; - border-radius: 0.2rem; -} - -#nice-back-to-top .icon-stack:hover { - background-color: #00818a; -} - -#nice-back-to-top .icon-stack i { - position: absolute; - top: 9px; - left: 50%; - -webkit-transform: translateX(-50%); - -ms-transform: translateX(-50%); - transform: translateX(-50%); - -webkit-transition: 0.15s ease 50ms; - -o-transition: 0.15s ease 50ms; - transition: 0.15s ease 50ms; -} - -#nice-back-to-top .icon-stack .back-to-top-text { - display: inline-block; - -webkit-transform: translateY(5px); - -ms-transform: translateY(5px); - transform: translateY(5px); - font-size: 11px; - text-transform: uppercase; - letter-spacing: 0.05em; - line-height: 1em; - -webkit-transition: -webkit-transform 0.35s cubic-bezier(0.165, 0.84, 0.44, 1); - transition: -webkit-transform 0.35s cubic-bezier(0.165, 0.84, 0.44, 1); - -o-transition: transform 0.35s cubic-bezier(0.165, 0.84, 0.44, 1); - transition: transform 0.35s cubic-bezier(0.165, 0.84, 0.44, 1); - transition: - transform 0.35s cubic-bezier(0.165, 0.84, 0.44, 1), - -webkit-transform 0.35s cubic-bezier(0.165, 0.84, 0.44, 1); -} - -#nice-back-to-top .icon-stack:hover i { - opacity: 0; - -webkit-transform: translateX(-50%) translateY(-10px); - -ms-transform: translateX(-50%) translateY(-10px); - transform: translateX(-50%) translateY(-10px); - -webkit-transition: all 0.45s cubic-bezier(0.165, 0.84, 0.44, 1); - -o-transition: all 0.45s cubic-bezier(0.165, 0.84, 0.44, 1); - transition: all 0.45s cubic-bezier(0.165, 0.84, 0.44, 1); -} - -#nice-back-to-top .icon-stack:hover .back-to-top-text { - -webkit-transform: translateY(0); - -ms-transform: translateY(0); - transform: translateY(0); - -webkit-transition: -webkit-transform 0.25s cubic-bezier(0.165, 0.84, 0.44, 1) 50ms; - transition: -webkit-transform 0.25s cubic-bezier(0.165, 0.84, 0.44, 1) 50ms; - -o-transition: transform 0.25s cubic-bezier(0.165, 0.84, 0.44, 1) 50ms; - transition: transform 0.25s cubic-bezier(0.165, 0.84, 0.44, 1) 50ms; - transition: - transform 0.25s cubic-bezier(0.165, 0.84, 0.44, 1) 50ms, - -webkit-transform 0.25s cubic-bezier(0.165, 0.84, 0.44, 1) 50ms; -} - /*-------------------------------------------------------------- Like style in blog post --------------------------------------------------------------*/ @@ -3351,104 +2917,6 @@ a:hover .overlay { padding-left: 3px; } -/*-------------------------------------------------------------- - top search style ---------------------------------------------------------------*/ - -.site-search-popup { - position: fixed; - width: 100%; - left: 0; - right: 0; - top: 0; - bottom: 0; - background-color: #283149; - z-index: 9999; - visibility: hidden; - opacity: 0; - -webkit-transform: translateY(10px); - -ms-transform: translateY(10px); - transform: translateY(10px); - -webkit-transition: all 0.3s; - -o-transition: all 0.3s; - transition: all 0.3s; -} - -.site-search-popup.show-popup { - visibility: visible; - opacity: 1; - -webkit-transform: translateY(0); - -ms-transform: translateY(0); - transform: translateY(0); -} - -.site-search-body { - position: absolute; - display: -ms-flexbox; - display: -webkit-box; - display: flex; - -ms-flex-pack: center; - -webkit-box-pack: center; - justify-content: center; - left: 0; - right: 0; - top: 50%; - -webkit-transform: translate(0, -50%); - -ms-transform: translate(0, -50%); - transform: translate(0, -50%); - -webkit-transition: all 400ms cubic-bezier(0.17, 0.67, 0.39, 1.09); - -o-transition: all 400ms cubic-bezier(0.17, 0.67, 0.39, 1.09); - transition: all 400ms cubic-bezier(0.17, 0.67, 0.39, 1.09); - /* easeOutSine */ -} - -.search-popup-close { - padding: 1rem 0; - text-align: right; - color: #bbbbbb; -} - -.popup-close-btn { - display: inline-block; - cursor: pointer; - line-height: 1; -} - -.popup-close-btn i { - display: block; -} - -.popup-close-btn:hover i { - -webkit-transform: rotate(90deg); - -ms-transform: rotate(90deg); - transform: rotate(90deg); -} - -/*-------------------------------------------------------------- - svg style ---------------------------------------------------------------*/ - -.svg-404 { - display: block; - width: 64px; - height: 64px; - background-image: url(''); - background-repeat: no-repeat; - background-position: center; - background-size: contain; - margin: 0 auto; -} - -.svg-search { - display: block; - width: 64px; - height: 64px; - background-image: url(''); - background-repeat: no-repeat; - background-position: center; - background-size: contain; -} - /*-------------------------------------------------------------- fixed widget style --------------------------------------------------------------*/ @@ -3514,41 +2982,6 @@ a:hover .overlay { } } -/*-------------------------------------------------------------- - list bookmarks style ---------------------------------------------------------------*/ - -.list-bookmarks { - margin: 2rem 0 0; -} - -.list-bookmarks .list-gogogo { - position: absolute; - left: 0; - top: 0; - width: 100%; - height: 100%; - z-index: 1; -} - -.list-bookmarks .list-grouped .list-content { - padding: 0.75rem 1rem 1rem; -} - -.list-bookmarks .list-favicon { - background-size: cover; - display: inline-block; - margin-left: 8px; - width: 12px; - height: 12px; -} - -@media (max-width: 767.98px) { - .list-bookmarks { - margin: 2.5rem 0 0; - } -} - /*-------------------------------------------------------------- comment style --------------------------------------------------------------*/ @@ -3680,13 +3113,12 @@ a:hover .overlay { } .comment-reply-link { - cursor: pointer; background-color: transparent; + transition: all 0.3s linear; } .comment-reply-link:hover { - transition: all 500ms; - font-weight: 800; + color: var(--color-primary); } @media only screen and (max-width: 767px) { diff --git a/src/assets/styles/reset.css b/src/assets/styles/reset.css index 575912c..60579ef 100644 --- a/src/assets/styles/reset.css +++ b/src/assets/styles/reset.css @@ -1,3 +1,6 @@ +/*-------------------------------------------------------------- + color variables +--------------------------------------------------------------*/ :root { --color-primary: #008c95; --color-dark: #151b2b; @@ -76,7 +79,6 @@ } /* -------------------------------------------------------------------------- */ - /* 0. CSS Reset /* -------------------------------------------------------------------------- */ @@ -190,7 +192,6 @@ table caption { } /* -------------------------------------------------------------------------- */ - /* 1. Document Setup /* -------------------------------------------------------------------------- */ diff --git a/src/components/comment/CommentItem.astro b/src/components/comment/CommentItem.astro index 6822dc0..adb332a 100644 --- a/src/components/comment/CommentItem.astro +++ b/src/components/comment/CommentItem.astro @@ -44,7 +44,7 @@ const { comment, config, depth, pending } = Astro.props; diff --git a/src/components/comment/artalk.ts b/src/components/comment/artalk.ts index 1d0e560..fc28384 100644 --- a/src/components/comment/artalk.ts +++ b/src/components/comment/artalk.ts @@ -8,8 +8,8 @@ import type { ErrorResp, } from '@/components/comment/types'; import { increaseViews } from '@/helpers/db/query'; -import { options } from '@/helpers/schema'; import { urlJoin } from '@/helpers/tools'; +import options from '@/options'; import { ARTALK_HOST } from 'astro:env/server'; import _ from 'lodash'; import { marked } from 'marked'; @@ -18,7 +18,7 @@ import { ELEMENT_NODE, transform, walk } from 'ultrahtml'; import sanitize from 'ultrahtml/transformers/sanitize'; // Access the artalk in internal docker host when it was deployed on zeabur. -const server = import.meta.env.PROD ? `http://${ARTALK_HOST}:23366` : options.settings.comments.server; +const server = options.isProd() ? `http://${ARTALK_HOST}:23366` : options.settings.comments.server; export const getConfig = async (): Promise => { const data = await fetch(urlJoin(server, '/api/v2/conf')) diff --git a/src/components/footer/Footer.astro b/src/components/footer/Footer.astro index c465936..2aa7e2f 100644 --- a/src/components/footer/Footer.astro +++ b/src/components/footer/Footer.astro @@ -1,24 +1,42 @@ --- +import options from '@/options'; import { DateTime } from 'luxon'; - -import { options } from '@/helpers/schema'; - -function currentYear(): number { - return DateTime.now().setZone(options.settings.timeZone).year; -} --- -