Merge pull request #13 from ray-template/main

sync main to dev
This commit is contained in:
yun 2023-09-28 11:41:59 +08:00 committed by GitHub
commit a82b28fb93
No known key found for this signature in database
GPG Key ID: 4AEE18F83AFDEB23
165 changed files with 2223 additions and 1187 deletions

View File

@ -13,5 +13,6 @@ visualizer.html
.env.*
src/locales/lang
.depcheckrc
src/components/RayChart/theme
src/components/RChart/theme
*.md
src/icons/*.svg

View File

@ -105,8 +105,13 @@ module.exports = {
'no-labels': 2, // 禁止标签声明
'no-lone-blocks': 2, // 禁止不必要的嵌套块
'no-multi-spaces': 1, // 禁止使用多余的空格
'no-multiple-empty-lines': [1, { max: 2 }], // 空行最多不能超过 `2` 行
'no-new-func': 1, // 禁止使用 `new Function`
'no-multiple-empty-lines': [
'error',
{
max: 2,
},
], // 空行最多不能超过 `2` 行
'no-new-func': 2, // 禁止使用 `new Function`
'no-new-object': 2, // 禁止使用 `new Object`
'no-new-require': 2, // 禁止使用 `new require`
'no-sparse-arrays': 2, // 禁止稀疏数组
@ -124,7 +129,6 @@ module.exports = {
'no-useless-call': 2, // 禁止不必要的 `call` 和 `apply`
'no-var': 'error', // 禁用 `var`
'no-with': 2, // 禁用 `with`
'no-undef': 0,
'use-isnan': 2, // 强制使用 isNaN 判断 NaN
'no-multi-assign': 2, // 禁止连续声明变量
'prefer-arrow-callback': 2, // 强制使用箭头函数作为回调
@ -143,15 +147,6 @@ module.exports = {
],
'vue/require-v-for-key': ['error'],
'vue/require-valid-default-prop': ['error'],
'no-use-before-define': [
'error',
{
functions: true,
classes: true,
variables: false,
allowNamedExports: false,
},
],
'vue/component-definition-name-casing': ['error', 'PascalCase'],
'vue/html-closing-bracket-newline': [
'error',

10
.vscode/settings.json vendored
View File

@ -9,5 +9,13 @@
"i18n-ally.displayLanguage": "zh-CN",
"i18n-ally.enabledFrameworks": ["vue", "react"],
"typescript.tsdk": "node_modules/typescript/lib",
"synthwave84.disableGlow": true
"alias-skip.mappings": {
"@": "/src",
"@use-utils": "/src/utils",
"@use-api": "/src/axios/api",
"@use-images": "/src/assets/images",
"@mock": "/mock"
},
"alias-skip.allowedsuffix": ["ts", "tsx"],
"alias-skip.rootpath": "package.json"
}

View File

@ -1,5 +1,69 @@
# CHANGE LOG
## 4.2.1
经过综合考虑,还是给模板增加 `cdn` 的配置。基于 `vite-plugin-cdn2` 插件实现。
### Feats
- 指令相关
- `v-copy` 指令将使用 `String` 强制转换传入的值
- 统一暴露节流、防抖指令的配置项类型 `import type { DebounceBindingOptions, ThrottleBindingOptions } from '@/directives/type'`
- 现在 `v-disabled` 指令生效时会降低一点元素的亮度
- `changeMenuModelValue` 方法添加节流锁,避免重复刷新 url 导致的一些问题
- 新增 `cdn`,缩减构建体积。如果不需要该配置,搜索 `viteCDNPlugin` 注释即可
## 4.2.0
针对分包做了全局的重新设计、调整。让包的名称更加语义化最重要的是重新抽离了一些全局可能常用的方法例如useI18n、useDayjs 等,在以前这些方法存放于对应的包中,其实这样很不合理,所以现在统一存放于 `src/hooks` 包中。并且该包以后统一存放 `hooks` 方法,并不是 `utils` 方法,做了一个本质的区分,所以 `xxxCopilot.ts` 文件中的方法并不会移动,维持存放在原有的模块下。
引入 `useGlobalVariable` 来管理全局变量。与 `pinia` `的使用场景不同useGlobalVariable` 是用于引入一些全局的响应式变量,这些变量不需要缓存,也不依赖任何插件。一个典型的应用是实现 `GlobalSpin`。使用该方法时,请谨慎使用,避免滥用,因为这些变量会被全局缓存且无法被回收。该方法存放的值,暂不支持缓存(如果有需要,可能后期会增加该功能)。
当项目插件或者需要配置项过多时候,会导致 `vite.config.ts` 文件变得异常臃肿。所以,在本次更新中,将插件的配置单独提出维护(`vite.plugin.confit.ts`)。系统的常用配置依旧在 `cfg.ts` 文件中。所以默认情况下,一般不需要修改 `vite.config.ts` 文件。
### Feats
- 新增 `useGlobalVariable` 管理全局变量(该变量可以是在注册插件之前被调用)
- `v-disbaled` 指令现在会尝试给元素添加 `disabled` 属性,如果该属性生效的话
- 注册指令操作现在不会中断程序执行,但是会抛出错误警告
- 抽离 `vite.plugin.confit.ts` 维护项目启动所需插件
## 4.1.9
### Feats
- 新增 RayQRCode 组件(二维码)
- 基于 awesome-qr 封装,继承其所有特性。并且拓展 状态、下载、自动更新等属性
- 自动卸载于释放内存,仅需要关注 text 内容填充
- 移除 qrcode.vue 依赖
- 更新 vue-hooks-plus 版本至 v1.8.2
- 移除 office 功能集成
- 统一包命名方式
- 更改 v-copy 实现细节(使用方式不变)
### Fixes
- 修复了一些小细节问题
## 4.1.8
### Feats
- 更新 `vite` 版本至 `v4.4.9`
- 更新 `vue-hooks-plus` 版本至 `v1.8.1`
- 更新了 RayTable 的一些事件的命名
- `RayChart` 组件做了一些调整
- 支持指定 observer 监听对象,默认为 chart 组件本身
- 默认开启 autoChangeTheme 功能
- 支持配置 throttleWait 节流等待时间,默认 500ms
- 支持通过配置 `desginConfig.echartTheme` 属性指定 `echart theme`。并且只需按照约定方式注册的主题,只需要指定主题名称,即可完成 `light` `dark` 两种主题指定
- RayChartInst 新增 dispose render 方法,允许手动渲染与卸载 chart 图
- 新增 animation 属性,如果为 true 则会强制触发渲染过渡动画。该配置受 `options.animation` 属性影响,如果该配置为 false 则不会启用过渡动画
- 移除反转色功能
- 新增图标页面
- 修改国际化图标
- 剔除无用代码,性能++++
## 4.1.7
### Feats

View File

@ -1,21 +0,0 @@
## 常见问题
### 路由
#### 缓存失效
> 如果出现缓存配置不生效的情况可以按照如下方法进行排查
- 查看 APP_KEEP_ALIVE setupKeepAlive 属性是否配置为 true
- 查看每个组件的 `name` 是否唯一,[`KeepAlive`](https://cn.vuejs.org/guide/built-ins/keep-alive.html) 组件重度依赖组件 `name` 作为唯一标识。详情可以查看官方文档
- 查看该页面的路由配置是否正确,比如:`path` 是否按照模板约定方式进行配置
#### 自动导入失败
> 模板采用自动导入路由模块方式。如果发现路由导入有误、或者导入报错,请查看文件命名是否有误。
### 国际化
#### 国际化切换错误、警告
> 模板二次封装 [`useI18n`](https://github.com/XiaoDaiGua-Ray/ray-template/blob/main/src/locales/useI18n.ts) 方法,首选该方法作为国际化语言切换方法。

121
README-US.md Normal file
View File

@ -0,0 +1,121 @@
<div align="center">
<a href="https://github.com/XiaoDaiGua-Ray/ray-template"> <img alt="Ray Template" width="200" height="200" src="https://usc1.contabostorage.com/c2e495d7890844d392e8ec0c6e5d77eb:alist/ray/ray.svg?sign=ZklU9Bh5b6oKp1X0LOhGwkx4g5mW4wk_w9Jt5zlZ5EQ=:0"> </a> <br> <br>
<a href="https://github.com/XiaoDaiGua-Ray/ray-template/blob/main/LICENSE"><img src="https://img.shields.io/github/license/XiaoDaiGua-Ray/ray-template" alt="LICENSE"></a>
</div>
<div align="center">
# Ray Template
A middle and backend template based on vite4.x & ts(x) & pinia & vue3.x
</div>
## ✨ Feature
- **Latest Technology Stack**Developed using front-end cutting-edge technologies such as vue3.x/vite4.x/pinia
- **TypeScript**The language for application-level JavaScript
- **App Theme**Configurable themes
- **Globalization**Built-in complete internationalization solution
- **Mock Data**Built-in Mock data scheme
- **Permissions**Built-in complete dynamic routing permission generation solution
- **Components**Secondary encapsulation of multiple commonly used components
- **Axios Request**Secondary encapsulation of the axios library, supporting functions such as cancellation, anti-shake, automatic repeat cancellation, etc.
- **Page Cache**Arbitrarily deep page cache
- **SVG**Built-in svg icon solution
- **Standalone Data Methods Views**Decoupled management of data, methods, and views allows for secondary development with confidence
## 🪄 Preview
- [Click to preview](https://xiaodaigua-ray.github.io/ray-template/#/)
- [Click to preview(Acceleration address)](https://ray-template.yunkuangao.com/#/)
## 🦾 Document
- [Document](https://xiaodaigua-ray.github.io/ray-template-doc/)
- [Document(Acceleration address)](https://ray-template.yunkuangao.com/ray-template-doc/)
## 🔋 Change Log
- [Change Log](https://github.com/XiaoDaiGua-Ray/xiaodaigua-ray.github.io/blob/main/CHANGELOG.md)
## 🪴 Prepare
- [Node](http://nodejs.org/) and [git](https://git-scm.com/) - Project development environment
- [Vite](https://vitejs.dev/) - Familiar with vite features
- [Vue3](https://v3.vuejs.org/) - Familiar with Vue3.x basic syntax
- [TypeScript](https://www.typescriptlang.org/) - Familiar with TypeScript basic syntax
- [Es6+](http://es6.ruanyifeng.com/) - Familiar with es6 basic syntax
- [Vue-Router-Next](https://next.router.vuejs.org/) - Familiar with the basic use of vue-router4.x
- [Naive-UI](https://www.naiveui.com) - UI basic usage
- [Mock.js](https://github.com/nuysoft/Mock) - Mockjs basic syntax
- [Pinia](https://pinia.vuejs.org/zh/introduction.html) - State manager pinia uses
- [TSX](https://github.com/vuejs/babel-plugin-jsx/blob/main/packages/babel-plugin-jsx/README-zh_CN.md) - TSX basic syntax
## 📦 Setup
### Get Project
```sh
# github
git clone https://github.com/XiaoDaiGua-Ray/ray-template.git
# If your download speed is very slow, you can switch to the proxy address below
git clone https://gh.yka.moe/https://github.com/XiaoDaiGua-Ray/ray-template.git
```
### Pull dependencies
```sh
pnpm i
```
### Startup project
```sh
pnpm dev
```
### Project packaging
```sh
pnpm build
```
### Preview project
```sh
pnpm preview
```
### Volumetric analysis
```sh
pnpm report
```
### Develop
Introduction and ease of use are the core ideas of this template. So you can safely delete all files under `views/demo` and `router/moduels/demo`, and you will have a clean project.
## 🪴 Project Activities
![Alt](https://repobeats.axiom.co/api/embed/fab6071297ab281913a42f07a2779b488cfd62b8.svg 'Repobeats analytics image')
### Contributor
Thanks for all their contributions 🐝!
<a href="https://github.com/XiaoDaiGua-Ray/ray-template/graphs/contributors">
<img src="https://contrib.rocks/image?repo=XiaoDaiGua-Ray/ray-template" />
</a>
## Browser Support
| [<img src="https://raw.githubusercontent.com/alrra/browser-logos/master/src/edge/edge_48x48.png" alt=" Edge" width="24px" height="24px" />](http://godban.github.io/browsers-support-badges/)</br>IE | [<img src="https://raw.githubusercontent.com/alrra/browser-logos/master/src/edge/edge_48x48.png" alt=" Edge" width="24px" height="24px" />](http://godban.github.io/browsers-support-badges/)</br>Edge | [<img src="https://raw.githubusercontent.com/alrra/browser-logos/master/src/firefox/firefox_48x48.png" alt="Firefox" width="24px" height="24px" />](http://godban.github.io/browsers-support-badges/)</br>Firefox | [<img src="https://raw.githubusercontent.com/alrra/browser-logos/master/src/chrome/chrome_48x48.png" alt="Chrome" width="24px" height="24px" />](http://godban.github.io/browsers-support-badges/)</br>Chrome | [<img src="https://raw.githubusercontent.com/alrra/browser-logos/master/src/safari/safari_48x48.png" alt="Safari" width="24px" height="24px" />](http://godban.github.io/browsers-support-badges/)</br>Safari |
| :--------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------: | :----------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------: | :---------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------: | :-----------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------: | :-----------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------: |
| not support | last 2 versions | last 2 versions | last 2 versions | last 2 versions |
## 📄 License
[MIT License](https://github.com/XiaoDaiGua-Ray/ray-template/blob/main/LICENSE) © 2022-PRESENT [Ray](https://github.com/XiaoDaiGua-Ray)

185
README.md
View File

@ -1,57 +1,49 @@
<div align="center"> <a href="https://github.com/XiaoDaiGua-Ray/ray-template"> <img alt="Ray Template" width="200" height="200" src="https://usc1.contabostorage.com/c2e495d7890844d392e8ec0c6e5d77eb:alist/ray/ray.svg?sign=ZklU9Bh5b6oKp1X0LOhGwkx4g5mW4wk_w9Jt5zlZ5EQ=:0"> </a> <br> <br>
<h1>Ray Template</h1>
<div align="center">
<a href="https://github.com/XiaoDaiGua-Ray/ray-template"> <img alt="Ray Template" width="200" height="200" src="https://usc1.contabostorage.com/c2e495d7890844d392e8ec0c6e5d77eb:alist/ray/ray.svg?sign=ZklU9Bh5b6oKp1X0LOhGwkx4g5mW4wk_w9Jt5zlZ5EQ=:0"> </a> <br> <br>
<a href="https://github.com/XiaoDaiGua-Ray/ray-template/blob/main/LICENSE"><img src="https://img.shields.io/github/license/XiaoDaiGua-Ray/ray-template" alt="LICENSE"></a>
</div>
<!-- ALL-CONTRIBUTORS-BADGE:START - Do not remove or modify this section -->
<div align="center">
[![All Contributors](https://img.shields.io/badge/all_contributors-1-orange.svg?style=flat-square)](#contributors-)
# Ray Template
<!-- ALL-CONTRIBUTORS-BADGE:END -->
简体中文 | [English](https://github.com/XiaoDaiGua-Ray/ray-template/blob/main/README-US.md)
## 前言
一个基于 vite4.x & ts(x) & pinia & vue3.x 的中后台模板
> 该项目模板采用 `vue3.x` `vite4.x` `pinia` `tsx` 进行开发。
> 使用 `naive ui` 作为组件库。
> 预设了最佳构建体验的配置与常用搬砖工具。意在提供一个简洁、快速上手的模板。
> 该模板不支持移动端设备。
</div>
## 感谢
## ✨ 特性
> 感谢 [`yun`](https://me.yka.moe/) 对于本人的支持。
## 预览地址
- [点击预览](https://xiaodaigua-ray.github.io/ray-template/#/)
- [点击预览(加速地址)](https://ray-template.yunkuangao.com/#/)
## 文档地址
- [文档](https://xiaodaigua-ray.github.io/ray-template-doc/)
- [文档(加速地址)](https://ray-template.yunkuangao.com/ray-template-doc/)
## 更新日志
- [日志](https://github.com/XiaoDaiGua-Ray/xiaodaigua-ray.github.io/blob/main/CHANGELOG.md)
## 常见问题
- [常见问题](https://github.com/XiaoDaiGua-Ray/ray-template/blob/main/COMMONPROBLEM.md)
## 特性
- **最新技术栈**:使用 Vue3.x/vite4.x 等前端前沿技术开发
- **最新技术栈**:使用 vue3.x/vite4.x/pinia 等前端前沿技术开发
- **TypeScript**:应用程序级 JavaScript 的语言
- **主题**:可配置的主题
- **国际化**:内置完善的国际化方案
- **Mock 数据**:内置 Mock 数据方案
- **权限**:内置完善的动态路由权限生成方案
- **组件**:二次封装了多个常用的组件
- **Axios 请求**:二次封装 axios 库
- **Axios 请求**:二次封装 axios 库,支持:取消、防抖、自动重复取消等功能
- **缓存**:任意深度页面缓存
- **SVG**:内置 svg icon 解决方案
- **独立的 Data Methods Views**:解耦管理的数据、方法、视图,放心二次开发
## 准备
## 🪄 预览地址
- [node](http://nodejs.org/) 和 [git](https://git-scm.com/) -项目开发环境
- [点击预览](https://xiaodaigua-ray.github.io/ray-template/#/)
- [点击预览(加速地址)](https://ray-template.yunkuangao.com/#/)
## 🦾 文档地址
- [文档](https://xiaodaigua-ray.github.io/ray-template-doc/)
- [文档(加速地址)](https://ray-template.yunkuangao.com/ray-template-doc/)
## 🔋 更新日志
- [更新日志](https://github.com/XiaoDaiGua-Ray/xiaodaigua-ray.github.io/blob/main/CHANGELOG.md)
## 🪴 准备
- [Node](http://nodejs.org/) 和 [git](https://git-scm.com/) - 项目开发环境
- [Vite](https://vitejs.dev/) - 熟悉 vite 特性
- [Vue3](https://v3.vuejs.org/) - 熟悉 Vue 基础语法
- [TypeScript](https://www.typescriptlang.org/) - 熟悉 TypeScript 基本语法
@ -62,16 +54,9 @@
- [Pinia](https://pinia.vuejs.org/zh/introduction.html) - 状态管理器 pinia 使用
- [TSX](https://github.com/vuejs/babel-plugin-jsx/blob/main/packages/babel-plugin-jsx/README-zh_CN.md) - tsx 基本语法
## 未来
## 📦 起步
> 根据个人时间空余情况,会不定时对该模板进行更新和迭代。希望将该工具的功能不断补全(虽然现在已经是足够日常开发和使用),将该模板打造为一个更加健全的中后台模板。如果你有好的想法和建议,可以直接联系我或者直接提 `issues` 即可。
## 提示
> 项目默认启用严格模式 `eslint`,但是由于 `vite-plugin-eslint` 插件优先级最高,所以如果出现自动导入类型错误提示,请优先解决其他问题。
> 建议开启 `vscode` 保存自动修复功能。
## 项目安装
### 获取项目
```sh
# github
@ -81,118 +66,58 @@ git clone https://github.com/XiaoDaiGua-Ray/ray-template.git
git clone https://gh.yka.moe/https://github.com/XiaoDaiGua-Ray/ray-template.git
```
## 拉取依赖
### 拉取依赖
```sh
# yarn
yarn
pnpm i
```
### 启动项目
```sh
# npm
npm install
pnpm dev
```
## 启动项目
### 项目打包
```sh
# yarn
yarn dev
pnpm build
```
### 预览项目
```sh
# npm
npm run dev
pnpm preview
```
## 项目打包
### 体积分析
```sh
# yarn
yarn build
pnpm report
```
```sh
# npm
### 开发
npm run build
```
简介、易上手是该模板的核心思路。所以你可以放心的直接删除 `views/demo` `router/moduels/demo` 下的所有文件,这样就是一个干净的项目了。
## 预览项目
## 🪴 项目活动
```sh
# yarn
![Alt](https://repobeats.axiom.co/api/embed/fab6071297ab281913a42f07a2779b488cfd62b8.svg 'Repobeats analytics image')
yarn preview
```
### 贡献者
```sh
# npm
感谢他们的所做的一切贡献 🐝
npm run preview
```
## 体积分析
```sh
# yarn
yarn report
```
```sh
# npm
npm run report
```
<a href="https://github.com/XiaoDaiGua-Ray/ray-template/graphs/contributors">
<img src="https://contrib.rocks/image?repo=XiaoDaiGua-Ray/ray-template" />
</a>
## 浏览器支持
> 仅支持现代浏览器,不支持 `IE`
| [<img src="https://raw.githubusercontent.com/alrra/browser-logos/master/src/edge/edge_48x48.png" alt=" Edge" width="24px" height="24px" />](http://godban.github.io/browsers-support-badges/)</br>IE | [<img src="https://raw.githubusercontent.com/alrra/browser-logos/master/src/edge/edge_48x48.png" alt=" Edge" width="24px" height="24px" />](http://godban.github.io/browsers-support-badges/)</br>Edge | [<img src="https://raw.githubusercontent.com/alrra/browser-logos/master/src/firefox/firefox_48x48.png" alt="Firefox" width="24px" height="24px" />](http://godban.github.io/browsers-support-badges/)</br>Firefox | [<img src="https://raw.githubusercontent.com/alrra/browser-logos/master/src/chrome/chrome_48x48.png" alt="Chrome" width="24px" height="24px" />](http://godban.github.io/browsers-support-badges/)</br>Chrome | [<img src="https://raw.githubusercontent.com/alrra/browser-logos/master/src/safari/safari_48x48.png" alt="Safari" width="24px" height="24px" />](http://godban.github.io/browsers-support-badges/)</br>Safari |
| :--------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------: | :----------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------: | :---------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------: | :-----------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------: | :-----------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------: |
| not support | last 2 versions | last 2 versions | last 2 versions | last 2 versions |
## 最后,希望大家搬砖愉快
## 📄 证书
## 贡献者
<!-- ALL-CONTRIBUTORS-LIST:START - Do not remove or modify this section -->
<!-- prettier-ignore-start -->
<!-- markdownlint-disable -->
<table>
<tbody>
<tr>
<td align="center" valign="top" width="14.28%"><a href="https://heartofyun.com"><img src="https://avatars.githubusercontent.com/u/40163747?v=4?s=100" width="100px;" alt="Cloud"/><br /><sub><b>Cloud</b></sub></a><br /><a href="#tool-yunkuangao" title="Tools">🔧</a></td>
</tr>
</tbody>
</table>
<!-- markdownlint-restore -->
<!-- prettier-ignore-end -->
<!-- ALL-CONTRIBUTORS-LIST:END -->
## Contributors ✨
Thanks goes to these wonderful people ([emoji key](https://allcontributors.org/docs/en/emoji-key)):
<!-- ALL-CONTRIBUTORS-LIST:START - Do not remove or modify this section -->
<!-- prettier-ignore-start -->
<!-- markdownlint-disable -->
<!-- markdownlint-restore -->
<!-- prettier-ignore-end -->
<!-- ALL-CONTRIBUTORS-LIST:END -->
This project follows the [all-contributors](https://github.com/all-contributors/all-contributors) specification. Contributions of any kind welcome!
## License
[MIT © Ray-2020](./LICENSE)
[MIT License](https://github.com/XiaoDaiGua-Ray/ray-template/blob/main/LICENSE) © 2022-PRESENT [Ray](https://github.com/XiaoDaiGua-Ray)

51
cfg.ts
View File

@ -38,15 +38,12 @@
import path from 'node:path'
import {
HTMLTitlePlugin,
buildOptions,
mixinCSSPlugin,
} from './vite-plugin/index'
import { APP_THEME } from './src/appConfig/designConfig'
import { PRE_LOADING_CONFIG, SIDE_BAR_LOGO } from './src/appConfig/appConfig'
import { htmlTitlePlugin, mixinCSSPlugin } from './vite-plugins/index'
import { APP_THEME } from './src/app-config/designConfig'
import { PRE_LOADING_CONFIG, SIDE_BAR_LOGO } from './src/app-config/appConfig'
import type { AppConfigExport } from '@/types/modules/cfg'
import type { BuildOptions } from 'vite'
const config: AppConfigExport = {
/** 公共基础路径配置, 如果为空则会默认以 '/' 填充 */
@ -82,7 +79,7 @@ const config: AppConfigExport = {
*
*
*/
title: HTMLTitlePlugin('Ray Template'),
title: htmlTitlePlugin(PRE_LOADING_CONFIG.title || 'Ray Template'),
/**
*
* HMR ()
@ -109,7 +106,39 @@ const config: AppConfigExport = {
*
*
*/
buildOptions: buildOptions,
buildOptions: (mode: string): BuildOptions => {
const outDirMap = {
test: 'dist/test-dist',
development: 'dist/development-dist',
production: 'dist/production-dist',
report: 'dist/report-dist',
}
const dirPath = outDirMap[mode] || 'dist/test-dist'
if (mode === 'production') {
return {
outDir: dirPath,
sourcemap: false,
terserOptions: {
compress: {
drop_console: true,
drop_debugger: true,
},
},
}
} else {
return {
outDir: dirPath,
sourcemap: true,
terserOptions: {
compress: {
drop_console: false,
drop_debugger: false,
},
},
}
}
},
/**
*
*
@ -135,6 +164,10 @@ const config: AppConfigExport = {
find: '@use-images',
replacement: path.resolve(__dirname, './src/assets/images'),
},
{
find: '@mock',
replacement: path.resolve(__dirname, './mock'),
},
],
}

View File

@ -1,26 +1,17 @@
import { defineMock } from 'vite-plugin-mock-dev-server'
import { pagination, stringify, response, array } from '@mock/shared/utils'
import { tableMock } from '@mock/shared/database'
import Mock from 'mockjs'
import { pagination, stringify, response } from '../shared/utils'
import { array } from '../shared/database'
export const getPersonList = defineMock({
url: '/api/list',
method: 'GET',
delay: 500,
response: (req, res) => {
const person = () => ({
id: Mock.Random.guid(),
address: Mock.Random.county(true),
email: Mock.Random.email(),
name: Mock.Random.cname(),
age: Mock.Random.integer(18, 60),
createDate: Mock.Random.date(),
})
const {
query: { page, pageSize, email },
} = req
let list = array(100).map(() => person())
let list = array(100).map(() => tableMock())
let length = list.length
if (!page || !pageSize) {

View File

@ -9,6 +9,22 @@
* @remark
*/
export function array(length: number) {
return new Array(length).fill(0)
import Mock from 'mockjs'
/**
*
* @param option
*
*
*/
export function tableMock(option?: object) {
return {
...option,
id: Mock.Random.guid(),
address: Mock.Random.county(true),
email: Mock.Random.email(),
name: Mock.Random.cname(),
age: Mock.Random.integer(18, 60),
createDate: Mock.Random.date(),
}
}

View File

@ -11,6 +11,10 @@
/* eslint-disable @typescript-eslint/no-explicit-any */
export function array(length: number) {
return new Array(length).fill(0)
}
/**
*
* @param pageCurrent

View File

@ -1,7 +1,7 @@
{
"name": "ray-template",
"private": false,
"version": "4.1.6",
"version": "4.2.1",
"type": "module",
"engines": {
"node": ">=16.0.0",
@ -25,6 +25,7 @@
},
"dependencies": {
"@vueuse/core": "^9.1.0",
"awesome-qr": "2.1.5-rc.0",
"axios": "^1.2.0",
"clipboard": "^2.0.11",
"crypto-js": "^4.1.1",
@ -37,11 +38,9 @@
"pinia": "^2.1.4",
"pinia-plugin-persistedstate": "^3.1.0",
"print-js": "^1.6.0",
"qrcode.vue": "^3.3.4",
"sass": "^1.54.3",
"screenfull": "^6.0.2",
"vue": "^3.3.4",
"vue-hooks-plus": "1.7.6",
"vue-hooks-plus": "1.8.2",
"vue-i18n": "^9.2.2",
"vue-router": "^4.2.4",
"vuedraggable": "^4.1.0",
@ -81,11 +80,13 @@
"postcss-px-to-viewport-8-plugin": "1.2.2",
"prettier": "^2.7.1",
"rollup-plugin-visualizer": "^5.8.3",
"sass": "1.54.3",
"svg-sprite-loader": "^6.0.11",
"typescript": "^5.0.2",
"unplugin-auto-import": "^0.15.0",
"unplugin-vue-components": "^0.25.1",
"vite": "^4.3.9",
"vite": "^4.4.9",
"vite-plugin-cdn2": "0.12.4",
"vite-plugin-compression": "^0.5.1",
"vite-plugin-ejs": "^1.6.4",
"vite-plugin-eslint": "1.8.1",

View File

@ -22,6 +22,8 @@ module.exports = {
unitPrecision: 3,
/** 指定需要转换成的视窗单位 */
viewportUnit: 'rem',
/** 制定字体转换单位 */
fontViewportUnit: 'rem',
/** 指定不转换为视窗单位的类 */
selectorBlackList: ['.ignore'],
/** 小于或等于 1px 不转换为视窗单位 */

View File

@ -22,7 +22,7 @@ import './index.scss'
import { NAvatar, NSpace } from 'naive-ui'
import { avatarProps, spaceProps } from 'naive-ui'
import { APP_CATCH_KEY } from '@/appConfig/appConfig'
import { APP_CATCH_KEY } from '@/app-config/appConfig'
import { getStorage } from '@/utils/cache'
import type { PropType } from 'vue'

View File

@ -27,25 +27,25 @@ const RayLink = defineComponent({
key: 'ray-js-note',
src: 'https://note.youdao.com/s/ObWEe2BB',
tooltip: 'Ray的前端学习笔记',
icon: 'https://usc1.contabostorage.com/c2e495d7890844d392e8ec0c6e5d77eb:image/longmao.jpeg',
icon: 'https://usc1.contabostorage.com/c2e495d7890844d392e8ec0c6e5d77eb:image/longmao.navigator.png',
},
{
key: 'ray-js-cover',
src: 'https://note.youdao.com/s/IC8xKPdB',
tooltip: 'Ray的面试题总结',
icon: 'https://usc1.contabostorage.com/c2e495d7890844d392e8ec0c6e5d77eb:image/longmao.jpeg',
icon: 'https://usc1.contabostorage.com/c2e495d7890844d392e8ec0c6e5d77eb:image/longmao.navigator.png',
},
{
key: 'ray-template-doc',
src: 'https://xiaodaigua-ray.github.io/ray-template-doc/',
tooltip: 'Ray Template Doc',
icon: 'https://usc1.contabostorage.com/c2e495d7890844d392e8ec0c6e5d77eb:image/longmao.jpeg',
icon: 'https://usc1.contabostorage.com/c2e495d7890844d392e8ec0c6e5d77eb:image/longmao.navigator.png',
},
{
key: 'ray-template-doc-out',
src: 'https://ray-template.yunkuangao.com/',
tooltip: 'Ray Template Doc (国内地址)',
icon: 'https://usc1.contabostorage.com/c2e495d7890844d392e8ec0c6e5d77eb:image/longmao.jpeg',
icon: 'https://usc1.contabostorage.com/c2e495d7890844d392e8ec0c6e5d77eb:image/longmao.navigator.png',
},
]

View File

@ -31,9 +31,9 @@ import {
import { useSetting } from '@/store'
import { naiveLocales } from '@/locales/helper'
const GlobalProvider = defineComponent({
export default defineComponent({
name: 'GlobalProvider',
setup() {
setup(_, { expose }) {
const settingStore = useSetting()
const modelPrimaryColorOverride = computed(
@ -54,6 +54,7 @@ const GlobalProvider = defineComponent({
configProviderProps: computed(() => ({
theme: modelThemeValue.value,
})),
notificationProviderProps: {},
},
)
@ -62,6 +63,8 @@ const GlobalProvider = defineComponent({
window.$loadingBar = loadingBar // 注入 `loadingBar`
window.$notification = notification // 注入 `notification`
expose()
return {
modelPrimaryColorOverride,
modelThemeValue,
@ -90,5 +93,3 @@ const GlobalProvider = defineComponent({
)
},
})
export default GlobalProvider

View File

@ -19,12 +19,14 @@
import { axiosCanceler } from '@/axios/helper/interceptor'
const AppRequestCanceler = defineComponent({
name: 'AppRequestCanceler',
setup() {
const AppRequestCancelerProvider = defineComponent({
name: 'AppRequestCancelerProvider',
setup(_, { expose }) {
onBeforeRouteUpdate(() => {
axiosCanceler.cancelAllRequest()
})
expose()
},
render() {
return (
@ -37,4 +39,4 @@ const AppRequestCanceler = defineComponent({
},
})
export default AppRequestCanceler
export default AppRequestCancelerProvider

View File

@ -20,7 +20,7 @@ import type { SettingState } from '@/store/modules/setting/type'
const AppStyleProvider = defineComponent({
name: 'AppStyleProvider',
setup() {
setup(_, { expose }) {
const settingStore = useSetting()
const { themeValue } = storeToRefs(settingStore)
@ -97,6 +97,8 @@ const AppStyleProvider = defineComponent({
immediate: true,
},
)
expose()
},
render() {
return <div class="app-style-provider"></div>

View File

@ -33,7 +33,11 @@ export const APP_KEEP_ALIVE: Readonly<AppKeepAlive> = {
maxKeepAliveLength: 5,
}
/** 首屏加载信息配置 */
/**
*
*
* title
*/
export const PRE_LOADING_CONFIG: PreloadingConfig = {
title: 'Ray Template',
tagColor: '#ff6700',

View File

@ -60,4 +60,10 @@ export const APP_THEME: AppTheme = {
* : <https://www.naiveui.com/zh-CN/dark/docs/customize-theme#%E4%BD%BF%E7%94%A8-peers-%E4%B8%BB%E9%A2%98%E5%8F%98%E9%87%8F>
*/
APP_NAIVE_UI_THEME_OVERRIDES: {},
/**
*
* echart
* json xxx-dark.json
*/
echartTheme: 'macarons',
}

View File

@ -29,13 +29,16 @@ import type { AppRawRequestConfig } from '@/axios/type'
/**
*
* setup 使
* vue component 使
* effect 使
* vue effect scope 使
*
* vue component 使
* vue effect 使
* @example
*
* // 请求函数
* const getUser = () => request({ url: 'http://localhost:3000/user' })
*
* // effect 中使用
* const { data } = useHookPlusRequest(getUser)
*/
function useRequest<

View File

@ -20,7 +20,7 @@
import { useAxiosInterceptor, axiosCanceler } from '@/axios/helper/interceptor'
import { appendRequestHeaders } from '@/axios/helper/axiosCopilot'
import { APP_CATCH_KEY } from '@/appConfig/appConfig'
import { APP_CATCH_KEY } from '@/app-config/appConfig'
import { getStorage } from '@/utils/cache'
import type {

View File

@ -17,7 +17,7 @@
*/
import axios from 'axios'
import { AXIOS_CONFIG } from '@/appConfig/requestConfig'
import { AXIOS_CONFIG } from '@/app-config/requestConfig'
import { useAxiosInterceptor, axiosCanceler } from '@/axios/helper/interceptor'
import {
setupResponseInterceptor,

View File

@ -13,7 +13,7 @@ import type {
ChartThemeRawArray,
ChartThemeRawModules,
LoadingOptions,
} from '@/components/RayChart/type'
} from '@/components/RChart/type'
/**
*
@ -26,12 +26,12 @@ import type {
* 1.
* 2.
* 3. json
* 4. @/components/RayChart/theme json
* 4. @/components/RChart/theme json
*/
export const setupChartTheme = () => {
// 获取所有主题
const themeRawModules: Record<string, ChartThemeRawModules> =
import.meta.glob('@/components/RayChart/theme/**/*.json', {
import.meta.glob('@/components/RChart/theme/**/*.json', {
eager: true,
})
const regx = /\/([^/]+)\.json$/

View File

@ -5,4 +5,10 @@
outline: none;
box-sizing: border-box;
transition: width 0.35s var(--r-bezier);
& .ray-chart__container {
width: 100%;
height: 100%;
box-sizing: border-box;
}
}

View File

@ -36,30 +36,38 @@ import {
PictorialBarChart,
} from 'echarts/charts' // 系列类型(后缀都为 `SeriesOption`)
import { LabelLayout, UniversalTransition } from 'echarts/features' // 标签自动布局, 全局过渡动画等特性
import { CanvasRenderer } from 'echarts/renderers' // `echarts` 渲染器
import {
CanvasRenderer,
// SVGRenderer,
} from 'echarts/renderers' // `echarts` 渲染器
import { useSetting } from '@/store'
import { cloneDeep, throttle } from 'lodash-es'
import { on, off, completeSize } from '@/utils/element'
import { call } from '@/utils/vue/index'
import { setupChartTheme, loadingOptions } from './helper'
import { LAYOUT_CONTENT_REF } from '@/appConfig/routerConfig'
import { APP_THEME } from '@/app-config/designConfig'
import type { PropType } from 'vue'
import type { EChartsInstance } from '@/types/modules/component'
import type { AnyFC, MaybeArray } from '@/types/modules/utils'
import type { DebouncedFunc } from 'lodash-es'
import type {
LoadingOptions,
AutoResize,
ChartTheme,
} from '@/components/RayChart/type'
import type { UseResizeObserverReturn } from '@vueuse/core'
} from '@/components/RChart/type'
import type {
UseResizeObserverReturn,
MaybeComputedElementRef,
MaybeElement,
} from '@vueuse/core'
import type { ECharts, EChartsCoreOption, SetOptionOpts } from 'echarts/core'
export type EChartsExtensionInstallRegisters = typeof CanvasRenderer
export type { RayChartInst } from './type'
const RayChart = defineComponent({
name: 'RayChart',
export default defineComponent({
name: 'RChart',
props: {
width: {
/**
@ -116,21 +124,19 @@ const RayChart = defineComponent({
type: Object as PropType<echarts.EChartsCoreOption>,
default: () => ({}),
},
success: {
onSuccess: {
/**
*
* chart
*
*
*
* () => EChartsInstance
* () => ECharts
*/
type: [Function, Array] as PropType<
MaybeArray<(e: EChartsInstance) => void>
>,
type: [Function, Array] as PropType<MaybeArray<(e: ECharts) => void>>,
default: null,
},
error: {
onError: {
/**
*
*
@ -148,13 +154,12 @@ const RayChart = defineComponent({
/**
*
*
*
* , `theme`
*
* 注意: 这个属性重度依赖此模板, .
* 注意: 这个属性重度依赖此模板
*/
type: Boolean,
default: false,
default: true,
},
use: {
/**
@ -180,15 +185,53 @@ const RayChart = defineComponent({
type: Object as PropType<LoadingOptions>,
default: () => loadingOptions(),
},
observer: {
/**
*
*
* autoResize
*
*/
type: Object as PropType<MaybeComputedElementRef<MaybeElement>>,
default: null,
},
throttleWait: {
/** 节流等待时间 */
type: Number,
default: 500,
},
animation: {
/** 是否强制启用渲染动画 */
type: Boolean,
default: true,
},
setChartOptions: {
/**
*
* options setOptions
*
*
* notMerge: false,
* lazyUpdate: true,
* silent: false,
* replaceMerge: [],
*
*
*/
type: Object as PropType<SetOptionOpts>,
default: () => ({}),
},
},
setup(props, { expose }) {
const settingStore = useSetting()
const { themeValue } = storeToRefs(settingStore)
const rayChartRef = ref<HTMLElement>() // `echart` 容器实例
const echartInstanceRef = ref<EChartsInstance>() // `echart` 拷贝实例, 解决直接使用响应式实例带来的问题
let echartInstance: EChartsInstance // `echart` 实例
let resizeThrottle: DebouncedFunc<AnyFC> // resize 防抖方法实例
let resizeOvserverReturn: UseResizeObserverReturn | undefined
const rayChartWrapperRef = ref<HTMLElement>()
const echartInstanceRef = ref<ECharts>() // `echart` 实例
let echartInstance: ECharts | null // `echart` 拷贝实例, 解决直接使用响应式实例带来的问题
let resizeThrottleReturn: DebouncedFunc<AnyFC> | null // resize 防抖方法实例
let resizeOvserverReturn: UseResizeObserverReturn | null
const { echartTheme } = APP_THEME
const cssVarsRef = computed(() => {
const cssVars = {
@ -198,9 +241,6 @@ const RayChart = defineComponent({
return cssVars
})
const modelLoadingOptions = computed(() =>
loadingOptions(props.loadingOptions),
)
/**
*
@ -237,10 +277,10 @@ const RayChart = defineComponent({
echarts.use([CanvasRenderer]) // 注册渲染器
try {
echarts.use(props.use)
echarts.use(props.use?.filter(Boolean))
} catch (e) {
console.error(
'Error: wrong property and method passed in extend attribute',
'register chart Core error: wrong property and method passed in extend attribute',
)
}
}
@ -253,10 +293,17 @@ const RayChart = defineComponent({
*
* ...
*/
const combineChartOptions = () => {
let options = cloneDeep(props.options)
const combineChartOptions = (ops: EChartsCoreOption) => {
let options = cloneDeep(ops)
const assign = (opts: object) => Object.assign({}, options, opts)
const assign = (opts: object) =>
Object.assign(
{
animation: true,
},
options,
opts,
)
if (props.showAria) {
options = assign({
@ -277,17 +324,16 @@ const RayChart = defineComponent({
* `echart`
*
*
*
* 使, `legend`
*/
const renderChart = (theme: ChartTheme = 'macarons') => {
const renderChart = (theme: ChartTheme = echartTheme) => {
/** 获取 dom 容器 */
const element = rayChartRef.value as HTMLElement
/** 获取配置项 */
const options = combineChartOptions()
const options = combineChartOptions(props.options)
/** 获取 dom 容器实际宽高 */
const { height, width } = element.getBoundingClientRect()
const { success, error } = props
const { onSuccess, onError } = props
try {
/** 注册主题 */
@ -305,19 +351,25 @@ const RayChart = defineComponent({
echartInstanceRef.value = echartInstance
/** 设置 options 配置项 */
options && echartInstance.setOption(options)
options && echartInstance.setOption({})
if (props.animation) {
setTimeout(() => {
options && echartInstance?.setOption(options)
})
}
/** 渲染成功回调 */
if (success) {
call(success, echartInstance)
if (onSuccess) {
call(onSuccess, echartInstance)
}
} catch (e) {
/** 渲染失败回调 */
if (error) {
call(error)
if (onError) {
call(onError)
}
console.error('RayChart render error: ', e)
console.error('RChart render error: ', e)
}
}
@ -329,9 +381,9 @@ const RayChart = defineComponent({
*/
const renderThemeChart = (bool?: boolean) => {
if (props.autoChangeTheme) {
bool ? renderChart('macarons-dark') : renderChart()
bool ? renderChart(`${echartTheme}-dark`) : renderChart()
return void 0
return
}
if (!props.theme) {
@ -357,10 +409,51 @@ const RayChart = defineComponent({
}
}
const mount = () => {
// 避免重复渲染
if (echartInstance?.getDom()) {
console.warn(
'RChart mount: There is a chart instance already initialized on the dom. Execution was interrupted',
)
return
}
if (props.autoChangeTheme) {
/** 注册 echarts */
renderThemeChart(themeValue.value)
} else {
props.theme ? renderChart(`${echartTheme}-dark`) : renderChart()
}
/** 注册事件 */
if (props.autoResize) {
resizeThrottleReturn = throttle(resizeChart, props.throttleWait)
/** 监听内容区域尺寸变化更新 chart */
resizeOvserverReturn = useResizeObserver(
props.observer || rayChartWrapperRef,
resizeThrottleReturn,
)
on(window, 'resize', resizeThrottleReturn)
}
}
const unmount = () => {
/** 卸载 echarts */
destroyChart()
/** 卸载事件柄 */
resizeThrottleReturn && off(window, 'resize', resizeThrottleReturn)
/** 注销防抖 */
resizeThrottleReturn?.cancel()
/** 注销 observer 监听 */
resizeOvserverReturn?.stop?.()
}
/** 监听全局主题变化, 然后重新渲染对应主题 echarts */
watch(
() => [themeValue.value],
([theme]) => {
() => themeValue.value,
(theme) => {
/**
*
* Q: 为什么需要重新卸载再渲染
@ -375,19 +468,19 @@ const RayChart = defineComponent({
},
)
/**
*
*
*
*
*/
watch(
() => props.showAria,
() => {
destroyChart()
/**
*
*
*
*
*/
if (props.autoChangeTheme || props.theme) {
themeValue.value ? renderChart('macarons-dark') : renderChart()
themeValue.value ? renderChart(`${echartTheme}-dark`) : renderChart()
} else {
renderChart()
}
@ -397,26 +490,41 @@ const RayChart = defineComponent({
/** 显示/隐藏加载动画 */
watch(
() => props.loading,
(newData) => {
newData
? echartInstance?.showLoading(modelLoadingOptions.value)
(ndata) => {
ndata
? echartInstance?.showLoading(props.loadingOptions)
: echartInstance?.hideLoading()
},
)
/** 监听 options 变化 */
if (props.watchOptions) {
watch(
() => props.watchOptions,
() => {
watch(
() => props.options,
(noptions) => {
if (props.watchOptions) {
/** 重新组合 options */
const options = combineChartOptions()
const options = combineChartOptions(noptions)
const setOpt = Object.assign({}, props.setChartOptions, {
notMerge: false,
lazyUpdate: true,
silent: false,
replaceMerge: [],
})
/** 如果 options 发生变动更新 echarts */
echartInstance?.setOption(options)
},
)
}
echartInstance?.setOption(options, setOpt)
}
},
{
deep: true,
},
)
expose({
echart: echartInstanceRef,
dispose: unmount,
render: mount,
})
onBeforeMount(async () => {
/** 注册 echarts 组件与渲染器 */
@ -425,53 +533,25 @@ const RayChart = defineComponent({
onMounted(() => {
nextTick(() => {
/** 注册 echarts */
if (props.autoChangeTheme) {
renderThemeChart(themeValue.value)
} else {
props.theme ? renderChart('macarons-dark') : renderChart()
}
/** 注册事件 */
if (props.autoResize) {
resizeThrottle = throttle(resizeChart, 500)
on(window, 'resize', resizeThrottle)
}
/** 监听内容区域尺寸变化更新 chart */
resizeOvserverReturn = useResizeObserver(
LAYOUT_CONTENT_REF.value as unknown as Ref<HTMLElement>,
resizeThrottle,
)
mount()
})
})
onBeforeUnmount(() => {
/** 卸载 echarts */
destroyChart()
/** 卸载事件柄 */
off(window, 'resize', resizeThrottle)
/** 注销防抖 */
resizeThrottle.cancel()
resizeOvserverReturn?.stop?.()
})
expose({
echart: echartInstanceRef,
unmount()
})
return {
rayChartRef,
cssVarsRef,
echartInstance: echartInstanceRef,
rayChartWrapperRef,
}
},
render() {
return (
<div class="ray-chart" style={[this.cssVarsRef]} ref="rayChartRef"></div>
<div class="ray-chart" style={[this.cssVarsRef]} ref="rayChartWrapperRef">
<div class="ray-chart__container" ref="rayChartRef"></div>
</div>
)
},
})
export default RayChart

View File

@ -9,6 +9,8 @@
* @remark
*/
import type { ECharts, EChartsCoreOption } from 'echarts/core'
export interface ChartThemeRawModules {
default: Record<string, UnknownObjectKey>
}
@ -41,3 +43,26 @@ export type AutoResize =
}
export type ChartTheme = 'macarons-dark' | string | object | 'macarons'
export interface RayChartInst {
/**
*
* echart
* 访 chart
*
* @default undefined
*/
echart: Ref<ECharts | undefined>
/**
*
* chart
* chart
*/
dispose: () => void
/**
*
* chart
* options props chart
*/
render: () => void
}

View File

@ -9,12 +9,22 @@
* @remark
*/
/**
*
* <https://www.naiveui.com/zh-CN/dark/components/grid>
*
*
* 使
*
* `NGrid` `NGridItem` , 使使 `NGridItem`
*/
import './index.scss'
import { collapseGridProps } from './props'
import { NCard, NGrid, NGridItem, NSpace } from 'naive-ui'
import RayIcon from '@/components/RayIcon'
import RayIcon from '@/components/RIcon'
import { call } from '@/utils/vue/index'
@ -76,14 +86,10 @@ const RayCollapseGrid = defineComponent({
>
{this.$slots.default?.()}
<NGridItem suffix class="ray-collapse-grid__suffix--btn">
{{
default: ({ overflow }: { overflow: boolean }) => (
<NSpace justify="end">
{this.$slots.action?.()}
{overflow ? this.CollapseIcon() : ''}
</NSpace>
),
}}
<NSpace justify="end">
{this.$slots.action?.()}
{this.CollapseIcon()}
</NSpace>
</NGridItem>
</NGrid>
),
@ -94,14 +100,3 @@ const RayCollapseGrid = defineComponent({
})
export default RayCollapseGrid
/**
*
* <https://www.naiveui.com/zh-CN/dark/components/grid>
*
*
*
* 使
*
* `NGrid` `NGridItem` , 使使 `NGridItem`
*/

View File

@ -0,0 +1,6 @@
import RayIframe from './src/index'
import type { RayIframeInst } from './src/index'
export default RayIframe
export type { RayIframeInst }

View File

@ -20,6 +20,10 @@ import type { PropType } from 'vue'
import type { MaybeArray } from '@/types/modules/utils'
import type { SpinProps } from 'naive-ui'
export interface RayIframeInst {
iframe: Ref<HTMLIFrameElement>
}
const RayIframe = defineComponent({
name: 'RayIframe',
props: {
@ -73,7 +77,7 @@ const RayIframe = defineComponent({
type: String,
default: null,
},
success: {
onSuccess: {
/**
*
* iframe
@ -84,7 +88,7 @@ const RayIframe = defineComponent({
>,
default: null,
},
error: {
onError: {
/**
*
* iframe
@ -119,20 +123,20 @@ const RayIframe = defineComponent({
const iframeLoadSuccess = (e: Event) => {
spinShow.value = false
const { success } = props
const { onSuccess } = props
if (success) {
call(success, iframeRef.value as HTMLIFrameElement, e)
if (onSuccess) {
call(onSuccess, iframeRef.value as HTMLIFrameElement, e)
}
}
const iframeLoadError = (e: Event) => {
spinShow.value = false
const { error } = props
const { onError } = props
if (error) {
call(error, e)
if (onError) {
call(onError, e)
}
}

View File

@ -0,0 +1,9 @@
import RayQRcode from './src/index'
export default RayQRcode
export type {
QRCodeStatus,
QRCodeLevel,
QRCodeRenderResponse,
QRCodeInst,
} from './src/type'

View File

@ -0,0 +1,26 @@
.ray-qrcode {
position: relative;
display: inline-flex;
& .ray-qrcode__error {
position: absolute;
width: 100%;
top: 50%;
left: 50%;
transform: translate(-50%, -50%);
z-index: 10;
background-color: rgba(0, 0, 0, 0.7);
width: 100%;
height: 100%;
@include flexCenter;
flex-direction: column;
gap: 18px 0;
& .ray-qrcode__error-content {
text-align: center;
font-size: 18px;
font-weight: 500;
color: #ffffff;
}
}
}

View File

@ -0,0 +1,188 @@
/**
*
* @author Ray <https://github.com/XiaoDaiGua-Ray>
*
* @date 2023-08-29
*
* @workspace ray-template
*
* @remark
*/
import './index.scss'
import { NButton, NSpin } from 'naive-ui'
import RayIcon from '@/components/RIcon/index'
import props from './props'
import { AwesomeQR } from 'awesome-qr'
import { isValueType, downloadBase64File } from '@use-utils/hook'
import { call } from '@/utils/vue/index'
import type { QRCodeRenderResponse, GIFBuffer } from './type'
const readGIFAsArrayBuffer = (url: string): Promise<GIFBuffer> => {
return new Promise((resolve, reject) => {
const xhr = new XMLHttpRequest()
xhr.responseType = 'blob'
xhr.onload = () => {
const reader = new FileReader()
reader.onloadend = () => {
resolve(reader.result)
}
reader.onerror = (e) => {
reject(e)
}
reader.onabort = (e) => {
reject(e)
}
reader.readAsArrayBuffer(xhr.response)
}
xhr.open('GET', url)
xhr.send()
})
}
export default defineComponent({
name: 'RayQRcode',
props,
setup(props, ctx) {
const { expose } = ctx
const qrcodeURL = ref<QRCodeRenderResponse>()
const spinOverrides = {
opacitySpinning: '0.1',
}
let gifBuffer: GIFBuffer
const getGIFImageByURL = async () => {
const { gifBackgroundURL } = props
if (!gifBackgroundURL) {
return
}
try {
gifBuffer = await readGIFAsArrayBuffer(gifBackgroundURL)
} catch (e) {
console.error(e)
}
}
const renderQRCode = () => {
const { gifBackground, ...ops } = props
new AwesomeQR({
...ops,
gifBackground: (gifBuffer as ArrayBuffer) ?? undefined,
})
.draw()
.then((res) => {
const { onSuccess } = props
if (onSuccess) {
call(onSuccess, res)
}
qrcodeURL.value = res
})
.catch((err) => {
const { onError } = props
if (onError) {
call(onError, err)
}
})
}
const errorActionClick = () => {
if (ctx.slots.errorAction) {
return
}
const { onReload } = props
if (onReload) {
call(onReload)
}
}
const downloadQRCode = (fileName?: string) => {
if (qrcodeURL.value && isValueType<string>(qrcodeURL.value, 'String')) {
downloadBase64File(
qrcodeURL.value,
fileName || new Date().getTime() + '.png',
)
}
}
watchEffect(() => {
if (props.watchText) {
nextTick().then(() => {
renderQRCode()
})
}
})
expose({
downloadQRCode,
})
onMounted(async () => {
await getGIFImageByURL()
renderQRCode()
})
return {
qrcodeURL,
spinOverrides,
errorActionClick,
}
},
render() {
return (
<div class="ray-qrcode">
<NSpin
show={this.status === 'loading'}
themeOverrides={this.spinOverrides}
description={this.loadingDescription}
>
<img src={this.qrcodeURL as string | undefined} />
</NSpin>
{this.status === 'error' ? (
<div class="ray-qrcode__error">
<div class="ray-qrcode__error-content">
{isValueType<string>(this.errorDescription, 'String')
? this.errorDescription
: () => this.errorDescription}
</div>
<div
class="ray-qrcode__error-btn"
onClick={this.errorActionClick.bind(this)}
>
{this.$slots.errorAction ? (
this.$slots.errorAction()
) : (
<>
<NButton text color="#ffffff">
{{
default: () => this.errorActionDescription,
icon: () => (
<RayIcon name="reload" size="16" color="#ffffff" />
),
}}
</NButton>
</>
)}
</div>
</div>
) : null}
</div>
)
},
})

View File

@ -0,0 +1,310 @@
/**
*
* @author Ray <https://github.com/XiaoDaiGua-Ray>
*
* @date 2023-08-29
*
* @workspace ray-template
*
* @remark
*/
import type { QRCodeStatus, QRCodeLevel } from './type'
import type { PropType, VNode } from 'vue'
import type { MaybeArray } from '@/types/modules/utils'
import type { Options } from 'awesome-qr'
const props = {
loadingDescription: {
/**
*
* Loading status description label
*
* @default undefined
*/
type: String,
},
watchText: {
/**
*
* Atuo watch QR code text
* If update text, then re-render QR code
*
* @default true
*/
type: Boolean,
default: true,
},
status: {
/**
*
* QR code status
*
* @default undefined
*/
type: String as PropType<QRCodeStatus>,
},
errorDescription: {
/**
*
* QR code error description label
*
* @default
*/
type: [String, Object] as PropType<string | VNode>,
default: '二维码已过期',
},
errorActionDescription: {
/**
*
* QR code error action description label
*
* @default
*/
type: String,
default: '重新加载',
},
text: {
/**
*
* Text to be encoded in the QR code
*/
type: String,
required: true,
},
size: {
/**
*
* Size of the QR code in pixel.
*
* @default 160
*/
type: Number,
default: 160,
},
margin: {
/**
*
* Size of margins around the QR code body in pixel.
*
* @default 12
*/
type: Number,
default: 12,
},
correctLevel: {
/**
*
* Error correction level of the QR code
* Accepts a value provided by _QRErrorCorrectLevel_
*
* @default 1
*/
type: Number as PropType<QRCodeLevel>,
default: 1,
validator: (value: unknown) => [0, 1, 2, 3].includes(value as number),
},
maskPattern: {
/**
*
* Specify the mask pattern to be used in QR code encoding
* Accepts a value provided by _QRMaskPattern_
*/
type: Number,
},
version: {
/**
*
* Specify the version to be used in QR code encoding
* Accepts an integer in range [1, 40]
*/
type: Number,
},
components: {
/**
*
* Options to control components in the QR code.
*
* @default {data:{scale...},...}
*/
type: Object as PropType<Options['components']>,
default: () => ({
data: {
scale: 1,
},
timing: {
scale: 1,
protectors: false,
},
alignment: {
scale: 1,
protectors: false,
},
cornerAlignment: {
scale: 1,
protectors: true,
},
}),
},
colorDark: {
/**
*
* Color of the blocks on the QR code
* Accepts a CSS &lt;color&gt;
*
* @default #000000
*/
type: String,
default: '#000000',
},
colorLight: {
/**
*
* Color of the blocks on the QR code
* Accepts a CSS &lt;color&gt;
*
* @default #ffffff
*/
type: String,
default: '#ffffff',
},
autoColor: {
/**
*
* Automatically calculate the _colorLight_ value from the QR code's background
*
* @default true
*/
type: Boolean,
default: true,
},
backgroundImage: {
/**
*
* Background image to be used in the QR code
* Accepts a `data:` string in web browsers or a Buffer in Node.js
*
* @default undefined
*/
type: String,
},
backgroundDimming: {
/**
*
* Color of the dimming mask above the background image
* Accepts a CSS &lt;color&gt;
*
* @default rgba(0, 0, 0, 0)
*/
type: String,
default: 'rgba(0, 0, 0, 0)',
},
gifBackgroundURL: {
/**
*
* GIF background image to be used in the QR code
*
* @default undefined
*/
type: String,
},
gifBackground: {
/**
*
* GIF background image to be used in the QR code
*
* @default undefined
*/
type: ArrayBuffer,
},
whiteMargin: {
/**
*
* Use a white margin instead of a transparent one which reveals the background of the QR code on margins
*
* @default true
*/
type: Boolean,
default: true,
},
logoImage: {
/**
*
* Logo image to be displayed at the center of the QR code
* Accepts a `data:` string in web browsers or a Buffer in Node.js
* When set to `undefined` or `null`, the logo is disabled
*
* @default undefined
*/
type: String,
},
logoScale: {
/**
*
* Ratio of the logo size to the QR code size
*
* @default 0.4
*/
type: Number,
default: 0.4,
},
logoMargin: {
/**
*
* Size of margins around the logo image in pixels
*
* @default 6
*/
type: Number,
default: 6,
},
logoCornerRadius: {
/**
* Corner radius of the logo image in pixels.
*
* @default 8
*/
type: Number,
default: 8,
},
dotScale: {
/**
*
* Ratio of the real size to the full size of the blocks.
* This can be helpful when you want to make more parts of the background visible.
*
* @default 1
*/
type: Number,
default: 1,
},
onSuccess: {
/**
*
* When the QR code is successfully generated, this callback is called
*/
type: [Function, Array] as PropType<
MaybeArray<(dataURL: ArrayBuffer | string | undefined) => void>
>,
default: null,
},
onError: {
/**
*
* When the QR code generation fails, this callback is called
*/
type: [Function, Array] as PropType<MaybeArray<(e: unknown) => void>>,
default: null,
},
onReload: {
/**
*
* When reload button is clicked, this callback is called
* This method will not execute if the errorAction slot is used
*/
type: [Function, Array] as PropType<MaybeArray<() => void>>,
default: null,
},
}
export default props

View File

@ -0,0 +1,28 @@
/**
*
* @author Ray <https://github.com/XiaoDaiGua-Ray>
*
* @date 2023-08-29
*
* @workspace ray-template
*
* @remark
*/
export type QRCodeStatus = 'error' | 'success' | 'loading'
export type QRCodeLevel = 0 | 1 | 2 | 3
export type QRCodeRenderResponse = string | ArrayBuffer | Buffer | undefined
export type QRCodeInst = {
/**
*
* @param fileName file name
*
* .png
*/
downloadQRCode: (fileName?: string) => void
}
export type GIFBuffer = string | ArrayBuffer | null

View File

@ -10,7 +10,7 @@
*/
import { NPopconfirm, NSpace, NButton, NPopover } from 'naive-ui'
import RayIcon from '@/components/RayIcon/index'
import RayIcon from '@/components/RIcon/index'
export type EmitterType = 'positive' | 'negative'

View File

@ -12,11 +12,11 @@
import './index.scss'
import { NPopover } from 'naive-ui'
import RayIcon from '@/components/RayIcon/index'
import RayIcon from '@/components/RIcon/index'
import screenfull from 'screenfull'
import type { TableSettingProvider } from '@/components/RayTable/src/type'
import type { TableSettingProvider } from '@/components/RTable/src/type'
const TableScreenfull = defineComponent({
name: 'TableScreenfull',

View File

@ -1,4 +1,4 @@
import type { ActionOptions } from '@/components/RayTable/src/type'
import type { ActionOptions } from '@/components/RTable/src/type'
export const setupSettingOptions = (options: ActionOptions[]) => {
const arr = options.map((curr) => {

View File

@ -19,7 +19,7 @@
import './index.scss'
import { NCard, NPopover, NEllipsis } from 'naive-ui'
import RayIcon from '@/components/RayIcon/index'
import RayIcon from '@/components/RIcon/index'
import VueDraggable from 'vuedraggable'
import { setupSettingOptions } from './hook'
@ -30,7 +30,7 @@ import type {
ActionOptions,
FixedType,
TableSettingFixedPopoverIcon,
} from '@/components/RayTable/src/type'
} from '@/components/RTable/src/type'
const TableSetting = defineComponent({
name: 'TableSetting',

View File

@ -12,11 +12,11 @@
import './index.scss'
import { NPopover, NCard } from 'naive-ui'
import RayIcon from '@/components/RayIcon/index'
import RayIcon from '@/components/RIcon/index'
import { call } from '@/utils/vue/index'
import type { TableSettingProvider } from '@/components/RayTable/src/type'
import type { TableSettingProvider } from '@/components/RTable/src/type'
import type { ComponentSize } from '@/types/modules/component'
import type { MaybeArray } from '@/types/modules/utils'

View File

@ -1,21 +1,10 @@
@keyframes scaleScreenfull {
0% {
transform: scale(1);
}
50% {
transform: scale(1.3);
}
100% {
transform: scale(1);
}
}
.ray-table {
& .ray-table-icon {
transition: transform 0.3s var(--r-bezier);
&:hover {
animation: scaleScreenfull 0.3s linear;
@include scaleAnimate();
animation: elementScale 0.3s linear;
animation-direction: alternate;
}
}

View File

@ -58,8 +58,7 @@ import type { ComponentSize } from '@/types/modules/component'
const RayTable = defineComponent({
name: 'RayTable',
props: props,
emits: ['update:columns', 'exportSuccess', 'exportError'],
setup(props, { emit, expose }) {
setup(props, { expose }) {
const rayTableInstance = ref<DataTableInst>()
const tableUUID = uuid(16) // 表格 id, 用于打印表格
@ -68,7 +67,15 @@ const RayTable = defineComponent({
const modelColumns = computed({
get: () => props.columns,
set: (arr) => {
emit('update:columns', arr)
const { onUpdateColumns, 'onUpdate:columns': _onUpdateColumns } = props
if (onUpdateColumns) {
call(onUpdateColumns, arr)
}
if (_onUpdateColumns) {
call(_onUpdateColumns, arr)
}
},
}) as unknown as WritableComputedRef<ActionOptions[]>
const menuConfig = reactive({
@ -76,7 +83,6 @@ const RayTable = defineComponent({
y: 0,
showMenu: false,
})
let prevRightClickIndex = -1 // 缓存上次点击索引位置
const cssVars = computed(() => {
const cssVar = {
'--ray-table-header-space': props.tableHeaderSpace,
@ -86,6 +92,7 @@ const RayTable = defineComponent({
})
const tableSize = ref(props.size)
const tableMethods = ref<Omit<DataTableInst, 'clearFilter'>>()
let prevRightClickIndex = -1 // 缓存上次点击索引位置
/** 注入相关属性 */
provide('tableSettingProvider', {
@ -111,17 +118,12 @@ const RayTable = defineComponent({
key: string | number,
option: DropdownOption,
) => {
const { onRightMenuClick, 'onUpdate:rightMenuClick': _onRightMenuClick } =
props
const { onRightMenuClick } = props
if (onRightMenuClick) {
call(onRightMenuClick, key, prevRightClickIndex, option)
}
if (_onRightMenuClick) {
call(_onRightMenuClick, key, prevRightClickIndex, option)
}
menuConfig.showMenu = false
}
@ -136,22 +138,24 @@ const RayTable = defineComponent({
const handleRowProps = (arr: ActionOptions, idx: number) => {
const interceptRowProps = props.rowProps?.(arr, idx)
const contextmenu = modelRightClickMenu.value.length
? (e: MouseEvent) => {
e.preventDefault()
prevRightClickIndex = idx
menuConfig.showMenu = false
nextTick().then(() => {
menuConfig.showMenu = true
menuConfig.x = e.clientX
menuConfig.y = e.clientY
})
}
: void 0
return {
...interceptRowProps,
onContextmenu: (e: MouseEvent) => {
e.preventDefault()
prevRightClickIndex = idx
menuConfig.showMenu = false
nextTick().then(() => {
menuConfig.showMenu = true
menuConfig.x = e.clientX
menuConfig.y = e.clientY
})
},
onContextmenu: contextmenu,
}
}
@ -164,6 +168,8 @@ const RayTable = defineComponent({
* `xlsx` , `file save`
*/
const handleExportPositive = async () => {
const { onExportSuccess, onExportError } = props
if (props.data.length && props.columns.length) {
try {
await exportFileToXLSX(
@ -174,9 +180,9 @@ const RayTable = defineComponent({
},
)
emit('exportSuccess')
onExportSuccess && call(onExportSuccess)
} catch (e) {
emit('exportError')
onExportError && call(onExportError)
}
}
}
@ -255,7 +261,6 @@ const RayTable = defineComponent({
}
},
render() {
console.log(this.action)
return (
<NCard
class="ray-table"
@ -277,21 +282,16 @@ const RayTable = defineComponent({
...this.$slots,
}}
</NDataTable>
{this.showMenu ? (
// 右键菜单
<NDropdown
show={this.showMenu}
placement="bottom-start"
trigger="manual"
x={this.x}
y={this.y}
options={this.modelRightClickMenu}
onClickoutside={() => (this.showMenu = false)}
onSelect={this.handleRightMenuSelect.bind(this)}
/>
) : (
''
)}
<NDropdown
show={this.showMenu}
placement="bottom-start"
trigger="manual"
x={this.x}
y={this.y}
options={this.modelRightClickMenu}
onClickoutside={() => (this.showMenu = false)}
onSelect={this.handleRightMenuSelect.bind(this)}
/>
</>
),
header: () => this.title || <div style="display: none;"></div>,

View File

@ -15,7 +15,7 @@ import type { PropType, VNode, VNodeChild } from 'vue'
import type { DropdownMixedOption } from './type'
import type PrintConfiguration from 'print-js'
import type { MaybeArray } from '@/types/modules/utils'
import type { DropdownOption } from 'naive-ui'
import type { DropdownOption, DataTableColumn } from 'naive-ui'
const rayTableProps = {
...dataTableProps, // 继承 `data table props`
@ -39,14 +39,6 @@ const rayTableProps = {
>,
default: null,
},
'onUpdate:rightMenuClick': {
type: [Function, Array] as PropType<
MaybeArray<
(key: string | number, index: number, option: DropdownOption) => void
>
>,
default: null,
},
title: {
/**
*
@ -77,16 +69,6 @@ const rayTableProps = {
type: Object as PropType<VNode>,
default: () => ({}),
},
showMenu: {
/**
*
*
*
*
*/
type: Boolean,
default: true,
},
exportTooltip: {
/**
*
@ -225,7 +207,30 @@ const rayTableProps = {
type: Boolean,
default: false,
},
/** 导出成功 */
onExportSuccess: {
type: [Function, Array] as PropType<MaybeArray<() => void>>,
default: null,
},
/** 导出失败 */
onExportError: {
type: [Function, Array] as PropType<MaybeArray<() => void>>,
default: null,
},
onUpdateColumns: {
type: [Function, Array] as PropType<
MaybeArray<(arr: DataTableColumn[]) => void>
>,
default: null,
},
'onUpdate:columns': {
type: [Function, Array] as PropType<
MaybeArray<(arr: DataTableColumn[]) => void>
>,
default: null,
},
} as const
export default rayTableProps
/**

View File

@ -24,7 +24,7 @@
</template>
<script lang="ts" setup>
import { useKeepAlive } from '@/store'
import { APP_KEEP_ALIVE } from '@/appConfig/appConfig'
import { APP_KEEP_ALIVE } from '@/app-config/appConfig'
import type { PropType } from 'vue'

View File

@ -1,3 +0,0 @@
import RayIframe from './src/index'
export default RayIframe

View File

@ -10,30 +10,9 @@
*/
import dayjs from 'dayjs'
import { DEFAULT_DAYJS_LOCAL, DAYJS_LOCAL_MAP } from '@/appConfig/localConfig'
import { DEFAULT_DAYJS_LOCAL } from '@/app-config/localConfig'
import 'dayjs/locale/zh-cn'
import type { DayjsLocal } from './type'
export const setupDayjs = () => {
dayjs.locale(DEFAULT_DAYJS_LOCAL)
}
/**
*
* dayjs hook
*
* :
* - locale: 切换 dayjs
*/
export const useDayjs = () => {
const locale = (key: DayjsLocal) => {
const mapkey = DAYJS_LOCAL_MAP[key]
mapkey ? dayjs.locale(mapkey) : dayjs.locale(DEFAULT_DAYJS_LOCAL)
}
return {
locale,
}
}

View File

@ -11,7 +11,6 @@
import { combineDirective } from './helper/combine'
import { forIn } from 'lodash-es'
import { isValueType } from '@/utils/hook'
import type { App } from 'vue'
import type { DirectiveModules } from '@/directives/type'
@ -33,17 +32,17 @@ export const setupDirectives = (app: App<Element>) => {
// 将所有的包提取出来(./modules/[file-name]/index.ts)
const directivesModules = combineDirective(directiveRawModules)
// 提取文件名(./modules/copy/index.ts => copy)
const reg = /(?<=modules\/).*(?=\/index\.ts)/
const regexExtractDirectiveName = /(?<=modules\/).*(?=\/index\.ts)/
// 匹配合法指令名称
const regexDirectiveName = /^([^-]+-)*[^-]+$/
forIn(directivesModules, (value, key) => {
const dname = key.match(reg)?.[0]
const dname = key.match(regexExtractDirectiveName)?.[0]
if (isValueType<string>(dname, 'String')) {
if (typeof dname === 'string' && regexDirectiveName.test(dname)) {
app.directive(dname, value?.())
} else {
throw new Error(
'directiveName is not string, please check your directive file name',
)
console.error(`[setupDirectives] ${dname} is not a valid directive name`)
}
})
}

View File

@ -20,35 +20,36 @@ import type { CopyElement } from './type'
import type { CustomDirectiveFC } from '@/directives/type'
const copyDirective: CustomDirectiveFC<CopyElement, string> = () => {
let clipboard: ClipboardJS | null
return {
mounted: (el, binding) => {
const value = binding.value
clipboard = new ClipboardJS(el, {
mounted: (el, { value }) => {
const clipboard = new ClipboardJS(el, {
text: () => String(value),
})
clipboard?.on('success', () => {
clipboard.on('success', () => {
window.$message.success('复制成功')
})
clipboard?.on('error', () => {
clipboard.on('error', () => {
window.$message.error('复制失败')
})
},
updated: (el, binding) => {
/** 其实这块代码写的挺蠢的, 但是我目前不知道怎么去优化, 阿巴阿巴阿巴 */
const value = binding.value
clipboard = new ClipboardJS(el, {
text: () => String(value),
})
el.$$clipboard = clipboard
},
beforeUnmount: () => {
clipboard?.destroy()
updated: (el, { value, oldValue }) => {
if (value !== oldValue) {
el.$$clipboard?.destroy()
clipboard = null
el.$$clipboard = new ClipboardJS(el, {
text: () => String(value),
})
}
},
beforeUnmount: (el: CopyElement) => {
if (el.$$clipboard) {
el.$$clipboard?.destroy()
el.$$clipboard = null
}
},
}
}

View File

@ -1,3 +1,6 @@
import type ClipboardJS from 'clipboard'
export interface CopyElement extends Element, UnknownObjectKey {
$value: string
$$clipboard: ClipboardJS | null
}

View File

@ -17,7 +17,6 @@
import { debounce } from 'lodash-es'
import { on, off } from '@use-utils/element'
import type { Directive } from 'vue'
import type { DebounceBindingOptions } from './type'
import type { AnyFC } from '@/types/modules/utils'
import type { DebouncedFunc } from 'lodash-es'
@ -30,16 +29,19 @@ const debounceDirective: CustomDirectiveFC<
let debounceFunction: DebouncedFunc<AnyFC> | null
return {
beforeMount: (el, binding) => {
const { func, trigger = 'click', wait = 500, options } = binding.value
beforeMount: (el, { value }) => {
const { func, trigger = 'click', wait = 500, options } = value
if (typeof func !== 'function') {
throw new Error('debounce directive value must be a function')
throw new TypeError('debounce directive value must be a function')
}
debounceFunction = debounce(func, wait, Object.assign({}, {}, options))
debounceFunction = debounce(func, wait, Object.assign({}, options))
on(el, trigger, debounceFunction)
},
beforeUnmount: (el, binding) => {
const { trigger = 'click' } = binding.value
beforeUnmount: (el, { value }) => {
const { trigger = 'click' } = value
if (debounceFunction) {
debounceFunction.cancel()

View File

@ -3,7 +3,7 @@ import type { AnyFC } from '@/types/modules/utils'
export interface DebounceBindingOptions {
func: AnyFC
trigger: string
wait: number
options: DebounceSettings
trigger?: string
wait?: number
options?: DebounceSettings
}

View File

@ -16,27 +16,35 @@
import { addClass, removeClass } from '@/utils/element'
import type { Directive } from 'vue'
import type { CustomDirectiveFC } from '@/directives/type'
const updateElementDisabledType = (el: HTMLElement, value: boolean) => {
if (el) {
const classes = 'ray-template__directive--disabled'
value ? addClass(el, classes) : removeClass(el, classes)
if (value) {
el.setAttribute('disabled', 'disabled')
addClass(el, classes)
} else {
el.removeAttribute('disabled')
removeClass(el, classes)
}
el?.setAttribute('disabled', value ? 'disabled' : '')
}
}
const disabledDirective: CustomDirectiveFC<HTMLElement, boolean> = () => {
return {
mounted: (el, binding) => {
const value = binding.value
mounted: (el, { value }) => {
updateElementDisabledType(el, value)
},
updated: (el, binding) => {
const value = binding.value
updated: (el, { value, oldValue }) => {
if (value === oldValue) {
return
}
updateElementDisabledType(el, value)
},

View File

@ -17,7 +17,6 @@
import { throttle } from 'lodash-es'
import { on, off } from '@use-utils/element'
import type { Directive } from 'vue'
import type { ThrottleBindingOptions } from './type'
import type { AnyFC } from '@/types/modules/utils'
import type { DebouncedFunc } from 'lodash-es'
@ -30,19 +29,19 @@ const throttleDirective: CustomDirectiveFC<
let throttleFunction: DebouncedFunc<AnyFC> | null
return {
beforeMount: (el, binding) => {
const { func, trigger = 'click', wait = 500, options } = binding.value
beforeMount: (el, { value }) => {
const { func, trigger = 'click', wait = 500, options } = value
if (typeof func !== 'function') {
throw new Error('throttle directive value must be a function')
throw new TypeError('throttle directive value must be a function')
}
throttleFunction = throttle(func, wait, Object.assign({}, {}, options))
throttleFunction = throttle(func, wait, Object.assign({}, options))
on(el, trigger, throttleFunction)
},
beforeUnmount: (el, binding) => {
const { trigger = 'click' } = binding.value
beforeUnmount: (el, { value }) => {
const { trigger = 'click' } = value
if (throttleFunction) {
throttleFunction.cancel()

View File

@ -3,7 +3,7 @@ import type { AnyFC } from '@/types/modules/utils'
export interface ThrottleBindingOptions {
func: AnyFC
trigger: string
wait: number
options: ThrottleSettings
trigger?: string
wait?: number
options?: ThrottleSettings
}

View File

@ -1,6 +1,9 @@
import type { Directive } from 'vue'
import type { App } from 'vue'
export type { DebounceBindingOptions } from './modules/debounce/type'
export type { ThrottleBindingOptions } from './modules/throttle/type'
export type CustomDirectiveFC<T, K> = () => Directive<T, K>
export interface DirectiveModules extends Object {

View File

@ -0,0 +1,18 @@
/**
*
* @author Ray <https://github.com/XiaoDaiGua-Ray>
*
* @date 2023-09-11
*
* @workspace ray-template
*
* @remark
*/
import {
setVariable,
getVariable,
globalVariableToRefs,
} from './useGlobalVariable'
export { setVariable, getVariable, globalVariableToRefs }

View File

@ -0,0 +1,40 @@
/**
*
* @author Ray <https://github.com/XiaoDaiGua-Ray>
*
* @date 2023-09-11
*
* @workspace ray-template
*
* @remark
*/
/**
*
* pinia 使
*
*
*
*/
/** 全局响应式变量 */
const variableState = reactive({
globalSpinning: false,
})
type VariableStateKey = keyof typeof variableState
export function setVariable(
key: VariableStateKey,
value: (typeof variableState)[VariableStateKey],
) {
variableState[key] = value
}
export function getVariable(key: VariableStateKey) {
return variableState[key] as (typeof variableState)[VariableStateKey]
}
export function globalVariableToRefs<K extends VariableStateKey>(key: K) {
return toRef<typeof variableState, K>(variableState, key)
}

16
src/hooks/web/index.ts Normal file
View File

@ -0,0 +1,16 @@
/**
*
* @author Ray <https://github.com/XiaoDaiGua-Ray>
*
* @date 2023-09-11
*
* @workspace ray-template
*
* @remark
*/
import { useI18n, t } from './useI18n'
import { useVueRouter } from '../web/useVueRouter'
import { useDayjs } from '../web/useDayjs'
export { useI18n, useVueRouter, useDayjs, t }

34
src/hooks/web/useDayjs.ts Normal file
View File

@ -0,0 +1,34 @@
/**
*
* @author Ray <https://github.com/XiaoDaiGua-Ray>
*
* @date 2023-09-11
*
* @workspace ray-template
*
* @remark
*/
import dayjs from 'dayjs'
import { DEFAULT_DAYJS_LOCAL, DAYJS_LOCAL_MAP } from '@/app-config/localConfig'
import type { DayjsLocal } from '@/dayjs/type'
/**
*
* dayjs hook
*
* :
* - locale: 切换 dayjs
*/
export const useDayjs = () => {
const locale = (key: DayjsLocal) => {
const mapkey = DAYJS_LOCAL_MAP[key]
mapkey ? dayjs.locale(mapkey) : dayjs.locale(DEFAULT_DAYJS_LOCAL)
}
return {
locale,
}
}

View File

@ -1,4 +1,15 @@
import { i18n } from './index'
/**
*
* @author Ray <https://github.com/XiaoDaiGua-Ray>
*
* @date 2023-09-11
*
* @workspace ray-template
*
* @remark
*/
import { i18n } from '@/locales/index'
import type { WritableComputedRef } from 'vue'

View File

@ -1,6 +1,7 @@
<svg t="1669090001868" class="icon" viewBox="0 0 1024 1024" version="1.1"
xmlns="http://www.w3.org/2000/svg" p-id="7911" width="200" height="200">
<svg xmlns="http://www.w3.org/2000/svg" aria-hidden="true" focusable="false" viewBox="0 0 24 24"
>
<path d="M0 0h24v24H0z" fill="none"></path>
<path
d="M918.954667 880.896c-0.618667-1.322667-154.688-334.378667-177.194667-382.421333-135.402667-288.917333-174.976-369.642667-196.821333-391.957334a31.829333 31.829333 0 0 0-13.013334-12.138666 32 32 0 0 0-42.944 14.293333L109.909333 865.706667a32 32 0 0 0 57.216 28.672l99.349334-198.421334h496.725333a49853.44 49853.44 0 0 1 97.536 211.605334 32.021333 32.021333 0 0 0 58.218667-26.666667zM521.002667 187.626667c39.850667 76.650667 126.698667 260.117333 212.458666 444.330666H298.517333L521.002667 187.626667z"
fill="currentColor" p-id="7912"></path>
d=" M12.87 15.07l-2.54-2.51.03-.03c1.74-1.94 2.98-4.17 3.71-6.53H17V4h-7V2H8v2H1v1.99h11.17C11.5 7.92 10.44 9.75 9 11.35 8.07 10.32 7.3 9.19 6.69 8h-2c.73 1.63 1.73 3.17 2.98 4.56l-5.09 5.02L4 19l5-5 3.11 3.11.76-2.04zM18.5 10h-2L12 22h2l1.12-3h4.75L21 22h2l-4.5-12zm-2.62 7l1.62-4.33L19.12 17h-3.24z "
></path>
</svg>

Before

Width:  |  Height:  |  Size: 722 B

After

Width:  |  Height:  |  Size: 477 B

View File

Before

Width:  |  Height:  |  Size: 930 B

After

Width:  |  Height:  |  Size: 930 B

View File

@ -11,8 +11,8 @@
import './index.scss'
import { NEllipsis } from 'naive-ui'
import RayIcon from '@/components/RayIcon/index'
import { NEllipsis, NPopover } from 'naive-ui'
import RayIcon from '@/components/RIcon/index'
const SiderBarLogo = defineComponent({
name: 'SiderBarLogo',
@ -37,9 +37,14 @@ const SiderBarLogo = defineComponent({
}
}
const TemplateLogo = ({ cursor }: { cursor: string }) => (
<RayIcon name={sideBarLogo!.icon as string} size="30" cursor={cursor} />
)
return {
sideBarLogo,
handleSideBarLogoClick,
TemplateLogo,
}
},
render() {
@ -47,27 +52,32 @@ const SiderBarLogo = defineComponent({
<div
class={[
'ray-menu__logo',
this.sideBarLogo?.url ? 'ray-menu__logo-url' : '',
this.sideBarLogo?.url ? 'ray-menu__logo-url' : null,
]}
onClick={this.handleSideBarLogoClick.bind(this)}
>
{this.sideBarLogo?.icon ? (
<RayIcon name={this.sideBarLogo.icon} size="30" />
) : (
''
)}
this.collapsed ? (
<NPopover placement="right">
{{
trigger: () => <this.TemplateLogo cursor="pointer" />,
default: () => this.sideBarLogo?.title,
}}
</NPopover>
) : (
<this.TemplateLogo cursor="pointer" />
)
) : null}
<h1
class={[
!this.collapsed ? 'ray-menu__logo-title--open' : '',
!this.collapsed ? 'ray-menu__logo-title--open' : null,
'ray-menu__logo-title',
]}
>
<NEllipsis>{this.sideBarLogo?.title}</NEllipsis>
</h1>
</div>
) : (
''
)
) : null
},
})

View File

@ -13,7 +13,7 @@ import { NMenu, NLayoutSider } from 'naive-ui'
import SiderBarLogo from './components/SiderBarLogo/index'
import { useMenu } from '@/store'
import { APP_MENU_CONFIG } from '@/appConfig/appConfig'
import { APP_MENU_CONFIG } from '@/app-config/appConfig'
import type { MenuInst } from 'naive-ui'
import type { NaiveMenuOptions } from '@/types/modules/component'

View File

@ -26,13 +26,13 @@
import './index.scss'
import { NScrollbar, NTag, NSpace, NLayoutHeader, NDropdown } from 'naive-ui'
import RayIcon from '@/components/RayIcon/index'
import RayIcon from '@/components/RIcon/index'
import { useMenu, useSetting } from '@/store'
import { uuid } from '@/utils/hook'
import { hasClass } from '@/utils/element'
import { redirectRouterToDashboard } from '@/router/helper/routerCopilot'
import { ROOT_ROUTE } from '@/appConfig/appConfig'
import { ROOT_ROUTE } from '@/app-config/appConfig'
import { queryElements } from '@use-utils/element'
import type { MenuOption, ScrollbarInst } from 'naive-ui'
@ -388,7 +388,7 @@ const MenuTag = defineComponent({
if (tags?.length) {
const [menuTag] = tags
nextTick(() => {
nextTick().then(() => {
menuTag.scrollIntoView?.()
})
}

View File

@ -12,7 +12,7 @@
import './index.scss'
import { NInput, NModal, NResult, NScrollbar, NSpace } from 'naive-ui'
import RayIcon from '@/components/RayIcon/index'
import RayIcon from '@/components/RIcon/index'
import { on, off, queryElements, addClass, removeClass } from '@/utils/element'
import { debounce } from 'lodash-es'

View File

@ -10,10 +10,10 @@
*/
import { NSpace, NSwitch, NTooltip } from 'naive-ui'
import RayIcon from '@/components/RayIcon'
import RayIcon from '@/components/RIcon'
import { useSetting } from '@/store'
import { useI18n } from '@/locales/useI18n'
import { useI18n } from '@/hooks/web/index'
const ThemeSwitch = defineComponent({
name: 'ThemeSwitch',

View File

@ -13,9 +13,9 @@ import {
} from 'naive-ui'
import ThemeSwitch from '@/layout/components/SiderBar/components/SettingDrawer/components/ThemeSwitch/index'
import { APP_THEME } from '@/appConfig/designConfig'
import { APP_THEME } from '@/app-config/designConfig'
import { useSetting } from '@/store'
import { useI18n } from '@/locales/useI18n'
import { useI18n } from '@/hooks/web/index'
import type { PropType } from 'vue'
import type { Placement } from '@/types/modules/component'
@ -48,7 +48,6 @@ const SettingDrawer = defineComponent({
primaryColorOverride,
menuTagSwitch,
breadcrumbSwitch,
invertSwitch,
footerSwitch,
contentTransition,
} = storeToRefs(settingStore)
@ -87,7 +86,6 @@ const SettingDrawer = defineComponent({
menuTagSwitch,
changeSwitcher,
breadcrumbSwitch,
invertSwitch,
footerSwitch,
contentTransitionOptions,
contentTransition,
@ -155,14 +153,6 @@ const SettingDrawer = defineComponent({
}
/>
</NDescriptionsItem>
<NDescriptionsItem label="反转色">
<NSwitch
v-model:value={this.invertSwitch}
onUpdateValue={(bool: boolean) =>
this.changeSwitcher(bool, 'invertSwitch')
}
/>
</NDescriptionsItem>
</NDescriptions>
</NSpace>
</NDrawerContent>

View File

@ -12,7 +12,7 @@
import './index.scss'
import { NTooltip } from 'naive-ui'
import RayIcon from '@/components/RayIcon/index'
import RayIcon from '@/components/RIcon/index'
import { tooltipProps } from 'naive-ui'

View File

@ -20,7 +20,7 @@
import './index.scss'
import { NLayoutHeader, NSpace, NTooltip, NDropdown } from 'naive-ui'
import RayIcon from '@/components/RayIcon/index'
import RayIcon from '@/components/RIcon/index'
import TootipIcon from '@/layout/components/SiderBar/components/TooltipIcon/index'
import SettingDrawer from './components/SettingDrawer/index'
import Breadcrumb from './components/Breadcrumb/index'
@ -28,10 +28,10 @@ import GlobalSeach from './components/GlobalSeach/index'
import AppAvatar from '@/app-components/app/AppAvatar/index'
import { useSetting } from '@/store'
import { LOCAL_OPTIONS } from '@/appConfig/localConfig'
import { LOCAL_OPTIONS } from '@/app-config/localConfig'
import { useAvatarOptions, avatarDropdownClick } from './hook'
import screenfull from 'screenfull'
import { useI18n } from '@/locales/useI18n'
import { useI18n } from '@/hooks/web/index'
import type { IconEventMapOptions, IconEventMap } from './type'

View File

@ -18,8 +18,8 @@
import './index.scss'
import { NSpin } from 'naive-ui'
import RayTransitionComponent from '@/components/RayTransitionComponent/index.vue'
import AppRequestCanceler from '@/app-components/provider/AppRequestCanceler/index'
import RTransitionComponent from '@/components/RTransitionComponent/index.vue'
import AppRequestCancelerProvider from '@/app-components/provider/AppRequestCancelerProvider/index'
import { useSetting } from '@/store'
@ -64,15 +64,13 @@ const ContentWrapper = defineComponent({
size="large"
themeOverrides={this.thmeOverridesSpin}
>
<AppRequestCanceler />
<AppRequestCancelerProvider />
{this.reloadRouteSwitch ? (
<RayTransitionComponent
<RTransitionComponent
class="content-wrapper"
transitionPropName={this.contentTransition + '-transform'}
/>
) : (
''
)}
) : null}
</NSpin>
)
},

View File

@ -19,7 +19,7 @@ import HeaderWrapper from './default/HeaderWrapper'
import FeatureWrapper from './default/FeatureWrapper'
import { useSetting } from '@/store'
import { LAYOUT_CONTENT_REF } from '@/appConfig/routerConfig'
import { LAYOUT_CONTENT_REF } from '@/app-config/routerConfig'
import { layoutHeaderCssVars } from '@/layout/layoutResize'
import useAppLockScreen from '@/app-components/app/AppLockScreen/appLockVar'
@ -60,9 +60,7 @@ const RLayout = defineComponent({
<HeaderWrapper ref="layoutSiderBarRef" />
{this.modelMenuTagSwitch ? (
<FeatureWrapper ref="layoutMenuTagRef" />
) : (
''
)}
) : null}
<NLayoutContent
ref="LAYOUT_CONTENT_REF"
class="r-layout-full__viewer-content"
@ -73,9 +71,7 @@ const RLayout = defineComponent({
{this.footerSwitch ? <FooterWrapper ref="layoutFooterRef" /> : ''}
</NLayoutContent>
</NLayout>
) : (
''
)
) : null
},
})

View File

@ -19,8 +19,8 @@
import { set } from 'lodash-es'
import { zhCN, dateZhCN } from 'naive-ui' // 导入 `naive ui` 中文包
import { getStorage } from '@use-utils/cache'
import { SYSTEM_DEFAULT_LOCAL } from '@/appConfig/localConfig'
import { APP_CATCH_KEY } from '@/appConfig/appConfig'
import { SYSTEM_DEFAULT_LOCAL } from '@/app-config/localConfig'
import { APP_CATCH_KEY } from '@/app-config/appConfig'
import type { Recordable } from '@/types/modules/helper'
import type {

View File

@ -24,7 +24,7 @@
*/
import { createI18n } from 'vue-i18n'
import { LOCAL_OPTIONS } from '@/appConfig/localConfig'
import { LOCAL_OPTIONS } from '@/app-config/localConfig'
import { getAppDefaultLanguage, getAppLocalMessages } from '@/locales/helper'
import type { App } from 'vue'

View File

@ -1,6 +1,6 @@
{
"Register": "注册",
"Signin": "登",
"Signin": "登",
"QRCodeSignin": "扫码登陆",
"NamePlaceholder": "请输入用户名",
"PasswordPlaceholder": "请输入密码",

View File

@ -1,27 +0,0 @@
/**
*
* @author Ray <https://github.com/XiaoDaiGua-Ray>
*
* @date 2023-03-22
*
* @workspace ray-template
*
* @remark
*/
/**
*
* onlyoffice
*
* ,
* , onlyoffice ,
* ,
*/
import { getAppEnvironment } from '@use-utils/hook'
import request from '@/axios/instance'
export const getOfficeDocumentApi = async (uuid: string) => {
const { VITE_APP_OFFICE_PROXY_URL } = getAppEnvironment()
const { get } = request
}

View File

@ -5,7 +5,7 @@
> router modules 包中的路由模块会与菜单一一映射,也就是说,路由模块的配置结构会影响菜单的展示。当你有子菜单需要配置时,你需要使用该组件。
```ts
import { t } from '@/locales/useI18n'
import { t } from '@/hooks/web/index'
import { LAYOUT } from '@/router/constant/index'
import type { AppRouteRecordRaw } from '@/router/type'

View File

@ -18,7 +18,7 @@
* router routerCopilot
*/
import { LAYOUT_CONTENT_REF } from '@/appConfig/routerConfig'
import { LAYOUT_CONTENT_REF } from '@/app-config/routerConfig'
import type { RouteLocationNormalized } from 'vue-router'
import type { AppRouteRecordRaw, RouteModules } from '@/router/type'

View File

@ -21,9 +21,9 @@
*/
import { getStorage } from '@/utils/cache'
import { APP_CATCH_KEY, ROOT_ROUTE } from '@/appConfig/appConfig'
import { APP_CATCH_KEY, ROOT_ROUTE } from '@/app-config/appConfig'
import { redirectRouterToDashboard } from '@/router/helper/routerCopilot'
import { WHITE_ROUTES } from '@/appConfig/routerConfig'
import { WHITE_ROUTES } from '@/app-config/routerConfig'
import { validRole } from '@/router/helper/routerCopilot'
import { isValueType } from '@/utils/hook'
@ -41,7 +41,7 @@ export const permissionRouter = (router: Router) => {
beforeEach((to, from, next) => {
const token = getStorage<string>(APP_CATCH_KEY.token)
const catchRoutePath = getStorage<string>(
const catchRoutePath = getStorage(
'menuKey',
'sessionStorage',
ROOT_ROUTE.path,

View File

@ -10,10 +10,10 @@
*/
import { permissionRouter } from './permission'
import { SETUP_ROUTER_ACTION, SUPER_ADMIN } from '@/appConfig/routerConfig'
import { SETUP_ROUTER_ACTION, SUPER_ADMIN } from '@/app-config/routerConfig'
import { useSignin } from '@/store'
import { useVueRouter } from '@/router/helper/useVueRouter'
import { ROOT_ROUTE } from '@/appConfig/appConfig'
import { useVueRouter } from '@/hooks/web/index'
import { ROOT_ROUTE } from '@/app-config/appConfig'
import { setStorage } from '@/utils/cache'
import { getAppEnvironment } from '@/utils/hook'

Some files were not shown because too many files have changed in this diff Show More