新增pdf小工具;

制作新编辑器
This commit is contained in:
Guarp 2025-03-07 18:26:36 +08:00
parent d93f89b1b6
commit 6204cb23d6
13 changed files with 1645 additions and 407 deletions

499
package-lock.json generated
View File

@ -8,7 +8,10 @@
"name": "mva-cyberv2",
"version": "0.0.0",
"dependencies": {
"@wangeditor/editor": "^5.1.23",
"@wangeditor/editor-for-vue": "^5.1.12",
"axios": "^1.7.9",
"jquery": "^3.7.1",
"js-cookie": "^3.0.5",
"marked": "^15.0.7",
"mermaid": "^11.4.1",
@ -72,6 +75,17 @@
"node": ">=6.0.0"
}
},
"node_modules/@babel/runtime": {
"version": "7.26.9",
"resolved": "https://registry.npmjs.org/@babel/runtime/-/runtime-7.26.9.tgz",
"integrity": "sha512-aA63XwOkcl4xxQa3HjPMqOP6LiK0ZDv3mUPYEFXkpHbaFjtGggE1A61FjFzJnB+p7/oy2gA8E+rcBNl/zC1tMg==",
"dependencies": {
"regenerator-runtime": "^0.14.0"
},
"engines": {
"node": ">=6.9.0"
}
},
"node_modules/@babel/types": {
"version": "7.26.8",
"resolved": "https://registry.npmjs.org/@babel/types/-/types-7.26.8.tgz",
@ -803,6 +817,11 @@
"win32"
]
},
"node_modules/@transloadit/prettier-bytes": {
"version": "0.0.7",
"resolved": "https://registry.npmjs.org/@transloadit/prettier-bytes/-/prettier-bytes-0.0.7.tgz",
"integrity": "sha512-VeJbUb0wEKbcwaSlj5n+LscBl9IPgLPkHVGBkh00cztv6X4L/TJXK58LzFuBKX7/GAfiGhIwH67YTLTlzvIzBA=="
},
"node_modules/@types/d3": {
"version": "7.4.3",
"resolved": "https://registry.npmjs.org/@types/d3/-/d3-7.4.3.tgz",
@ -1031,6 +1050,11 @@
"integrity": "sha512-AYnb1nQyY49te+VRAVgmzfcgjYS91mY5P0TKUDCLEM+gNnA+3T6rWITXRLYCpahpqSQbN5cE+gHpnPyXjHWxcw==",
"dev": true
},
"node_modules/@types/event-emitter": {
"version": "0.3.5",
"resolved": "https://registry.npmjs.org/@types/event-emitter/-/event-emitter-0.3.5.tgz",
"integrity": "sha512-zx2/Gg0Eg7gwEiOIIh5w9TrhKKTeQh7CPCOPNc0el4pLSwzebA8SmnHwZs2dWlLONvyulykSwGSQxQHLhjGLvQ=="
},
"node_modules/@types/geojson": {
"version": "7946.0.16",
"resolved": "https://registry.npmjs.org/@types/geojson/-/geojson-7946.0.16.tgz",
@ -1042,6 +1066,56 @@
"integrity": "sha512-ScaPdn1dQczgbl0QFTeTOmVHFULt394XJgOQNoyVhZ6r2vLnMLJfBPd53SB52T/3G36VI1/g2MZaX0cwDuXsfw==",
"optional": true
},
"node_modules/@uppy/companion-client": {
"version": "2.2.2",
"resolved": "https://registry.npmjs.org/@uppy/companion-client/-/companion-client-2.2.2.tgz",
"integrity": "sha512-5mTp2iq97/mYSisMaBtFRry6PTgZA6SIL7LePteOV5x0/DxKfrZW3DEiQERJmYpHzy7k8johpm2gHnEKto56Og==",
"dependencies": {
"@uppy/utils": "^4.1.2",
"namespace-emitter": "^2.0.1"
}
},
"node_modules/@uppy/core": {
"version": "2.3.4",
"resolved": "https://registry.npmjs.org/@uppy/core/-/core-2.3.4.tgz",
"integrity": "sha512-iWAqppC8FD8mMVqewavCz+TNaet6HPXitmGXpGGREGrakZ4FeuWytVdrelydzTdXx6vVKkOmI2FLztGg73sENQ==",
"dependencies": {
"@transloadit/prettier-bytes": "0.0.7",
"@uppy/store-default": "^2.1.1",
"@uppy/utils": "^4.1.3",
"lodash.throttle": "^4.1.1",
"mime-match": "^1.0.2",
"namespace-emitter": "^2.0.1",
"nanoid": "^3.1.25",
"preact": "^10.5.13"
}
},
"node_modules/@uppy/store-default": {
"version": "2.1.1",
"resolved": "https://registry.npmjs.org/@uppy/store-default/-/store-default-2.1.1.tgz",
"integrity": "sha512-xnpTxvot2SeAwGwbvmJ899ASk5tYXhmZzD/aCFsXePh/v8rNvR2pKlcQUH7cF/y4baUGq3FHO/daKCok/mpKqQ=="
},
"node_modules/@uppy/utils": {
"version": "4.1.3",
"resolved": "https://registry.npmjs.org/@uppy/utils/-/utils-4.1.3.tgz",
"integrity": "sha512-nTuMvwWYobnJcytDO3t+D6IkVq/Qs4Xv3vyoEZ+Iaf8gegZP+rEyoaFT2CK5XLRMienPyqRqNbIfRuFaOWSIFw==",
"dependencies": {
"lodash.throttle": "^4.1.1"
}
},
"node_modules/@uppy/xhr-upload": {
"version": "2.1.3",
"resolved": "https://registry.npmjs.org/@uppy/xhr-upload/-/xhr-upload-2.1.3.tgz",
"integrity": "sha512-YWOQ6myBVPs+mhNjfdWsQyMRWUlrDLMoaG7nvf/G6Y3GKZf8AyjFDjvvJ49XWQ+DaZOftGkHmF1uh/DBeGivJQ==",
"dependencies": {
"@uppy/companion-client": "^2.2.2",
"@uppy/utils": "^4.1.2",
"nanoid": "^3.1.25"
},
"peerDependencies": {
"@uppy/core": "^2.3.3"
}
},
"node_modules/@vitejs/plugin-vue": {
"version": "5.2.1",
"resolved": "https://registry.npmjs.org/@vitejs/plugin-vue/-/plugin-vue-5.2.1.tgz",
@ -1151,6 +1225,156 @@
"resolved": "https://registry.npmjs.org/@vue/shared/-/shared-3.5.13.tgz",
"integrity": "sha512-/hnE/qP5ZoGpol0a5mDi45bOd7t3tjYJBjsgCsivow7D48cJeV5l05RD82lPqi7gRiphZM37rnhW1l6ZoCNNnQ=="
},
"node_modules/@wangeditor/basic-modules": {
"version": "1.1.7",
"resolved": "https://registry.npmjs.org/@wangeditor/basic-modules/-/basic-modules-1.1.7.tgz",
"integrity": "sha512-cY9CPkLJaqF05STqfpZKWG4LpxTMeGSIIF1fHvfm/mz+JXatCagjdkbxdikOuKYlxDdeqvOeBmsUBItufDLXZg==",
"dependencies": {
"is-url": "^1.2.4"
},
"peerDependencies": {
"@wangeditor/core": "1.x",
"dom7": "^3.0.0",
"lodash.throttle": "^4.1.1",
"nanoid": "^3.2.0",
"slate": "^0.72.0",
"snabbdom": "^3.1.0"
}
},
"node_modules/@wangeditor/code-highlight": {
"version": "1.0.3",
"resolved": "https://registry.npmjs.org/@wangeditor/code-highlight/-/code-highlight-1.0.3.tgz",
"integrity": "sha512-iazHwO14XpCuIWJNTQTikqUhGKyqj+dUNWJ9288Oym9M2xMVHvnsOmDU2sgUDWVy+pOLojReMPgXCsvvNlOOhw==",
"dependencies": {
"prismjs": "^1.23.0"
},
"peerDependencies": {
"@wangeditor/core": "1.x",
"dom7": "^3.0.0",
"slate": "^0.72.0",
"snabbdom": "^3.1.0"
}
},
"node_modules/@wangeditor/core": {
"version": "1.1.19",
"resolved": "https://registry.npmjs.org/@wangeditor/core/-/core-1.1.19.tgz",
"integrity": "sha512-KevkB47+7GhVszyYF2pKGKtCSj/YzmClsD03C3zTt+9SR2XWT5T0e3yQqg8baZpcMvkjs1D8Dv4fk8ok/UaS2Q==",
"dependencies": {
"@types/event-emitter": "^0.3.3",
"event-emitter": "^0.3.5",
"html-void-elements": "^2.0.0",
"i18next": "^20.4.0",
"scroll-into-view-if-needed": "^2.2.28",
"slate-history": "^0.66.0"
},
"peerDependencies": {
"@uppy/core": "^2.1.1",
"@uppy/xhr-upload": "^2.0.3",
"dom7": "^3.0.0",
"is-hotkey": "^0.2.0",
"lodash.camelcase": "^4.3.0",
"lodash.clonedeep": "^4.5.0",
"lodash.debounce": "^4.0.8",
"lodash.foreach": "^4.5.0",
"lodash.isequal": "^4.5.0",
"lodash.throttle": "^4.1.1",
"lodash.toarray": "^4.4.0",
"nanoid": "^3.2.0",
"slate": "^0.72.0",
"snabbdom": "^3.1.0"
}
},
"node_modules/@wangeditor/editor": {
"version": "5.1.23",
"resolved": "https://registry.npmjs.org/@wangeditor/editor/-/editor-5.1.23.tgz",
"integrity": "sha512-0RxfeVTuK1tktUaPROnCoFfaHVJpRAIE2zdS0mpP+vq1axVQpLjM8+fCvKzqYIkH0Pg+C+44hJpe3VVroSkEuQ==",
"dependencies": {
"@uppy/core": "^2.1.1",
"@uppy/xhr-upload": "^2.0.3",
"@wangeditor/basic-modules": "^1.1.7",
"@wangeditor/code-highlight": "^1.0.3",
"@wangeditor/core": "^1.1.19",
"@wangeditor/list-module": "^1.0.5",
"@wangeditor/table-module": "^1.1.4",
"@wangeditor/upload-image-module": "^1.0.2",
"@wangeditor/video-module": "^1.1.4",
"dom7": "^3.0.0",
"is-hotkey": "^0.2.0",
"lodash.camelcase": "^4.3.0",
"lodash.clonedeep": "^4.5.0",
"lodash.debounce": "^4.0.8",
"lodash.foreach": "^4.5.0",
"lodash.isequal": "^4.5.0",
"lodash.throttle": "^4.1.1",
"lodash.toarray": "^4.4.0",
"nanoid": "^3.2.0",
"slate": "^0.72.0",
"snabbdom": "^3.1.0"
}
},
"node_modules/@wangeditor/editor-for-vue": {
"version": "5.1.12",
"resolved": "https://registry.npmjs.org/@wangeditor/editor-for-vue/-/editor-for-vue-5.1.12.tgz",
"integrity": "sha512-0Ds3D8I+xnpNWezAeO7HmPRgTfUxHLMd9JKcIw+QzvSmhC5xUHbpCcLU+KLmeBKTR/zffnS5GQo6qi3GhTMJWQ==",
"peerDependencies": {
"@wangeditor/editor": ">=5.1.0",
"vue": "^3.0.5"
}
},
"node_modules/@wangeditor/list-module": {
"version": "1.0.5",
"resolved": "https://registry.npmjs.org/@wangeditor/list-module/-/list-module-1.0.5.tgz",
"integrity": "sha512-uDuYTP6DVhcYf7mF1pTlmNn5jOb4QtcVhYwSSAkyg09zqxI1qBqsfUnveeDeDqIuptSJhkh81cyxi+MF8sEPOQ==",
"peerDependencies": {
"@wangeditor/core": "1.x",
"dom7": "^3.0.0",
"slate": "^0.72.0",
"snabbdom": "^3.1.0"
}
},
"node_modules/@wangeditor/table-module": {
"version": "1.1.4",
"resolved": "https://registry.npmjs.org/@wangeditor/table-module/-/table-module-1.1.4.tgz",
"integrity": "sha512-5saanU9xuEocxaemGdNi9t8MCDSucnykEC6jtuiT72kt+/Hhh4nERYx1J20OPsTCCdVr7hIyQenFD1iSRkIQ6w==",
"peerDependencies": {
"@wangeditor/core": "1.x",
"dom7": "^3.0.0",
"lodash.isequal": "^4.5.0",
"lodash.throttle": "^4.1.1",
"nanoid": "^3.2.0",
"slate": "^0.72.0",
"snabbdom": "^3.1.0"
}
},
"node_modules/@wangeditor/upload-image-module": {
"version": "1.0.2",
"resolved": "https://registry.npmjs.org/@wangeditor/upload-image-module/-/upload-image-module-1.0.2.tgz",
"integrity": "sha512-z81lk/v71OwPDYeQDxj6cVr81aDP90aFuywb8nPD6eQeECtOymrqRODjpO6VGvCVxVck8nUxBHtbxKtjgcwyiA==",
"peerDependencies": {
"@uppy/core": "^2.0.3",
"@uppy/xhr-upload": "^2.0.3",
"@wangeditor/basic-modules": "1.x",
"@wangeditor/core": "1.x",
"dom7": "^3.0.0",
"lodash.foreach": "^4.5.0",
"slate": "^0.72.0",
"snabbdom": "^3.1.0"
}
},
"node_modules/@wangeditor/video-module": {
"version": "1.1.4",
"resolved": "https://registry.npmjs.org/@wangeditor/video-module/-/video-module-1.1.4.tgz",
"integrity": "sha512-ZdodDPqKQrgx3IwWu4ZiQmXI8EXZ3hm2/fM6E3t5dB8tCaIGWQZhmqd6P5knfkRAd3z2+YRSRbxOGfoRSp/rLg==",
"peerDependencies": {
"@uppy/core": "^2.1.4",
"@uppy/xhr-upload": "^2.0.7",
"@wangeditor/core": "1.x",
"dom7": "^3.0.0",
"nanoid": "^3.2.0",
"slate": "^0.72.0",
"snabbdom": "^3.1.0"
}
},
"node_modules/acorn": {
"version": "8.14.0",
"resolved": "https://registry.npmjs.org/acorn/-/acorn-8.14.0.tgz",
@ -1220,6 +1444,11 @@
"node": ">= 10"
}
},
"node_modules/compute-scroll-into-view": {
"version": "1.0.20",
"resolved": "https://registry.npmjs.org/compute-scroll-into-view/-/compute-scroll-into-view-1.0.20.tgz",
"integrity": "sha512-UCB0ioiyj8CRjtrvaceBLqqhZCVP+1B8+NWQhmdsm0VXOJtobBCf1dBQmebCCo34qZmUwZfIH2MZLqNHazrfjg=="
},
"node_modules/confbox": {
"version": "0.1.8",
"resolved": "https://registry.npmjs.org/confbox/-/confbox-0.1.8.tgz",
@ -1281,6 +1510,18 @@
"resolved": "https://registry.npmjs.org/layout-base/-/layout-base-2.0.1.tgz",
"integrity": "sha512-dp3s92+uNI1hWIpPGH3jK2kxE2lMjdXdr+DH8ynZHpd6PUlH6x6cbuXnoMmiNumznqaNO31xu9e79F0uuZ0JFg=="
},
"node_modules/d": {
"version": "1.0.2",
"resolved": "https://registry.npmjs.org/d/-/d-1.0.2.tgz",
"integrity": "sha512-MOqHvMWF9/9MX6nza0KgvFH4HpMU0EF5uUDXqX/BtxtU8NfB0QzRtJ8Oe/6SuS4kbhyzVJwjd97EA4PKrzJ8bw==",
"dependencies": {
"es5-ext": "^0.10.64",
"type": "^2.7.2"
},
"engines": {
"node": ">=0.12"
}
},
"node_modules/d3": {
"version": "7.9.0",
"resolved": "https://registry.npmjs.org/d3/-/d3-7.9.0.tgz",
@ -1732,6 +1973,14 @@
"node": ">=0.4.0"
}
},
"node_modules/dom7": {
"version": "3.0.0",
"resolved": "https://registry.npmjs.org/dom7/-/dom7-3.0.0.tgz",
"integrity": "sha512-oNlcUdHsC4zb7Msx7JN3K0Nro1dzJ48knvBOnDPKJ2GV9wl1i5vydJZUSyOfrkKFDZEud/jBsTk92S/VGSAe/g==",
"dependencies": {
"ssr-window": "^3.0.0-alpha.1"
}
},
"node_modules/dompurify": {
"version": "3.2.4",
"resolved": "https://registry.npmjs.org/dompurify/-/dompurify-3.2.4.tgz",
@ -1751,6 +2000,43 @@
"url": "https://github.com/fb55/entities?sponsor=1"
}
},
"node_modules/es5-ext": {
"version": "0.10.64",
"resolved": "https://registry.npmjs.org/es5-ext/-/es5-ext-0.10.64.tgz",
"integrity": "sha512-p2snDhiLaXe6dahss1LddxqEm+SkuDvV8dnIQG0MWjyHpcMNfXKPE+/Cc0y+PhxJX3A4xGNeFCj5oc0BUh6deg==",
"hasInstallScript": true,
"dependencies": {
"es6-iterator": "^2.0.3",
"es6-symbol": "^3.1.3",
"esniff": "^2.0.1",
"next-tick": "^1.1.0"
},
"engines": {
"node": ">=0.10"
}
},
"node_modules/es6-iterator": {
"version": "2.0.3",
"resolved": "https://registry.npmjs.org/es6-iterator/-/es6-iterator-2.0.3.tgz",
"integrity": "sha512-zw4SRzoUkd+cl+ZoE15A9o1oQd920Bb0iOJMQkQhl3jNc03YqVjAhG7scf9C5KWRU/R13Orf588uCC6525o02g==",
"dependencies": {
"d": "1",
"es5-ext": "^0.10.35",
"es6-symbol": "^3.1.1"
}
},
"node_modules/es6-symbol": {
"version": "3.1.4",
"resolved": "https://registry.npmjs.org/es6-symbol/-/es6-symbol-3.1.4.tgz",
"integrity": "sha512-U9bFFjX8tFiATgtkJ1zg25+KviIXpgRvRHS8sau3GfhVzThRQrOeksPeT0BWW2MNZs1OEWJ1DPXOQMn0KKRkvg==",
"dependencies": {
"d": "^1.0.2",
"ext": "^1.7.0"
},
"engines": {
"node": ">=0.12"
}
},
"node_modules/esbuild": {
"version": "0.24.2",
"resolved": "https://registry.npmjs.org/esbuild/-/esbuild-0.24.2.tgz",
@ -1791,11 +2077,42 @@
"@esbuild/win32-x64": "0.24.2"
}
},
"node_modules/esniff": {
"version": "2.0.1",
"resolved": "https://registry.npmjs.org/esniff/-/esniff-2.0.1.tgz",
"integrity": "sha512-kTUIGKQ/mDPFoJ0oVfcmyJn4iBDRptjNVIzwIFR7tqWXdVI9xfA2RMwY/gbSpJG3lkdWNEjLap/NqVHZiJsdfg==",
"dependencies": {
"d": "^1.0.1",
"es5-ext": "^0.10.62",
"event-emitter": "^0.3.5",
"type": "^2.7.2"
},
"engines": {
"node": ">=0.10"
}
},
"node_modules/estree-walker": {
"version": "2.0.2",
"resolved": "https://registry.npmjs.org/estree-walker/-/estree-walker-2.0.2.tgz",
"integrity": "sha512-Rfkk/Mp/DL7JVje3u18FxFujQlTNR2q6QfMSMB7AvCBx91NGj/ba3kCfza0f6dVDbw7YlRf/nDrn7pQrCCyQ/w=="
},
"node_modules/event-emitter": {
"version": "0.3.5",
"resolved": "https://registry.npmjs.org/event-emitter/-/event-emitter-0.3.5.tgz",
"integrity": "sha512-D9rRn9y7kLPnJ+hMq7S/nhvoKwwvVJahBi2BPmx3bvbsEdK3W9ii8cBSGjP+72/LnM4n6fo3+dkCX5FeTQruXA==",
"dependencies": {
"d": "1",
"es5-ext": "~0.10.14"
}
},
"node_modules/ext": {
"version": "1.7.0",
"resolved": "https://registry.npmjs.org/ext/-/ext-1.7.0.tgz",
"integrity": "sha512-6hxeJYaL110a9b5TEJSj0gojyHQAmA2ch5Os+ySCiA1QGdS697XWY1pzsrSjqA9LDEEgdB/KypIlR59RcLuHYw==",
"dependencies": {
"type": "^2.7.2"
}
},
"node_modules/follow-redirects": {
"version": "1.15.9",
"resolved": "https://registry.npmjs.org/follow-redirects/-/follow-redirects-1.15.9.tgz",
@ -1858,6 +2175,23 @@
"resolved": "https://registry.npmjs.org/hachure-fill/-/hachure-fill-0.5.2.tgz",
"integrity": "sha512-3GKBOn+m2LX9iq+JC1064cSFprJY4jL1jCXTcpnfER5HYE2l/4EfWSGzkPa/ZDBmYI0ZOEj5VHV/eKnPGkHuOg=="
},
"node_modules/html-void-elements": {
"version": "2.0.1",
"resolved": "https://registry.npmjs.org/html-void-elements/-/html-void-elements-2.0.1.tgz",
"integrity": "sha512-0quDb7s97CfemeJAnW9wC0hw78MtW7NU3hqtCD75g2vFlDLt36llsYD7uB7SUzojLMP24N5IatXf7ylGXiGG9A==",
"funding": {
"type": "github",
"url": "https://github.com/sponsors/wooorm"
}
},
"node_modules/i18next": {
"version": "20.6.1",
"resolved": "https://registry.npmjs.org/i18next/-/i18next-20.6.1.tgz",
"integrity": "sha512-yCMYTMEJ9ihCwEQQ3phLo7I/Pwycf8uAx+sRHwwk5U9Aui/IZYgQRyMqXafQOw5QQ7DM1Z+WyEXWIqSuJHhG2A==",
"dependencies": {
"@babel/runtime": "^7.12.0"
}
},
"node_modules/iconv-lite": {
"version": "0.6.3",
"resolved": "https://registry.npmjs.org/iconv-lite/-/iconv-lite-0.6.3.tgz",
@ -1869,6 +2203,15 @@
"node": ">=0.10.0"
}
},
"node_modules/immer": {
"version": "9.0.21",
"resolved": "https://registry.npmjs.org/immer/-/immer-9.0.21.tgz",
"integrity": "sha512-bc4NBHqOqSfRW7POMkHd51LvClaeMXpm8dx0e8oE2GORbq5aRK7Bxl4FyzVLdGtLmvLKL7BTDBG5ACQm4HWjTA==",
"funding": {
"type": "opencollective",
"url": "https://opencollective.com/immer"
}
},
"node_modules/internmap": {
"version": "2.0.3",
"resolved": "https://registry.npmjs.org/internmap/-/internmap-2.0.3.tgz",
@ -1877,6 +2220,29 @@
"node": ">=12"
}
},
"node_modules/is-hotkey": {
"version": "0.2.0",
"resolved": "https://registry.npmjs.org/is-hotkey/-/is-hotkey-0.2.0.tgz",
"integrity": "sha512-UknnZK4RakDmTgz4PI1wIph5yxSs/mvChWs9ifnlXsKuXgWmOkY/hAE0H/k2MIqH0RlRye0i1oC07MCRSD28Mw=="
},
"node_modules/is-plain-object": {
"version": "5.0.0",
"resolved": "https://registry.npmjs.org/is-plain-object/-/is-plain-object-5.0.0.tgz",
"integrity": "sha512-VRSzKkbMm5jMDoKLbltAkFQ5Qr7VDiTFGXxYFXXowVj387GeGNOCsOH6Msy00SGZ3Fp84b1Naa1psqgcCIEP5Q==",
"engines": {
"node": ">=0.10.0"
}
},
"node_modules/is-url": {
"version": "1.2.4",
"resolved": "https://registry.npmjs.org/is-url/-/is-url-1.2.4.tgz",
"integrity": "sha512-ITvGim8FhRiYe4IQ5uHSkj7pVaPDrCTkNd3yq3cV7iZAcJdHTUMPMEHcqSOy9xZ9qFenQCvi+2wjH9a1nXqHww=="
},
"node_modules/jquery": {
"version": "3.7.1",
"resolved": "https://registry.npmjs.org/jquery/-/jquery-3.7.1.tgz",
"integrity": "sha512-m4avr8yL8kmFN8psrbFFFmB/If14iN5o9nw/NgnnM+kybDJpRsAynV2BsfpTYrTRysYUdADVD7CkUUizgkpLfg=="
},
"node_modules/js-cookie": {
"version": "3.0.5",
"resolved": "https://registry.npmjs.org/js-cookie/-/js-cookie-3.0.5.tgz",
@ -1958,6 +2324,42 @@
"resolved": "https://registry.npmjs.org/lodash-es/-/lodash-es-4.17.21.tgz",
"integrity": "sha512-mKnC+QJ9pWVzv+C4/U3rRsHapFfHvQFoFB92e52xeyGMcX6/OlIl78je1u8vePzYZSkkogMPJ2yjxxsb89cxyw=="
},
"node_modules/lodash.camelcase": {
"version": "4.3.0",
"resolved": "https://registry.npmjs.org/lodash.camelcase/-/lodash.camelcase-4.3.0.tgz",
"integrity": "sha512-TwuEnCnxbc3rAvhf/LbG7tJUDzhqXyFnv3dtzLOPgCG/hODL7WFnsbwktkD7yUV0RrreP/l1PALq/YSg6VvjlA=="
},
"node_modules/lodash.clonedeep": {
"version": "4.5.0",
"resolved": "https://registry.npmjs.org/lodash.clonedeep/-/lodash.clonedeep-4.5.0.tgz",
"integrity": "sha512-H5ZhCF25riFd9uB5UCkVKo61m3S/xZk1x4wA6yp/L3RFP6Z/eHH1ymQcGLo7J3GMPfm0V/7m1tryHuGVxpqEBQ=="
},
"node_modules/lodash.debounce": {
"version": "4.0.8",
"resolved": "https://registry.npmjs.org/lodash.debounce/-/lodash.debounce-4.0.8.tgz",
"integrity": "sha512-FT1yDzDYEoYWhnSGnpE/4Kj1fLZkDFyqRb7fNt6FdYOSxlUWAtp42Eh6Wb0rGIv/m9Bgo7x4GhQbm5Ys4SG5ow=="
},
"node_modules/lodash.foreach": {
"version": "4.5.0",
"resolved": "https://registry.npmjs.org/lodash.foreach/-/lodash.foreach-4.5.0.tgz",
"integrity": "sha512-aEXTF4d+m05rVOAUG3z4vZZ4xVexLKZGF0lIxuHZ1Hplpk/3B6Z1+/ICICYRLm7c41Z2xiejbkCkJoTlypoXhQ=="
},
"node_modules/lodash.isequal": {
"version": "4.5.0",
"resolved": "https://registry.npmjs.org/lodash.isequal/-/lodash.isequal-4.5.0.tgz",
"integrity": "sha512-pDo3lu8Jhfjqls6GkMgpahsF9kCyayhgykjyLMNFTKWrpVdAQtYyB4muAMWozBB4ig/dtWAmsMxLEI8wuz+DYQ==",
"deprecated": "This package is deprecated. Use require('node:util').isDeepStrictEqual instead."
},
"node_modules/lodash.throttle": {
"version": "4.1.1",
"resolved": "https://registry.npmjs.org/lodash.throttle/-/lodash.throttle-4.1.1.tgz",
"integrity": "sha512-wIkUCfVKpVsWo3JSZlc+8MB5it+2AN5W8J7YVMST30UrvcQNZ1Okbj+rbVniijTWE6FGYy4XJq/rHkas8qJMLQ=="
},
"node_modules/lodash.toarray": {
"version": "4.4.0",
"resolved": "https://registry.npmjs.org/lodash.toarray/-/lodash.toarray-4.4.0.tgz",
"integrity": "sha512-QyffEA3i5dma5q2490+SgCvDN0pXLmRGSyAANuVi0HQ01Pkfr9fuoKQW8wm1wGBnJITs/mS7wQvS6VshUEBFCw=="
},
"node_modules/magic-string": {
"version": "0.30.17",
"resolved": "https://registry.npmjs.org/magic-string/-/magic-string-0.30.17.tgz",
@ -2023,6 +2425,14 @@
"node": ">= 0.6"
}
},
"node_modules/mime-match": {
"version": "1.0.2",
"resolved": "https://registry.npmjs.org/mime-match/-/mime-match-1.0.2.tgz",
"integrity": "sha512-VXp/ugGDVh3eCLOBCiHZMYWQaTNUHv2IJrut+yXA6+JbLPXHglHwfS/5A5L0ll+jkCY7fIzRJcH6OIunF+c6Cg==",
"dependencies": {
"wildcard": "^1.1.0"
}
},
"node_modules/mime-types": {
"version": "2.1.35",
"resolved": "https://registry.npmjs.org/mime-types/-/mime-types-2.1.35.tgz",
@ -2050,6 +2460,11 @@
"resolved": "https://registry.npmjs.org/ms/-/ms-2.1.3.tgz",
"integrity": "sha512-6FlzubTLZG3J2a/NVCAleEhjzq5oxgHyaCU9yYXvcLsvoVaHJq/s5xXI6/XXP6tz7R9xAOtHnSO/tXtF3WRTlA=="
},
"node_modules/namespace-emitter": {
"version": "2.0.1",
"resolved": "https://registry.npmjs.org/namespace-emitter/-/namespace-emitter-2.0.1.tgz",
"integrity": "sha512-N/sMKHniSDJBjfrkbS/tpkPj4RAbvW3mr8UAzvlMHyun93XEm83IAvhWtJVHo+RHn/oO8Job5YN4b+wRjSVp5g=="
},
"node_modules/nanoid": {
"version": "3.3.8",
"resolved": "https://registry.npmjs.org/nanoid/-/nanoid-3.3.8.tgz",
@ -2067,6 +2482,11 @@
"node": "^10 || ^12 || ^13.7 || ^14 || >=15.0.1"
}
},
"node_modules/next-tick": {
"version": "1.1.0",
"resolved": "https://registry.npmjs.org/next-tick/-/next-tick-1.1.0.tgz",
"integrity": "sha512-CXdUiJembsNjuToQvxayPZF9Vqht7hewsvy2sOWafLvi2awflj9mOC6bHIg50orX8IJvWKY9wYQ/zB2kogPslQ=="
},
"node_modules/package-manager-detector": {
"version": "0.2.9",
"resolved": "https://registry.npmjs.org/package-manager-detector/-/package-manager-detector-0.2.9.tgz",
@ -2138,11 +2558,33 @@
"node": "^10 || ^12 || >=14"
}
},
"node_modules/preact": {
"version": "10.26.4",
"resolved": "https://registry.npmjs.org/preact/-/preact-10.26.4.tgz",
"integrity": "sha512-KJhO7LBFTjP71d83trW+Ilnjbo+ySsaAgCfXOXUlmGzJ4ygYPWmysm77yg4emwfmoz3b22yvH5IsVFHbhUaH5w==",
"funding": {
"type": "opencollective",
"url": "https://opencollective.com/preact"
}
},
"node_modules/prismjs": {
"version": "1.29.0",
"resolved": "https://registry.npmjs.org/prismjs/-/prismjs-1.29.0.tgz",
"integrity": "sha512-Kx/1w86q/epKcmte75LNrEoT+lX8pBpavuAbvJWRXar7Hz8jrtF+e3vY751p0R8H9HdArwaCTNDDzHg/ScJK1Q==",
"engines": {
"node": ">=6"
}
},
"node_modules/proxy-from-env": {
"version": "1.1.0",
"resolved": "https://registry.npmjs.org/proxy-from-env/-/proxy-from-env-1.1.0.tgz",
"integrity": "sha512-D+zkORCbA9f1tdWRK0RaCR3GPv50cMxcrz4X8k5LTSUD1Dkw47mKJEZQNunItRTkWwgtaUSo1RVFRIG9ZXiFYg=="
},
"node_modules/regenerator-runtime": {
"version": "0.14.1",
"resolved": "https://registry.npmjs.org/regenerator-runtime/-/regenerator-runtime-0.14.1.tgz",
"integrity": "sha512-dYnhHh0nJoMfnkZs6GmmhFknAGRrLznOu5nc9ML+EJxGvrx6H7teuevqVqCuPcPK//3eDrrjQhehXVx9cnkGdw=="
},
"node_modules/robust-predicates": {
"version": "3.0.2",
"resolved": "https://registry.npmjs.org/robust-predicates/-/robust-predicates-3.0.2.tgz",
@ -2207,6 +2649,43 @@
"resolved": "https://registry.npmjs.org/safer-buffer/-/safer-buffer-2.1.2.tgz",
"integrity": "sha512-YZo3K82SD7Riyi0E1EQPojLz7kpepnSQI9IyPbHHg1XXXevb5dJI7tpyN2ADxGcQbHG7vcyRHk0cbwqcQriUtg=="
},
"node_modules/scroll-into-view-if-needed": {
"version": "2.2.31",
"resolved": "https://registry.npmjs.org/scroll-into-view-if-needed/-/scroll-into-view-if-needed-2.2.31.tgz",
"integrity": "sha512-dGCXy99wZQivjmjIqihaBQNjryrz5rueJY7eHfTdyWEiR4ttYpsajb14rn9s5d4DY4EcY6+4+U/maARBXJedkA==",
"dependencies": {
"compute-scroll-into-view": "^1.0.20"
}
},
"node_modules/slate": {
"version": "0.72.8",
"resolved": "https://registry.npmjs.org/slate/-/slate-0.72.8.tgz",
"integrity": "sha512-/nJwTswQgnRurpK+bGJFH1oM7naD5qDmHd89JyiKNT2oOKD8marW0QSBtuFnwEbL5aGCS8AmrhXQgNOsn4osAw==",
"dependencies": {
"immer": "^9.0.6",
"is-plain-object": "^5.0.0",
"tiny-warning": "^1.0.3"
}
},
"node_modules/slate-history": {
"version": "0.66.0",
"resolved": "https://registry.npmjs.org/slate-history/-/slate-history-0.66.0.tgz",
"integrity": "sha512-6MWpxGQZiMvSINlCbMW43E2YBSVMCMCIwQfBzGssjWw4kb0qfvj0pIdblWNRQZD0hR6WHP+dHHgGSeVdMWzfng==",
"dependencies": {
"is-plain-object": "^5.0.0"
},
"peerDependencies": {
"slate": ">=0.65.3"
}
},
"node_modules/snabbdom": {
"version": "3.6.2",
"resolved": "https://registry.npmjs.org/snabbdom/-/snabbdom-3.6.2.tgz",
"integrity": "sha512-ig5qOnCDbugFntKi6c7Xlib8bA6xiJVk8O+WdFrV3wxbMqeHO0hXFQC4nAhPVWfZfi8255lcZkNhtIBINCc4+Q==",
"engines": {
"node": ">=12.17.0"
}
},
"node_modules/source-map-js": {
"version": "1.2.1",
"resolved": "https://registry.npmjs.org/source-map-js/-/source-map-js-1.2.1.tgz",
@ -2215,6 +2694,11 @@
"node": ">=0.10.0"
}
},
"node_modules/ssr-window": {
"version": "3.0.0",
"resolved": "https://registry.npmjs.org/ssr-window/-/ssr-window-3.0.0.tgz",
"integrity": "sha512-q+8UfWDg9Itrg0yWK7oe5p/XRCJpJF9OBtXfOPgSJl+u3Xd5KI328RUEvUqSMVM9CiQUEf1QdBzJMkYGErj9QA=="
},
"node_modules/stylis": {
"version": "4.3.6",
"resolved": "https://registry.npmjs.org/stylis/-/stylis-4.3.6.tgz",
@ -2229,6 +2713,11 @@
"url": "https://github.com/sponsors/limonte"
}
},
"node_modules/tiny-warning": {
"version": "1.0.3",
"resolved": "https://registry.npmjs.org/tiny-warning/-/tiny-warning-1.0.3.tgz",
"integrity": "sha512-lBN9zLN/oAf68o3zNXYrdCt1kP8WsiGW8Oo2ka41b2IM5JL/S1CTyX1rW0mb/zSuJun0ZUrDxx4sqvYS2FWzPA=="
},
"node_modules/tinyexec": {
"version": "0.3.2",
"resolved": "https://registry.npmjs.org/tinyexec/-/tinyexec-0.3.2.tgz",
@ -2242,6 +2731,11 @@
"node": ">=6.10"
}
},
"node_modules/type": {
"version": "2.7.3",
"resolved": "https://registry.npmjs.org/type/-/type-2.7.3.tgz",
"integrity": "sha512-8j+1QmAbPvLZow5Qpi6NCaN8FB60p/6x8/vfNqOk/hC+HuvFZhL4+WfekuhQLiqFZXOgQdrs3B+XxEmCc6b3FQ=="
},
"node_modules/ufo": {
"version": "1.5.4",
"resolved": "https://registry.npmjs.org/ufo/-/ufo-1.5.4.tgz",
@ -2418,6 +2912,11 @@
"vue": "^3.0.2"
}
},
"node_modules/wildcard": {
"version": "1.1.2",
"resolved": "https://registry.npmjs.org/wildcard/-/wildcard-1.1.2.tgz",
"integrity": "sha512-DXukZJxpHA8LuotRwL0pP1+rS6CS7FF2qStDDE1C7DDg2rLud2PXRMuEDYIPhgEezwnlHNL4c+N6MfMTjCGTng=="
},
"node_modules/yaml": {
"version": "2.7.0",
"resolved": "https://registry.npmjs.org/yaml/-/yaml-2.7.0.tgz",

View File

@ -9,7 +9,10 @@
"preview": "vite preview"
},
"dependencies": {
"@wangeditor/editor": "^5.1.23",
"@wangeditor/editor-for-vue": "^5.1.12",
"axios": "^1.7.9",
"jquery": "^3.7.1",
"js-cookie": "^3.0.5",
"marked": "^15.0.7",
"mermaid": "^11.4.1",

View File

@ -0,0 +1,115 @@
<!DOCTYPE html>
<html lang="en">
<head>
<meta charset="UTF-8">
<meta name="viewport" content="width=device-width, initial-scale=1.0">
<title>PDF Page Extractor and Cover Generator</title>
<script src="https://cdnjs.cloudflare.com/ajax/libs/pdf-lib/1.17.1/pdf-lib.min.js"></script>
<script src="https://cdnjs.cloudflare.com/ajax/libs/pdf.js/2.10.377/pdf.min.js"></script>
<style>
#pdf-render {
max-width: 100%;
height: auto;
border: 1px solid #ddd;
margin-top: 20px;
}
</style>
</head>
<body>
<h2>Upload a PDF and Extract Specific Pages</h2>
<input type="file" id="pdfFile" accept="application/pdf">
<br><br>
<label for="startPage">Start Page:</label>
<input type="number" id="startPage" min="1">
<label for="endPage">End Page:</label>
<input type="number" id="endPage" min="1">
<br><br>
<button onclick="extractPages()">Extract Pages</button>
<br><br>
<a id="downloadLink" style="display: none;">Download Extracted PDF</a>
<h2>Generate PDF Cover Page</h2>
<input type="file" id="pdf-upload" accept="application/pdf">
<canvas id="pdf-render"></canvas>
<script>
async function extractPages() {
const fileInput = document.getElementById('pdfFile');
const startPage = parseInt(document.getElementById('startPage').value);
const endPage = parseInt(document.getElementById('endPage').value);
if (!fileInput.files.length) {
alert('Please upload a PDF file.');
return;
}
if (isNaN(startPage) || isNaN(endPage) || startPage <= 0 || endPage < startPage) {
alert('Please enter a valid page range.');
return;
}
const file = fileInput.files[0];
const arrayBuffer = await file.arrayBuffer();
const pdfDoc = await PDFLib.PDFDocument.load(arrayBuffer);
const totalPages = pdfDoc.getPageCount();
if (startPage > totalPages || endPage > totalPages) {
alert('Page range exceeds the total number of pages in the PDF.');
return;
}
const newPdfDoc = await PDFLib.PDFDocument.create();
for (let i = startPage - 1; i < endPage; i++) {
const [copiedPage] = await newPdfDoc.copyPages(pdfDoc, [i]);
newPdfDoc.addPage(copiedPage);
}
const pdfBytes = await newPdfDoc.save();
const blob = new Blob([pdfBytes], { type: 'application/pdf' });
const downloadLink = document.getElementById('downloadLink');
downloadLink.href = URL.createObjectURL(blob);
downloadLink.download = 'extracted_pages.pdf';
downloadLink.style.display = 'block';
downloadLink.textContent = 'Download Extracted PDF';
}
document.getElementById('pdf-upload').addEventListener('change', async function(event) {
const file = event.target.files[0];
if (file && file.type === "application/pdf") {
const fileReader = new FileReader();
fileReader.onload = async function() {
const typedArray = new Uint8Array(this.result);
const loadingTask = pdfjsLib.getDocument(typedArray);
try {
const pdf = await loadingTask.promise;
const page = await pdf.getPage(1); // 获取第一页
const viewport = page.getViewport({ scale: 1.5 });
const canvas = document.getElementById('pdf-render');
const context = canvas.getContext('2d');
canvas.width = viewport.width;
canvas.height = viewport.height;
// 渲染页面
const renderContext = {
canvasContext: context,
viewport: viewport
};
await page.render(renderContext).promise;
console.log('封面生成成功');
} catch (error) {
console.error('无法加载 PDF 文件:', error);
}
};
fileReader.readAsArrayBuffer(file);
} else {
alert('请选择一个有效的 PDF 文件');
}
});
</script>
</body>
</html>

View File

@ -36,7 +36,8 @@ onMounted(async () => {
} else {
AuthService.logout();
}
updateGlobalTheme(store.state.theme)
updateGlobalTheme(store.state.theme);
})
// Vuex

View File

@ -2,9 +2,15 @@ import { createApp } from 'vue'
import App from './App.vue'
import store from './store'
import router from './router'
import $ from 'jquery';
import './style.css'
const app = createApp(App);
app.config.globalProperties.$ = $;
window.$ = $;
window.jQuery = $;
app.use(router);
app.use(store);
app.mount('#app')

View File

@ -1,426 +1,543 @@
<script setup>
import GeneralRenderer from "../components/GeneralRenderer.vue";
import {onMounted, onUnmounted, ref, watch} from "vue";
import store from "../store/index.js";
import swal from "../utils/sweetalert.js";
import getCurrentTime from "../utils/getCurrentTime.js";
import Swal from "sweetalert2";
import api from "../utils/axios.js";
const contentInput = ref(store.state.editStore.blog || '');
const titleInput = ref(store.state.editStore.blogTitle || '')
const portMode = ref('both');
const windowWidth = ref(0);
const isMobileMode = ref(false);
const isMenuOpen = ref(false);
const funcButtons = ref([
{name: 'h1', func: '# [cur]'},
{name: 'h2', func: '## [cur]'},
{name: 'h3', func: '### [cur]'},
{name: '<s>abc</s>', func: '~~[cur]~~'},
{name: '<b>abc</b>', func: '**[cur]**'},
{name: '<i>abc</i>', func: '*[cur]*'},
{name: '<code>abc</code>', func: '\`[cur]\`'},
{name: '●', func: '- '},
{name: 'url', func: '[[cur]](https://example.com)'},
{name: 'img', func: '![图片说明](https://example.com/img1.png)'},
{name: 'mth', func: '$[cur]$'},
{name: 'Mth', func: '$$[cur]$$'},
])
function clickFuncBtn(func) {
const textarea = document.querySelector('textarea'); // textarea
const startPos = textarea.selectionStart; //
const endPos = textarea.selectionEnd; //
const selectedText = textarea.value.slice(startPos, endPos); //
let newText;
if (selectedText) {
newText = func.replace('[cur]', selectedText);
} else {
newText = func.replace('[cur]', '请在此填写内容');
}
textarea.setRangeText(newText, startPos, endPos, 'select'); //
contentInput.value = textarea.value;
const curPos = textarea.value.indexOf('[cur]');
textarea.selectionStart = curPos;
textarea.selectionEnd = curPos;
textarea.focus();
}
const checkWindowSize = () => {
windowWidth.value = window.innerWidth;
if (windowWidth.value < 705) {
isMobileMode.value = true;
} else {
isMobileMode.value = false;
isMenuOpen.value = false;
}
};
const saveDocument = () => {
store.commit('saveEdit', {
blog: contentInput.value,
blogTitle: titleInput.value,
blogSaveTime: getCurrentTime()
});
swal.tip('success', '保存成功')
};
const handleKeydown = (event) => {
// Ctrl + S
if (event.ctrlKey && event.key === 's') {
event.preventDefault(); //
saveDocument(); //
}
};
const submitDocument = async () => {
if (!titleInput.value) {
swal.tip('info', '标题为必填项');
return;
}
if (!contentInput.value) {
swal.tip('info', '内容为必填项');
return;
}
store.commit('saveEdit', {
blog: contentInput.value,
blogTitle: titleInput.value,
blogSaveTime: getCurrentTime()
});
const result = await Swal.fire({
title: '是否允许评论',
html:
'<label class="swal2-switch checkbox-label">' +
'<input type="checkbox" id="swal-switch" class="swal2-switch-input">' ,
showCancelButton: true,
confirmButtonText: '提交',
cancelButtonText: '取消',
})
if (result.isConfirmed) {
const isChecked = document.getElementById('swal-switch').checked;
const formData = new FormData();
formData.append('title', titleInput.value);
formData.append('content', contentInput.value);
formData.append('allowComments', isChecked);
store.commit('startLoading');
api.post('/blogs', formData, {
headers: {
'Content-Type': 'multipart/form-data', // Content-Type
},
}).then(result => {
if (result.code === 1) {
swal.tip('success', `提交成功id${ ('为'+ result.blogId) || '读取失败...' }`);
store.commit('stopLoading');
return;
}
swal.tip('error', `错误(其他状态码 ${result.code}`)
store.commit('stopLoading');
}).catch(result => {
swal.tip('error', `错误 请求失败 ${result.code}`)
store.commit('stopLoading');
})
}
}
watch(portMode, async () => {
setTimeout(() => {
contentInput.value = contentInput.value + ' ';
}, 1)
setTimeout(() => {
contentInput.value = contentInput.value.slice(0, -1);
}, 2)
});
watch(contentInput, () => {
if (store.state.editAutoSave.on && store.state.editAutoSave.interval === 114514) {
store.commit('saveEdit', {
blog: contentInput.value,
blogTitle: titleInput.value,
blogSaveTime: getCurrentTime()
});
}
})
watch(titleInput, () => {
if (store.state.editAutoSave.on && store.state.editAutoSave.interval === 114514) {
store.commit('saveEdit', {
blog: contentInput.value,
blogTitle: titleInput.value,
blogSaveTime: getCurrentTime()
});
}
})
onMounted(() => {
checkWindowSize();
window.addEventListener('resize', checkWindowSize);
window.addEventListener('keydown', handleKeydown);
autoSave = setInterval(()=>{
if (! store.state.editAutoSave.on || store.state.editAutoSave.interval === 114514) {
return;
}
store.commit('saveEdit', {
blog: contentInput.value,
blogTitle: titleInput.value,
blogSaveTime: getCurrentTime()
});
}, store.state.editAutoSave.interval);
});
let autoSave
onUnmounted(() => {
clearInterval(autoSave);
autoSave = undefined;
window.removeEventListener('keydown', handleKeydown);
});
</script>
<template>
<div class="container" :class="{'compact-form': isMobileMode}">
<div class="header">
<input placeholder="输入标题" v-model="titleInput">
</div>
<div class="top">
<div class="function-btn" :style="{flex: isMobileMode?1:3}">
<button v-if="! isMobileMode" v-for="btn in funcButtons" v-html="btn.name" @click="clickFuncBtn(btn.func)"/>
<button v-if="isMobileMode" @click="isMenuOpen = ! isMenuOpen"></button>
<div v-if="isMobileMode && isMenuOpen" class="function-btn-menu">
<button v-for="btn in funcButtons" v-html="btn.name" @click="clickFuncBtn(btn.func)"/>
</div>
</div>
<div class="port-btn">
<button @click="portMode = 'both'" :class="{onMode: portMode === 'both'}"></button>
<button @click="portMode = 'edit'" :class="{onMode: portMode === 'edit'}"></button>
<button @click="portMode = 'view'" :class="{onMode: portMode === 'view'}"></button>
</div>
<div class="doc-btn">
<button @click="saveDocument">保存</button>
<button @click="submitDocument">提交</button>
</div>
</div>
<div class="middle">
<div v-if="portMode !== 'view'" class="left">
<textarea v-model="contentInput"></textarea>
</div>
<div v-if="portMode !== 'edit'" class="right">
<GeneralRenderer :content-input="contentInput"/>
</div>
</div>
<div class="bottom">
<div class="characters">总字符数: {{ contentInput.length }}</div>
<div class="auto-save-switch" @click="store.commit('toggleAutoSave')">自动保存: {{ store.state.editAutoSave.on ? '' : '' }} </div>
<div class="save-time-display">{{ store.state.editStore.blogSaveTime ? `上次保存 [${store.state.editStore.blogSaveTime}` : ''}}]</div>
<div class="container">
<div class="editor-container">
<Toolbar
class="tool-bar"
:editor="editorRef"
:defaultConfig="toolbarConfig"
:mode="mode"
/>
<Editor
:defaultConfig="editorConfig"
:mode="mode"
v-model="valueHtml"
@onCreated="handleCreated"
@onChange="handleChange"
@onDestroyed="handleDestroyed"
@onFocus="handleFocus"
@onBlur="handleBlur"
@customAlert="customAlert"
@customPaste="customPaste"
/>
</div>
<button class="submit-btn" @click="submitBlog">提交</button>
</div>
</template>
<script setup>
import '@wangeditor/editor/dist/css/style.css';
import {ref, shallowRef, onMounted, onBeforeUnmount} from 'vue';
import {Editor, Toolbar} from '@wangeditor/editor-for-vue';
import api from "../utils/axios.js";
import swal from "../utils/sweetalert.js";
import Swal from "sweetalert2";
// shallowRef
const editorRef = shallowRef();
// HTML
const valueHtml = ref('<p>hello</p>');
const imagesCache = ref([]);
//
const toolbarConfig = {};
const editorConfig = {
placeholder: '请输入内容...',
MENU_CONF: {},
};
//
const mode = 'default';
toolbarConfig.excludeKeys = ["insertImage", "group-video", "fullScreen", "insertTable"];
toolbarConfig.modalAppendToBody = true;
editorConfig.MENU_CONF.uploadImage = {
fieldName: "image", //
async customUpload(file, insertFn) {
const index = imagesCache.value.length; //
//
const imageSize = await getImageSize(file);
//
imagesCache.value.push({
file,
originalWidth: imageSize.width,
originalHeight: imageSize.height
});
// URL
const objectURL = URL.createObjectURL(file);
insertFn(objectURL, `image-${index}`, objectURL); //
}
};
// 🎯
const getImageSize = (file) => {
return new Promise((resolve) => {
const img = new Image();
img.src = URL.createObjectURL(file);
img.onload = () => {
resolve({width: img.width, height: img.height});
};
});
};
// Ajax
onMounted(() => {
// setTimeout(() => {
// valueHtml.value = '<p> Ajax </p>';
// // valueHtml.value = valueHtml.value.replaceAll(
// // '<holder>',
// // ''
// // );
// }, 1500);
});
//
onBeforeUnmount(() => {
const editor = editorRef.value;
if (editor == null) return;
editor.destroy();
});
async function showInputPopup() {
try {
while (1) {
const result = await Swal.fire({
title: '请输入标题',
input: 'text',
inputLabel: '标题',
inputPlaceholder: '请输入您的标题...',
showCancelButton: true,
cancelButtonText: '取消',
confirmButtonText: '确定',
inputValidator: (value) => {
if (!value) {
return '标题不能为空!'
}
}
})
//
if (!result.isConfirmed) {
return -1;
}
const title = result.value;
if (title) {
const result = await swal.window('info', `确定吗?`, `用"${title}"作为标题`, '确定', '重输');
if (result.isConfirmed) {
return title;
}
}
}
} catch (error) {
console.error('输入弹窗出错:', error)
}
}
// 🚀
const submitBlog = async () => {
if (!editorRef.value) return;
const title = await showInputPopup();
if (title === -1) {
return;
}
const response = await swal.window('info', '允许评论吗?', '其他用户可以在你的博客下留言', '允许', '不允许');
let allowComments = response.isConfirmed;
let content = editorRef.value.getHtml(); // HTML
const images = [...imagesCache.value]; //
// `<img>`
const imgTags = content.match(/<img[^>]+>/g) || [];
// blob URL
const urlToIndexMap = new Map();
let uniqueIndex = 0;
// blob URL
imgTags.forEach((imgTag) => {
const srcMatch = imgTag.match(/src=["']([^"']+)["']/);
if (srcMatch) {
const src = srcMatch[1]; // blob URL
if (!urlToIndexMap.has(src)) {
urlToIndexMap.set(src, uniqueIndex);
uniqueIndex++;
}
}
});
// `<img>`
imgTags.forEach((imgTag) => {
const srcMatch = imgTag.match(/src=["']([^"']+)["']/);
if (srcMatch) {
const src = srcMatch[1];
const index = urlToIndexMap.get(src); // blob URL
const styleMatch = imgTag.match(/style=["']([^"']+)["']/);
let width = "", height = "";
// images blob URL
const imageData = images.find((img) => URL.createObjectURL(img.file) === src);
const originalWidth = imageData?.originalWidth || 0;
const originalHeight = imageData?.originalHeight || 0;
if (styleMatch && styleMatch[1]) {
const styleStr = styleMatch[1];
const widthMatch = styleStr.match(/width:\s*([\d.]+)px/);
const heightMatch = styleStr.match(/height:\s*([\d.]+)px/);
const percentWidthMatch = styleStr.match(/width:\s*([\d.]+)%/);
if (widthMatch) {
width = widthMatch[1];
} else if (percentWidthMatch && originalWidth) {
width = ((parseFloat(percentWidthMatch[1]) / 100) * originalWidth).toFixed(2);
}
if (heightMatch) {
height = heightMatch[1];
} else if (width && originalWidth && originalHeight) {
height = ((width / originalWidth) * originalHeight).toFixed(2);
}
} else {
width = originalWidth;
height = originalHeight;
}
// 使
content = content.replace(imgTag, `<preholder image ${index} width=${width} height=${height}>`);
}
});
// 2
const formData = new FormData();
formData.append("title", title);
formData.append("content", content);
formData.append("allow_comments", allowComments);
images.forEach((imgData, index) => {
formData.append(`images[${index}]`, imgData.file); //
});
console.log(Object.fromEntries(formData.entries()));
// 3
api.post('/blogs', formData).then(response => {
if (response.status !== 200) {
swal.tip('error', `404'}`);
return;
}
if (response.code === 0) {
swal.window('success', `提交成功, 博客id${response.blogId || '未找到(blogId字段)'}`);
return;
}
swal.tip('error', '提交失败, code字段不为0')
}).catch((e) => {
swal.tip('error', `错误${e.message}`)
});
};
//
const handleCreated = (editor) => {
// console.log('created', editor);
editorRef.value = editor; // editor
};
const handleChange = (editor) => {
// console.log('change:', editor.getHtml());
};
const handleDestroyed = (editor) => {
// console.log('destroyed', editor);
};
const handleFocus = (editor) => {
// console.log('focus', editor);
};
const handleBlur = (editor) => {
// console.log('blur', editor);
};
const customAlert = (info, type) => {
// alert(`${type} - ${info}`);
};
const customPaste = (editor, event, callback) => {
// console.log('ClipboardEvent ', event);
// editor.insertText('xxx');
// callback(false); //
};
//
const insertText = () => {
const editor = editorRef.value;
if (editor == null) return;
editor.insertText('hello world');
};
const printHtml = () => {
const editor = editorRef.value;
if (editor == null) return;
console.log(editor.getHtml());
};
const disable = () => {
const editor = editorRef.value;
if (editor == null) return;
editor.disable();
};
</script>
<style scoped>
.container {
background: #131313;
width: calc(100% - 40px);
height: calc(100vh - 100px);
padding: 20px;
max-width: none;
height: calc(100vh - 65px);
max-width: 1300px;
width: 100%;
display: flex;
flex-direction: column;
align-items: center;
}
.container.compact-form {
width: 100%;
height: calc(100vh - 60px);
padding: 0;
}
.theme-light .container {
background: #e0e0e0;
}
.header {
flex: 0 0 60px;
width: 100%;
display: flex;
align-items: center;
}
.header input {
border: none;
outline: none;
padding: 15px;
width: 100%;
height: calc(100% - 30px);
font-size: initial;
background: #2a2a2a;
color: white;
}
.theme-light .header input {
background: white;
color: black;
}
.top {
flex: 0 0 40px;
width: 100%;
background: #3d3d3d;
display: flex;
flex-direction: row;
gap: 10px;
align-content: space-between;
}
.theme-light .top {
background: #f1f1f1;
}
.function-btn {
flex: 2;
display: flex;
flex-direction: row;
align-items: center;
justify-content: flex-start;
margin-left: 5px;
gap: 5px;
}
.function-btn-menu {
position: absolute;
display: flex;
flex-direction: row;
align-items: center;
flex-wrap: wrap;
max-height: 100px;
gap: 3px;
padding: 3px;
top: 190px;
left: 30px;
//width: 100px;
//height: 50px;
background: rgba(0, 0, 0, 0.12);
gap: 20px;
}
.port-btn {
flex: 1;
display: flex;
flex-direction: row;
align-items: center;
justify-content: center;
gap: 5px;
}
.doc-btn {
flex: 1;
display: flex;
flex-direction: row;
align-items: center;
justify-content: flex-end;
margin-right: 5px;
gap: 5px;
}
.doc-btn button {
width: auto !important;
}
.top button {
color: white;
background: #131313;
border: #007bff solid 2px;
width: 30px;
height: 30px;
display: flex;
justify-content: center;
align-items: center;
.submit-btn {
position: absolute;
right: 0;
bottom: 0;
width: 80px;
height: 40px;
border: #ffb74d solid 3px;
border-radius: 5px;
margin: 5px;
opacity: 0.3;
transition: all 0.2s ease;
}
.submit-btn:hover {
opacity: 1;
background: #ffb74d;
}
.editor-container {
width: 99%;
height: 100vh;
max-height: 100%;
margin-top: 10px;
display: flex;
flex-direction: column;
border: 1px solid #5d5d5d;
overflow: hidden;
}
.top button.onMode {
background: #007bff;
border: #007bff solid 2px;
}
.theme-light .top button {
color: black;
background: white;
border: #ffb74d solid 2px;
}
.theme-light .top button.onMode {
background: #ffb74d;
border: #ffb74d solid 2px;
.theme-light .editor-container {
border: 1px solid #c4c4c4;
}
.bottom {
flex: 0 0 25px;
width: 100%;
</style>
<style>
div[data-slate-editor] {
max-height: 0;
}
.w-e-text-container {
background: #2a2a2a;
color: gray;
font-size: small;
display: flex;
align-items: center;
justify-content: space-between;
}
.theme-light .bottom {
background: white;
outline: gray solid 1px;
.w-e-toolbar {
background: #1c1c1c;
border-bottom: 1px solid #595959;
}
.bottom .characters {
margin: 0 5px;
.w-e-toolbar button {
color: #e5e5e5;
background-color: #262626;
}
.bottom .save-time-display {
margin: 0 5px;
.w-e-bar-item button:hover {
color: white;
background-color: #494949;
}
.bottom .auto-save-switch:hover {
background: rgba(128, 128, 128, 0.1);
cursor: pointer;
.w-e-bar-item .disabled:hover {
background-color: #494949;
}
.middle {
width: 100%;
height: 0;
max-height: calc(100vh - 125px);
flex: 1;
display: flex;
align-items: center;
.w-e-bar-item-group .w-e-bar-item-menus-container {
background-color: #262626;
border: #464646 solid 1px;
}
.w-e-toolbar svg {
fill: #e5e5e5;
}
.w-e-drop-panel {
background: #262626;
}
.w-e-panel-content-table {
background-color: #262626;
color: white;
}
.w-e-panel-content-table td {
background-color: #262626;
}
.w-e-panel-content-table td.active {
background-color: #494949;
}
#w-e-textarea-1 {
background: #000000;
color: white;
}
.w-e-bar-divider {
background: #595959;
}
.w-e-select-list {
background: #262626;
}
.w-e-select-list ul {
background: #262626;
color: white;
}
.w-e-select-list ul .selected {
background: #494949;
}
.w-e-select-list ul li:hover {
background: #595959;
}
.w-e-drop-panel, .w-e-select-list {
border: #464646 solid 1px;
}
.w-e-panel-content-color li {
border: #464646 solid 1px;
}
.w-e-panel-content-color li .color-block {
border: #464646 solid 1px;
}
.w-e-hover-bar {
background: black;
}
/* 亮色模式 */
.theme-light .w-e-text-container {
background: #ffffff;
}
.theme-light .w-e-toolbar {
background: #f5f5f5;
border-bottom: 1px solid #d4d4d4;
}
.theme-light .w-e-toolbar button {
color: #333333;
background-color: #ebebeb;
}
.theme-light .w-e-bar-item .disabled svg {
fill: #b6b6b6;
}
.theme-light .w-e-bar-item .disabled {
color: #b6b6b6;
}
.theme-light .w-e-bar-item button:hover {
color: black;
background-color: #dcdcdc;
}
.theme-light .w-e-bar-item .disabled:hover {
color: #b6b6b6;
background-color: #dcdcdc;
}
.theme-light .w-e-bar-item-group .w-e-bar-item-menus-container {
background-color: #ebebeb;
border: #d4d4d4 solid 1px;
}
.theme-light .w-e-toolbar svg {
fill: #333333;
}
.theme-light .w-e-drop-panel {
background: #ebebeb;
}
.theme-light .w-e-panel-content-table {
background-color: #ffffff;
color: black;
}
.theme-light .w-e-panel-content-table td {
background-color: #ffffff;
}
.theme-light .w-e-panel-content-table td.active {
background-color: #dcdcdc;
}
.theme-light #w-e-textarea-1 {
background: #ffffff;
color: black;
}
.theme-light .w-e-bar-divider {
background: #d4d4d4;
}
.theme-light .w-e-select-list {
background: #ebebeb;
}
.theme-light .w-e-select-list ul {
background: #ebebeb;
color: black;
}
.theme-light .w-e-select-list ul .selected {
background: #dcdcdc;
}
.theme-light .w-e-select-list ul li:hover {
background: #d4d4d4;
}
.theme-light .w-e-drop-panel, .theme-light .w-e-select-list {
border: #d4d4d4 solid 1px;
}
.theme-light .w-e-panel-content-color li {
border: #d4d4d4 solid 1px;
}
.theme-light .w-e-panel-content-color li .color-block {
border: #d4d4d4 solid 1px;
}
.theme-light .w-e-hover-bar {
background: white;
}
.middle .left {
flex: 1;
height: 100%;
display: flex;
justify-content: center;
align-items: center;
border: rgba(128, 128, 128, 0.5) solid 1px;
}
.left textarea {
width: calc(100% - 2 * (15px));
height: calc(100% - 2 * (15px));
resize: none;
padding: 15px;
font-size: 17px;
font-family: sans-serif,serif;
border: none;
transition: all 0.2s ease;
}
.left textarea {
background: #1a1a1a;
color: white;
}
.theme-light .left textarea {
background: white;
color: black;
}
.left textarea:focus {
outline: #007bff solid 1px;
}
.theme-light .left textarea:focus {
outline: #ffb74d solid 1px;
}
.middle .right {
flex: 1;
width: calc(100% - 2 * (15px));
height: 100%;
background: #1a1a1a;
border: rgba(128, 128, 128, 0.5) solid 1px;
color: white;
overflow: auto;
padding: 0 15px;
transition: all 0.2s ease;
}
.theme-light .right {
background: white;
color: black;
}
</style>

426
src/pages/MdLatexEditor.vue Normal file
View File

@ -0,0 +1,426 @@
<script setup>
import GeneralRenderer from "../components/GeneralRenderer.vue";
import {onMounted, onUnmounted, ref, watch} from "vue";
import store from "../store/index.js";
import swal from "../utils/sweetalert.js";
import getCurrentTime from "../utils/getCurrentTime.js";
import Swal from "sweetalert2";
import api from "../utils/axios.js";
const contentInput = ref(store.state.editStore.blog || '');
const titleInput = ref(store.state.editStore.blogTitle || '')
const portMode = ref('both');
const windowWidth = ref(0);
const isMobileMode = ref(false);
const isMenuOpen = ref(false);
const funcButtons = ref([
{name: 'h1', func: '# [cur]'},
{name: 'h2', func: '## [cur]'},
{name: 'h3', func: '### [cur]'},
{name: '<s>abc</s>', func: '~~[cur]~~'},
{name: '<b>abc</b>', func: '**[cur]**'},
{name: '<i>abc</i>', func: '*[cur]*'},
{name: '<code>abc</code>', func: '\`[cur]\`'},
{name: '●', func: '- '},
{name: 'url', func: '[[cur]](https://example.com)'},
{name: 'img', func: '![图片说明](https://example.com/img1.png)'},
{name: 'mth', func: '$[cur]$'},
{name: 'Mth', func: '$$[cur]$$'},
])
function clickFuncBtn(func) {
const textarea = document.querySelector('textarea'); // textarea
const startPos = textarea.selectionStart; //
const endPos = textarea.selectionEnd; //
const selectedText = textarea.value.slice(startPos, endPos); //
let newText;
if (selectedText) {
newText = func.replace('[cur]', selectedText);
} else {
newText = func.replace('[cur]', '请在此填写内容');
}
textarea.setRangeText(newText, startPos, endPos, 'select'); //
contentInput.value = textarea.value;
const curPos = textarea.value.indexOf('[cur]');
textarea.selectionStart = curPos;
textarea.selectionEnd = curPos;
textarea.focus();
}
const checkWindowSize = () => {
windowWidth.value = window.innerWidth;
if (windowWidth.value < 705) {
isMobileMode.value = true;
} else {
isMobileMode.value = false;
isMenuOpen.value = false;
}
};
const saveDocument = () => {
store.commit('saveEdit', {
blog: contentInput.value,
blogTitle: titleInput.value,
blogSaveTime: getCurrentTime()
});
swal.tip('success', '保存成功')
};
const handleKeydown = (event) => {
// Ctrl + S
if (event.ctrlKey && event.key === 's') {
event.preventDefault(); //
saveDocument(); //
}
};
const submitDocument = async () => {
if (!titleInput.value) {
swal.tip('info', '标题为必填项');
return;
}
if (!contentInput.value) {
swal.tip('info', '内容为必填项');
return;
}
store.commit('saveEdit', {
blog: contentInput.value,
blogTitle: titleInput.value,
blogSaveTime: getCurrentTime()
});
const result = await Swal.fire({
title: '是否允许评论',
html:
'<label class="swal2-switch checkbox-label">' +
'<input type="checkbox" id="swal-switch" class="swal2-switch-input">' ,
showCancelButton: true,
confirmButtonText: '提交',
cancelButtonText: '取消',
})
if (result.isConfirmed) {
const isChecked = document.getElementById('swal-switch').checked;
const formData = new FormData();
formData.append('title', titleInput.value);
formData.append('content', contentInput.value);
formData.append('allowComments', isChecked);
store.commit('startLoading');
api.post('/blogs', formData, {
headers: {
'Content-Type': 'multipart/form-data', // Content-Type
},
}).then(result => {
if (result.code === 1) {
swal.tip('success', `提交成功id${ ('为'+ result.blogId) || '读取失败...' }`);
store.commit('stopLoading');
return;
}
swal.tip('error', `错误(其他状态码 ${result.code}`)
store.commit('stopLoading');
}).catch(result => {
swal.tip('error', `错误 请求失败 ${result.code}`)
store.commit('stopLoading');
})
}
}
watch(portMode, async () => {
setTimeout(() => {
contentInput.value = contentInput.value + ' ';
}, 1)
setTimeout(() => {
contentInput.value = contentInput.value.slice(0, -1);
}, 2)
});
watch(contentInput, () => {
if (store.state.editAutoSave.on && store.state.editAutoSave.interval === 114514) {
store.commit('saveEdit', {
blog: contentInput.value,
blogTitle: titleInput.value,
blogSaveTime: getCurrentTime()
});
}
})
watch(titleInput, () => {
if (store.state.editAutoSave.on && store.state.editAutoSave.interval === 114514) {
store.commit('saveEdit', {
blog: contentInput.value,
blogTitle: titleInput.value,
blogSaveTime: getCurrentTime()
});
}
})
onMounted(() => {
checkWindowSize();
window.addEventListener('resize', checkWindowSize);
window.addEventListener('keydown', handleKeydown);
autoSave = setInterval(()=>{
if (! store.state.editAutoSave.on || store.state.editAutoSave.interval === 114514) {
return;
}
store.commit('saveEdit', {
blog: contentInput.value,
blogTitle: titleInput.value,
blogSaveTime: getCurrentTime()
});
}, store.state.editAutoSave.interval);
});
let autoSave
onUnmounted(() => {
clearInterval(autoSave);
autoSave = undefined;
window.removeEventListener('keydown', handleKeydown);
});
</script>
<template>
<div class="container" :class="{'compact-form': isMobileMode}">
<div class="header">
<input placeholder="输入标题" v-model="titleInput">
</div>
<div class="top">
<div class="function-btn" :style="{flex: isMobileMode?1:3}">
<button v-if="! isMobileMode" v-for="btn in funcButtons" v-html="btn.name" @click="clickFuncBtn(btn.func)"/>
<button v-if="isMobileMode" @click="isMenuOpen = ! isMenuOpen"></button>
<div v-if="isMobileMode && isMenuOpen" class="function-btn-menu">
<button v-for="btn in funcButtons" v-html="btn.name" @click="clickFuncBtn(btn.func)"/>
</div>
</div>
<div class="port-btn">
<button @click="portMode = 'both'" :class="{onMode: portMode === 'both'}"></button>
<button @click="portMode = 'edit'" :class="{onMode: portMode === 'edit'}"></button>
<button @click="portMode = 'view'" :class="{onMode: portMode === 'view'}"></button>
</div>
<div class="doc-btn">
<button @click="saveDocument">保存</button>
<button @click="submitDocument">提交</button>
</div>
</div>
<div class="middle">
<div v-if="portMode !== 'view'" class="left">
<textarea v-model="contentInput"></textarea>
</div>
<div v-if="portMode !== 'edit'" class="right">
<GeneralRenderer :content-input="contentInput"/>
</div>
</div>
<div class="bottom">
<div class="characters">总字符数: {{ contentInput.length }}</div>
<div class="auto-save-switch" @click="store.commit('toggleAutoSave')">自动保存: {{ store.state.editAutoSave.on ? '' : '' }} </div>
<div class="save-time-display">{{ store.state.editStore.blogSaveTime ? `上次保存 [${store.state.editStore.blogSaveTime}` : ''}}]</div>
</div>
</div>
</template>
<style scoped>
.container {
background: #131313;
width: calc(100% - 40px);
height: calc(100vh - 100px);
padding: 20px;
max-width: none;
display: flex;
flex-direction: column;
align-items: center;
}
.container.compact-form {
width: 100%;
height: calc(100vh - 60px);
padding: 0;
}
.theme-light .container {
background: #e0e0e0;
}
.header {
flex: 0 0 60px;
width: 100%;
display: flex;
align-items: center;
}
.header input {
border: none;
outline: none;
padding: 15px;
width: 100%;
height: calc(100% - 30px);
font-size: initial;
background: #2a2a2a;
color: white;
}
.theme-light .header input {
background: white;
color: black;
}
.top {
flex: 0 0 40px;
width: 100%;
background: #3d3d3d;
display: flex;
flex-direction: row;
gap: 10px;
align-content: space-between;
}
.theme-light .top {
background: #f1f1f1;
}
.function-btn {
flex: 2;
display: flex;
flex-direction: row;
align-items: center;
justify-content: flex-start;
margin-left: 5px;
gap: 5px;
}
.function-btn-menu {
position: absolute;
display: flex;
flex-direction: row;
flex-wrap: wrap;
max-height: 100px;
gap: 3px;
padding: 3px;
top: 190px;
left: 30px;
//width: 100px;
//height: 50px;
background: rgba(0, 0, 0, 0.12);
}
.port-btn {
flex: 1;
display: flex;
flex-direction: row;
align-items: center;
justify-content: center;
gap: 5px;
}
.doc-btn {
flex: 1;
display: flex;
flex-direction: row;
align-items: center;
justify-content: flex-end;
margin-right: 5px;
gap: 5px;
}
.doc-btn button {
width: auto !important;
}
.top button {
color: white;
background: #131313;
border: #007bff solid 2px;
width: 30px;
height: 30px;
display: flex;
justify-content: center;
align-items: center;
border-radius: 5px;
overflow: hidden;
}
.top button.onMode {
background: #007bff;
border: #007bff solid 2px;
}
.theme-light .top button {
color: black;
background: white;
border: #ffb74d solid 2px;
}
.theme-light .top button.onMode {
background: #ffb74d;
border: #ffb74d solid 2px;
}
.bottom {
flex: 0 0 25px;
width: 100%;
background: #2a2a2a;
color: gray;
font-size: small;
display: flex;
align-items: center;
justify-content: space-between;
}
.theme-light .bottom {
background: white;
outline: gray solid 1px;
}
.bottom .characters {
margin: 0 5px;
}
.bottom .save-time-display {
margin: 0 5px;
}
.bottom .auto-save-switch:hover {
background: rgba(128, 128, 128, 0.1);
cursor: pointer;
}
.middle {
width: 100%;
height: 0;
max-height: calc(100vh - 125px);
flex: 1;
display: flex;
align-items: center;
background: white;
}
.middle .left {
flex: 1;
height: 100%;
display: flex;
justify-content: center;
align-items: center;
border: rgba(128, 128, 128, 0.5) solid 1px;
}
.left textarea {
width: calc(100% - 2 * (15px));
height: calc(100% - 2 * (15px));
resize: none;
padding: 15px;
font-size: 17px;
font-family: sans-serif,serif;
border: none;
transition: all 0.2s ease;
}
.left textarea {
background: #1a1a1a;
color: white;
}
.theme-light .left textarea {
background: white;
color: black;
}
.left textarea:focus {
outline: #007bff solid 1px;
}
.theme-light .left textarea:focus {
outline: #ffb74d solid 1px;
}
.middle .right {
flex: 1;
width: calc(100% - 2 * (15px));
height: 100%;
background: #1a1a1a;
border: rgba(128, 128, 128, 0.5) solid 1px;
color: white;
overflow: auto;
padding: 0 15px;
transition: all 0.2s ease;
}
.theme-light .right {
background: white;
color: black;
}
</style>

21
src/pages/Test_page.vue Normal file
View File

@ -0,0 +1,21 @@
<script setup>
import {ref} from "vue";
const blogDisplay = ref('<h2>654654654</h2><ul><li><strong>砍砍价考核表计划表</strong></li></ul><ol><li><strong>1第三方代发</strong></li><li><strong>水电费水电费sd f收到f</strong></li><li><strong>收到f sd </strong></li><li><strong>11234</strong></li></ol>')
</script>
<template>
<div class="container">
<div v-html="blogDisplay"></div>
</div>
</template>
<style scoped>
.container {
display: flex;
flex-direction: column;
align-items: flex-start;
width: 100%;
}
</style>

View File

@ -1,5 +1,5 @@
<script setup>
import { ref } from 'vue';
import {onMounted, ref} from 'vue';
import ToolsBox from "../components/Tools_box.vue";
const categories = ref([
@ -12,8 +12,22 @@ const searchQuery = ref('');
const tools = ref([
{ id: 1, title: 'GPA在线计算器', description: '手动输入在线算', image: null, category: ['计算'] },
{ id: 2, title: 'PDF页面提取器', description: '提取指定页和封面', image: null, category: ['提取', 'pdf'] },
]);
onMounted(() => {
const tags = {};
tools.value.forEach(tool => {
tool.category.forEach(tag => {
tags[tag] = tags[tag]+1 || 1;
})
})
function topNKeys(obj, n) {
let sortedEntries = Object.entries(obj).sort((a, b) => b[1] - a[1]);
return sortedEntries.slice(0, n).map(entry => entry[0]);
}
categories.value = topNKeys(tags, 10).map(tag => { return { name: tag, active: false }})
})
// Toggle function for categories, only one category can be active at a time
const toggleCategory = (category) => {

View File

@ -16,6 +16,7 @@ const pageLoading = ref(false);
const sendCD = ref(0);
const userInput = ref('');
let timer = null;
async function refreshBoard(page, pageSize) {
if (!page) {
@ -56,6 +57,7 @@ async function refreshBoard(page, pageSize) {
async function goPage(page) {
const messageTemp = messages.value
const pageTemp = currentPage.value;
pageLoading.value = true;
currentPage.value = page;
if (await refreshBoard(page) === 0) {
@ -65,6 +67,7 @@ async function goPage(page) {
}
swal.tip('error', '加载留言板失败...')
messages.value = messageTemp;
currentPage.value = pageTemp;
pageLoading.value = false;
}
@ -128,16 +131,16 @@ watch(sendCD, (newValue) => {
});
onMounted(async () => {
sendCD.value = store.state.demosLocal.board?.sendCD || 0;
await refreshBoard();
timer = setInterval(refreshBoard, 7000)
timer = setInterval(refreshBoard, 7000);
});
let timer
onBeforeUnmount(() => {
clearInterval(timer);
timer = undefined;
})
if (timer) {
clearInterval(timer);
timer = null; //
}
});
</script>

View File

@ -0,0 +1,23 @@
<script setup>
</script>
<template>
<div class="container">
<iframe src="/static/isolatedPages/pdfExtractor/index.html" width="100%" height="100%"></iframe>
</div>
</template>
<style scoped>
.container {
width: 100%;
height: calc(100vh - 40px);
padding: 0;
}
iframe {
border: none;
width: 100%;
height: 100%;
margin-bottom: 20px;
}
</style>

View File

@ -19,10 +19,13 @@ import Pod_page from "../pages/demoPages/podExercise/Pod_page.vue";
import Pod_quiz from "../pages/demoPages/podExercise/Quiz.vue";
import Tools_home from "../pages/Tools_home.vue";
import GpaCalculator_page from "../pages/toolPages/gpaCalculator/gpaCalculator_page.vue";
import PdfEx_page from "../pages/toolPages/pdfExtractor/pdfEx_page.vue";
import About from "../pages/About.vue";
import Editor from "../pages/Editor.vue";
import NotFound from "../pages/errorPages/notFound.vue";
import Test_page from "../pages/Test_page.vue";
const routes = [
{path: '/404',
name: '404',
@ -76,6 +79,8 @@ const routes = [
children: [
{path: "1", component: GpaCalculator_page},
{path: "gpa", component: GpaCalculator_page},
{path: "2", component: PdfEx_page},
{path: "pdf-extractor", component: PdfEx_page},
]
}, {
path: '/about',
@ -100,6 +105,11 @@ const routes = [
name: 'Editor',
component: Editor,
meta: {title: '编辑器'},
},{
path: '/test_page',
name: 'Test',
component: Test_page,
meta: {title: '测试页'},
},
];

View File

@ -126,7 +126,7 @@ code {
/* 选中的滚动条滑块效果 */
::-webkit-scrollbar-thumb:hover {
background-color: #888;
background-color: #565656;
}
.theme-light ::-webkit-scrollbar-thumb:hover {