新增加载动画

This commit is contained in:
ray_wuhao 2023-04-04 15:09:53 +08:00
parent ff91d9f1cc
commit 340c41a016
21 changed files with 744 additions and 68 deletions

View File

@ -1,7 +1,7 @@
#开发环境
NODE_ENV = 'development'
VITE_APP_URL = 'api/'
VITE_APP_URL = '/api'
# office 服务代理地址
VITE_APP_OFFICE_PROXY_URL = '/office/'

View File

@ -1,7 +1,7 @@
#生产环境
NODE_ENV = 'production'
VITE_APP_URL = 'api/'
VITE_APP_URL = '/'
# office 服务代理地址
VITE_APP_OFFICE_PROXY_URL = 'https://office.yka.one/'

View File

@ -1,7 +1,7 @@
#测试环境
NODE_ENV = 'test'
VITE_APP_URL = 'api/'
VITE_APP_URL = 'https://testray.yka.moe/doc-json/'
# office 服务代理地址
VITE_APP_OFFICE_PROXY_URL = 'https://office.yka.one/'

View File

@ -5,6 +5,12 @@
### Fixes
- 修复移动端登陆页显示问题
- 改进了一些方法逻辑的问题
- 新增加载动画
### Feats
- 修改移动端自适应配置方案(现在使用 postcss-px-to-viewport),默认不启用
## 3.1.5

View File

@ -6,8 +6,127 @@
<meta name="viewport" content="width=device-width, initial-scale=1.0" />
<title>Vite + Vue + TS</title>
</head>
<style>
#pre-loading-animation {
position: fixed;
left: 0;
right: 0;
top: 0;
bottom: 0;
background-color: #ffffff;
color: #2d8cf0;
text-align: center;
}
.ray-template--dark #pre-loading-animation {
background-color: #2a3146;
}
#pre-loading-animation .pre-loading-animation__wrapper {
width: 100%;
height: 100%;
display: flex;
flex-direction: column;
justify-content: center;
align-items: center;
}
#pre-loading-animation
.pre-loading-animation__wrapper
.pre-loading-animation__wrapper-title {
font-size: 30px;
padding-bottom: 48px;
}
#pre-loading-animation
.pre-loading-animation__wrapper
.pre-loading-animation__wrapper-loader {
margin: 0 0 2em;
height: 100px;
width: 20%;
text-align: center;
padding: 1em;
margin: 0 auto 1em;
display: inline-block;
vertical-align: top;
}
#pre-loading-animation
.pre-loading-animation__wrapper
.pre-loading-animation__wrapper-loader
svg
path,
#pre-loading-animation
.pre-loading-animation__wrapper
.pre-loading-animation__wrapper-loader
svg
rect {
fill: #ff6700;
}
#pre-loading-animation
.pre-loading-animation__wrapper
.pre-loading-animation__wrapper-loader
svg {
transform: scale(2);
}
</style>
<body>
<div id="app"></div>
<div id="pre-loading-animation">
<div class="pre-loading-animation__wrapper">
<div class="pre-loading-animation__wrapper-title">Ray Template</div>
<div class="pre-loading-animation__wrapper-loader">
<svg
version="1.1"
id="Layer_1"
xmlns="http://www.w3.org/2000/svg"
xmlns:xlink="http://www.w3.org/1999/xlink"
x="0px"
y="0px"
width="24px"
height="30px"
viewBox="0 0 24 30"
style="enable-background: new 0 0 50 50"
xml:space="preserve"
>
<rect x="0" y="0" width="4" height="10" fill="#333">
<animateTransform
attributeType="xml"
attributeName="transform"
type="translate"
values="0 0; 0 20; 0 0"
begin="0"
dur="0.6s"
repeatCount="indefinite"
/>
</rect>
<rect x="10" y="0" width="4" height="10" fill="#333">
<animateTransform
attributeType="xml"
attributeName="transform"
type="translate"
values="0 0; 0 20; 0 0"
begin="0.2s"
dur="0.6s"
repeatCount="indefinite"
/>
</rect>
<rect x="20" y="0" width="4" height="10" fill="#333">
<animateTransform
attributeType="xml"
attributeName="transform"
type="translate"
values="0 0; 0 20; 0 0"
begin="0.4s"
dur="0.6s"
repeatCount="indefinite"
/>
</rect>
</svg>
</div>
</div>
</div>
<script type="module" src="/src/main.ts"></script>
</body>
</html>

View File

@ -66,7 +66,7 @@
"husky": "^8.0.3",
"lint-staged": "^13.1.0",
"postcss": "^8.1.0",
"postcss-pxtorem": "^6.0.0",
"postcss-px-to-viewport": "^1.1.1",
"prettier": "^2.7.1",
"rollup-plugin-visualizer": "^5.8.3",
"svg-sprite-loader": "^6.0.11",

View File

@ -11,11 +11,21 @@ module.exports = {
],
grid: true,
},
'postcss-pxtorem': {
rootValue: 16, // 根元素字体大小或根据 `input` 参数返回根元素字体大小
unitPrecision: 5,
propList: ['font', 'font-size', 'line-height', 'letter-spacing'], // 可以从 `px` 更改为 `rem` 的属性
selectorBlackList: [], // 要忽略并保留为 `px` 的选择器
},
// 'postcss-px-to-viewport': {
// /** 视窗的宽度(设计稿的宽度) */
// viewportWidth: 1920,
// /** 视窗的高度(设计稿高度, 一般无需指定) */
// viewportHeight: 1080,
// /** 指定 px 转换为视窗单位值的小数位数 */
// unitPrecision: 3,
// /** 指定需要转换成的视窗单位 */
// viewportUnit: 'vw',
// /** 指定不转换为视窗单位的类 */
// selectorBlackList: ['.ignore'],
// /** 小于或等于 1px 不转换为视窗单位 */
// minPixelValue: 1,
// /** 允许在媒体查询中转换 px */
// mediaQuery: false,
// },
},
}

View File

@ -5,7 +5,7 @@ import GlobalSpin from '@/spin/index'
import { getCache } from '@/utils/cache'
import { get } from 'lodash-es'
import { useSetting } from '@/store'
import { addClass, removeClass } from '@/utils/element'
import { addClass, removeClass, addStyle } from '@/utils/element'
const App = defineComponent({
name: 'App',
@ -29,7 +29,19 @@ const App = defineComponent({
body.style.setProperty('--ray-theme-primary-color', _p || primaryColor)
}
/** 隐藏加载动画 */
const hiddenLoadingAnimation = () => {
const el = document.getElementById('pre-loading-animation')
if (el) {
addStyle(el, {
display: 'none',
})
}
}
syncPrimaryColorToBody()
hiddenLoadingAnimation()
watch(
() => themeValue.value,

View File

@ -18,11 +18,11 @@ import SettingDrawer from './components/SettingDrawer/index'
import Breadcrumb from './components/Breadcrumb/index'
import { useSetting } from '@/store'
import { useSignin } from '@/store'
import { localOptions } from '@/language/index'
import { useAvatarOptions } from './hook'
import { getCache } from '@/utils/cache'
import screenfull from 'screenfull'
import { logout } from '@/utils/user'
import type { IconEventMapOptions, IconEventMap } from './type'
@ -37,9 +37,11 @@ const SiderBar = defineComponent({
name: 'SiderBar',
setup() {
const settingStore = useSetting()
const signinStore = useSignin()
const { t } = useI18n()
const { updateLocale, changeSwitcher } = settingStore
const { logout } = signinStore
const { drawerPlacement, breadcrumbSwitch } = storeToRefs(settingStore)
const showSettings = ref(false)
const person = getCache('person')
@ -145,7 +147,11 @@ const SiderBar = defineComponent({
align="center"
justify="space-between"
>
<NSpace align="center" itemStyle={this.spaceItemStyle}>
<NSpace
align="center"
wrapItem={false}
itemStyle={this.spaceItemStyle}
>
{this.leftIconOptions.map((curr) => (
<NTooltip>
{{
@ -163,7 +169,11 @@ const SiderBar = defineComponent({
))}
{this.breadcrumbSwitch ? <Breadcrumb /> : ''}
</NSpace>
<NSpace align="center" itemStyle={this.spaceItemStyle}>
<NSpace
align="center"
wrapItem={false}
itemStyle={this.spaceItemStyle}
>
{this.rightTooltipIconOptions.map((curr) => (
<RayTooltipIcon
iconName={curr.name}

View File

@ -1,6 +1,6 @@
import './index.scss'
import { NLayout, NLayoutContent, NSpin } from 'naive-ui'
import { NLayout, NLayoutContent } from 'naive-ui'
import RayTransitionComponent from '@/components/RayTransitionComponent/index.vue'
import LayoutMenu from './components/Menu/index'
import SiderBar from './components/SiderBar/index'
@ -15,7 +15,6 @@ const Layout = defineComponent({
const { height: windowHeight } = useWindowSize()
const {
themeValue,
reloadRouteSwitch: modelReloadRoute,
menuTagSwitch: modelMenuTagSwitch,
} = storeToRefs(settingStore)

View File

@ -26,7 +26,8 @@ import { useSignin } from '@/store'
import { whiteRoutes, superAdmin } from './configuration'
export const validRole = (options: IMenuOptions) => {
const { role } = storeToRefs(useSignin())
const { signinCallback } = storeToRefs(useSignin())
const role = computed(() => signinCallback.value.role)
const { meta, name } = options
const hidden =

View File

@ -34,6 +34,7 @@ export const permissionRouter = (router: Router) => {
rootRoute: { path },
} = __APP_CFG__
/** 如果没有权限, 则重定向至首页 */
const redirectToDashboard = (next: NavigationGuardNext) => {
next(path)
@ -43,7 +44,8 @@ export const permissionRouter = (router: Router) => {
beforeEach((to, from, next) => {
const token = getCache('token')
const route = getCache('menuKey')
const { role } = storeToRefs(useSignin())
const { signinCallback } = storeToRefs(useSignin())
const role = computed(() => signinCallback.value.role)
const { meta } = to
const hasRole = () => {

View File

@ -1,14 +1,33 @@
import piniaPluginPersistedstate from 'pinia-plugin-persistedstate'
/**
*
* @author Ray <https://github.com/XiaoDaiGua-Ray>
*
* @date 2023-01-03
*
* @workspace ray-template
*
* @remark
*/
import type { App } from 'vue'
/**
*
* pinia
* function
*
* 官网地址: https://prazdevs.github.io/pinia-plugin-persistedstate/zh/guide/
*/
import piniaPluginPersistedstate from 'pinia-plugin-persistedstate'
export { useSetting } from './modules/setting' // import { useSetting } from '@/store' 即可使用
export { useMenu } from './modules/menu/index'
export { useSignin } from './modules/signin'
const store = createPinia()
import type { App } from 'vue'
/** 设置并且注册 pinia */
export const setupStore = (app: App<Element>) => {
const store = createPinia()
app.use(store)
store.use(piniaPluginPersistedstate)

View File

@ -12,25 +12,34 @@
/**
*
* 便, pinia
*
* 可以存储: 头像, ,
*
* ,
* 使 sessionStorage
*/
import { isEmpty } from 'lodash-es'
import { logout } from '@/utils/user'
import { removeCache } from '@/utils/cache'
export interface SigninForm extends IUnknownObjectKey {
name: string
pwd: string
}
export interface SigninCallback extends IUnknownObjectKey {
role: string
name: string
}
export const useSignin = defineStore(
'signin',
() => {
const state = reactive({
role: '',
/**
*
* ()
* role , 如果需要更改请同步更改: router/basic.tsrouter/permission.ts
*/
signinCallback: {} as SigninCallback,
})
/**
@ -42,7 +51,10 @@ export const useSignin = defineStore(
*/
const signin = (signinForm: SigninForm) => {
if (!isEmpty(signinForm)) {
state.role = 'admin'
state.signinCallback = {
role: 'admin',
name: signinForm.name,
}
return 0
} else {
@ -50,14 +62,29 @@ export const useSignin = defineStore(
}
}
/**
*
* 退
* 300ms
*/
const logout = () => {
window.$message.info('账号退出中...')
removeCache('all-sessionStorage')
setTimeout(() => window.location.reload(), 300)
}
return {
...toRefs(state),
signin,
logout,
}
},
{
persist: {
key: 'piniaSigninStore',
paths: ['signinCallback'],
storage: sessionStorage,
},
},
)

View File

@ -18,8 +18,9 @@ export const setCache = <T = unknown>(
/**
*
* @param key key
*
* @returns
*
* @remark 'no'
*/
export const getCache = (key: string, type: CacheType = 'sessionStorage') => {
const data =
@ -33,6 +34,11 @@ export const getCache = (key: string, type: CacheType = 'sessionStorage') => {
/**
*
* @param key key
*
* key:
* - all: 删除所有缓存值
* - all-sessionStorage: 删除所有 sessionStorage
* - all-localStorage: 删除所有 localStorage
*/
export const removeCache = (
key: string | 'all' | 'all-sessionStorage' | 'all-localStorage',

View File

@ -4,6 +4,14 @@ import AES from 'crypto-js/aes'
import MD5 from 'crypto-js/md5'
import BASE64 from 'crypto-js/enc-base64'
/**
*
* 使, demo 使
* ,
*
* 手动补上官网地址: http://github.com/brix/crypto-js
*/
/**
*
* @param message

View File

@ -11,17 +11,21 @@ export const getElementChildNodes = (
el: HTMLElement,
target?: string[] | string,
) => {
let nodes = Array.from(el.childNodes)
if (el) {
let nodes = Array.from(el.childNodes)
if (Array.isArray(target)) {
nodes = nodes.filter((el) => target.includes(el.nodeName))
} else {
if (target) {
nodes = nodes.filter((el) => el.nodeName === target)
if (Array.isArray(target)) {
nodes = nodes.filter((el) => target.includes(el.nodeName))
} else {
if (target) {
nodes = nodes.filter((el) => el.nodeName === target)
}
}
}
return nodes
return nodes
} else {
return []
}
}
/**
@ -65,7 +69,7 @@ export const off = (
/**
*
* @param element Target element dom
* @param className className: 'xxx xxx' | 'xxx'
* @param className className: 'xxx xxx' | 'xxx' ( css )
*
* @remark className(: 'xxx xxx' | 'xxx')
*/
@ -84,7 +88,7 @@ export const addClass = (element: HTMLElement, className: string) => {
/**
*
* @param element Target element dom
* @param className className: 'xxx xxx' | 'xxx'
* @param className className: 'xxx xxx' | 'xxx' ( css )
*
* @remark className(: 'xxx xxx' | 'xxx')
* @remark removeAllClass class name
@ -113,11 +117,11 @@ export const removeClass = (
/**
*
* @param element Target element dom
* @param className className: 'xxx xxx' | 'xxx'
* @param className className: 'xxx xxx' | 'xxx' ( css )
*
* @returns boolean
*
* @remark className(: 'xxx xxx' | 'xxx')
* @remark className(: 'xxx xxx' | 'xxx' )
*/
export const hasClass = (element: HTMLElement, className: string) => {
const elementClassName = element.className

View File

@ -72,7 +72,7 @@ export const uuid = (length = 16, radix?: number) => {
for (i = 0; i < 36; i++) {
if (!arr[i]) {
r = 0 | (Math.random() * 16)
r = 0 | (Math.random() * radix)
arr[i] = sad[i === 19 ? (r & 0x3) | 0x8 : r]
}

View File

@ -1,13 +0,0 @@
import { removeCache } from '@/utils/cache'
/**
*
* @remark 退 localStorage
*/
export const logout = () => {
window.$message.info('账号退出中...')
removeCache('all-sessionStorage')
setTimeout(() => window.location.reload(), 300)
}

View File

@ -74,25 +74,29 @@ export const exportFileToXLSX = async (
config: ExportXLSXConfig = {},
) => {
await new Promise<void>((resolve, reject) => {
if (dataSource?.length) {
const sheetHeader = setupSheetHeader(columns ?? []) // 获取所有列(设置为 `excel` 表头)
const sheetData = utils.json_to_sheet(dataSource) // 将所有数据转换为表格数据类型
const workBook = utils.book_new()
const filename = config.filename
? config.filename + '.xlsx'
: dayjs().format('YYYY-MM-DD') + '导出表格.xlsx'
if (Array.isArray(dataSource)) {
if (dataSource.length) {
const sheetHeader = setupSheetHeader(columns ?? []) // 获取所有列(设置为 `excel` 表头)
const sheetData = utils.json_to_sheet(dataSource) // 将所有数据转换为表格数据类型
const workBook = utils.book_new()
const filename = config.filename
? config.filename + '.xlsx'
: dayjs().format('YYYY-MM-DD') + '导出表格.xlsx'
utils.book_append_sheet(workBook, sheetData, 'Data')
utils.book_append_sheet(workBook, sheetData, 'Data')
const range = utils.decode_range(sheetData['!ref'] as string) // 获取所有单元格
const range = utils.decode_range(sheetData['!ref'] as string) // 获取所有单元格
if (columns?.length) {
transformSheetHeader(range, sheetData, sheetHeader)
if (columns?.length) {
transformSheetHeader(range, sheetData, sheetHeader)
}
writeFileXLSX(workBook, filename) // 输出表格
resolve()
} else {
resolve()
}
writeFileXLSX(workBook, filename) // 输出表格
resolve()
} else {
reject()
}

File diff suppressed because one or more lines are too long