Upload 339 files
Browse filesThis view is limited to 50 files because it contains too many changes.
See raw diff
- .gitattributes +22 -0
- frontend/src/App.vue +137 -0
- frontend/src/assets/fonts/AlibabaPuHuiTi.woff2 +3 -0
- frontend/src/assets/fonts/CangerXiaowanzi.woff2 +3 -0
- frontend/src/assets/fonts/DeYiHei.woff2 +3 -0
- frontend/src/assets/fonts/FangZhengFangSong.woff2 +3 -0
- frontend/src/assets/fonts/FangZhengHeiTi.woff2 +3 -0
- frontend/src/assets/fonts/FangZhengKaiTi.woff2 +3 -0
- frontend/src/assets/fonts/FangZhengShuSong.woff2 +3 -0
- frontend/src/assets/fonts/FengguangMingrui.woff2 +3 -0
- frontend/src/assets/fonts/LXGWWenKai.woff2 +3 -0
- frontend/src/assets/fonts/MiSans.woff2 +3 -0
- frontend/src/assets/fonts/RuiziZhenyan.woff2 +3 -0
- frontend/src/assets/fonts/ShetuModernSquare.woff2 +3 -0
- frontend/src/assets/fonts/SourceHanSans.woff2 +3 -0
- frontend/src/assets/fonts/SourceHanSerif.woff2 +3 -0
- frontend/src/assets/fonts/SucaiJishiCoolSquare.woff2 +3 -0
- frontend/src/assets/fonts/SucaiJishiKangkang.woff2 +3 -0
- frontend/src/assets/fonts/TuniuRounded.woff2 +3 -0
- frontend/src/assets/fonts/WenDingPLKaiTi.woff2 +3 -0
- frontend/src/assets/fonts/YousheTitleBlack.woff2 +3 -0
- frontend/src/assets/fonts/ZcoolHappy.woff2 +3 -0
- frontend/src/assets/fonts/ZhuQueFangSong.woff2 +3 -0
- frontend/src/assets/fonts/ZizhiQuXiMai.woff2 +3 -0
- frontend/src/assets/styles/font.scss +9 -0
- frontend/src/assets/styles/global.scss +138 -0
- frontend/src/assets/styles/mixin.scss +42 -0
- frontend/src/assets/styles/prosemirror.scss +102 -0
- frontend/src/assets/styles/variable.scss +13 -0
- frontend/src/components.d.ts +7 -0
- frontend/src/components/Button.vue +116 -0
- frontend/src/components/ButtonGroup.vue +86 -0
- frontend/src/components/Checkbox.vue +109 -0
- frontend/src/components/CheckboxButton.vue +21 -0
- frontend/src/components/ColorButton.vue +42 -0
- frontend/src/components/ColorListButton.vue +58 -0
- frontend/src/components/ColorPicker/Alpha.vue +107 -0
- frontend/src/components/ColorPicker/Checkboard.vue +60 -0
- frontend/src/components/ColorPicker/EditableInput.vue +69 -0
- frontend/src/components/ColorPicker/Hue.vue +117 -0
- frontend/src/components/ColorPicker/Saturation.vue +108 -0
- frontend/src/components/ColorPicker/index.vue +443 -0
- frontend/src/components/Contextmenu/MenuContent.vue +137 -0
- frontend/src/components/Contextmenu/index.vue +80 -0
- frontend/src/components/Contextmenu/types.ts +14 -0
- frontend/src/components/Divider.vue +34 -0
- frontend/src/components/Drawer.vue +126 -0
- frontend/src/components/FileInput.vue +45 -0
- frontend/src/components/FullscreenSpin.vue +71 -0
- frontend/src/components/GradientBar.vue +149 -0
.gitattributes
CHANGED
|
@@ -33,3 +33,25 @@ saved_model/**/* filter=lfs diff=lfs merge=lfs -text
|
|
| 33 |
*.zip filter=lfs diff=lfs merge=lfs -text
|
| 34 |
*.zst filter=lfs diff=lfs merge=lfs -text
|
| 35 |
*tfevents* filter=lfs diff=lfs merge=lfs -text
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
| 33 |
*.zip filter=lfs diff=lfs merge=lfs -text
|
| 34 |
*.zst filter=lfs diff=lfs merge=lfs -text
|
| 35 |
*tfevents* filter=lfs diff=lfs merge=lfs -text
|
| 36 |
+
frontend/src/assets/fonts/AlibabaPuHuiTi.woff2 filter=lfs diff=lfs merge=lfs -text
|
| 37 |
+
frontend/src/assets/fonts/CangerXiaowanzi.woff2 filter=lfs diff=lfs merge=lfs -text
|
| 38 |
+
frontend/src/assets/fonts/DeYiHei.woff2 filter=lfs diff=lfs merge=lfs -text
|
| 39 |
+
frontend/src/assets/fonts/FangZhengFangSong.woff2 filter=lfs diff=lfs merge=lfs -text
|
| 40 |
+
frontend/src/assets/fonts/FangZhengHeiTi.woff2 filter=lfs diff=lfs merge=lfs -text
|
| 41 |
+
frontend/src/assets/fonts/FangZhengKaiTi.woff2 filter=lfs diff=lfs merge=lfs -text
|
| 42 |
+
frontend/src/assets/fonts/FangZhengShuSong.woff2 filter=lfs diff=lfs merge=lfs -text
|
| 43 |
+
frontend/src/assets/fonts/FengguangMingrui.woff2 filter=lfs diff=lfs merge=lfs -text
|
| 44 |
+
frontend/src/assets/fonts/LXGWWenKai.woff2 filter=lfs diff=lfs merge=lfs -text
|
| 45 |
+
frontend/src/assets/fonts/MiSans.woff2 filter=lfs diff=lfs merge=lfs -text
|
| 46 |
+
frontend/src/assets/fonts/RuiziZhenyan.woff2 filter=lfs diff=lfs merge=lfs -text
|
| 47 |
+
frontend/src/assets/fonts/ShetuModernSquare.woff2 filter=lfs diff=lfs merge=lfs -text
|
| 48 |
+
frontend/src/assets/fonts/SourceHanSans.woff2 filter=lfs diff=lfs merge=lfs -text
|
| 49 |
+
frontend/src/assets/fonts/SourceHanSerif.woff2 filter=lfs diff=lfs merge=lfs -text
|
| 50 |
+
frontend/src/assets/fonts/SucaiJishiCoolSquare.woff2 filter=lfs diff=lfs merge=lfs -text
|
| 51 |
+
frontend/src/assets/fonts/SucaiJishiKangkang.woff2 filter=lfs diff=lfs merge=lfs -text
|
| 52 |
+
frontend/src/assets/fonts/TuniuRounded.woff2 filter=lfs diff=lfs merge=lfs -text
|
| 53 |
+
frontend/src/assets/fonts/WenDingPLKaiTi.woff2 filter=lfs diff=lfs merge=lfs -text
|
| 54 |
+
frontend/src/assets/fonts/YousheTitleBlack.woff2 filter=lfs diff=lfs merge=lfs -text
|
| 55 |
+
frontend/src/assets/fonts/ZcoolHappy.woff2 filter=lfs diff=lfs merge=lfs -text
|
| 56 |
+
frontend/src/assets/fonts/ZhuQueFangSong.woff2 filter=lfs diff=lfs merge=lfs -text
|
| 57 |
+
frontend/src/assets/fonts/ZizhiQuXiMai.woff2 filter=lfs diff=lfs merge=lfs -text
|
frontend/src/App.vue
ADDED
|
@@ -0,0 +1,137 @@
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
| 1 |
+
<template>
|
| 2 |
+
<div id="app">
|
| 3 |
+
<!-- 未登录状态显示登录页面 -->
|
| 4 |
+
<Login v-if="!authStore.isLoggedIn" />
|
| 5 |
+
|
| 6 |
+
<!-- 已登录但数据加载中 -->
|
| 7 |
+
<FullscreenSpin v-else-if="!dataLoaded" tip="数据加载中,请稍等..." loading :mask="false" />
|
| 8 |
+
|
| 9 |
+
<!-- 已登录且数据已加载 -->
|
| 10 |
+
<template v-else>
|
| 11 |
+
<Screen v-if="screening" />
|
| 12 |
+
<Editor v-else-if="_isPC" />
|
| 13 |
+
<Mobile v-else />
|
| 14 |
+
</template>
|
| 15 |
+
</div>
|
| 16 |
+
</template>
|
| 17 |
+
|
| 18 |
+
<script lang="ts" setup>
|
| 19 |
+
import { onMounted, ref } from 'vue'
|
| 20 |
+
import { storeToRefs } from 'pinia'
|
| 21 |
+
import { useScreenStore, useMainStore, useSnapshotStore, useSlidesStore, useAuthStore } from '@/store'
|
| 22 |
+
import { LOCALSTORAGE_KEY_DISCARDED_DB } from '@/configs/storage'
|
| 23 |
+
import { deleteDiscardedDB } from '@/utils/database'
|
| 24 |
+
import { isPC } from '@/utils/common'
|
| 25 |
+
import api from '@/services'
|
| 26 |
+
import dataSyncService from '@/services/dataSyncService'
|
| 27 |
+
|
| 28 |
+
import Editor from './views/Editor/index.vue'
|
| 29 |
+
import Screen from './views/Screen/index.vue'
|
| 30 |
+
import Mobile from './views/Mobile/index.vue'
|
| 31 |
+
import Login from './views/Login.vue'
|
| 32 |
+
import FullscreenSpin from '@/components/FullscreenSpin.vue'
|
| 33 |
+
|
| 34 |
+
const _isPC = isPC()
|
| 35 |
+
const dataLoaded = ref(false)
|
| 36 |
+
|
| 37 |
+
const mainStore = useMainStore()
|
| 38 |
+
const slidesStore = useSlidesStore()
|
| 39 |
+
const snapshotStore = useSnapshotStore()
|
| 40 |
+
const authStore = useAuthStore()
|
| 41 |
+
const { databaseId } = storeToRefs(mainStore)
|
| 42 |
+
const { screening } = storeToRefs(useScreenStore())
|
| 43 |
+
|
| 44 |
+
if (import.meta.env.MODE !== 'development') {
|
| 45 |
+
window.onbeforeunload = () => false
|
| 46 |
+
}
|
| 47 |
+
|
| 48 |
+
// 初始化应用数据
|
| 49 |
+
const initializeApp = async () => {
|
| 50 |
+
try {
|
| 51 |
+
// 初始化 DataSyncService(在 Pinia 可用后)
|
| 52 |
+
await dataSyncService.initialize()
|
| 53 |
+
|
| 54 |
+
// 如果用户已登录,加载用户的PPT数据
|
| 55 |
+
if (authStore.isLoggedIn) {
|
| 56 |
+
const pptList = await api.getPPTList()
|
| 57 |
+
|
| 58 |
+
// 如果用户有PPT,加载第一个PPT;否则创建默认PPT
|
| 59 |
+
if (pptList.length > 0) {
|
| 60 |
+
const firstPPT = await api.getPPT(pptList[0].name)
|
| 61 |
+
slidesStore.setSlides(firstPPT.slides)
|
| 62 |
+
slidesStore.setTitle(firstPPT.title)
|
| 63 |
+
if (firstPPT.theme) {
|
| 64 |
+
slidesStore.setTheme(firstPPT.theme)
|
| 65 |
+
}
|
| 66 |
+
// 设置当前PPT ID以便自动保存
|
| 67 |
+
dataSyncService.setCurrentPPTId(pptList[0].name)
|
| 68 |
+
}
|
| 69 |
+
else {
|
| 70 |
+
// 创建默认演示文稿
|
| 71 |
+
const defaultPPT = await api.createPPT('我的第一个演示文稿')
|
| 72 |
+
slidesStore.setSlides(defaultPPT.ppt.slides)
|
| 73 |
+
slidesStore.setTitle(defaultPPT.ppt.title)
|
| 74 |
+
slidesStore.setTheme(defaultPPT.ppt.theme)
|
| 75 |
+
dataSyncService.setCurrentPPTId(defaultPPT.pptId)
|
| 76 |
+
}
|
| 77 |
+
}
|
| 78 |
+
else {
|
| 79 |
+
// 未登录状态,加载默认示例数据
|
| 80 |
+
const slides = await api.getFileData('slides')
|
| 81 |
+
slidesStore.setSlides(slides)
|
| 82 |
+
}
|
| 83 |
+
|
| 84 |
+
await deleteDiscardedDB()
|
| 85 |
+
snapshotStore.initSnapshotDatabase()
|
| 86 |
+
dataLoaded.value = true
|
| 87 |
+
}
|
| 88 |
+
catch (error) {
|
| 89 |
+
// 如果是认证错误,清除登录状态
|
| 90 |
+
if (error === 'Authentication failed') {
|
| 91 |
+
authStore.logout()
|
| 92 |
+
}
|
| 93 |
+
else {
|
| 94 |
+
// 其他错误,加载默认数据
|
| 95 |
+
try {
|
| 96 |
+
const slides = await api.getFileData('slides')
|
| 97 |
+
slidesStore.setSlides(slides)
|
| 98 |
+
dataLoaded.value = true
|
| 99 |
+
}
|
| 100 |
+
catch (fallbackError) {
|
| 101 |
+
// 创建一个空的默认幻灯片
|
| 102 |
+
slidesStore.setSlides([{
|
| 103 |
+
id: 'default-slide',
|
| 104 |
+
elements: [],
|
| 105 |
+
background: { type: 'solid', color: '#ffffff' }
|
| 106 |
+
}])
|
| 107 |
+
dataLoaded.value = true
|
| 108 |
+
}
|
| 109 |
+
}
|
| 110 |
+
}
|
| 111 |
+
}
|
| 112 |
+
|
| 113 |
+
onMounted(async () => {
|
| 114 |
+
// 初始化认证状态
|
| 115 |
+
await authStore.initAuth()
|
| 116 |
+
|
| 117 |
+
// 初始化应用数据
|
| 118 |
+
await initializeApp()
|
| 119 |
+
})
|
| 120 |
+
|
| 121 |
+
// 应用注销时向 localStorage 中记录下本次 indexedDB 的数据库ID,用于之后清除数据库
|
| 122 |
+
window.addEventListener('unload', () => {
|
| 123 |
+
const discardedDB = localStorage.getItem(LOCALSTORAGE_KEY_DISCARDED_DB)
|
| 124 |
+
const discardedDBList: string[] = discardedDB ? JSON.parse(discardedDB) : []
|
| 125 |
+
|
| 126 |
+
discardedDBList.push(databaseId.value)
|
| 127 |
+
|
| 128 |
+
const newDiscardedDB = JSON.stringify(discardedDBList)
|
| 129 |
+
localStorage.setItem(LOCALSTORAGE_KEY_DISCARDED_DB, newDiscardedDB)
|
| 130 |
+
})
|
| 131 |
+
</script>
|
| 132 |
+
|
| 133 |
+
<style lang="scss">
|
| 134 |
+
#app {
|
| 135 |
+
height: 100%;
|
| 136 |
+
}
|
| 137 |
+
</style>
|
frontend/src/assets/fonts/AlibabaPuHuiTi.woff2
ADDED
|
@@ -0,0 +1,3 @@
|
|
|
|
|
|
|
|
|
|
|
|
|
| 1 |
+
version https://git-lfs.github.com/spec/v1
|
| 2 |
+
oid sha256:e03857d7e181a9201baee2edef8dc6dba054dcb42a4b763cf75bb3dbdee2b321
|
| 3 |
+
size 4150196
|
frontend/src/assets/fonts/CangerXiaowanzi.woff2
ADDED
|
@@ -0,0 +1,3 @@
|
|
|
|
|
|
|
|
|
|
|
|
|
| 1 |
+
version https://git-lfs.github.com/spec/v1
|
| 2 |
+
oid sha256:18e5bccdcfa630d8a953ae610cc066f0a2941690fc84aced856062cd2a27d88d
|
| 3 |
+
size 695832
|
frontend/src/assets/fonts/DeYiHei.woff2
ADDED
|
@@ -0,0 +1,3 @@
|
|
|
|
|
|
|
|
|
|
|
|
|
| 1 |
+
version https://git-lfs.github.com/spec/v1
|
| 2 |
+
oid sha256:ecd63faa61348c4f2480b5fddd8c48fd76f12542328eaa530b2a69badff68ab9
|
| 3 |
+
size 1361616
|
frontend/src/assets/fonts/FangZhengFangSong.woff2
ADDED
|
@@ -0,0 +1,3 @@
|
|
|
|
|
|
|
|
|
|
|
|
|
| 1 |
+
version https://git-lfs.github.com/spec/v1
|
| 2 |
+
oid sha256:d50e0c75d1688cb49aa3e50dc95ba9eb7a695f6593d805149f4abb78a3dd133b
|
| 3 |
+
size 1538112
|
frontend/src/assets/fonts/FangZhengHeiTi.woff2
ADDED
|
@@ -0,0 +1,3 @@
|
|
|
|
|
|
|
|
|
|
|
|
|
| 1 |
+
version https://git-lfs.github.com/spec/v1
|
| 2 |
+
oid sha256:481eee4596a4125253ebf81edf01102298ea85cdb12b850d27863503ccb32518
|
| 3 |
+
size 1189808
|
frontend/src/assets/fonts/FangZhengKaiTi.woff2
ADDED
|
@@ -0,0 +1,3 @@
|
|
|
|
|
|
|
|
|
|
|
|
|
| 1 |
+
version https://git-lfs.github.com/spec/v1
|
| 2 |
+
oid sha256:a43bcc7319f7c570920c1a7302e2d73cb65378897bdb73a8445cf84559655015
|
| 3 |
+
size 1656040
|
frontend/src/assets/fonts/FangZhengShuSong.woff2
ADDED
|
@@ -0,0 +1,3 @@
|
|
|
|
|
|
|
|
|
|
|
|
|
| 1 |
+
version https://git-lfs.github.com/spec/v1
|
| 2 |
+
oid sha256:b143be48bb5699ae271a1f81c41872499634e896b9c66c1fe3b28bcc4708a4e0
|
| 3 |
+
size 1185016
|
frontend/src/assets/fonts/FengguangMingrui.woff2
ADDED
|
@@ -0,0 +1,3 @@
|
|
|
|
|
|
|
|
|
|
|
|
|
| 1 |
+
version https://git-lfs.github.com/spec/v1
|
| 2 |
+
oid sha256:227d498f6b1ae157e079dccb6cd563f9b89b789b99be9dd660888f928c398660
|
| 3 |
+
size 321284
|
frontend/src/assets/fonts/LXGWWenKai.woff2
ADDED
|
@@ -0,0 +1,3 @@
|
|
|
|
|
|
|
|
|
|
|
|
|
| 1 |
+
version https://git-lfs.github.com/spec/v1
|
| 2 |
+
oid sha256:5cf730147cd4923546015110f9085216b023df388e2c5ca270b5db17d8de496b
|
| 3 |
+
size 1826736
|
frontend/src/assets/fonts/MiSans.woff2
ADDED
|
@@ -0,0 +1,3 @@
|
|
|
|
|
|
|
|
|
|
|
|
|
| 1 |
+
version https://git-lfs.github.com/spec/v1
|
| 2 |
+
oid sha256:7d7a4ba4faf18306e446787c1ab1bd1e90c9f27bfa937cd8eb3469c7504e563f
|
| 3 |
+
size 4847960
|
frontend/src/assets/fonts/RuiziZhenyan.woff2
ADDED
|
@@ -0,0 +1,3 @@
|
|
|
|
|
|
|
|
|
|
|
|
|
| 1 |
+
version https://git-lfs.github.com/spec/v1
|
| 2 |
+
oid sha256:fac4031d38f8497766ac3019b094c2fc91326c4257307b33a6c99b72cb5a529d
|
| 3 |
+
size 616772
|
frontend/src/assets/fonts/ShetuModernSquare.woff2
ADDED
|
@@ -0,0 +1,3 @@
|
|
|
|
|
|
|
|
|
|
|
|
|
| 1 |
+
version https://git-lfs.github.com/spec/v1
|
| 2 |
+
oid sha256:d6a319f07a79112133b61163eeaa87e632113fccb18a3655a9fbe07a912f1097
|
| 3 |
+
size 790024
|
frontend/src/assets/fonts/SourceHanSans.woff2
ADDED
|
@@ -0,0 +1,3 @@
|
|
|
|
|
|
|
|
|
|
|
|
|
| 1 |
+
version https://git-lfs.github.com/spec/v1
|
| 2 |
+
oid sha256:baef3d9c86508d957eee369b1289a1a918a6f1f394f10bd1129332c84f91c671
|
| 3 |
+
size 7511656
|
frontend/src/assets/fonts/SourceHanSerif.woff2
ADDED
|
@@ -0,0 +1,3 @@
|
|
|
|
|
|
|
|
|
|
|
|
|
| 1 |
+
version https://git-lfs.github.com/spec/v1
|
| 2 |
+
oid sha256:4a39a09928cb92aca1e1bf72d841718ecb9c9549c1d1fa95620b8b42f49434db
|
| 3 |
+
size 10413080
|
frontend/src/assets/fonts/SucaiJishiCoolSquare.woff2
ADDED
|
@@ -0,0 +1,3 @@
|
|
|
|
|
|
|
|
|
|
|
|
|
| 1 |
+
version https://git-lfs.github.com/spec/v1
|
| 2 |
+
oid sha256:fade334be8d8e9bb68d43d43f7ab811b51433bc92d5201b30be191e026e0c913
|
| 3 |
+
size 218068
|
frontend/src/assets/fonts/SucaiJishiKangkang.woff2
ADDED
|
@@ -0,0 +1,3 @@
|
|
|
|
|
|
|
|
|
|
|
|
|
| 1 |
+
version https://git-lfs.github.com/spec/v1
|
| 2 |
+
oid sha256:8b874c03c6a6006b7d4f2dbaf01f3c46b51554def44a0ed1f01d126c7ff58ff2
|
| 3 |
+
size 500448
|
frontend/src/assets/fonts/TuniuRounded.woff2
ADDED
|
@@ -0,0 +1,3 @@
|
|
|
|
|
|
|
|
|
|
|
|
|
| 1 |
+
version https://git-lfs.github.com/spec/v1
|
| 2 |
+
oid sha256:f07953ca9d512298bc9a770d0236e992211ab3cebd6f2afd90e6b0da9da76da4
|
| 3 |
+
size 108684
|
frontend/src/assets/fonts/WenDingPLKaiTi.woff2
ADDED
|
@@ -0,0 +1,3 @@
|
|
|
|
|
|
|
|
|
|
|
|
|
| 1 |
+
version https://git-lfs.github.com/spec/v1
|
| 2 |
+
oid sha256:5c91fd0d539e35631dd5086451bc37732fa19e9eb9f7202eb881d6616f6d3634
|
| 3 |
+
size 1962288
|
frontend/src/assets/fonts/YousheTitleBlack.woff2
ADDED
|
@@ -0,0 +1,3 @@
|
|
|
|
|
|
|
|
|
|
|
|
|
| 1 |
+
version https://git-lfs.github.com/spec/v1
|
| 2 |
+
oid sha256:58d9eeac1664dbadefc309a26e6496f3d03915ec32d78edc66bce4760db74d77
|
| 3 |
+
size 642944
|
frontend/src/assets/fonts/ZcoolHappy.woff2
ADDED
|
@@ -0,0 +1,3 @@
|
|
|
|
|
|
|
|
|
|
|
|
|
| 1 |
+
version https://git-lfs.github.com/spec/v1
|
| 2 |
+
oid sha256:ee4391316d9e560dbf96d92c7a9a12a73b0ac39b4f77662b43c2bd4fd16dcde7
|
| 3 |
+
size 937400
|
frontend/src/assets/fonts/ZhuQueFangSong.woff2
ADDED
|
@@ -0,0 +1,3 @@
|
|
|
|
|
|
|
|
|
|
|
|
|
| 1 |
+
version https://git-lfs.github.com/spec/v1
|
| 2 |
+
oid sha256:81dd160d3c4608a70cc9eb794e4aaf8b1a6b07dec1a5284c48e402f0ffba1459
|
| 3 |
+
size 2598904
|
frontend/src/assets/fonts/ZizhiQuXiMai.woff2
ADDED
|
@@ -0,0 +1,3 @@
|
|
|
|
|
|
|
|
|
|
|
|
|
| 1 |
+
version https://git-lfs.github.com/spec/v1
|
| 2 |
+
oid sha256:2cce3fff22f295562a2970044c32ba330f32529503ca77c3d8c26649934f65f3
|
| 3 |
+
size 678188
|
frontend/src/assets/styles/font.scss
ADDED
|
@@ -0,0 +1,9 @@
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
| 1 |
+
$fonts: 'SourceHanSans', 'SourceHanSerif', 'FangZhengHeiTi', 'FangZhengKaiTi', 'FangZhengShuSong', 'FangZhengFangSong', 'AlibabaPuHuiTi', 'ZhuQueFangSong', 'LXGWWenKai', 'WenDingPLKaiTi', 'DeYiHei', 'MiSans', 'CangerXiaowanzi', 'YousheTitleBlack', 'FengguangMingrui', 'ShetuModernSquare', 'ZcoolHappy', 'ZizhiQuXiMai', 'SucaiJishiKangkang', 'SucaiJishiCoolSquare', 'TuniuRounded', 'RuiziZhenyan';
|
| 2 |
+
|
| 3 |
+
@each $font in $fonts {
|
| 4 |
+
@font-face {
|
| 5 |
+
font-display: swap;
|
| 6 |
+
font-family: $font;
|
| 7 |
+
src: url('https://asset.pptist.cn/font/#{$font}.woff2') format('woff2');
|
| 8 |
+
}
|
| 9 |
+
}
|
frontend/src/assets/styles/global.scss
ADDED
|
@@ -0,0 +1,138 @@
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
| 1 |
+
html, body, div, span, applet, object, iframe,
|
| 2 |
+
h1, h2, h3, h4, h5, h6, p, blockquote, pre,
|
| 3 |
+
a, abbr, acronym, address, big, cite, code,
|
| 4 |
+
del, dfn, em, img, ins, kbd, q, s, samp,
|
| 5 |
+
small, strike, strong, sub, sup, tt, var,
|
| 6 |
+
b, u, i, center,
|
| 7 |
+
dl, dt, dd, ol, ul, li,
|
| 8 |
+
fieldset, form, label, legend,
|
| 9 |
+
table, caption, tbody, tfoot, thead, tr, th, td,
|
| 10 |
+
article, aside, canvas, details, embed,
|
| 11 |
+
figure, figcaption, footer, header, hgroup,
|
| 12 |
+
menu, nav, output, ruby, section, summary,
|
| 13 |
+
time, mark, audio, video {
|
| 14 |
+
margin: 0;
|
| 15 |
+
padding: 0;
|
| 16 |
+
border: 0;
|
| 17 |
+
font-size: 100%;
|
| 18 |
+
vertical-align: baseline;
|
| 19 |
+
box-sizing: border-box;
|
| 20 |
+
}
|
| 21 |
+
|
| 22 |
+
*::before,
|
| 23 |
+
*::after {
|
| 24 |
+
box-sizing: border-box;
|
| 25 |
+
}
|
| 26 |
+
|
| 27 |
+
article,
|
| 28 |
+
aside,
|
| 29 |
+
details,
|
| 30 |
+
figcaption,
|
| 31 |
+
figure,
|
| 32 |
+
footer,
|
| 33 |
+
header,
|
| 34 |
+
hgroup,
|
| 35 |
+
menu,
|
| 36 |
+
nav,
|
| 37 |
+
section {
|
| 38 |
+
display: block;
|
| 39 |
+
}
|
| 40 |
+
|
| 41 |
+
html,
|
| 42 |
+
body {
|
| 43 |
+
width: 100%;
|
| 44 |
+
height: 100%;
|
| 45 |
+
overflow: hidden;
|
| 46 |
+
background-color: #fff;
|
| 47 |
+
color: $textColor;
|
| 48 |
+
}
|
| 49 |
+
|
| 50 |
+
body {
|
| 51 |
+
font-family: -apple-system, BlinkMacSystemFont, 'PingFang SC', 'Hiragino Sans GB', 'Microsoft YaHei', 'Helvetica Neue', Helvetica, Arial, sans-serif, 'Apple Color Emoji', 'Segoe UI Emoji', 'Segoe UI Symbol';
|
| 52 |
+
}
|
| 53 |
+
|
| 54 |
+
ol,
|
| 55 |
+
ul {
|
| 56 |
+
list-style: none;
|
| 57 |
+
}
|
| 58 |
+
|
| 59 |
+
blockquote, q {
|
| 60 |
+
quotes: none;
|
| 61 |
+
}
|
| 62 |
+
|
| 63 |
+
blockquote::before,
|
| 64 |
+
blockquote::after,
|
| 65 |
+
q::before,
|
| 66 |
+
q::after {
|
| 67 |
+
content: '';
|
| 68 |
+
}
|
| 69 |
+
|
| 70 |
+
table {
|
| 71 |
+
border-collapse: collapse;
|
| 72 |
+
border-spacing: 0;
|
| 73 |
+
}
|
| 74 |
+
|
| 75 |
+
a {
|
| 76 |
+
text-decoration: none;
|
| 77 |
+
color: $themeColor;
|
| 78 |
+
}
|
| 79 |
+
|
| 80 |
+
img {
|
| 81 |
+
vertical-align: middle;
|
| 82 |
+
border-style: none;
|
| 83 |
+
}
|
| 84 |
+
|
| 85 |
+
hr {
|
| 86 |
+
box-sizing: content-box;
|
| 87 |
+
height: 0;
|
| 88 |
+
overflow: visible;
|
| 89 |
+
}
|
| 90 |
+
|
| 91 |
+
mark.active {
|
| 92 |
+
background-color: #ff9632;
|
| 93 |
+
}
|
| 94 |
+
|
| 95 |
+
input,
|
| 96 |
+
button,
|
| 97 |
+
select,
|
| 98 |
+
optgroup,
|
| 99 |
+
textarea {
|
| 100 |
+
color: inherit;
|
| 101 |
+
}
|
| 102 |
+
|
| 103 |
+
button,
|
| 104 |
+
input {
|
| 105 |
+
overflow: visible;
|
| 106 |
+
}
|
| 107 |
+
|
| 108 |
+
button,
|
| 109 |
+
select {
|
| 110 |
+
text-transform: none;
|
| 111 |
+
}
|
| 112 |
+
|
| 113 |
+
textarea {
|
| 114 |
+
overflow: auto;
|
| 115 |
+
resize: vertical;
|
| 116 |
+
}
|
| 117 |
+
|
| 118 |
+
a,
|
| 119 |
+
area,
|
| 120 |
+
button,
|
| 121 |
+
[role='button'],
|
| 122 |
+
input:not([type='range']),
|
| 123 |
+
label,
|
| 124 |
+
select,
|
| 125 |
+
summary,
|
| 126 |
+
textarea {
|
| 127 |
+
touch-action: manipulation;
|
| 128 |
+
}
|
| 129 |
+
|
| 130 |
+
::-webkit-scrollbar {
|
| 131 |
+
width: 5px;
|
| 132 |
+
height: 5px;
|
| 133 |
+
background-color: transparent;
|
| 134 |
+
}
|
| 135 |
+
::-webkit-scrollbar-thumb {
|
| 136 |
+
background-color: #e1e1e1;
|
| 137 |
+
border-radius: 3px;
|
| 138 |
+
}
|
frontend/src/assets/styles/mixin.scss
ADDED
|
@@ -0,0 +1,42 @@
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
| 1 |
+
@mixin ellipsis-oneline() {
|
| 2 |
+
overflow: hidden;
|
| 3 |
+
white-space: nowrap;
|
| 4 |
+
text-overflow: ellipsis;
|
| 5 |
+
}
|
| 6 |
+
|
| 7 |
+
@mixin ellipsis-multiline($line: 2) {
|
| 8 |
+
word-wrap: break-word;
|
| 9 |
+
overflow: hidden;
|
| 10 |
+
text-overflow: ellipsis;
|
| 11 |
+
display: -webkit-box;
|
| 12 |
+
-webkit-line-clamp: $line;
|
| 13 |
+
-webkit-box-orient: vertical;
|
| 14 |
+
}
|
| 15 |
+
|
| 16 |
+
@mixin flex-grid-layout() {
|
| 17 |
+
display: flex;
|
| 18 |
+
flex-wrap: wrap;
|
| 19 |
+
align-content: flex-start;
|
| 20 |
+
}
|
| 21 |
+
|
| 22 |
+
@mixin flex-grid-layout-children($col, $colWidth) {
|
| 23 |
+
width: $colWidth;
|
| 24 |
+
margin-bottom: calc(#{100 - $col * $colWidth} / #{$col - 1});
|
| 25 |
+
|
| 26 |
+
&:not(:nth-child(#{$col}n)) {
|
| 27 |
+
margin-right: calc(#{100 - $col * $colWidth} / #{$col - 1});
|
| 28 |
+
}
|
| 29 |
+
}
|
| 30 |
+
|
| 31 |
+
@mixin overflow-overlay() {
|
| 32 |
+
overflow: auto;
|
| 33 |
+
overflow: overlay;
|
| 34 |
+
}
|
| 35 |
+
|
| 36 |
+
@mixin absolute-0() {
|
| 37 |
+
position: absolute;
|
| 38 |
+
top: 0;
|
| 39 |
+
right: 0;
|
| 40 |
+
bottom: 0;
|
| 41 |
+
left: 0;
|
| 42 |
+
}
|
frontend/src/assets/styles/prosemirror.scss
ADDED
|
@@ -0,0 +1,102 @@
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
| 1 |
+
.ProseMirror, .ProseMirror-static {
|
| 2 |
+
outline: 0;
|
| 3 |
+
border: 0;
|
| 4 |
+
font-size: 16px;
|
| 5 |
+
word-break: break-word;
|
| 6 |
+
white-space: normal;
|
| 7 |
+
|
| 8 |
+
&:not(.ProseMirror-static) {
|
| 9 |
+
user-select: text;
|
| 10 |
+
}
|
| 11 |
+
|
| 12 |
+
::selection {
|
| 13 |
+
background-color: rgba($themeColor, 0.25);
|
| 14 |
+
color: inherit;
|
| 15 |
+
}
|
| 16 |
+
|
| 17 |
+
p {
|
| 18 |
+
margin: 0;
|
| 19 |
+
margin-top: var(--paragraphSpace);
|
| 20 |
+
}
|
| 21 |
+
p:first-child {
|
| 22 |
+
margin-top: 0;
|
| 23 |
+
}
|
| 24 |
+
|
| 25 |
+
ul, ol, li {
|
| 26 |
+
margin: 0;
|
| 27 |
+
margin-top: var(--paragraphSpace);
|
| 28 |
+
}
|
| 29 |
+
ul {
|
| 30 |
+
list-style-type: disc;
|
| 31 |
+
padding-inline-start: 1.25em;
|
| 32 |
+
|
| 33 |
+
li {
|
| 34 |
+
list-style-type: inherit;
|
| 35 |
+
padding: 0.125em 0;
|
| 36 |
+
}
|
| 37 |
+
}
|
| 38 |
+
|
| 39 |
+
ol {
|
| 40 |
+
list-style-type: decimal;
|
| 41 |
+
padding-inline-start: 1.25em;
|
| 42 |
+
|
| 43 |
+
li {
|
| 44 |
+
list-style-type: inherit;
|
| 45 |
+
padding: 0.125em 0;
|
| 46 |
+
}
|
| 47 |
+
}
|
| 48 |
+
|
| 49 |
+
code {
|
| 50 |
+
background-color: #f1f1f1;
|
| 51 |
+
padding: 2px 6px;
|
| 52 |
+
margin: 0 1px;
|
| 53 |
+
border-radius: 4px;
|
| 54 |
+
font-family: SFMono-Regular, Consolas, 'Liberation Mono', Menlo, monospace;
|
| 55 |
+
}
|
| 56 |
+
|
| 57 |
+
sup {
|
| 58 |
+
vertical-align: super;
|
| 59 |
+
font-size: smaller;
|
| 60 |
+
}
|
| 61 |
+
sub {
|
| 62 |
+
vertical-align: sub;
|
| 63 |
+
font-size: smaller;
|
| 64 |
+
}
|
| 65 |
+
|
| 66 |
+
blockquote {
|
| 67 |
+
overflow: hidden;
|
| 68 |
+
padding: 0 1.2em;
|
| 69 |
+
margin: 0.6em 0;
|
| 70 |
+
font-style: italic;
|
| 71 |
+
border-left: 4px solid #e0e0e0;
|
| 72 |
+
}
|
| 73 |
+
|
| 74 |
+
[data-indent='1'] {
|
| 75 |
+
padding-left: 1em;
|
| 76 |
+
}
|
| 77 |
+
[data-indent='2'] {
|
| 78 |
+
padding-left: 2em;
|
| 79 |
+
}
|
| 80 |
+
[data-indent='3'] {
|
| 81 |
+
padding-left: 3em;
|
| 82 |
+
}
|
| 83 |
+
[data-indent='4'] {
|
| 84 |
+
padding-left: 4em;
|
| 85 |
+
}
|
| 86 |
+
[data-indent='5'] {
|
| 87 |
+
padding-left: 5em;
|
| 88 |
+
}
|
| 89 |
+
[data-indent='6'] {
|
| 90 |
+
padding-left: 6em;
|
| 91 |
+
}
|
| 92 |
+
[data-indent='7'] {
|
| 93 |
+
padding-left: 7em;
|
| 94 |
+
}
|
| 95 |
+
[data-indent='8'] {
|
| 96 |
+
padding-left: 8em;
|
| 97 |
+
}
|
| 98 |
+
}
|
| 99 |
+
|
| 100 |
+
.ProseMirror-selectednode {
|
| 101 |
+
outline: none !important;
|
| 102 |
+
}
|
frontend/src/assets/styles/variable.scss
ADDED
|
@@ -0,0 +1,13 @@
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
| 1 |
+
$themeColor: #d14424;
|
| 2 |
+
$themeHoverColor: #de6949;
|
| 3 |
+
$textColor: #41464b;
|
| 4 |
+
$borderColor: #e5e7eb;
|
| 5 |
+
$lightGray: #f9f9f9;
|
| 6 |
+
|
| 7 |
+
$boxShadow: 0 4px 6px -1px rgba(0, 0, 0, .1), 0 2px 4px -2px rgba(0, 0, 0, .1);
|
| 8 |
+
|
| 9 |
+
$transitionDelay: .2s;
|
| 10 |
+
$transitionDelayFast: .1s;
|
| 11 |
+
$transitionDelaySlow: .3s;
|
| 12 |
+
|
| 13 |
+
$borderRadius: 2px;
|
frontend/src/components.d.ts
ADDED
|
@@ -0,0 +1,7 @@
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
| 1 |
+
import type { Icons } from '@/plugins/icon'
|
| 2 |
+
|
| 3 |
+
declare module 'vue' {
|
| 4 |
+
export type GlobalComponents = Icons
|
| 5 |
+
}
|
| 6 |
+
|
| 7 |
+
export {}
|
frontend/src/components/Button.vue
ADDED
|
@@ -0,0 +1,116 @@
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
| 1 |
+
<template>
|
| 2 |
+
<button
|
| 3 |
+
class="button"
|
| 4 |
+
:class="{
|
| 5 |
+
'disabled': disabled,
|
| 6 |
+
'checked': !disabled && checked,
|
| 7 |
+
'default': !disabled && type === 'default',
|
| 8 |
+
'primary': !disabled && type === 'primary',
|
| 9 |
+
'checkbox': !disabled && type === 'checkbox',
|
| 10 |
+
'radio': !disabled && type === 'radio',
|
| 11 |
+
'small': size === 'small',
|
| 12 |
+
'first': first,
|
| 13 |
+
'last': last,
|
| 14 |
+
}"
|
| 15 |
+
@click="handleClick()"
|
| 16 |
+
>
|
| 17 |
+
<slot></slot>
|
| 18 |
+
</button>
|
| 19 |
+
</template>
|
| 20 |
+
|
| 21 |
+
<script lang="ts" setup>
|
| 22 |
+
const props = withDefaults(defineProps<{
|
| 23 |
+
checked?: boolean
|
| 24 |
+
disabled?: boolean
|
| 25 |
+
type?: 'default' | 'primary' | 'checkbox' | 'radio'
|
| 26 |
+
size?: 'small' | 'normal'
|
| 27 |
+
first?: boolean
|
| 28 |
+
last?: boolean
|
| 29 |
+
}>(), {
|
| 30 |
+
checked: false,
|
| 31 |
+
disabled: false,
|
| 32 |
+
type: 'default',
|
| 33 |
+
size: 'normal',
|
| 34 |
+
first: false,
|
| 35 |
+
last: false,
|
| 36 |
+
})
|
| 37 |
+
|
| 38 |
+
const emit = defineEmits<{
|
| 39 |
+
(event: 'click'): void
|
| 40 |
+
}>()
|
| 41 |
+
|
| 42 |
+
const handleClick = () => {
|
| 43 |
+
if (props.disabled) return
|
| 44 |
+
emit('click')
|
| 45 |
+
}
|
| 46 |
+
</script>
|
| 47 |
+
|
| 48 |
+
<style lang="scss" scoped>
|
| 49 |
+
.button {
|
| 50 |
+
height: 32px;
|
| 51 |
+
line-height: 32px;
|
| 52 |
+
outline: 0;
|
| 53 |
+
font-size: 13px;
|
| 54 |
+
padding: 0 15px;
|
| 55 |
+
text-align: center;
|
| 56 |
+
color: $textColor;
|
| 57 |
+
border-radius: $borderRadius;
|
| 58 |
+
user-select: none;
|
| 59 |
+
letter-spacing: 1px;
|
| 60 |
+
cursor: pointer;
|
| 61 |
+
|
| 62 |
+
&.small {
|
| 63 |
+
height: 24px;
|
| 64 |
+
line-height: 24px;
|
| 65 |
+
padding: 0 7px;
|
| 66 |
+
letter-spacing: 0;
|
| 67 |
+
font-size: 12px;
|
| 68 |
+
}
|
| 69 |
+
|
| 70 |
+
&.default {
|
| 71 |
+
background-color: #fff;
|
| 72 |
+
border: 1px solid #d9d9d9;
|
| 73 |
+
color: $textColor;
|
| 74 |
+
|
| 75 |
+
&:hover {
|
| 76 |
+
color: $themeColor;
|
| 77 |
+
border-color: $themeColor;
|
| 78 |
+
}
|
| 79 |
+
}
|
| 80 |
+
&.primary {
|
| 81 |
+
background-color: $themeColor;
|
| 82 |
+
border: 1px solid $themeColor;
|
| 83 |
+
color: #fff;
|
| 84 |
+
|
| 85 |
+
&:hover {
|
| 86 |
+
background-color: $themeHoverColor;
|
| 87 |
+
border-color: $themeHoverColor;
|
| 88 |
+
}
|
| 89 |
+
}
|
| 90 |
+
&.checkbox, &.radio {
|
| 91 |
+
background-color: #fff;
|
| 92 |
+
border: 1px solid #d9d9d9;
|
| 93 |
+
color: $textColor;
|
| 94 |
+
|
| 95 |
+
&:not(.checked):hover {
|
| 96 |
+
color: $themeColor;
|
| 97 |
+
}
|
| 98 |
+
}
|
| 99 |
+
&.checked {
|
| 100 |
+
color: #fff;
|
| 101 |
+
background-color: $themeColor;
|
| 102 |
+
border-color: $themeColor;
|
| 103 |
+
|
| 104 |
+
&:hover {
|
| 105 |
+
background-color: $themeHoverColor;
|
| 106 |
+
border-color: $themeHoverColor;
|
| 107 |
+
}
|
| 108 |
+
}
|
| 109 |
+
&.disabled {
|
| 110 |
+
background-color: #f5f5f5;
|
| 111 |
+
border: 1px solid #d9d9d9;
|
| 112 |
+
color: #b7b7b7;
|
| 113 |
+
cursor: default;
|
| 114 |
+
}
|
| 115 |
+
}
|
| 116 |
+
</style>
|
frontend/src/components/ButtonGroup.vue
ADDED
|
@@ -0,0 +1,86 @@
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
| 1 |
+
<template>
|
| 2 |
+
<div class="button-group" :class="{ 'passive': passive }" ref="groupRef">
|
| 3 |
+
<slot></slot>
|
| 4 |
+
</div>
|
| 5 |
+
</template>
|
| 6 |
+
|
| 7 |
+
<script lang="ts" setup>
|
| 8 |
+
withDefaults(defineProps<{
|
| 9 |
+
passive?: boolean
|
| 10 |
+
}>(), {
|
| 11 |
+
passive: false,
|
| 12 |
+
})
|
| 13 |
+
</script>
|
| 14 |
+
|
| 15 |
+
<style lang="scss" scoped>
|
| 16 |
+
.button-group {
|
| 17 |
+
display: flex;
|
| 18 |
+
align-items: center;
|
| 19 |
+
|
| 20 |
+
::v-deep(button.button) {
|
| 21 |
+
border-radius: 0;
|
| 22 |
+
border-left-width: 1px;
|
| 23 |
+
border-right-width: 0;
|
| 24 |
+
display: inline-block;
|
| 25 |
+
}
|
| 26 |
+
|
| 27 |
+
&:not(.passive) {
|
| 28 |
+
::v-deep(button.button) {
|
| 29 |
+
&:not(:last-child, .radio, .checkbox):hover {
|
| 30 |
+
position: relative;
|
| 31 |
+
|
| 32 |
+
&::after {
|
| 33 |
+
content: '';
|
| 34 |
+
width: 1px;
|
| 35 |
+
height: calc(100% + 2px);
|
| 36 |
+
background-color: $themeColor;
|
| 37 |
+
position: absolute;
|
| 38 |
+
top: -1px;
|
| 39 |
+
right: -1px;
|
| 40 |
+
}
|
| 41 |
+
}
|
| 42 |
+
|
| 43 |
+
&:first-child {
|
| 44 |
+
border-top-left-radius: $borderRadius;
|
| 45 |
+
border-bottom-left-radius: $borderRadius;
|
| 46 |
+
border-left-width: 1px;
|
| 47 |
+
}
|
| 48 |
+
|
| 49 |
+
&:last-child {
|
| 50 |
+
border-top-right-radius: $borderRadius;
|
| 51 |
+
border-bottom-right-radius: $borderRadius;
|
| 52 |
+
border-right-width: 1px;
|
| 53 |
+
}
|
| 54 |
+
}
|
| 55 |
+
}
|
| 56 |
+
&.passive {
|
| 57 |
+
::v-deep(button.button) {
|
| 58 |
+
&:not(.last, .radio, .checkbox):hover {
|
| 59 |
+
position: relative;
|
| 60 |
+
|
| 61 |
+
&::after {
|
| 62 |
+
content: '';
|
| 63 |
+
width: 1px;
|
| 64 |
+
height: calc(100% + 2px);
|
| 65 |
+
background-color: $themeColor;
|
| 66 |
+
position: absolute;
|
| 67 |
+
top: -1px;
|
| 68 |
+
right: -1px;
|
| 69 |
+
}
|
| 70 |
+
}
|
| 71 |
+
|
| 72 |
+
&.first {
|
| 73 |
+
border-top-left-radius: $borderRadius;
|
| 74 |
+
border-bottom-left-radius: $borderRadius;
|
| 75 |
+
border-left-width: 1px;
|
| 76 |
+
}
|
| 77 |
+
|
| 78 |
+
&.last {
|
| 79 |
+
border-top-right-radius: $borderRadius;
|
| 80 |
+
border-bottom-right-radius: $borderRadius;
|
| 81 |
+
border-right-width: 1px;
|
| 82 |
+
}
|
| 83 |
+
}
|
| 84 |
+
}
|
| 85 |
+
}
|
| 86 |
+
</style>
|
frontend/src/components/Checkbox.vue
ADDED
|
@@ -0,0 +1,109 @@
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
| 1 |
+
<template>
|
| 2 |
+
<label
|
| 3 |
+
class="checkbox"
|
| 4 |
+
:class="{
|
| 5 |
+
'checked': value,
|
| 6 |
+
'disabled': disabled,
|
| 7 |
+
}"
|
| 8 |
+
@change="$event => handleChange($event)"
|
| 9 |
+
>
|
| 10 |
+
<span class="checkbox-input"></span>
|
| 11 |
+
<input class="checkbox-original" type="checkbox" :checked="value">
|
| 12 |
+
<span class="checkbox-label">
|
| 13 |
+
<slot></slot>
|
| 14 |
+
</span>
|
| 15 |
+
</label>
|
| 16 |
+
</template>
|
| 17 |
+
|
| 18 |
+
<script lang="ts" setup>
|
| 19 |
+
const props = withDefaults(defineProps<{
|
| 20 |
+
value: boolean
|
| 21 |
+
disabled?: boolean
|
| 22 |
+
}>(), {
|
| 23 |
+
disabled: false,
|
| 24 |
+
})
|
| 25 |
+
|
| 26 |
+
const emit = defineEmits<{
|
| 27 |
+
(event: 'update:value', payload: boolean): void
|
| 28 |
+
}>()
|
| 29 |
+
|
| 30 |
+
const handleChange = (e: Event) => {
|
| 31 |
+
if (props.disabled) return
|
| 32 |
+
emit('update:value', (e.target as HTMLInputElement).checked)
|
| 33 |
+
}
|
| 34 |
+
</script>
|
| 35 |
+
|
| 36 |
+
<style lang="scss" scoped>
|
| 37 |
+
.checkbox {
|
| 38 |
+
height: 20px;
|
| 39 |
+
display: flex;
|
| 40 |
+
align-items: center;
|
| 41 |
+
cursor: pointer;
|
| 42 |
+
|
| 43 |
+
&:not(.disabled).checked {
|
| 44 |
+
.checkbox-input {
|
| 45 |
+
background-color: $themeColor;
|
| 46 |
+
border-color: $themeColor;
|
| 47 |
+
}
|
| 48 |
+
.checkbox-input::after {
|
| 49 |
+
transform: rotate(45deg) scaleY(1);
|
| 50 |
+
}
|
| 51 |
+
|
| 52 |
+
.checkbox-label {
|
| 53 |
+
color: $themeColor;
|
| 54 |
+
}
|
| 55 |
+
}
|
| 56 |
+
|
| 57 |
+
&.disabled {
|
| 58 |
+
color: #b7b7b7;
|
| 59 |
+
cursor: default;
|
| 60 |
+
|
| 61 |
+
.checkbox-input {
|
| 62 |
+
background-color: #f5f5f5;
|
| 63 |
+
}
|
| 64 |
+
}
|
| 65 |
+
}
|
| 66 |
+
|
| 67 |
+
.checkbox-input {
|
| 68 |
+
display: inline-block;
|
| 69 |
+
position: relative;
|
| 70 |
+
border: 1px solid #d9d9d9;
|
| 71 |
+
border-radius: $borderRadius;
|
| 72 |
+
width: 16px;
|
| 73 |
+
height: 16px;
|
| 74 |
+
background-color: #fff;
|
| 75 |
+
vertical-align: middle;
|
| 76 |
+
transition: border-color .15s cubic-bezier(.71, -.46, .29, 1.46), background-color .15s cubic-bezier(.71, -.46, .29, 1.46);
|
| 77 |
+
z-index: 1;
|
| 78 |
+
|
| 79 |
+
&::after {
|
| 80 |
+
content: '';
|
| 81 |
+
border: 2px solid #fff;
|
| 82 |
+
border-left: 0;
|
| 83 |
+
border-top: 0;
|
| 84 |
+
height: 9px;
|
| 85 |
+
left: 4px;
|
| 86 |
+
position: absolute;
|
| 87 |
+
top: 1px;
|
| 88 |
+
transform: rotate(45deg) scaleY(0);
|
| 89 |
+
width: 6px;
|
| 90 |
+
transition: transform .15s ease-in .05s;
|
| 91 |
+
transform-origin: center;
|
| 92 |
+
}
|
| 93 |
+
}
|
| 94 |
+
.checkbox-original {
|
| 95 |
+
opacity: 0;
|
| 96 |
+
outline: 0;
|
| 97 |
+
position: absolute;
|
| 98 |
+
margin: 0;
|
| 99 |
+
width: 0;
|
| 100 |
+
height: 0;
|
| 101 |
+
z-index: -1;
|
| 102 |
+
}
|
| 103 |
+
.checkbox-label {
|
| 104 |
+
margin-left: 5px;
|
| 105 |
+
line-height: 20px;
|
| 106 |
+
font-size: 13px;
|
| 107 |
+
user-select: none;
|
| 108 |
+
}
|
| 109 |
+
</style>
|
frontend/src/components/CheckboxButton.vue
ADDED
|
@@ -0,0 +1,21 @@
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
| 1 |
+
<template>
|
| 2 |
+
<Button
|
| 3 |
+
:checked="checked"
|
| 4 |
+
:disabled="disabled"
|
| 5 |
+
type="checkbox"
|
| 6 |
+
>
|
| 7 |
+
<slot></slot>
|
| 8 |
+
</Button>
|
| 9 |
+
</template>
|
| 10 |
+
|
| 11 |
+
<script lang="ts" setup>
|
| 12 |
+
import Button from './Button.vue'
|
| 13 |
+
|
| 14 |
+
withDefaults(defineProps<{
|
| 15 |
+
checked?: boolean
|
| 16 |
+
disabled?: boolean
|
| 17 |
+
}>(), {
|
| 18 |
+
checked: false,
|
| 19 |
+
disabled: false,
|
| 20 |
+
})
|
| 21 |
+
</script>
|
frontend/src/components/ColorButton.vue
ADDED
|
@@ -0,0 +1,42 @@
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
| 1 |
+
<template>
|
| 2 |
+
<Button class="color-btn">
|
| 3 |
+
<div class="color-block">
|
| 4 |
+
<div class="content" :style="{ backgroundColor: color }"></div>
|
| 5 |
+
</div>
|
| 6 |
+
<IconPlatte class="color-btn-icon" />
|
| 7 |
+
</Button>
|
| 8 |
+
</template>
|
| 9 |
+
|
| 10 |
+
<script lang="ts" setup>
|
| 11 |
+
import Button from './Button.vue'
|
| 12 |
+
|
| 13 |
+
defineProps<{
|
| 14 |
+
color: string
|
| 15 |
+
}>()
|
| 16 |
+
</script>
|
| 17 |
+
|
| 18 |
+
<style lang="scss" scoped>
|
| 19 |
+
.color-btn {
|
| 20 |
+
width: 100%;
|
| 21 |
+
display: flex !important;
|
| 22 |
+
align-items: center;
|
| 23 |
+
justify-content: center;
|
| 24 |
+
padding: 0 !important;
|
| 25 |
+
}
|
| 26 |
+
.color-block {
|
| 27 |
+
height: 20px;
|
| 28 |
+
margin-left: 8px;
|
| 29 |
+
flex: 1;
|
| 30 |
+
outline: 1px dashed rgba($color: #666, $alpha: .12);
|
| 31 |
+
background-image: url();
|
| 32 |
+
}
|
| 33 |
+
.content {
|
| 34 |
+
width: 100%;
|
| 35 |
+
height: 100%;
|
| 36 |
+
}
|
| 37 |
+
.color-btn-icon {
|
| 38 |
+
width: 32px;
|
| 39 |
+
font-size: 13px;
|
| 40 |
+
color: #bfbfbf;
|
| 41 |
+
}
|
| 42 |
+
</style>
|
frontend/src/components/ColorListButton.vue
ADDED
|
@@ -0,0 +1,58 @@
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
| 1 |
+
<template>
|
| 2 |
+
<Button class="color-btn">
|
| 3 |
+
<div class="blocks">
|
| 4 |
+
<div class="color-block" v-for="(color, index) in colors" :key="index">
|
| 5 |
+
<div class="content" :style="{ backgroundColor: color }"></div>
|
| 6 |
+
</div>
|
| 7 |
+
</div>
|
| 8 |
+
<IconPlatte class="color-btn-icon" />
|
| 9 |
+
</Button>
|
| 10 |
+
</template>
|
| 11 |
+
|
| 12 |
+
<script lang="ts" setup>
|
| 13 |
+
import { computed } from 'vue'
|
| 14 |
+
import Button from './Button.vue'
|
| 15 |
+
|
| 16 |
+
const props = defineProps<{
|
| 17 |
+
colors: string[]
|
| 18 |
+
}>()
|
| 19 |
+
|
| 20 |
+
const colors = computed(() => {
|
| 21 |
+
if (props.colors.length > 12) return props.colors.slice(0, 12)
|
| 22 |
+
return props.colors
|
| 23 |
+
})
|
| 24 |
+
</script>
|
| 25 |
+
|
| 26 |
+
<style lang="scss" scoped>
|
| 27 |
+
.color-btn {
|
| 28 |
+
width: 100%;
|
| 29 |
+
display: flex !important;
|
| 30 |
+
align-items: center;
|
| 31 |
+
justify-content: center;
|
| 32 |
+
padding: 0 !important;
|
| 33 |
+
}
|
| 34 |
+
.blocks {
|
| 35 |
+
display: flex;
|
| 36 |
+
flex: 1;
|
| 37 |
+
margin-left: 8px;
|
| 38 |
+
outline: 1px dashed rgba($color: #666, $alpha: .12);
|
| 39 |
+
}
|
| 40 |
+
.color-block {
|
| 41 |
+
height: 20px;
|
| 42 |
+
flex: 1;
|
| 43 |
+
background-image: url();
|
| 44 |
+
|
| 45 |
+
& + & {
|
| 46 |
+
margin-left: 2px;
|
| 47 |
+
}
|
| 48 |
+
}
|
| 49 |
+
.content {
|
| 50 |
+
width: 100%;
|
| 51 |
+
height: 100%;
|
| 52 |
+
}
|
| 53 |
+
.color-btn-icon {
|
| 54 |
+
width: 32px;
|
| 55 |
+
font-size: 13px;
|
| 56 |
+
color: #bfbfbf;
|
| 57 |
+
}
|
| 58 |
+
</style>
|
frontend/src/components/ColorPicker/Alpha.vue
ADDED
|
@@ -0,0 +1,107 @@
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
| 1 |
+
<template>
|
| 2 |
+
<div class="alpha">
|
| 3 |
+
<div class="alpha-checkboard-wrap">
|
| 4 |
+
<Checkboard />
|
| 5 |
+
</div>
|
| 6 |
+
<div class="alpha-gradient" :style="{ background: gradientColor }"></div>
|
| 7 |
+
<div
|
| 8 |
+
class="alpha-container"
|
| 9 |
+
ref="alphaRef"
|
| 10 |
+
@mousedown="$event => handleMouseDown($event)"
|
| 11 |
+
>
|
| 12 |
+
<div class="alpha-pointer" :style="{ left: color.a * 100 + '%' }">
|
| 13 |
+
<div class="alpha-picker"></div>
|
| 14 |
+
</div>
|
| 15 |
+
</div>
|
| 16 |
+
</div>
|
| 17 |
+
</template>
|
| 18 |
+
|
| 19 |
+
<script lang="ts" setup>
|
| 20 |
+
import { computed, onUnmounted, ref } from 'vue'
|
| 21 |
+
|
| 22 |
+
import Checkboard from './Checkboard.vue'
|
| 23 |
+
import type { ColorFormats } from 'tinycolor2'
|
| 24 |
+
|
| 25 |
+
const props = defineProps<{
|
| 26 |
+
value: ColorFormats.RGBA
|
| 27 |
+
}>()
|
| 28 |
+
|
| 29 |
+
const emit = defineEmits<{
|
| 30 |
+
(event: 'colorChange', payload: ColorFormats.RGBA): void
|
| 31 |
+
}>()
|
| 32 |
+
|
| 33 |
+
const color = computed(() => props.value)
|
| 34 |
+
|
| 35 |
+
const gradientColor = computed(() => {
|
| 36 |
+
const rgbaStr = [color.value.r, color.value.g, color.value.b].join(',')
|
| 37 |
+
return `linear-gradient(to right, rgba(${rgbaStr}, 0) 0%, rgba(${rgbaStr}, 1) 100%)`
|
| 38 |
+
})
|
| 39 |
+
|
| 40 |
+
const alphaRef = ref<HTMLElement>()
|
| 41 |
+
const handleChange = (e: MouseEvent) => {
|
| 42 |
+
e.preventDefault()
|
| 43 |
+
if (!alphaRef.value) return
|
| 44 |
+
const containerWidth = alphaRef.value.clientWidth
|
| 45 |
+
const xOffset = alphaRef.value.getBoundingClientRect().left + window.pageXOffset
|
| 46 |
+
const left = e.pageX - xOffset
|
| 47 |
+
let a
|
| 48 |
+
|
| 49 |
+
if (left < 0) a = 0
|
| 50 |
+
else if (left > containerWidth) a = 1
|
| 51 |
+
else a = Math.round(left * 100 / containerWidth) / 100
|
| 52 |
+
|
| 53 |
+
if (color.value.a !== a) {
|
| 54 |
+
emit('colorChange', {
|
| 55 |
+
r: color.value.r,
|
| 56 |
+
g: color.value.g,
|
| 57 |
+
b: color.value.b,
|
| 58 |
+
a: a,
|
| 59 |
+
})
|
| 60 |
+
}
|
| 61 |
+
}
|
| 62 |
+
|
| 63 |
+
const unbindEventListeners = () => {
|
| 64 |
+
window.removeEventListener('mousemove', handleChange)
|
| 65 |
+
window.removeEventListener('mouseup', unbindEventListeners)
|
| 66 |
+
}
|
| 67 |
+
const handleMouseDown = (e: MouseEvent) => {
|
| 68 |
+
handleChange(e)
|
| 69 |
+
window.addEventListener('mousemove', handleChange)
|
| 70 |
+
window.addEventListener('mouseup', unbindEventListeners)
|
| 71 |
+
}
|
| 72 |
+
onUnmounted(unbindEventListeners)
|
| 73 |
+
</script>
|
| 74 |
+
|
| 75 |
+
<style lang="scss" scoped>
|
| 76 |
+
.alpha {
|
| 77 |
+
@include absolute-0();
|
| 78 |
+
}
|
| 79 |
+
.alpha-checkboard-wrap {
|
| 80 |
+
overflow: hidden;
|
| 81 |
+
|
| 82 |
+
@include absolute-0();
|
| 83 |
+
}
|
| 84 |
+
.alpha-gradient {
|
| 85 |
+
@include absolute-0();
|
| 86 |
+
}
|
| 87 |
+
.alpha-container {
|
| 88 |
+
cursor: pointer;
|
| 89 |
+
position: relative;
|
| 90 |
+
z-index: 2;
|
| 91 |
+
height: 100%;
|
| 92 |
+
margin: 0 3px;
|
| 93 |
+
}
|
| 94 |
+
.alpha-pointer {
|
| 95 |
+
z-index: 2;
|
| 96 |
+
position: absolute;
|
| 97 |
+
}
|
| 98 |
+
.alpha-picker {
|
| 99 |
+
cursor: pointer;
|
| 100 |
+
width: 4px;
|
| 101 |
+
height: 8px;
|
| 102 |
+
box-shadow: 0 0 2px rgba(0, 0, 0, .6);
|
| 103 |
+
background: #fff;
|
| 104 |
+
margin-top: 1px;
|
| 105 |
+
transform: translateX(-2px);
|
| 106 |
+
}
|
| 107 |
+
</style>
|
frontend/src/components/ColorPicker/Checkboard.vue
ADDED
|
@@ -0,0 +1,60 @@
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
| 1 |
+
<template>
|
| 2 |
+
<div class="checkerboard" :style="bgStyle"></div>
|
| 3 |
+
</template>
|
| 4 |
+
|
| 5 |
+
<script lang="ts" setup>
|
| 6 |
+
import { computed } from 'vue'
|
| 7 |
+
|
| 8 |
+
const props = withDefaults(defineProps<{
|
| 9 |
+
size?: number
|
| 10 |
+
white?: string
|
| 11 |
+
grey?: string
|
| 12 |
+
}>(), {
|
| 13 |
+
size: 8,
|
| 14 |
+
white: '#fff',
|
| 15 |
+
grey: '#e6e6e6',
|
| 16 |
+
})
|
| 17 |
+
|
| 18 |
+
interface CheckboardCache {
|
| 19 |
+
[key: string]: string | null
|
| 20 |
+
}
|
| 21 |
+
const checkboardCache: CheckboardCache = {}
|
| 22 |
+
|
| 23 |
+
const renderCheckboard = (white: string, grey: string, size: number) => {
|
| 24 |
+
const canvas = document.createElement('canvas')
|
| 25 |
+
canvas.width = canvas.height = size * 2
|
| 26 |
+
const ctx = canvas.getContext('2d')
|
| 27 |
+
|
| 28 |
+
if (!ctx) return null
|
| 29 |
+
|
| 30 |
+
ctx.fillStyle = white
|
| 31 |
+
ctx.fillRect(0, 0, canvas.width, canvas.height)
|
| 32 |
+
ctx.fillStyle = grey
|
| 33 |
+
ctx.fillRect(0, 0, size, size)
|
| 34 |
+
ctx.translate(size, size)
|
| 35 |
+
ctx.fillRect(0, 0, size, size)
|
| 36 |
+
return canvas.toDataURL()
|
| 37 |
+
}
|
| 38 |
+
|
| 39 |
+
const getCheckboard = (white: string, grey: string, size: number) => {
|
| 40 |
+
const key = white + ',' + grey + ',' + size
|
| 41 |
+
if (checkboardCache[key]) return checkboardCache[key]
|
| 42 |
+
|
| 43 |
+
const checkboard = renderCheckboard(white, grey, size)
|
| 44 |
+
checkboardCache[key] = checkboard
|
| 45 |
+
return checkboard
|
| 46 |
+
}
|
| 47 |
+
|
| 48 |
+
const bgStyle = computed(() => {
|
| 49 |
+
const checkboard = getCheckboard(props.white, props.grey, props.size)
|
| 50 |
+
return { backgroundImage: `url(${checkboard})` }
|
| 51 |
+
})
|
| 52 |
+
</script>
|
| 53 |
+
|
| 54 |
+
<style lang="scss" scoped>
|
| 55 |
+
.checkerboard {
|
| 56 |
+
background-size: contain;
|
| 57 |
+
|
| 58 |
+
@include absolute-0();
|
| 59 |
+
}
|
| 60 |
+
</style>
|
frontend/src/components/ColorPicker/EditableInput.vue
ADDED
|
@@ -0,0 +1,69 @@
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
| 1 |
+
<template>
|
| 2 |
+
<div class="editable-input">
|
| 3 |
+
<input
|
| 4 |
+
class="input-content"
|
| 5 |
+
:value="val"
|
| 6 |
+
@input="$event => handleInput($event)"
|
| 7 |
+
>
|
| 8 |
+
</div>
|
| 9 |
+
</template>
|
| 10 |
+
|
| 11 |
+
<script lang="ts" setup>
|
| 12 |
+
import { computed } from 'vue'
|
| 13 |
+
import tinycolor, { type ColorFormats } from 'tinycolor2'
|
| 14 |
+
|
| 15 |
+
const props = defineProps<{
|
| 16 |
+
value: ColorFormats.RGBA
|
| 17 |
+
}>()
|
| 18 |
+
|
| 19 |
+
const emit = defineEmits<{
|
| 20 |
+
(event: 'colorChange', payload: ColorFormats.RGBA): void
|
| 21 |
+
}>()
|
| 22 |
+
|
| 23 |
+
const val = computed(() => {
|
| 24 |
+
let _hex = ''
|
| 25 |
+
if (props.value.a < 1) _hex = tinycolor(props.value).toHex8String().toUpperCase()
|
| 26 |
+
else _hex = tinycolor(props.value).toHexString().toUpperCase()
|
| 27 |
+
return _hex.replace('#', '')
|
| 28 |
+
})
|
| 29 |
+
|
| 30 |
+
const handleInput = (e: Event) => {
|
| 31 |
+
const value = (e.target as HTMLInputElement).value
|
| 32 |
+
if (value.length >= 6) {
|
| 33 |
+
const color = tinycolor(value)
|
| 34 |
+
if (color.isValid()) {
|
| 35 |
+
emit('colorChange', color.toRgb())
|
| 36 |
+
}
|
| 37 |
+
}
|
| 38 |
+
}
|
| 39 |
+
</script>
|
| 40 |
+
|
| 41 |
+
<style lang="scss" scoped>
|
| 42 |
+
.editable-input {
|
| 43 |
+
width: 100%;
|
| 44 |
+
position: relative;
|
| 45 |
+
overflow: hidden;
|
| 46 |
+
text-align: center;
|
| 47 |
+
font-size: 14px;
|
| 48 |
+
|
| 49 |
+
&::after {
|
| 50 |
+
content: '#';
|
| 51 |
+
position: absolute;
|
| 52 |
+
left: 0;
|
| 53 |
+
top: 50%;
|
| 54 |
+
transform: translateY(-50%);
|
| 55 |
+
color: #999;
|
| 56 |
+
}
|
| 57 |
+
}
|
| 58 |
+
.input-content {
|
| 59 |
+
width: 100%;
|
| 60 |
+
padding: 3px;
|
| 61 |
+
border: 0;
|
| 62 |
+
border-bottom: 1px solid #ddd;
|
| 63 |
+
outline: none;
|
| 64 |
+
text-align: center;
|
| 65 |
+
}
|
| 66 |
+
.input-label {
|
| 67 |
+
text-transform: capitalize;
|
| 68 |
+
}
|
| 69 |
+
</style>
|
frontend/src/components/ColorPicker/Hue.vue
ADDED
|
@@ -0,0 +1,117 @@
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
| 1 |
+
<template>
|
| 2 |
+
<div class="hue">
|
| 3 |
+
<div
|
| 4 |
+
class="hue-container"
|
| 5 |
+
ref="hueRef"
|
| 6 |
+
@mousedown="$event => handleMouseDown($event)"
|
| 7 |
+
>
|
| 8 |
+
<div
|
| 9 |
+
class="hue-pointer"
|
| 10 |
+
:style="{ left: pointerLeft }"
|
| 11 |
+
>
|
| 12 |
+
<div class="hue-picker"></div>
|
| 13 |
+
</div>
|
| 14 |
+
</div>
|
| 15 |
+
</div>
|
| 16 |
+
</template>
|
| 17 |
+
|
| 18 |
+
<script lang="ts" setup>
|
| 19 |
+
import { computed, onUnmounted, ref, watch } from 'vue'
|
| 20 |
+
import tinycolor, { type ColorFormats } from 'tinycolor2'
|
| 21 |
+
|
| 22 |
+
const props = defineProps<{
|
| 23 |
+
value: ColorFormats.RGBA
|
| 24 |
+
hue: number
|
| 25 |
+
}>()
|
| 26 |
+
|
| 27 |
+
const emit = defineEmits<{
|
| 28 |
+
(event: 'colorChange', payload: ColorFormats.HSLA): void
|
| 29 |
+
}>()
|
| 30 |
+
|
| 31 |
+
const oldHue = ref(0)
|
| 32 |
+
const pullDirection = ref('')
|
| 33 |
+
|
| 34 |
+
const color = computed(() => {
|
| 35 |
+
const hsla = tinycolor(props.value).toHsl()
|
| 36 |
+
if (props.hue !== -1) hsla.h = props.hue
|
| 37 |
+
return hsla
|
| 38 |
+
})
|
| 39 |
+
|
| 40 |
+
const pointerLeft = computed(() => {
|
| 41 |
+
if (color.value.h === 0 && pullDirection.value === 'right') return '100%'
|
| 42 |
+
return color.value.h * 100 / 360 + '%'
|
| 43 |
+
})
|
| 44 |
+
|
| 45 |
+
watch(() => props.value, () => {
|
| 46 |
+
const hsla = tinycolor(props.value).toHsl()
|
| 47 |
+
const h = hsla.s === 0 ? props.hue : hsla.h
|
| 48 |
+
if (h !== 0 && h - oldHue.value > 0) pullDirection.value = 'right'
|
| 49 |
+
if (h !== 0 && h - oldHue.value < 0) pullDirection.value = 'left'
|
| 50 |
+
oldHue.value = h
|
| 51 |
+
})
|
| 52 |
+
|
| 53 |
+
const hueRef = ref<HTMLElement>()
|
| 54 |
+
const handleChange = (e: MouseEvent) => {
|
| 55 |
+
e.preventDefault()
|
| 56 |
+
if (!hueRef.value) return
|
| 57 |
+
|
| 58 |
+
const containerWidth = hueRef.value.clientWidth
|
| 59 |
+
const xOffset = hueRef.value.getBoundingClientRect().left + window.pageXOffset
|
| 60 |
+
const left = e.pageX - xOffset
|
| 61 |
+
let h, percent
|
| 62 |
+
|
| 63 |
+
if (left < 0) h = 0
|
| 64 |
+
else if (left > containerWidth) h = 360
|
| 65 |
+
else {
|
| 66 |
+
percent = left * 100 / containerWidth
|
| 67 |
+
h = 360 * percent / 100
|
| 68 |
+
}
|
| 69 |
+
if (props.hue === -1 || color.value.h !== h) {
|
| 70 |
+
emit('colorChange', {
|
| 71 |
+
h,
|
| 72 |
+
l: color.value.l,
|
| 73 |
+
s: color.value.s,
|
| 74 |
+
a: color.value.a,
|
| 75 |
+
})
|
| 76 |
+
}
|
| 77 |
+
}
|
| 78 |
+
|
| 79 |
+
const unbindEventListeners = () => {
|
| 80 |
+
window.removeEventListener('mousemove', handleChange)
|
| 81 |
+
window.removeEventListener('mouseup', unbindEventListeners)
|
| 82 |
+
}
|
| 83 |
+
const handleMouseDown = (e: MouseEvent) => {
|
| 84 |
+
handleChange(e)
|
| 85 |
+
window.addEventListener('mousemove', handleChange)
|
| 86 |
+
window.addEventListener('mouseup', unbindEventListeners)
|
| 87 |
+
}
|
| 88 |
+
onUnmounted(unbindEventListeners)
|
| 89 |
+
</script>
|
| 90 |
+
|
| 91 |
+
<style lang="scss" scoped>
|
| 92 |
+
.hue {
|
| 93 |
+
background: linear-gradient(to right, #f00 0%, #ff0 17%, #0f0 33%, #0ff 50%, #00f 67%, #f0f 83%, #f00 100%);
|
| 94 |
+
|
| 95 |
+
@include absolute-0();
|
| 96 |
+
}
|
| 97 |
+
.hue-container {
|
| 98 |
+
cursor: pointer;
|
| 99 |
+
margin: 0 2px;
|
| 100 |
+
position: relative;
|
| 101 |
+
height: 100%;
|
| 102 |
+
}
|
| 103 |
+
.hue-pointer {
|
| 104 |
+
z-index: 2;
|
| 105 |
+
position: absolute;
|
| 106 |
+
top: 0;
|
| 107 |
+
}
|
| 108 |
+
.hue-picker {
|
| 109 |
+
cursor: pointer;
|
| 110 |
+
margin-top: 1px;
|
| 111 |
+
width: 4px;
|
| 112 |
+
height: 8px;
|
| 113 |
+
box-shadow: 0 0 2px rgba(0, 0, 0, .6);
|
| 114 |
+
background: #fff;
|
| 115 |
+
transform: translateX(-2px);
|
| 116 |
+
}
|
| 117 |
+
</style>
|
frontend/src/components/ColorPicker/Saturation.vue
ADDED
|
@@ -0,0 +1,108 @@
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
| 1 |
+
<template>
|
| 2 |
+
<div
|
| 3 |
+
class="saturation"
|
| 4 |
+
ref="saturationRef"
|
| 5 |
+
:style="{ background: bgColor }"
|
| 6 |
+
@mousedown="$event => handleMouseDown($event)"
|
| 7 |
+
>
|
| 8 |
+
<div class="saturation-white"></div>
|
| 9 |
+
<div class="saturation-black"></div>
|
| 10 |
+
<div class="saturation-pointer"
|
| 11 |
+
:style="{
|
| 12 |
+
top: pointerTop,
|
| 13 |
+
left: pointerLeft,
|
| 14 |
+
}"
|
| 15 |
+
>
|
| 16 |
+
<div class="saturation-circle"></div>
|
| 17 |
+
</div>
|
| 18 |
+
</div>
|
| 19 |
+
</template>
|
| 20 |
+
|
| 21 |
+
<script lang="ts" setup>
|
| 22 |
+
import { computed, onUnmounted, ref } from 'vue'
|
| 23 |
+
import tinycolor, { type ColorFormats } from 'tinycolor2'
|
| 24 |
+
import { throttle, clamp } from 'lodash'
|
| 25 |
+
|
| 26 |
+
const props = defineProps<{
|
| 27 |
+
value: ColorFormats.RGBA
|
| 28 |
+
hue: number
|
| 29 |
+
}>()
|
| 30 |
+
|
| 31 |
+
const emit = defineEmits<{
|
| 32 |
+
(event: 'colorChange', payload: ColorFormats.HSVA): void
|
| 33 |
+
}>()
|
| 34 |
+
|
| 35 |
+
const color = computed(() => {
|
| 36 |
+
const hsva = tinycolor(props.value).toHsv()
|
| 37 |
+
if (props.hue !== -1) hsva.h = props.hue
|
| 38 |
+
return hsva
|
| 39 |
+
})
|
| 40 |
+
|
| 41 |
+
const bgColor = computed(() => `hsl(${color.value.h}, 100%, 50%)`)
|
| 42 |
+
const pointerTop = computed(() => (-(color.value.v * 100) + 1) + 100 + '%')
|
| 43 |
+
const pointerLeft = computed(() => color.value.s * 100 + '%')
|
| 44 |
+
|
| 45 |
+
const emitChangeEvent = throttle(function(param: ColorFormats.HSVA) {
|
| 46 |
+
emit('colorChange', param)
|
| 47 |
+
}, 20, { leading: true, trailing: false })
|
| 48 |
+
|
| 49 |
+
const saturationRef = ref<HTMLElement>()
|
| 50 |
+
const handleChange = (e: MouseEvent) => {
|
| 51 |
+
e.preventDefault()
|
| 52 |
+
if (!saturationRef.value) return
|
| 53 |
+
|
| 54 |
+
const containerWidth = saturationRef.value.clientWidth
|
| 55 |
+
const containerHeight = saturationRef.value.clientHeight
|
| 56 |
+
const xOffset = saturationRef.value.getBoundingClientRect().left + window.pageXOffset
|
| 57 |
+
const yOffset = saturationRef.value.getBoundingClientRect().top + window.pageYOffset
|
| 58 |
+
const left = clamp(e.pageX - xOffset, 0, containerWidth)
|
| 59 |
+
const top = clamp(e.pageY - yOffset, 0, containerHeight)
|
| 60 |
+
const saturation = left / containerWidth
|
| 61 |
+
const bright = clamp(-(top / containerHeight) + 1, 0, 1)
|
| 62 |
+
|
| 63 |
+
emitChangeEvent({
|
| 64 |
+
h: color.value.h,
|
| 65 |
+
s: saturation,
|
| 66 |
+
v: bright,
|
| 67 |
+
a: color.value.a,
|
| 68 |
+
})
|
| 69 |
+
}
|
| 70 |
+
|
| 71 |
+
const unbindEventListeners = () => {
|
| 72 |
+
window.removeEventListener('mousemove', handleChange)
|
| 73 |
+
window.removeEventListener('mouseup', unbindEventListeners)
|
| 74 |
+
}
|
| 75 |
+
const handleMouseDown = (e: MouseEvent) => {
|
| 76 |
+
handleChange(e)
|
| 77 |
+
window.addEventListener('mousemove', handleChange)
|
| 78 |
+
window.addEventListener('mouseup', unbindEventListeners)
|
| 79 |
+
}
|
| 80 |
+
onUnmounted(unbindEventListeners)
|
| 81 |
+
</script>
|
| 82 |
+
|
| 83 |
+
<style lang="scss" scoped>
|
| 84 |
+
.saturation,
|
| 85 |
+
.saturation-white,
|
| 86 |
+
.saturation-black {
|
| 87 |
+
@include absolute-0();
|
| 88 |
+
|
| 89 |
+
cursor: pointer;
|
| 90 |
+
}
|
| 91 |
+
.saturation-white {
|
| 92 |
+
background: linear-gradient(to right, #fff, rgba(255, 255, 255, 0));
|
| 93 |
+
}
|
| 94 |
+
.saturation-black {
|
| 95 |
+
background: linear-gradient(to top, #000, rgba(0, 0, 0, 0));
|
| 96 |
+
}
|
| 97 |
+
.saturation-pointer {
|
| 98 |
+
cursor: pointer;
|
| 99 |
+
position: absolute;
|
| 100 |
+
}
|
| 101 |
+
.saturation-circle {
|
| 102 |
+
width: 4px;
|
| 103 |
+
height: 4px;
|
| 104 |
+
box-shadow: 0 0 0 1.5px #fff, inset 0 0 1px 1px rgba(0, 0, 0, .3), 0 0 1px 2px rgba(0, 0, 0, .4);
|
| 105 |
+
border-radius: 50%;
|
| 106 |
+
transform: translate(-2px, -2px);
|
| 107 |
+
}
|
| 108 |
+
</style>
|
frontend/src/components/ColorPicker/index.vue
ADDED
|
@@ -0,0 +1,443 @@
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
| 1 |
+
<template>
|
| 2 |
+
<div class="color-picker">
|
| 3 |
+
<div class="picker-saturation-wrap">
|
| 4 |
+
<Saturation :value="color" :hue="hue" @colorChange="value => changeColor(value)" />
|
| 5 |
+
</div>
|
| 6 |
+
<div class="picker-controls">
|
| 7 |
+
<div class="picker-color-wrap">
|
| 8 |
+
<div class="picker-current-color" :style="{ background: currentColor }"></div>
|
| 9 |
+
<Checkboard />
|
| 10 |
+
</div>
|
| 11 |
+
<div class="picker-sliders">
|
| 12 |
+
<div class="picker-hue-wrap">
|
| 13 |
+
<Hue :value="color" :hue="hue" @colorChange="value => changeColor(value)" />
|
| 14 |
+
</div>
|
| 15 |
+
<div class="picker-alpha-wrap">
|
| 16 |
+
<Alpha :value="color" @colorChange="value => changeColor(value)" />
|
| 17 |
+
</div>
|
| 18 |
+
</div>
|
| 19 |
+
</div>
|
| 20 |
+
|
| 21 |
+
<div class="picker-field">
|
| 22 |
+
<EditableInput class="input" :value="color" @colorChange="value => changeColor(value)" />
|
| 23 |
+
<div class="straw" @click="openEyeDropper()"><IconNeedle /></div>
|
| 24 |
+
<div class="transparent" @click="selectPresetColor('#00000000')">
|
| 25 |
+
<Checkboard />
|
| 26 |
+
</div>
|
| 27 |
+
</div>
|
| 28 |
+
|
| 29 |
+
<div class="picker-presets">
|
| 30 |
+
<div
|
| 31 |
+
class="picker-presets-color"
|
| 32 |
+
v-for="c in themeColors"
|
| 33 |
+
:key="c"
|
| 34 |
+
:style="{ background: c }"
|
| 35 |
+
@click="selectPresetColor(c)"
|
| 36 |
+
></div>
|
| 37 |
+
</div>
|
| 38 |
+
|
| 39 |
+
<div class="picker-gradient-presets">
|
| 40 |
+
<div
|
| 41 |
+
class="picker-gradient-col"
|
| 42 |
+
v-for="(col, index) in presetColors"
|
| 43 |
+
:key="index"
|
| 44 |
+
>
|
| 45 |
+
<div class="picker-gradient-color"
|
| 46 |
+
v-for="c in col"
|
| 47 |
+
:key="c"
|
| 48 |
+
:style="{ background: c }"
|
| 49 |
+
@click="selectPresetColor(c)"
|
| 50 |
+
></div>
|
| 51 |
+
</div>
|
| 52 |
+
</div>
|
| 53 |
+
|
| 54 |
+
<div class="picker-presets">
|
| 55 |
+
<div
|
| 56 |
+
v-for="c in standardColors"
|
| 57 |
+
:key="c"
|
| 58 |
+
class="picker-presets-color"
|
| 59 |
+
:style="{ background: c }"
|
| 60 |
+
@click="selectPresetColor(c)"
|
| 61 |
+
></div>
|
| 62 |
+
</div>
|
| 63 |
+
|
| 64 |
+
<div class="recent-colors-title" v-if="recentColors.length">最近使用:</div>
|
| 65 |
+
<div class="picker-presets">
|
| 66 |
+
<div
|
| 67 |
+
v-for="c in recentColors"
|
| 68 |
+
:key="c"
|
| 69 |
+
class="picker-presets-color alpha"
|
| 70 |
+
@click="selectPresetColor(c)"
|
| 71 |
+
>
|
| 72 |
+
<div class="picker-presets-color-content" :style="{ background: c }"></div>
|
| 73 |
+
</div>
|
| 74 |
+
</div>
|
| 75 |
+
</div>
|
| 76 |
+
</template>
|
| 77 |
+
|
| 78 |
+
<script lang="ts" setup>
|
| 79 |
+
import { computed, onMounted, ref, watch } from 'vue'
|
| 80 |
+
import tinycolor, { type ColorFormats } from 'tinycolor2'
|
| 81 |
+
import { debounce } from 'lodash'
|
| 82 |
+
import { toCanvas } from 'html-to-image'
|
| 83 |
+
import message from '@/utils/message'
|
| 84 |
+
|
| 85 |
+
import Alpha from './Alpha.vue'
|
| 86 |
+
import Checkboard from './Checkboard.vue'
|
| 87 |
+
import Hue from './Hue.vue'
|
| 88 |
+
import Saturation from './Saturation.vue'
|
| 89 |
+
import EditableInput from './EditableInput.vue'
|
| 90 |
+
|
| 91 |
+
const props = withDefaults(defineProps<{
|
| 92 |
+
modelValue?: string
|
| 93 |
+
}>(), {
|
| 94 |
+
modelValue: '#e86b99',
|
| 95 |
+
})
|
| 96 |
+
|
| 97 |
+
const emit = defineEmits<{
|
| 98 |
+
(event: 'update:modelValue', payload: string): void
|
| 99 |
+
}>()
|
| 100 |
+
|
| 101 |
+
const RECENT_COLORS = 'RECENT_COLORS'
|
| 102 |
+
|
| 103 |
+
const presetColorConfig = [
|
| 104 |
+
['#7f7f7f', '#f2f2f2'],
|
| 105 |
+
['#0d0d0d', '#808080'],
|
| 106 |
+
['#1c1a10', '#ddd8c3'],
|
| 107 |
+
['#0e243d', '#c6d9f0'],
|
| 108 |
+
['#233f5e', '#dae5f0'],
|
| 109 |
+
['#632623', '#f2dbdb'],
|
| 110 |
+
['#4d602c', '#eaf1de'],
|
| 111 |
+
['#3f3150', '#e6e0ec'],
|
| 112 |
+
['#1e5867', '#d9eef3'],
|
| 113 |
+
['#99490f', '#fee9da'],
|
| 114 |
+
]
|
| 115 |
+
|
| 116 |
+
const gradient = (startColor: string, endColor: string, step: number) => {
|
| 117 |
+
const _startColor = tinycolor(startColor).toRgb()
|
| 118 |
+
const _endColor = tinycolor(endColor).toRgb()
|
| 119 |
+
|
| 120 |
+
const rStep = (_endColor.r - _startColor.r) / step
|
| 121 |
+
const gStep = (_endColor.g - _startColor.g) / step
|
| 122 |
+
const bStep = (_endColor.b - _startColor.b) / step
|
| 123 |
+
const gradientColorArr = []
|
| 124 |
+
|
| 125 |
+
for (let i = 0; i < step; i++) {
|
| 126 |
+
const gradientColor = tinycolor({
|
| 127 |
+
r: _startColor.r + rStep * i,
|
| 128 |
+
g: _startColor.g + gStep * i,
|
| 129 |
+
b: _startColor.b + bStep * i,
|
| 130 |
+
}).toRgbString()
|
| 131 |
+
gradientColorArr.push(gradientColor)
|
| 132 |
+
}
|
| 133 |
+
return gradientColorArr
|
| 134 |
+
}
|
| 135 |
+
|
| 136 |
+
const getPresetColors = () => {
|
| 137 |
+
const presetColors = []
|
| 138 |
+
for (const color of presetColorConfig) {
|
| 139 |
+
presetColors.push(gradient(color[1], color[0], 5))
|
| 140 |
+
}
|
| 141 |
+
return presetColors
|
| 142 |
+
}
|
| 143 |
+
|
| 144 |
+
const themeColors = ['#000000', '#ffffff', '#eeece1', '#1e497b', '#4e81bb', '#e2534d', '#9aba60', '#8165a0', '#47acc5', '#f9974c']
|
| 145 |
+
const standardColors = ['#c21401', '#ff1e02', '#ffc12a', '#ffff3a', '#90cf5b', '#00af57', '#00afee', '#0071be', '#00215f', '#72349d']
|
| 146 |
+
|
| 147 |
+
const hue = ref(-1)
|
| 148 |
+
const recentColors = ref<string[]>([])
|
| 149 |
+
|
| 150 |
+
const color = computed({
|
| 151 |
+
get() {
|
| 152 |
+
return tinycolor(props.modelValue).toRgb()
|
| 153 |
+
},
|
| 154 |
+
set(rgba: ColorFormats.RGBA) {
|
| 155 |
+
const rgbaString = `rgba(${[rgba.r, rgba.g, rgba.b, rgba.a].join(',')})`
|
| 156 |
+
emit('update:modelValue', rgbaString)
|
| 157 |
+
},
|
| 158 |
+
})
|
| 159 |
+
|
| 160 |
+
const presetColors = getPresetColors()
|
| 161 |
+
|
| 162 |
+
const currentColor = computed(() => {
|
| 163 |
+
return `rgba(${[color.value.r, color.value.g, color.value.b, color.value.a].join(',')})`
|
| 164 |
+
})
|
| 165 |
+
|
| 166 |
+
const selectPresetColor = (colorString: string) => {
|
| 167 |
+
hue.value = tinycolor(colorString).toHsl().h
|
| 168 |
+
emit('update:modelValue', colorString)
|
| 169 |
+
}
|
| 170 |
+
|
| 171 |
+
// 每次选择非预设颜色时,需要将该颜色加入到最近使用列表中
|
| 172 |
+
const updateRecentColorsCache = debounce(function() {
|
| 173 |
+
const _color = tinycolor(color.value).toRgbString()
|
| 174 |
+
if (!recentColors.value.includes(_color)) {
|
| 175 |
+
recentColors.value = [_color, ...recentColors.value]
|
| 176 |
+
|
| 177 |
+
const maxLength = 10
|
| 178 |
+
if (recentColors.value.length > maxLength) {
|
| 179 |
+
recentColors.value = recentColors.value.slice(0, maxLength)
|
| 180 |
+
}
|
| 181 |
+
}
|
| 182 |
+
}, 300, { trailing: true })
|
| 183 |
+
|
| 184 |
+
onMounted(() => {
|
| 185 |
+
const recentColorsCache = localStorage.getItem(RECENT_COLORS)
|
| 186 |
+
if (recentColorsCache) recentColors.value = JSON.parse(recentColorsCache)
|
| 187 |
+
})
|
| 188 |
+
|
| 189 |
+
watch(recentColors, () => {
|
| 190 |
+
const recentColorsCache = JSON.stringify(recentColors.value)
|
| 191 |
+
localStorage.setItem(RECENT_COLORS, recentColorsCache)
|
| 192 |
+
})
|
| 193 |
+
|
| 194 |
+
const changeColor = (value: ColorFormats.RGBA | ColorFormats.HSLA | ColorFormats.HSVA) => {
|
| 195 |
+
if ('h' in value) {
|
| 196 |
+
hue.value = value.h
|
| 197 |
+
color.value = tinycolor(value).toRgb()
|
| 198 |
+
}
|
| 199 |
+
else {
|
| 200 |
+
hue.value = tinycolor(value).toHsl().h
|
| 201 |
+
color.value = value
|
| 202 |
+
}
|
| 203 |
+
|
| 204 |
+
updateRecentColorsCache()
|
| 205 |
+
}
|
| 206 |
+
|
| 207 |
+
// 打开取色吸管
|
| 208 |
+
// 检查环境是否支持原生取色吸管,支持则使用原生吸管,否则使用自定义吸管
|
| 209 |
+
const openEyeDropper = () => {
|
| 210 |
+
const isSupportedEyeDropper = 'EyeDropper' in window
|
| 211 |
+
|
| 212 |
+
if (isSupportedEyeDropper) browserEyeDropper()
|
| 213 |
+
else customEyeDropper()
|
| 214 |
+
}
|
| 215 |
+
|
| 216 |
+
// 原生取色吸管
|
| 217 |
+
const browserEyeDropper = () => {
|
| 218 |
+
message.success('按 ESC 键关闭取色吸管', { duration: 0 })
|
| 219 |
+
|
| 220 |
+
// eslint-disable-next-line
|
| 221 |
+
const eyeDropper = new (window as any).EyeDropper()
|
| 222 |
+
eyeDropper.open().then((result: { sRGBHex: string }) => {
|
| 223 |
+
const tColor = tinycolor(result.sRGBHex)
|
| 224 |
+
hue.value = tColor.toHsl().h
|
| 225 |
+
color.value = tColor.toRgb()
|
| 226 |
+
|
| 227 |
+
message.closeAll()
|
| 228 |
+
updateRecentColorsCache()
|
| 229 |
+
}).catch(() => {
|
| 230 |
+
message.closeAll()
|
| 231 |
+
})
|
| 232 |
+
}
|
| 233 |
+
|
| 234 |
+
// 基于 Canvas 的自定义取色吸管
|
| 235 |
+
const customEyeDropper = () => {
|
| 236 |
+
const targetRef: HTMLElement | null = document.querySelector('.canvas')
|
| 237 |
+
if (!targetRef) return
|
| 238 |
+
|
| 239 |
+
const maskRef = document.createElement('div')
|
| 240 |
+
maskRef.style.cssText = 'position: fixed; top: 0; left: 0; bottom: 0; right: 0; z-index: 9999; cursor: wait;'
|
| 241 |
+
document.body.appendChild(maskRef)
|
| 242 |
+
|
| 243 |
+
const colorBlockRef = document.createElement('div')
|
| 244 |
+
colorBlockRef.style.cssText = 'position: absolute; top: -100px; left: -100px; width: 16px; height: 16px; border: 1px solid #000; z-index: 999'
|
| 245 |
+
maskRef.appendChild(colorBlockRef)
|
| 246 |
+
|
| 247 |
+
const { left, top, width, height } = targetRef.getBoundingClientRect()
|
| 248 |
+
|
| 249 |
+
const filter = (node: HTMLElement) => {
|
| 250 |
+
if (node.tagName && node.tagName.toUpperCase() === 'FOREIGNOBJECT') return false
|
| 251 |
+
if (node.classList && node.classList.contains('operate')) return false
|
| 252 |
+
return true
|
| 253 |
+
}
|
| 254 |
+
|
| 255 |
+
toCanvas(targetRef, { filter, fontEmbedCSS: '', width, height, canvasWidth: width, canvasHeight: height, pixelRatio: 1 }).then(canvasRef => {
|
| 256 |
+
canvasRef.style.cssText = `position: absolute; top: ${top}px; left: ${left}px; cursor: crosshair;`
|
| 257 |
+
maskRef.style.cursor = 'default'
|
| 258 |
+
maskRef.appendChild(canvasRef)
|
| 259 |
+
|
| 260 |
+
const ctx = canvasRef.getContext('2d')
|
| 261 |
+
if (!ctx) return
|
| 262 |
+
|
| 263 |
+
let currentColor = ''
|
| 264 |
+
const handleMousemove = (e: MouseEvent) => {
|
| 265 |
+
const x = e.x
|
| 266 |
+
const y = e.y
|
| 267 |
+
|
| 268 |
+
const mouseX = x - left
|
| 269 |
+
const mouseY = y - top
|
| 270 |
+
|
| 271 |
+
const [r, g, b, a] = ctx.getImageData(mouseX, mouseY, 1, 1).data
|
| 272 |
+
currentColor = `rgba(${r}, ${g}, ${b}, ${(a / 255).toFixed(2)})`
|
| 273 |
+
|
| 274 |
+
colorBlockRef.style.left = x + 10 + 'px'
|
| 275 |
+
colorBlockRef.style.top = y + 10 + 'px'
|
| 276 |
+
colorBlockRef.style.backgroundColor = currentColor
|
| 277 |
+
}
|
| 278 |
+
const handleMouseleave = () => {
|
| 279 |
+
currentColor = ''
|
| 280 |
+
colorBlockRef.style.left = '-100px'
|
| 281 |
+
colorBlockRef.style.top = '-100px'
|
| 282 |
+
colorBlockRef.style.backgroundColor = ''
|
| 283 |
+
}
|
| 284 |
+
const handleMousedown = (e: MouseEvent) => {
|
| 285 |
+
if (currentColor && e.button === 0) {
|
| 286 |
+
const tColor = tinycolor(currentColor)
|
| 287 |
+
hue.value = tColor.toHsl().h
|
| 288 |
+
color.value = tColor.toRgb()
|
| 289 |
+
|
| 290 |
+
updateRecentColorsCache()
|
| 291 |
+
}
|
| 292 |
+
document.body.removeChild(maskRef)
|
| 293 |
+
|
| 294 |
+
canvasRef.removeEventListener('mousemove', handleMousemove)
|
| 295 |
+
canvasRef.removeEventListener('mouseleave', handleMouseleave)
|
| 296 |
+
window.removeEventListener('mousedown', handleMousedown)
|
| 297 |
+
}
|
| 298 |
+
|
| 299 |
+
canvasRef.addEventListener('mousemove', handleMousemove)
|
| 300 |
+
canvasRef.addEventListener('mouseleave', handleMouseleave)
|
| 301 |
+
window.addEventListener('mousedown', handleMousedown)
|
| 302 |
+
}).catch(() => {
|
| 303 |
+
message.error('取色吸管初始化失败')
|
| 304 |
+
document.body.removeChild(maskRef)
|
| 305 |
+
})
|
| 306 |
+
}
|
| 307 |
+
</script>
|
| 308 |
+
|
| 309 |
+
<style lang="scss" scoped>
|
| 310 |
+
.color-picker {
|
| 311 |
+
position: relative;
|
| 312 |
+
width: 240px;
|
| 313 |
+
background: #fff;
|
| 314 |
+
user-select: none;
|
| 315 |
+
margin-bottom: -10px;
|
| 316 |
+
}
|
| 317 |
+
.picker-saturation-wrap {
|
| 318 |
+
width: 100%;
|
| 319 |
+
padding-bottom: 50%;
|
| 320 |
+
position: relative;
|
| 321 |
+
overflow: hidden;
|
| 322 |
+
}
|
| 323 |
+
.picker-controls {
|
| 324 |
+
display: flex;
|
| 325 |
+
}
|
| 326 |
+
.picker-sliders {
|
| 327 |
+
padding: 4px 0;
|
| 328 |
+
flex: 1;
|
| 329 |
+
}
|
| 330 |
+
.picker-hue-wrap {
|
| 331 |
+
position: relative;
|
| 332 |
+
height: 10px;
|
| 333 |
+
}
|
| 334 |
+
.picker-alpha-wrap {
|
| 335 |
+
position: relative;
|
| 336 |
+
height: 10px;
|
| 337 |
+
margin-top: 4px;
|
| 338 |
+
overflow: hidden;
|
| 339 |
+
}
|
| 340 |
+
.picker-color-wrap {
|
| 341 |
+
width: 24px;
|
| 342 |
+
height: 24px;
|
| 343 |
+
position: relative;
|
| 344 |
+
margin-top: 4px;
|
| 345 |
+
margin-right: 4px;
|
| 346 |
+
outline: 1px dashed rgba($color: #666, $alpha: .12);
|
| 347 |
+
|
| 348 |
+
.checkerboard {
|
| 349 |
+
background-size: auto;
|
| 350 |
+
}
|
| 351 |
+
}
|
| 352 |
+
.picker-current-color {
|
| 353 |
+
@include absolute-0();
|
| 354 |
+
|
| 355 |
+
z-index: 2;
|
| 356 |
+
}
|
| 357 |
+
|
| 358 |
+
.picker-field {
|
| 359 |
+
display: flex;
|
| 360 |
+
margin-bottom: 8px;
|
| 361 |
+
|
| 362 |
+
.transparent {
|
| 363 |
+
width: 24px;
|
| 364 |
+
height: 24px;
|
| 365 |
+
margin-top: 4px;
|
| 366 |
+
margin-left: 8px;
|
| 367 |
+
position: relative;
|
| 368 |
+
cursor: pointer;
|
| 369 |
+
|
| 370 |
+
&::after {
|
| 371 |
+
content: '';
|
| 372 |
+
width: 26px;
|
| 373 |
+
height: 2px;
|
| 374 |
+
position: absolute;
|
| 375 |
+
top: 11px;
|
| 376 |
+
left: -1px;
|
| 377 |
+
transform: rotate(-45deg);
|
| 378 |
+
background-color: #f00;
|
| 379 |
+
}
|
| 380 |
+
|
| 381 |
+
.checkerboard {
|
| 382 |
+
background-size: auto;
|
| 383 |
+
}
|
| 384 |
+
}
|
| 385 |
+
|
| 386 |
+
.straw {
|
| 387 |
+
width: 24px;
|
| 388 |
+
height: 24px;
|
| 389 |
+
margin-top: 4px;
|
| 390 |
+
margin-left: 8px;
|
| 391 |
+
display: flex;
|
| 392 |
+
justify-content: center;
|
| 393 |
+
align-items: center;
|
| 394 |
+
font-size: 20px;
|
| 395 |
+
background-color: #f5f5f5;
|
| 396 |
+
outline: 1px solid #f1f1f1;
|
| 397 |
+
cursor: pointer;
|
| 398 |
+
}
|
| 399 |
+
.input {
|
| 400 |
+
flex: 1;
|
| 401 |
+
}
|
| 402 |
+
}
|
| 403 |
+
|
| 404 |
+
.picker-presets {
|
| 405 |
+
@include flex-grid-layout();
|
| 406 |
+
}
|
| 407 |
+
.picker-presets-color {
|
| 408 |
+
@include flex-grid-layout-children(10, 7%);
|
| 409 |
+
|
| 410 |
+
height: 0;
|
| 411 |
+
padding-bottom: 7%;
|
| 412 |
+
flex-shrink: 0;
|
| 413 |
+
position: relative;
|
| 414 |
+
cursor: pointer;
|
| 415 |
+
|
| 416 |
+
&.alpha {
|
| 417 |
+
background-image: url();
|
| 418 |
+
}
|
| 419 |
+
}
|
| 420 |
+
.picker-presets-color-content {
|
| 421 |
+
@include absolute-0();
|
| 422 |
+
}
|
| 423 |
+
.picker-gradient-presets {
|
| 424 |
+
@include flex-grid-layout();
|
| 425 |
+
}
|
| 426 |
+
.picker-gradient-col {
|
| 427 |
+
@include flex-grid-layout-children(10, 7%);
|
| 428 |
+
|
| 429 |
+
display: flex;
|
| 430 |
+
flex-direction: column;
|
| 431 |
+
}
|
| 432 |
+
.picker-gradient-color {
|
| 433 |
+
width: 100%;
|
| 434 |
+
height: 16px;
|
| 435 |
+
position: relative;
|
| 436 |
+
cursor: pointer;
|
| 437 |
+
}
|
| 438 |
+
|
| 439 |
+
.recent-colors-title {
|
| 440 |
+
font-size: 12px;
|
| 441 |
+
margin-bottom: 4px;
|
| 442 |
+
}
|
| 443 |
+
</style>
|
frontend/src/components/Contextmenu/MenuContent.vue
ADDED
|
@@ -0,0 +1,137 @@
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
| 1 |
+
<template>
|
| 2 |
+
<ul class="menu-content">
|
| 3 |
+
<template v-for="(menu, index) in menus" :key="menu.text || index">
|
| 4 |
+
<li
|
| 5 |
+
v-if="!menu.hide"
|
| 6 |
+
class="menu-item"
|
| 7 |
+
@click.stop="handleClickMenuItem(menu)"
|
| 8 |
+
:class="{'divider': menu.divider, 'disable': menu.disable}"
|
| 9 |
+
>
|
| 10 |
+
<div
|
| 11 |
+
class="menu-item-content"
|
| 12 |
+
:class="{
|
| 13 |
+
'has-children': menu.children,
|
| 14 |
+
'has-handler': menu.handler,
|
| 15 |
+
}"
|
| 16 |
+
v-if="!menu.divider"
|
| 17 |
+
>
|
| 18 |
+
<span class="text">{{menu.text}}</span>
|
| 19 |
+
<span class="sub-text" v-if="menu.subText && !menu.children">{{menu.subText}}</span>
|
| 20 |
+
|
| 21 |
+
<menu-content
|
| 22 |
+
class="sub-menu"
|
| 23 |
+
:menus="menu.children"
|
| 24 |
+
v-if="menu.children && menu.children.length"
|
| 25 |
+
:handleClickMenuItem="handleClickMenuItem"
|
| 26 |
+
/>
|
| 27 |
+
</div>
|
| 28 |
+
</li>
|
| 29 |
+
</template>
|
| 30 |
+
</ul>
|
| 31 |
+
</template>
|
| 32 |
+
|
| 33 |
+
<script lang="ts" setup>
|
| 34 |
+
import type { ContextmenuItem } from './types'
|
| 35 |
+
|
| 36 |
+
defineProps<{
|
| 37 |
+
menus: ContextmenuItem[]
|
| 38 |
+
handleClickMenuItem: (item: ContextmenuItem) => void
|
| 39 |
+
}>()
|
| 40 |
+
</script>
|
| 41 |
+
|
| 42 |
+
<style lang="scss" scoped>
|
| 43 |
+
$menuWidth: 180px;
|
| 44 |
+
$menuHeight: 30px;
|
| 45 |
+
$subMenuWidth: 120px;
|
| 46 |
+
|
| 47 |
+
.menu-content {
|
| 48 |
+
width: $menuWidth;
|
| 49 |
+
padding: 5px 0;
|
| 50 |
+
background: #fff;
|
| 51 |
+
border: 1px solid $borderColor;
|
| 52 |
+
box-shadow: $boxShadow;
|
| 53 |
+
border-radius: $borderRadius;
|
| 54 |
+
list-style: none;
|
| 55 |
+
margin: 0;
|
| 56 |
+
}
|
| 57 |
+
.menu-item {
|
| 58 |
+
padding: 0 20px;
|
| 59 |
+
color: #555;
|
| 60 |
+
font-size: 12px;
|
| 61 |
+
transition: all $transitionDelayFast;
|
| 62 |
+
white-space: nowrap;
|
| 63 |
+
height: $menuHeight;
|
| 64 |
+
line-height: $menuHeight;
|
| 65 |
+
background-color: #fff;
|
| 66 |
+
cursor: pointer;
|
| 67 |
+
|
| 68 |
+
&:not(.disable):hover > .menu-item-content > .sub-menu {
|
| 69 |
+
display: block;
|
| 70 |
+
}
|
| 71 |
+
|
| 72 |
+
&:not(.disable):hover > .has-children.has-handler::after {
|
| 73 |
+
transform: scale(1);
|
| 74 |
+
}
|
| 75 |
+
|
| 76 |
+
&:hover:not(.disable) {
|
| 77 |
+
background-color: rgba($color: $themeColor, $alpha: .2);
|
| 78 |
+
}
|
| 79 |
+
|
| 80 |
+
&.divider {
|
| 81 |
+
height: 1px;
|
| 82 |
+
overflow: hidden;
|
| 83 |
+
margin: 5px;
|
| 84 |
+
background-color: #e5e5e5;
|
| 85 |
+
line-height: 0;
|
| 86 |
+
padding: 0;
|
| 87 |
+
}
|
| 88 |
+
|
| 89 |
+
&.disable {
|
| 90 |
+
color: #b1b1b1;
|
| 91 |
+
cursor: no-drop;
|
| 92 |
+
}
|
| 93 |
+
}
|
| 94 |
+
.menu-item-content {
|
| 95 |
+
display: flex;
|
| 96 |
+
align-items: center;
|
| 97 |
+
justify-content: space-between;
|
| 98 |
+
position: relative;
|
| 99 |
+
|
| 100 |
+
&.has-children::before {
|
| 101 |
+
content: '';
|
| 102 |
+
display: inline-block;
|
| 103 |
+
width: 8px;
|
| 104 |
+
height: 8px;
|
| 105 |
+
border-width: 1px;
|
| 106 |
+
border-style: solid;
|
| 107 |
+
border-color: #666 #666 transparent transparent;
|
| 108 |
+
position: absolute;
|
| 109 |
+
right: 0;
|
| 110 |
+
top: 50%;
|
| 111 |
+
transform: translateY(-50%) rotate(45deg);
|
| 112 |
+
}
|
| 113 |
+
&.has-children.has-handler::after {
|
| 114 |
+
content: '';
|
| 115 |
+
display: inline-block;
|
| 116 |
+
width: 1px;
|
| 117 |
+
height: 24px;
|
| 118 |
+
background-color: rgba($color: #fff, $alpha: .3);
|
| 119 |
+
position: absolute;
|
| 120 |
+
right: 18px;
|
| 121 |
+
top: 3px;
|
| 122 |
+
transform: scale(0);
|
| 123 |
+
transition: transform $transitionDelay;
|
| 124 |
+
}
|
| 125 |
+
|
| 126 |
+
.sub-text {
|
| 127 |
+
opacity: 0.6;
|
| 128 |
+
}
|
| 129 |
+
.sub-menu {
|
| 130 |
+
width: $subMenuWidth;
|
| 131 |
+
position: absolute;
|
| 132 |
+
display: none;
|
| 133 |
+
left: 112%;
|
| 134 |
+
top: -6px;
|
| 135 |
+
}
|
| 136 |
+
}
|
| 137 |
+
</style>
|
frontend/src/components/Contextmenu/index.vue
ADDED
|
@@ -0,0 +1,80 @@
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
| 1 |
+
<template>
|
| 2 |
+
<div
|
| 3 |
+
class="mask"
|
| 4 |
+
@contextmenu.prevent="removeContextmenu()"
|
| 5 |
+
@mousedown.left="removeContextmenu()"
|
| 6 |
+
></div>
|
| 7 |
+
|
| 8 |
+
<div
|
| 9 |
+
class="contextmenu"
|
| 10 |
+
:style="{
|
| 11 |
+
left: style.left + 'px',
|
| 12 |
+
top: style.top + 'px',
|
| 13 |
+
}"
|
| 14 |
+
@contextmenu.prevent
|
| 15 |
+
>
|
| 16 |
+
<MenuContent
|
| 17 |
+
:menus="menus"
|
| 18 |
+
:handleClickMenuItem="handleClickMenuItem"
|
| 19 |
+
/>
|
| 20 |
+
</div>
|
| 21 |
+
</template>
|
| 22 |
+
|
| 23 |
+
<script lang="ts" setup>
|
| 24 |
+
import { computed } from 'vue'
|
| 25 |
+
import type { ContextmenuItem, Axis } from './types'
|
| 26 |
+
|
| 27 |
+
import MenuContent from './MenuContent.vue'
|
| 28 |
+
|
| 29 |
+
const props = defineProps<{
|
| 30 |
+
axis: Axis
|
| 31 |
+
el: HTMLElement
|
| 32 |
+
menus: ContextmenuItem[]
|
| 33 |
+
removeContextmenu: () => void
|
| 34 |
+
}>()
|
| 35 |
+
|
| 36 |
+
const style = computed(() => {
|
| 37 |
+
const MENU_WIDTH = 180
|
| 38 |
+
const MENU_HEIGHT = 30
|
| 39 |
+
const DIVIDER_HEIGHT = 11
|
| 40 |
+
const PADDING = 5
|
| 41 |
+
|
| 42 |
+
const { x, y } = props.axis
|
| 43 |
+
const menuCount = props.menus.filter(menu => !(menu.divider || menu.hide)).length
|
| 44 |
+
const dividerCount = props.menus.filter(menu => menu.divider).length
|
| 45 |
+
|
| 46 |
+
const menuWidth = MENU_WIDTH
|
| 47 |
+
const menuHeight = menuCount * MENU_HEIGHT + dividerCount * DIVIDER_HEIGHT + PADDING * 2
|
| 48 |
+
|
| 49 |
+
const screenWidth = document.body.clientWidth
|
| 50 |
+
const screenHeight = document.body.clientHeight
|
| 51 |
+
|
| 52 |
+
return {
|
| 53 |
+
left: screenWidth <= x + menuWidth ? x - menuWidth : x,
|
| 54 |
+
top: screenHeight <= y + menuHeight ? y - menuHeight : y,
|
| 55 |
+
}
|
| 56 |
+
})
|
| 57 |
+
|
| 58 |
+
const handleClickMenuItem = (item: ContextmenuItem) => {
|
| 59 |
+
if (item.disable) return
|
| 60 |
+
if (item.children && !item.handler) return
|
| 61 |
+
if (item.handler) item.handler(props.el)
|
| 62 |
+
props.removeContextmenu()
|
| 63 |
+
}
|
| 64 |
+
</script>
|
| 65 |
+
|
| 66 |
+
<style lang="scss">
|
| 67 |
+
.mask {
|
| 68 |
+
position: fixed;
|
| 69 |
+
left: 0;
|
| 70 |
+
top: 0;
|
| 71 |
+
width: 100vw;
|
| 72 |
+
height: 100vh;
|
| 73 |
+
z-index: 9998;
|
| 74 |
+
}
|
| 75 |
+
.contextmenu {
|
| 76 |
+
position: fixed;
|
| 77 |
+
z-index: 9999;
|
| 78 |
+
user-select: none;
|
| 79 |
+
}
|
| 80 |
+
</style>
|
frontend/src/components/Contextmenu/types.ts
ADDED
|
@@ -0,0 +1,14 @@
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
| 1 |
+
export interface ContextmenuItem {
|
| 2 |
+
text?: string
|
| 3 |
+
subText?: string
|
| 4 |
+
divider?: boolean
|
| 5 |
+
disable?: boolean
|
| 6 |
+
hide?: boolean
|
| 7 |
+
children?: ContextmenuItem[]
|
| 8 |
+
handler?: (el: HTMLElement) => void
|
| 9 |
+
}
|
| 10 |
+
|
| 11 |
+
export interface Axis {
|
| 12 |
+
x: number
|
| 13 |
+
y: number
|
| 14 |
+
}
|
frontend/src/components/Divider.vue
ADDED
|
@@ -0,0 +1,34 @@
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
| 1 |
+
<template>
|
| 2 |
+
<div :class="['divider', type]"
|
| 3 |
+
:style="{
|
| 4 |
+
margin: type === 'horizontal' ? `${margin >= 0 ? margin : 24}px 0` : `0 ${margin >= 0 ? margin : 8}px`
|
| 5 |
+
}"
|
| 6 |
+
></div>
|
| 7 |
+
</template>
|
| 8 |
+
|
| 9 |
+
<script lang="ts" setup>
|
| 10 |
+
withDefaults(defineProps<{
|
| 11 |
+
type?: 'horizontal' | 'vertical'
|
| 12 |
+
margin?: number
|
| 13 |
+
}>(), {
|
| 14 |
+
type: 'horizontal',
|
| 15 |
+
margin: -1,
|
| 16 |
+
})
|
| 17 |
+
</script>
|
| 18 |
+
|
| 19 |
+
<style lang="scss" scoped>
|
| 20 |
+
.divider {
|
| 21 |
+
&.horizontal {
|
| 22 |
+
width: 100%;
|
| 23 |
+
margin: 24px 0;
|
| 24 |
+
border-block-start: 1px solid rgba(5, 5, 5, .06);
|
| 25 |
+
}
|
| 26 |
+
&.vertical {
|
| 27 |
+
position: relative;
|
| 28 |
+
height: 1em;
|
| 29 |
+
display: inline-block;
|
| 30 |
+
margin: 0 8px;
|
| 31 |
+
border-inline-start: 1px solid rgba(5, 5, 5, .06);
|
| 32 |
+
}
|
| 33 |
+
}
|
| 34 |
+
</style>
|
frontend/src/components/Drawer.vue
ADDED
|
@@ -0,0 +1,126 @@
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
| 1 |
+
<template>
|
| 2 |
+
<Teleport to="body">
|
| 3 |
+
<Transition :name="`drawer-slide-${placement}`"
|
| 4 |
+
@afterLeave="contentVisible = false"
|
| 5 |
+
@before-enter="contentVisible = true"
|
| 6 |
+
>
|
| 7 |
+
<div :class="['drawer', placement]" v-show="visible" :style="{ width: props.width + 'px' }">
|
| 8 |
+
<div class="header">
|
| 9 |
+
<slot name="title"></slot>
|
| 10 |
+
<span class="close-btn" @click="emit('update:visible', false)"><IconClose /></span>
|
| 11 |
+
</div>
|
| 12 |
+
<div class="content" v-if="contentVisible" :style="contentStyle">
|
| 13 |
+
<slot></slot>
|
| 14 |
+
</div>
|
| 15 |
+
</div>
|
| 16 |
+
</Transition>
|
| 17 |
+
</Teleport>
|
| 18 |
+
</template>
|
| 19 |
+
|
| 20 |
+
<script lang="ts" setup>
|
| 21 |
+
import { computed, ref, type CSSProperties } from 'vue'
|
| 22 |
+
|
| 23 |
+
const props = withDefaults(defineProps<{
|
| 24 |
+
visible: boolean
|
| 25 |
+
width?: number
|
| 26 |
+
contentStyle?: CSSProperties
|
| 27 |
+
placement?: 'left' | 'right'
|
| 28 |
+
}>(), {
|
| 29 |
+
width: 320,
|
| 30 |
+
placement: 'right',
|
| 31 |
+
})
|
| 32 |
+
|
| 33 |
+
const emit = defineEmits<{
|
| 34 |
+
(event: 'update:visible', payload: boolean): void
|
| 35 |
+
}>()
|
| 36 |
+
|
| 37 |
+
const contentVisible = ref(false)
|
| 38 |
+
|
| 39 |
+
const contentStyle = computed(() => {
|
| 40 |
+
return {
|
| 41 |
+
width: props.width + 'px',
|
| 42 |
+
...(props.contentStyle || {})
|
| 43 |
+
}
|
| 44 |
+
})
|
| 45 |
+
</script>
|
| 46 |
+
|
| 47 |
+
<style lang="scss" scoped>
|
| 48 |
+
.drawer {
|
| 49 |
+
height: 100%;
|
| 50 |
+
position: fixed;
|
| 51 |
+
top: 0;
|
| 52 |
+
bottom: 0;
|
| 53 |
+
z-index: 5000;
|
| 54 |
+
background: #fff;
|
| 55 |
+
display: flex;
|
| 56 |
+
flex-direction: column;
|
| 57 |
+
|
| 58 |
+
&.left {
|
| 59 |
+
left: 0;
|
| 60 |
+
box-shadow: 3px 0 6px -4px rgba(0, 0, 0, 0.12), 9px 0 28px 8px rgba(0, 0, 0, 0.05);
|
| 61 |
+
}
|
| 62 |
+
&.right {
|
| 63 |
+
right: 0;
|
| 64 |
+
box-shadow: -3px 0 6px -4px rgba(0, 0, 0, 0.12), -9px 0 28px 8px rgba(0, 0, 0, 0.05);
|
| 65 |
+
}
|
| 66 |
+
}
|
| 67 |
+
|
| 68 |
+
.header {
|
| 69 |
+
height: 50px;
|
| 70 |
+
padding: 0 15px;
|
| 71 |
+
position: relative;
|
| 72 |
+
display: flex;
|
| 73 |
+
align-items: center;
|
| 74 |
+
|
| 75 |
+
.close-btn {
|
| 76 |
+
width: 20px;
|
| 77 |
+
height: 20px;
|
| 78 |
+
display: flex;
|
| 79 |
+
justify-content: center;
|
| 80 |
+
align-items: center;
|
| 81 |
+
position: absolute;
|
| 82 |
+
top: 15px;
|
| 83 |
+
right: 15px;
|
| 84 |
+
cursor: pointer;
|
| 85 |
+
}
|
| 86 |
+
}
|
| 87 |
+
.content {
|
| 88 |
+
padding: 0 15px;
|
| 89 |
+
overflow: auto;
|
| 90 |
+
flex: 1;
|
| 91 |
+
}
|
| 92 |
+
|
| 93 |
+
.drawer-slide-right-enter-active {
|
| 94 |
+
animation: drawer-slide-right-enter .25s both ease;
|
| 95 |
+
}
|
| 96 |
+
.drawer-slide-right-leave-active {
|
| 97 |
+
animation: drawer-slide-right-leave .25s both ease;
|
| 98 |
+
}
|
| 99 |
+
.drawer-slide-left-enter-active {
|
| 100 |
+
animation: drawer-slide-left-enter .25s both ease;
|
| 101 |
+
}
|
| 102 |
+
.drawer-slide-left-leave-active {
|
| 103 |
+
animation: drawer-slide-left-leave .25s both ease;
|
| 104 |
+
}
|
| 105 |
+
|
| 106 |
+
@keyframes drawer-slide-right-enter {
|
| 107 |
+
from {
|
| 108 |
+
transform: translateX(100%);
|
| 109 |
+
}
|
| 110 |
+
}
|
| 111 |
+
@keyframes drawer-slide-right-leave {
|
| 112 |
+
to {
|
| 113 |
+
transform: translateX(100%);
|
| 114 |
+
}
|
| 115 |
+
}
|
| 116 |
+
@keyframes drawer-slide-left-enter {
|
| 117 |
+
from {
|
| 118 |
+
transform: translateX(-100%);
|
| 119 |
+
}
|
| 120 |
+
}
|
| 121 |
+
@keyframes drawer-slide-left-leave {
|
| 122 |
+
to {
|
| 123 |
+
transform: translateX(-100%);
|
| 124 |
+
}
|
| 125 |
+
}
|
| 126 |
+
</style>
|
frontend/src/components/FileInput.vue
ADDED
|
@@ -0,0 +1,45 @@
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
| 1 |
+
<template>
|
| 2 |
+
<div class="file-input" @click="handleClick()">
|
| 3 |
+
<slot></slot>
|
| 4 |
+
<input
|
| 5 |
+
class="input"
|
| 6 |
+
type="file"
|
| 7 |
+
name="upload"
|
| 8 |
+
ref="inputRef"
|
| 9 |
+
:accept="accept"
|
| 10 |
+
@change="$event => handleChange($event)"
|
| 11 |
+
>
|
| 12 |
+
</div>
|
| 13 |
+
</template>
|
| 14 |
+
|
| 15 |
+
<script lang="ts" setup>
|
| 16 |
+
import { ref } from 'vue'
|
| 17 |
+
|
| 18 |
+
withDefaults(defineProps<{
|
| 19 |
+
accept?: string
|
| 20 |
+
}>(), {
|
| 21 |
+
accept: 'image/*',
|
| 22 |
+
})
|
| 23 |
+
|
| 24 |
+
const emit = defineEmits<{
|
| 25 |
+
(event: 'change', payload: FileList): void
|
| 26 |
+
}>()
|
| 27 |
+
|
| 28 |
+
const inputRef = ref<HTMLInputElement>()
|
| 29 |
+
|
| 30 |
+
const handleClick = () => {
|
| 31 |
+
if (!inputRef.value) return
|
| 32 |
+
inputRef.value.value = ''
|
| 33 |
+
inputRef.value.click()
|
| 34 |
+
}
|
| 35 |
+
const handleChange = (e: Event) => {
|
| 36 |
+
const files = (e.target as HTMLInputElement).files
|
| 37 |
+
if (files) emit('change', files)
|
| 38 |
+
}
|
| 39 |
+
</script>
|
| 40 |
+
|
| 41 |
+
<style lang="scss" scoped>
|
| 42 |
+
.input {
|
| 43 |
+
display: none;
|
| 44 |
+
}
|
| 45 |
+
</style>
|
frontend/src/components/FullscreenSpin.vue
ADDED
|
@@ -0,0 +1,71 @@
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
| 1 |
+
<template>
|
| 2 |
+
<div class="fullscreen-spin" :class="{ 'mask': mask }" v-if="loading">
|
| 3 |
+
<div class="spin">
|
| 4 |
+
<div class="spinner"></div>
|
| 5 |
+
<div class="text">{{tip}}</div>
|
| 6 |
+
</div>
|
| 7 |
+
</div>
|
| 8 |
+
</template>
|
| 9 |
+
|
| 10 |
+
<script lang="ts" setup>
|
| 11 |
+
withDefaults(defineProps<{
|
| 12 |
+
loading?: boolean
|
| 13 |
+
mask?: boolean
|
| 14 |
+
tip?: string
|
| 15 |
+
}>(), {
|
| 16 |
+
loading: false,
|
| 17 |
+
mask: true,
|
| 18 |
+
tip: '',
|
| 19 |
+
})
|
| 20 |
+
</script>
|
| 21 |
+
|
| 22 |
+
<style lang="scss" scoped>
|
| 23 |
+
.fullscreen-spin {
|
| 24 |
+
position: fixed;
|
| 25 |
+
top: 0;
|
| 26 |
+
bottom: 0;
|
| 27 |
+
left: 0;
|
| 28 |
+
right: 0;
|
| 29 |
+
z-index: 100;
|
| 30 |
+
display: flex;
|
| 31 |
+
justify-content: center;
|
| 32 |
+
align-items: center;
|
| 33 |
+
|
| 34 |
+
&.mask {
|
| 35 |
+
background-color: rgba($color: #f1f1f1, $alpha: .7);
|
| 36 |
+
}
|
| 37 |
+
}
|
| 38 |
+
.spin {
|
| 39 |
+
width: 200px;
|
| 40 |
+
height: 200px;
|
| 41 |
+
position: fixed;
|
| 42 |
+
top: 50%;
|
| 43 |
+
left: 50%;
|
| 44 |
+
margin-top: -100px;
|
| 45 |
+
margin-left: -100px;
|
| 46 |
+
display: flex;
|
| 47 |
+
flex-direction: column;
|
| 48 |
+
justify-content: center;
|
| 49 |
+
align-items: center;
|
| 50 |
+
}
|
| 51 |
+
.spinner {
|
| 52 |
+
width: 36px;
|
| 53 |
+
height: 36px;
|
| 54 |
+
border: 3px solid $themeColor;
|
| 55 |
+
border-top-color: transparent;
|
| 56 |
+
border-radius: 50%;
|
| 57 |
+
animation: spinner .8s linear infinite;
|
| 58 |
+
}
|
| 59 |
+
.text {
|
| 60 |
+
margin-top: 20px;
|
| 61 |
+
color: $themeColor;
|
| 62 |
+
}
|
| 63 |
+
@keyframes spinner {
|
| 64 |
+
0% {
|
| 65 |
+
transform: rotate(0deg);
|
| 66 |
+
}
|
| 67 |
+
100% {
|
| 68 |
+
transform: rotate(360deg);
|
| 69 |
+
}
|
| 70 |
+
}
|
| 71 |
+
</style>
|
frontend/src/components/GradientBar.vue
ADDED
|
@@ -0,0 +1,149 @@
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
| 1 |
+
<template>
|
| 2 |
+
<div class="gradient-bar">
|
| 3 |
+
<div class="bar" ref="barRef" :style="{ backgroundImage: gradientStyle }" @click="$event => addPoint($event)"></div>
|
| 4 |
+
<div class="point"
|
| 5 |
+
:class="{ 'active': index === i }"
|
| 6 |
+
v-for="(item, i) in points"
|
| 7 |
+
:key="item.pos + '-' + i"
|
| 8 |
+
:style="{
|
| 9 |
+
backgroundColor: item.color,
|
| 10 |
+
left: `calc(${item.pos}% - 5px)`,
|
| 11 |
+
}"
|
| 12 |
+
@mousedown.left="movePoint(i)"
|
| 13 |
+
@click.right="removePoint(i)"
|
| 14 |
+
></div>
|
| 15 |
+
</div>
|
| 16 |
+
</template>
|
| 17 |
+
|
| 18 |
+
<script lang="ts" setup>
|
| 19 |
+
import type { GradientColor } from '@/types/slides'
|
| 20 |
+
import { ref, computed, watchEffect } from 'vue'
|
| 21 |
+
|
| 22 |
+
const props = defineProps<{
|
| 23 |
+
value: GradientColor[]
|
| 24 |
+
index: number
|
| 25 |
+
}>()
|
| 26 |
+
|
| 27 |
+
const emit = defineEmits<{
|
| 28 |
+
(event: 'update:value', payload: GradientColor[]): void
|
| 29 |
+
(event: 'update:index', payload: number): void
|
| 30 |
+
}>()
|
| 31 |
+
|
| 32 |
+
const points = ref<GradientColor[]>([])
|
| 33 |
+
|
| 34 |
+
const barRef = ref<HTMLElement>()
|
| 35 |
+
|
| 36 |
+
watchEffect(() => {
|
| 37 |
+
points.value = props.value
|
| 38 |
+
if (props.index > props.value.length - 1) emit('update:index', 0)
|
| 39 |
+
})
|
| 40 |
+
|
| 41 |
+
const gradientStyle = computed(() => {
|
| 42 |
+
const list = points.value.map(item => `${item.color} ${item.pos}%`)
|
| 43 |
+
return `linear-gradient(to right, ${list.join(',')})`
|
| 44 |
+
})
|
| 45 |
+
|
| 46 |
+
const removePoint = (index: number) => {
|
| 47 |
+
if (props.value.length <= 2) return
|
| 48 |
+
|
| 49 |
+
let targetIndex = 0
|
| 50 |
+
|
| 51 |
+
if (index === props.index) {
|
| 52 |
+
targetIndex = (index - 1 < 0) ? 0 : index - 1
|
| 53 |
+
}
|
| 54 |
+
else if (props.index === props.value.length - 1) {
|
| 55 |
+
targetIndex = props.value.length - 2
|
| 56 |
+
}
|
| 57 |
+
|
| 58 |
+
const values = props.value.filter((item, _index) => _index !== index)
|
| 59 |
+
emit('update:index', targetIndex)
|
| 60 |
+
emit('update:value', values)
|
| 61 |
+
}
|
| 62 |
+
|
| 63 |
+
const movePoint = (index: number) => {
|
| 64 |
+
let isMouseDown = true
|
| 65 |
+
|
| 66 |
+
document.onmousemove = e => {
|
| 67 |
+
if (!isMouseDown) return
|
| 68 |
+
if (!barRef.value) return
|
| 69 |
+
|
| 70 |
+
let pos = Math.round((e.clientX - barRef.value.getBoundingClientRect().left) / barRef.value.clientWidth * 100)
|
| 71 |
+
if (pos > 100) pos = 100
|
| 72 |
+
if (pos < 0) pos = 0
|
| 73 |
+
|
| 74 |
+
points.value = points.value.map((item, _index) => {
|
| 75 |
+
if (_index === index) return { ...item, pos }
|
| 76 |
+
return item
|
| 77 |
+
})
|
| 78 |
+
}
|
| 79 |
+
document.onmouseup = () => {
|
| 80 |
+
isMouseDown = false
|
| 81 |
+
|
| 82 |
+
const point = points.value[index]
|
| 83 |
+
const _points = [...points.value]
|
| 84 |
+
_points.splice(index, 1)
|
| 85 |
+
|
| 86 |
+
let targetIndex = 0
|
| 87 |
+
for (let i = 0; i < _points.length; i++) {
|
| 88 |
+
if (point.pos > _points[i].pos) targetIndex = i + 1
|
| 89 |
+
}
|
| 90 |
+
|
| 91 |
+
_points.splice(targetIndex, 0, point)
|
| 92 |
+
|
| 93 |
+
emit('update:index', targetIndex)
|
| 94 |
+
emit('update:value', _points)
|
| 95 |
+
|
| 96 |
+
document.onmousemove = null
|
| 97 |
+
document.onmouseup = null
|
| 98 |
+
}
|
| 99 |
+
}
|
| 100 |
+
|
| 101 |
+
const addPoint = (e: MouseEvent) => {
|
| 102 |
+
if (props.value.length >= 6) return
|
| 103 |
+
if (!barRef.value) return
|
| 104 |
+
const pos = Math.round((e.clientX - barRef.value.getBoundingClientRect().left) / barRef.value.clientWidth * 100)
|
| 105 |
+
|
| 106 |
+
let targetIndex = 0
|
| 107 |
+
for (let i = 0; i < props.value.length; i++) {
|
| 108 |
+
if (pos > props.value[i].pos) targetIndex = i + 1
|
| 109 |
+
}
|
| 110 |
+
const color = props.value[targetIndex - 1] ? props.value[targetIndex - 1].color : props.value[targetIndex].color
|
| 111 |
+
const values = [...props.value]
|
| 112 |
+
values.splice(targetIndex, 0, { pos, color })
|
| 113 |
+
emit('update:index', targetIndex)
|
| 114 |
+
emit('update:value', values)
|
| 115 |
+
}
|
| 116 |
+
</script>
|
| 117 |
+
|
| 118 |
+
<style lang="scss" scoped>
|
| 119 |
+
.gradient-bar {
|
| 120 |
+
width: calc(100% - 10px);
|
| 121 |
+
height: 18px;
|
| 122 |
+
padding: 1px 0;
|
| 123 |
+
margin: 3px 0;
|
| 124 |
+
position: relative;
|
| 125 |
+
left: 5px;
|
| 126 |
+
|
| 127 |
+
.bar {
|
| 128 |
+
height: 16px;
|
| 129 |
+
border: 1px solid #d9d9d9;
|
| 130 |
+
}
|
| 131 |
+
.point {
|
| 132 |
+
width: 10px;
|
| 133 |
+
height: 18px;
|
| 134 |
+
background-color: #fff;
|
| 135 |
+
position: absolute;
|
| 136 |
+
top: 0;
|
| 137 |
+
border: 2px solid #fff;
|
| 138 |
+
outline: 1px solid #d9d9d9;
|
| 139 |
+
box-shadow: 0 0 2px 2px #d9d9d9;
|
| 140 |
+
border-radius: 1px;
|
| 141 |
+
cursor: pointer;
|
| 142 |
+
|
| 143 |
+
&.active {
|
| 144 |
+
outline: 1px solid $themeColor;
|
| 145 |
+
box-shadow: 0 0 2px 2px $themeColor;
|
| 146 |
+
}
|
| 147 |
+
}
|
| 148 |
+
}
|
| 149 |
+
</style>
|