From 875626c8d26da53ef80e64f7033be3e18e0705ed Mon Sep 17 00:00:00 2001 From: ray_wuhao <443547225@qq.com> Date: Sat, 11 Mar 2023 18:14:34 +0800 Subject: [PATCH] =?UTF-8?q?v3.1.3=EF=BC=8C=E8=A1=A5=E5=85=85RayTable?= =?UTF-8?q?=E5=85=A8=E5=B1=8F=E3=80=81=E5=B0=BA=E5=AF=B8=E4=BF=AE=E6=94=B9?= =?UTF-8?q?=E5=8A=9F=E8=83=BD?= MIME-Version: 1.0 Content-Type: text/plain; charset=UTF-8 Content-Transfer-Encoding: 8bit --- cfg.ts | 22 ++- src/components/RayLink/index.tsx | 10 +- .../src/components/TableAction/index.tsx | 67 ++++++---- .../src/components/TableScreenfull/index.tsx | 65 +++++++++ .../src/components/TableSetting/index.scss | 8 +- .../src/components/TableSetting/index.tsx | 3 +- .../src/components/TableSize/index.scss | 45 +++++++ .../src/components/TableSize/index.tsx | 125 ++++++++++++++++++ src/components/RayTable/src/index.tsx | 36 ++++- src/components/RayTable/src/type.ts | 2 + src/icons/adjustment.svg | 6 + src/layout/index.tsx | 30 ++++- src/router/index.ts | 1 + src/spin/index.tsx | 2 +- src/styles/theme.scss | 11 ++ src/types/cfg.ts | 1 + src/utils/element.ts | 24 +++- src/views/table/index.tsx | 14 +- vite-plugin/index.ts | 23 ++++ vite.config.ts | 6 +- 20 files changed, 431 insertions(+), 70 deletions(-) create mode 100644 src/components/RayTable/src/components/TableScreenfull/index.tsx create mode 100644 src/components/RayTable/src/components/TableSize/index.scss create mode 100644 src/components/RayTable/src/components/TableSize/index.tsx create mode 100644 src/icons/adjustment.svg create mode 100644 src/styles/theme.scss diff --git a/cfg.ts b/cfg.ts index deb67638..e9776aaa 100644 --- a/cfg.ts +++ b/cfg.ts @@ -1,6 +1,10 @@ import path from 'node:path' -import { HTMLTitlePlugin, buildOptions } from './vite-plugin/index' +import { + HTMLTitlePlugin, + buildOptions, + mixinCSSPlugin, +} from './vite-plugin/index' import type { AppConfigExport } from './src/types/cfg' @@ -20,6 +24,22 @@ const config: AppConfigExport = { url: '/dashboard', jumpType: 'station', }, + /** + * + * 预处理全局需要注入的 css 文件 + * + * 预设: + * - ./src/styles/mixins.scss + * - ./src/styles/setting.scss + * - ./src/styles/theme.scss + * + * 如果需要删除或者修改, 需要同步修改目录下的 css 文件 + */ + mixinCSS: mixinCSSPlugin([ + './src/styles/mixins.scss', + './src/styles/setting.scss', + './src/styles/theme.scss', + ]), /** * * 版权信息 diff --git a/src/components/RayLink/index.tsx b/src/components/RayLink/index.tsx index 3f413677..98df02d2 100644 --- a/src/components/RayLink/index.tsx +++ b/src/components/RayLink/index.tsx @@ -21,31 +21,31 @@ const RayLink = defineComponent({ key: 'yun-cloud-images', src: 'https://yunkuangao.com/', tooltip: '云图床', - icon: 'https://yunkuangao.com/images/20170801_005902048_iOS.md.jpg', + icon: 'https://yunkuangao.me/wp-content/uploads/2022/05/cropped-cropped-QQ%E5%9B%BE%E7%89%8720220511113928.jpg', }, { key: 'ray-js-note', src: 'https://note.youdao.com/s/ObWEe2BB', tooltip: 'Ray的前端学习笔记', - icon: 'https://yunkuangao.com/images/longmao.jpeg', + icon: 'https://usc1.contabostorage.com/c2e495d7890844d392e8ec0c6e5d77eb:image/longmao.jpeg', }, { key: 'ray-js-cover', src: 'https://note.youdao.com/s/IC8xKPdB', tooltip: 'Ray的面试题总结', - icon: 'https://yunkuangao.com/images/longmao.jpeg', + icon: 'https://usc1.contabostorage.com/c2e495d7890844d392e8ec0c6e5d77eb:image/longmao.jpeg', }, { key: 'ray-template-doc', src: 'https://xiaodaigua-ray.github.io/ray-template-doc/', tooltip: 'Ray Template Doc', - icon: 'https://yunkuangao.com/images/longmao.jpeg', + icon: 'https://usc1.contabostorage.com/c2e495d7890844d392e8ec0c6e5d77eb:image/longmao.jpeg', }, { key: 'ray-template-doc-out', src: 'https://ray-template.yunkuangao.com/', tooltip: 'Ray Template Doc (国内地址)', - icon: 'https://yunkuangao.com/images/longmao.jpeg', + icon: 'https://usc1.contabostorage.com/c2e495d7890844d392e8ec0c6e5d77eb:image/longmao.jpeg', }, ] diff --git a/src/components/RayTable/src/components/TableAction/index.tsx b/src/components/RayTable/src/components/TableAction/index.tsx index 36ab396a..319352fa 100644 --- a/src/components/RayTable/src/components/TableAction/index.tsx +++ b/src/components/RayTable/src/components/TableAction/index.tsx @@ -9,7 +9,7 @@ * @remark 今天也是元气满满撸代码的一天 */ -import { NPopconfirm, NSpace, NButton } from 'naive-ui' +import { NPopconfirm, NSpace, NButton, NPopover } from 'naive-ui' import RayIcon from '@/components/RayIcon/index' export type EmitterType = 'positive' | 'negative' @@ -65,6 +65,10 @@ const TableAction = defineComponent({ type: Number, default: 18, }, + popoverContent: { + type: String, + required: true, + }, }, emits: ['positive', 'negative'], setup(_, { emit }) { @@ -83,37 +87,44 @@ const TableAction = defineComponent({ }, render() { return ( - + {{ trigger: () => ( - - ), - default: () => this.tooltip, - action: () => ( - - - {this.negativeText} - - - {this.positiveText} - - + + {{ + trigger: () => ( + + ), + default: () => this.tooltip, + action: () => ( + + + {this.negativeText} + + + {this.positiveText} + + + ), + }} + ), + default: () => this.popoverContent, }} - + ) }, }) diff --git a/src/components/RayTable/src/components/TableScreenfull/index.tsx b/src/components/RayTable/src/components/TableScreenfull/index.tsx new file mode 100644 index 00000000..086e3c6b --- /dev/null +++ b/src/components/RayTable/src/components/TableScreenfull/index.tsx @@ -0,0 +1,65 @@ +/** + * + * @author Ray + * + * @date 2023-03-11 + * + * @workspace ray-template + * + * @remark 今天也是元气满满撸代码的一天 + */ + +import { NPopover } from 'naive-ui' +import RayIcon from '@/components/RayIcon/index' + +import screenfull from 'screenfull' + +import type { TableSettingProvider } from '@/components/RayTable/src/type' + +const TableScreenfull = defineComponent({ + name: 'TableScreenfull', + setup() { + const tableSettingProvider = inject( + 'tableSettingProvider', + {} as TableSettingProvider, + ) + + const rayTableUUID = computed(() => tableSettingProvider.rayTableUUID) + let currentTableIsFullscreen = screenfull.isFullscreen // 缓存当前是否处于全屏状态 + + const handleScreenfull = () => { + const el = document.getElementById(rayTableUUID.value) + + currentTableIsFullscreen = !currentTableIsFullscreen + + if (el && screenfull.isEnabled && currentTableIsFullscreen) { + screenfull.request(el) + } else { + screenfull.exit() + } + } + + return { + handleScreenfull, + } + }, + render() { + return ( + + {{ + trigger: () => ( + + ), + default: () => '全屏表格', + }} + + ) + }, +}) + +export default TableScreenfull diff --git a/src/components/RayTable/src/components/TableSetting/index.scss b/src/components/RayTable/src/components/TableSetting/index.scss index 23d8e6cb..b0f3a0e0 100644 --- a/src/components/RayTable/src/components/TableSetting/index.scss +++ b/src/components/RayTable/src/components/TableSetting/index.scss @@ -1,7 +1,3 @@ -$iconSpace: 5px; -$width: 140px; -$activedColor: #2080f0; - .ray-table__setting:hover { transform: rotate(180deg); transition: transform 0.3s var(--r-bezier); @@ -32,12 +28,12 @@ $activedColor: #2080f0; &.draggable-item--dark { &:hover { - background-color: rgba(255, 255, 255, 0.08); + background-color: $hoverDarkBackgroundColor; } } &:hover { - background-color: #e8f2fd; + background-color: $hoverLightBackgroundColor; & .draggable-item__d--icon { opacity: 1; diff --git a/src/components/RayTable/src/components/TableSetting/index.tsx b/src/components/RayTable/src/components/TableSetting/index.tsx index 63be1169..14e64ab5 100644 --- a/src/components/RayTable/src/components/TableSetting/index.tsx +++ b/src/components/RayTable/src/components/TableSetting/index.tsx @@ -10,7 +10,8 @@ */ import './index.scss' -import { NCard, NPopover, NEllipsis, NButton } from 'naive-ui' + +import { NCard, NPopover, NEllipsis } from 'naive-ui' import RayIcon from '@/components/RayIcon/index' import VueDraggable from 'vuedraggable' import { setupSettingOptions } from './hook' diff --git a/src/components/RayTable/src/components/TableSize/index.scss b/src/components/RayTable/src/components/TableSize/index.scss new file mode 100644 index 00000000..2968e449 --- /dev/null +++ b/src/components/RayTable/src/components/TableSize/index.scss @@ -0,0 +1,45 @@ +.ray-table__table-size { + padding: 0 !important; + + & .n-card__content { + padding: 0 !important; + margin: 0 !important; + + & .table-size__dropdown { + box-sizing: border-box; + padding: 4px 0; + background-color: transparent; + + & .table-size__dropdown-wrapper { + position: relative; + z-index: 1; + display: flex; + flex-direction: column; + + & .dropdown-item { + height: 34px; + line-height: 34px; + text-align: center; + cursor: pointer; + padding: 0 16px; + transition: background-color 0.3s var(--r-bezier), color 0.3s var(--r-bezier); + + &.dropdown-item--active, + &:hover { + background-color: $hoverLightBackgroundColor; + color: $activedColor; + } + } + } + } + } +} + +.ray-template--dark { + & .table-size__dropdown-wrapper { + & .dropdown-item:hover { + background-color: $hoverDarkBackgroundColor; + color: $activedColor; + } + } +} diff --git a/src/components/RayTable/src/components/TableSize/index.tsx b/src/components/RayTable/src/components/TableSize/index.tsx new file mode 100644 index 00000000..cfa373f8 --- /dev/null +++ b/src/components/RayTable/src/components/TableSize/index.tsx @@ -0,0 +1,125 @@ +/** + * + * @author Ray + * + * @date 2023-03-10 + * + * @workspace ray-template + * + * @remark 今天也是元气满满撸代码的一天 + */ + +import './index.scss' + +import { NPopover, NCard } from 'naive-ui' +import RayIcon from '@/components/RayIcon/index' + +import type { TableSettingProvider } from '@/components/RayTable/src/type' + +const TableSize = defineComponent({ + name: 'TableSize', + emits: ['changeSize'], + setup(_, { emit }) { + const tableSettingProvider = inject( + 'tableSettingProvider', + {} as TableSettingProvider, + ) + + const popoverShow = ref(false) + const currentSize = ref(tableSettingProvider.size) + const size = computed({ + get: () => tableSettingProvider.size, + set: (val) => { + currentSize.value = val + }, + }) + const sizeOptions = ref([ + { + label: '默认', + key: 'medium', + }, + { + label: '紧凑', + key: 'small', + }, + { + label: '宽松', + key: 'large', + }, + ]) + + const handleDropdownClick = (key: ComponentSize) => { + sizeOptions.value.forEach((curr) => { + if (curr.key === key) { + size.value = key + + popoverShow.value = false + + emit('changeSize', key) + } + }) + } + + return { + sizeOptions, + currentSize, + handleDropdownClick, + popoverShow, + } + }, + render() { + return ( + + {{ + trigger: () => ( + + {{ + trigger: () => ( + + ), + default: () => '表格密度', + }} + + ), + default: () => ( + +
+
+ {this.sizeOptions.map((curr) => ( +
+
{curr.label}
+
+ ))} +
+
+
+ ), + }} +
+ ) + }, +}) + +export default TableSize diff --git a/src/components/RayTable/src/index.tsx b/src/components/RayTable/src/index.tsx index 11f3f68b..c34203ee 100644 --- a/src/components/RayTable/src/index.tsx +++ b/src/components/RayTable/src/index.tsx @@ -15,7 +15,7 @@ * * 完全继承 `NDataTable`, 该组件继承 `NDataTable Props` 属性和方法 * - * 实现: 抬头, 操作栏, 右键菜单功能拓展, 输出 `excel` + * 实现: 抬头, 操作栏, 右键菜单功能拓展, 输出 `excel`, 表格尺寸调整 * * 右键菜单功能, 需要同时启用 `showMenu` 与配置菜单选项才能正常使用 * @@ -27,9 +27,12 @@ */ import './index.scss' + import { NDataTable, NCard, NDropdown, NDivider } from 'naive-ui' import TableSetting from './components/TableSetting/index' import TableAction from './components/TableAction/index' +import TableSize from './components/TableSize/index' +import TableScreenfull from './components/TableScreenfull/index' import props from './props' import print from 'print-js' @@ -47,6 +50,7 @@ const RayTable = defineComponent({ emits: ['update:columns', 'menuSelect', 'exportSuccess', 'exportError'], setup(props, { emit }) { const tableUUID = uuid() + const rayTableUUID = uuid() const modelRightClickMenu = computed(() => props.rightClickMenu) const modelColumns = computed({ get: () => props.columns, @@ -59,7 +63,7 @@ const RayTable = defineComponent({ y: 0, showMenu: false, }) - let prevRightClickIndex = -1 + let prevRightClickIndex = -1 // 缓存上次点击索引位置 const cssVars = computed(() => { const cssVar = { '--ray-table-header-space': props.tableHeaderSpace, @@ -67,6 +71,7 @@ const RayTable = defineComponent({ return cssVar }) + const tableSize = ref(props.size) /** * @@ -75,6 +80,8 @@ const RayTable = defineComponent({ provide('tableSettingProvider', { modelRightClickMenu, modelColumns, + size: props.size, + rayTableUUID, }) const handleColumnsUpdate = (arr: ActionOptions[]) => { @@ -175,8 +182,13 @@ const RayTable = defineComponent({ print(options) } + const handleChangeTableSize = (size: ComponentSize) => { + tableSize.value = size + } + return { tableUUID, + rayTableUUID, handleColumnsUpdate, ...toRefs(menuConfig), handleRowProps, @@ -184,11 +196,18 @@ const RayTable = defineComponent({ handleExportPositive, handlePrintPositive, cssVars, + handleChangeTableSize, + tableSize, } }, render() { return ( - + {{ default: () => ( <> @@ -196,6 +215,7 @@ const RayTable = defineComponent({ id={this.tableUUID} {...this.$props} rowProps={this.handleRowProps.bind(this)} + size={this.tableSize} > {{ empty: () => this.$slots?.empty?.(), @@ -227,6 +247,7 @@ const RayTable = defineComponent({ + {/* 表格尺寸调整 */} + + + {/* 全屏表格 */} + + {/* 表格列操作 */} export interface TableSettingProvider { modelRightClickMenu: RightClickMenu modelColumns: SettingOptions + size: ComponentSize + rayTableUUID: string } export interface ExportExcelProvider { diff --git a/src/icons/adjustment.svg b/src/icons/adjustment.svg new file mode 100644 index 00000000..bdca828e --- /dev/null +++ b/src/icons/adjustment.svg @@ -0,0 +1,6 @@ + + + \ No newline at end of file diff --git a/src/layout/index.tsx b/src/layout/index.tsx index 25250b4c..9ff51e65 100644 --- a/src/layout/index.tsx +++ b/src/layout/index.tsx @@ -1,12 +1,13 @@ import './index.scss' -import { NLayout, NLayoutContent } from 'naive-ui' +import { NLayout, NLayoutContent, NSpin } from 'naive-ui' import RayTransitionComponent from '@/components/RayTransitionComponent/index.vue' import LayoutMenu from './components/Menu/index' import SiderBar from './components/SiderBar/index' import MenuTag from './components/MenuTag/index' import { useSetting } from '@/store' +import { addClass, removeClass } from '@/utils/element' const Layout = defineComponent({ name: 'Layout', @@ -38,19 +39,42 @@ const Layout = defineComponent({ layout: { copyright }, } = __APP_CFG__ + watch( + () => themeValue.value, + (newData) => { + /** + * + * 初始化时根据当前主题色进行初始化 body 的 class 属性 + * + * 根据 themeValue 进行初始化 + */ + const body = document.body + const darkClassName = 'ray-template--dark' + const lightClassName = 'ray-template--light' + + newData + ? removeClass(body, lightClassName) + : removeClass(body, darkClassName) + + addClass(body, newData ? darkClassName : lightClassName) + }, + { + immediate: true, + }, + ) + return { windowHeight, modelReloadRoute, modelMenuTagSwitch, cssVarsRef, copyright, - themeValue, } }, render() { return (
diff --git a/src/router/index.ts b/src/router/index.ts index ba653965..ad174956 100644 --- a/src/router/index.ts +++ b/src/router/index.ts @@ -22,6 +22,7 @@ export const setupRouter = (app: App) => { /** * * @remark 路由切换启用顶部加载条 + * @remark 路由切换启用加载动画 */ export const setupRouterLoadingBar = () => { router.beforeEach(() => { diff --git a/src/spin/index.tsx b/src/spin/index.tsx index 8e3a0b64..ac34432f 100644 --- a/src/spin/index.tsx +++ b/src/spin/index.tsx @@ -40,7 +40,7 @@ const GlobalSpin = defineComponent({ > {{ default: () => this.$slots.default?.(), - description: () => this.$slots.description?.(), + description: () => 'loading...', }} ) diff --git a/src/styles/theme.scss b/src/styles/theme.scss new file mode 100644 index 00000000..dced985b --- /dev/null +++ b/src/styles/theme.scss @@ -0,0 +1,11 @@ +/** + * 明暗主题变量 + * + * 全局自定义组件使用变量 + */ + +$iconSpace: 5px; +$width: 140px; +$activedColor: #2d8cf0; +$hoverLightBackgroundColor: rgba(45, 140, 240, 0.1); +$hoverDarkBackgroundColor: rgba(45, 140, 240, 0.15); diff --git a/src/types/cfg.ts b/src/types/cfg.ts index 214c74c2..480cde4c 100644 --- a/src/types/cfg.ts +++ b/src/types/cfg.ts @@ -27,6 +27,7 @@ export interface Config { title: HTMLTitle copyright?: LayoutCopyright sideBarLogo?: LayoutSideBarLogo + mixinCSS?: string } export type Recordable = Record diff --git a/src/utils/element.ts b/src/utils/element.ts index 18548faa..c63bccc6 100644 --- a/src/utils/element.ts +++ b/src/utils/element.ts @@ -87,16 +87,26 @@ export const addClass = (element: HTMLElement, className: string) => { * @param className 所需删除className,可: 'xxx xxx' | 'xxx'格式删除 * * @remark 删除元素className(可: 'xxx xxx' | 'xxx'格式删除) + * @remark 如果输入值为 removeAllClass 则会删除该元素所有 class name */ -export const removeClass = (element: HTMLElement, className: string) => { +export const removeClass = ( + element: HTMLElement, + className: string | 'removeAllClass', +) => { if (element) { - const classes = className.trim().split(' ') + if (className === 'removeAllClass') { + const classList = element.classList - classes.forEach((item) => { - if (item) { - element.classList.remove(item) - } - }) + classList.forEach((curr) => classList.remove(curr)) + } else { + const classes = className.trim().split(' ') + + classes.forEach((item) => { + if (item) { + element.classList.remove(item) + } + }) + } } } diff --git a/src/views/table/index.tsx b/src/views/table/index.tsx index 1794fe93..44629220 100644 --- a/src/views/table/index.tsx +++ b/src/views/table/index.tsx @@ -180,6 +180,7 @@ const TableView = defineComponent({ 点击导出按钮即可导出 excel 表格,默认以列为表头输出 点击打印按钮即可打印该表格 右键菜单 + 全屏表格 ( <> - 搜索 + 搜索 重置 ), @@ -240,17 +241,6 @@ const TableView = defineComponent({ tableFooter: () => '表格的底部内容区域,有时候你可能会用上', }} - ) }, diff --git a/vite-plugin/index.ts b/vite-plugin/index.ts index ce50701b..1b57c8ba 100644 --- a/vite-plugin/index.ts +++ b/vite-plugin/index.ts @@ -156,3 +156,26 @@ export const useViteBuildPlugin = (options?: BuildOptions) => { return Object.assign({}, defaultPlugin, options) } + +/** + * + * @param options 预处理 css 文件 + * @returns additionalData string + * + * @remark 辅助处理需要全局注入的 css 样式文件, 会在构建期间完成注入 + */ +export const mixinCSSPlugin = (options?: string[]) => { + const defaultOptions = [] + + if (Array.isArray(options)) { + defaultOptions.push(...options) + } + + const mixisString = defaultOptions.reduce((pre, curr) => { + const temp = `@import "${curr}";` + + return (pre += temp) + }, '') + + return mixisString as string +} diff --git a/vite.config.ts b/vite.config.ts index 755f54d6..b6be6536 100644 --- a/vite.config.ts +++ b/vite.config.ts @@ -23,7 +23,8 @@ import config from './cfg' import pkg from './package.json' const { dependencies, devDependencies, name, version } = pkg -const { server, buildOptions, alias, title, copyright, sideBarLogo } = config +const { server, buildOptions, alias, title, copyright, sideBarLogo, mixinCSS } = + config /** * @@ -140,8 +141,7 @@ export default defineConfig(async ({ mode }) => { css: { preprocessorOptions: { scss: { - additionalData: - '@import "./src/styles/mixins.scss"; @import "./src/styles/setting.scss";', // 全局 `mixin` + additionalData: mixinCSS, }, }, modules: {