mirror of
https://gitee.com/nocobase/nocobase.git
synced 2025-05-05 21:49:25 +08:00
* refactor: update umi version 3.x to version 4.x
* refactor: update react-router-dom version to 6.x
* refactor(react-router-dom): change Layout Component `props.children` to `<Outlet />`
* refactor(react-router-dom): change <Route /> props and <RouteSwitch /> correct
* refactor(react-router-dom): replace `<Redirect />` to `<Navigate replace />`
* refactor(react-router-dom): replace `useHistory` to `useNavigate`
* refactor(react-router-dom): replace `useRouteMatch` to `useParams`
* refactor(react-router-dom & dumi): fix <RouteSwitch /> & umi document bug
* refactor(react-router-dom): `useRoutes` Optimize `<RouteSwitch />` code
* refactor(react-router-dom): update `Route` types and docs
* refactor(react-router-dom): optimize RouteSwitch code
* refactor(react-router-dom): `useLocation` no generics type
* refactor(react-router-dom): add `less v3.9.0` to `resolutions` to solve the error of `gulp-less`
* refactor(react-router-dom): fix `<RouteSwitch />` `props.routes` as an array is not handled
* chore: upgrade `dumi` and refactor docs
* fix: completed code review, add `targets` to solve browser compatibility & removed `chainWebpack`
* refactor(dumi): upgraded dumi under `packages/core/client`
* refactor(dumi): delete `packages/core/dumi-theme-nocobase`
* refactor(dumi): degrade `react` & replace `dumi-theme-antd` to `dumi-theme-nocobase`
* refactor(dumi): solve conflicts between multiple dumi applications
* fix: login page error in react 17
* refactor(dumi): remove less resolutions
* refactor(dumi): umi add `msfu: true` config
* fix: merge bug
* fix: self code review
* fix: code reivew and test bug
* refactor: upgrade react to 18
* refactor: degrade react types to 17
* chore: fix ci error
* fix: support routerBase & fix workflow page params
* fix(doc): menu externel link
* fix: build error
* fix: delete
* fix: vitest error
* fix: react-router new code replace
* fix: vitest markdown error
* fix: title is none when refresh
* fix: merge error
* fix: sidebar width is wrong
* fix: useProps error
* fix: side-menu-width
* fix: menu selectId is wrong & useProps is string
* fix: menu selected first default & side menu hide when change
* fix: test error & v0.10 change log
* fix: new compnent doc modify
* fix: set umi `fastRefresh=false`
* refactor: application v2
* fix: improve code
* fix: bug
* fix: page = 0 error
* fix: workflow navigate error
* feat: plugin manager
* fix: afterAdd
* feat: update docs
* feat: update docs
* fix: page tab change not work
* fix: login redirect query param doesn't work
* fix: bug and doc
* feat: update docs
* fix: ci error
* fix: merge main
* feat: update docs
* feat: update docs
* feat: update docs
* chore(versions): 😊 publish v0.10.0-alpha.1
* fix: translations
* chore: backend node test max old space size
* docs: add useSearchParams
---------
Co-authored-by: chenos <chenlinxh@gmail.com>
Co-authored-by: ChengLei Shao <chareice@live.com>
147 lines
4.3 KiB
TypeScript
147 lines
4.3 KiB
TypeScript
import { css } from '@emotion/css';
|
|
import { Dropdown, Menu, Modal } from 'antd';
|
|
import React, { createContext, useState } from 'react';
|
|
import { useTranslation } from 'react-i18next';
|
|
import { useNavigate } from 'react-router-dom';
|
|
import { useACLRoleContext, useAPIClient, useCurrentUserContext } from '..';
|
|
import { useCurrentAppInfo } from '../appInfo/CurrentAppInfoProvider';
|
|
import { ChangePassword } from './ChangePassword';
|
|
import { EditProfile } from './EditProfile';
|
|
import { LanguageSettings } from './LanguageSettings';
|
|
import { SwitchRole } from './SwitchRole';
|
|
import { ThemeSettings } from './ThemeSettings';
|
|
const ApplicationVersion = () => {
|
|
const data = useCurrentAppInfo();
|
|
return (
|
|
<Menu.Item key="version" disabled>
|
|
Version {data?.data?.version}
|
|
</Menu.Item>
|
|
);
|
|
};
|
|
|
|
/**
|
|
* @note If you want to change here, Note the Setting block on the mobile side
|
|
*/
|
|
export const SettingsMenu: React.FC<{
|
|
redirectUrl?: string;
|
|
}> = (props) => {
|
|
const { redirectUrl = '' } = props;
|
|
const { allowAll, snippets } = useACLRoleContext();
|
|
const appAllowed = allowAll || snippets?.includes('app');
|
|
const navigate = useNavigate();
|
|
const api = useAPIClient();
|
|
const { t } = useTranslation();
|
|
const silenceApi = useAPIClient();
|
|
const check = async () => {
|
|
return await new Promise((resolve) => {
|
|
const heartbeat = setInterval(() => {
|
|
silenceApi
|
|
.silent()
|
|
.resource('app')
|
|
.getInfo()
|
|
.then((res) => {
|
|
console.log(res);
|
|
if (res?.status === 200) {
|
|
resolve('ok');
|
|
clearInterval(heartbeat);
|
|
}
|
|
return res;
|
|
})
|
|
.catch(() => {
|
|
// ignore
|
|
});
|
|
}, 3000);
|
|
});
|
|
};
|
|
return (
|
|
<Menu>
|
|
<ApplicationVersion />
|
|
<Menu.Divider />
|
|
<EditProfile />
|
|
<ChangePassword />
|
|
<Menu.Divider />
|
|
<SwitchRole />
|
|
<LanguageSettings />
|
|
<ThemeSettings />
|
|
<Menu.Divider />
|
|
{appAllowed && (
|
|
<>
|
|
<Menu.Item
|
|
key="cache"
|
|
onClick={async () => {
|
|
await api.resource('app').clearCache();
|
|
window.location.reload();
|
|
}}
|
|
>
|
|
{t('Clear cache')}
|
|
</Menu.Item>
|
|
<Menu.Item
|
|
key="reboot"
|
|
onClick={async () => {
|
|
Modal.confirm({
|
|
title: t('Reboot application'),
|
|
content: t(
|
|
'The will interrupt service, it may take a few seconds to restart. Are you sure to continue?',
|
|
),
|
|
okText: t('Reboot'),
|
|
okButtonProps: {
|
|
danger: true,
|
|
},
|
|
onOk: async () => {
|
|
await api.resource('app').reboot();
|
|
await check();
|
|
window.location.reload();
|
|
},
|
|
});
|
|
}}
|
|
>
|
|
{t('Reboot application')}
|
|
</Menu.Item>
|
|
<Menu.Divider />
|
|
</>
|
|
)}
|
|
<Menu.Item
|
|
key="signout"
|
|
onClick={async () => {
|
|
await api.auth.signOut();
|
|
navigate(`/signin?redirect=${encodeURIComponent(redirectUrl)}`);
|
|
}}
|
|
>
|
|
{t('Sign out')}
|
|
</Menu.Item>
|
|
</Menu>
|
|
);
|
|
};
|
|
export const DropdownVisibleContext = createContext(null);
|
|
export const CurrentUser = () => {
|
|
const [visible, setVisible] = useState(false);
|
|
const { data } = useCurrentUserContext();
|
|
return (
|
|
<div style={{ display: 'inline-flex', verticalAlign: 'top' }}>
|
|
<DropdownVisibleContext.Provider value={{ visible, setVisible }}>
|
|
<Dropdown
|
|
open={visible}
|
|
onOpenChange={(visible) => {
|
|
setVisible(visible);
|
|
}}
|
|
overlay={<SettingsMenu />}
|
|
>
|
|
<span
|
|
className={css`
|
|
max-width: 160px;
|
|
overflow: hidden;
|
|
display: inline-block;
|
|
line-height: 12px;
|
|
white-space: nowrap;
|
|
text-overflow: ellipsis;
|
|
`}
|
|
style={{ cursor: 'pointer', border: 0, padding: '16px', color: 'rgba(255, 255, 255, 0.65)' }}
|
|
>
|
|
{data?.data?.nickname || data?.data?.email}
|
|
</span>
|
|
</Dropdown>
|
|
</DropdownVisibleContext.Provider>
|
|
</div>
|
|
);
|
|
};
|