preferences-drawer.vue 17 KB

123456789101112131415161718192021222324252627282930313233343536373839404142434445464748495051525354555657585960616263646566676869707172737475767778798081828384858687888990919293949596979899100101102103104105106107108109110111112113114115116117118119120121122123124125126127128129130131132133134135136137138139140141142143144145146147148149150151152153154155156157158159160161162163164165166167168169170171172173174175176177178179180181182183184185186187188189190191192193194195196197198199200201202203204205206207208209210211212213214215216217218219220221222223224225226227228229230231232233234235236237238239240241242243244245246247248249250251252253254255256257258259260261262263264265266267268269270271272273274275276277278279280281282283284285286287288289290291292293294295296297298299300301302303304305306307308309310311312313314315316317318319320321322323324325326327328329330331332333334335336337338339340341342343344345346347348349350351352353354355356357358359360361362363364365366367368369370371372373374375376377378379380381382383384385386387388389390391392393394395396397398399400401402403404405406407408409410411412413414415416417418419420421422423424425426427428429430431432433434435436437438439440441442443444445446447448449450451452453454455456457458459460461462463464465466467468469470471472473474475476477478479480481482483484485486487488489490491492493
  1. <script setup lang="ts">
  2. import type { SupportedLanguagesType } from '@vben/locales';
  3. import type {
  4. BreadcrumbStyleType,
  5. BuiltinThemeType,
  6. ContentCompactType,
  7. LayoutHeaderMenuAlignType,
  8. LayoutHeaderModeType,
  9. LayoutType,
  10. NavigationStyleType,
  11. PreferencesButtonPositionType,
  12. ThemeModeType,
  13. } from '@vben/types';
  14. import type { SegmentedItem } from '@vben-core/shadcn-ui';
  15. import { computed, ref } from 'vue';
  16. import { Copy, Pin, PinOff, RotateCw } from '@vben/icons';
  17. import { $t, loadLocaleMessages } from '@vben/locales';
  18. import {
  19. clearPreferencesCache,
  20. preferences,
  21. resetPreferences,
  22. usePreferences,
  23. } from '@vben/preferences';
  24. import { useVbenDrawer } from '@vben-core/popup-ui';
  25. import {
  26. VbenButton,
  27. VbenIconButton,
  28. VbenSegmented,
  29. } from '@vben-core/shadcn-ui';
  30. import { globalShareState } from '@vben-core/shared/global-state';
  31. import { useClipboard } from '@vueuse/core';
  32. import {
  33. Animation,
  34. Block,
  35. Breadcrumb,
  36. BuiltinTheme,
  37. ColorMode,
  38. Content,
  39. Copyright,
  40. Footer,
  41. FontSize,
  42. General,
  43. GlobalShortcutKeys,
  44. Header,
  45. Layout,
  46. Navigation,
  47. Radius,
  48. Sidebar,
  49. Tabbar,
  50. Theme,
  51. Widget,
  52. } from './blocks';
  53. const emit = defineEmits<{ clearPreferencesAndLogout: [] }>();
  54. const message = globalShareState.getMessage();
  55. const appLocale = defineModel<SupportedLanguagesType>('appLocale');
  56. const appDynamicTitle = defineModel<boolean>('appDynamicTitle');
  57. const appLayout = defineModel<LayoutType>('appLayout');
  58. const appColorGrayMode = defineModel<boolean>('appColorGrayMode');
  59. const appColorWeakMode = defineModel<boolean>('appColorWeakMode');
  60. const appContentCompact = defineModel<ContentCompactType>('appContentCompact');
  61. const appWatermark = defineModel<boolean>('appWatermark');
  62. const appWatermarkContent = defineModel<string>('appWatermarkContent');
  63. const appEnableCheckUpdates = defineModel<boolean>('appEnableCheckUpdates');
  64. const appEnableStickyPreferencesNavigationBar = defineModel<boolean>(
  65. 'appEnableStickyPreferencesNavigationBar',
  66. );
  67. const appPreferencesButtonPosition = defineModel<PreferencesButtonPositionType>(
  68. 'appPreferencesButtonPosition',
  69. );
  70. const transitionProgress = defineModel<boolean>('transitionProgress');
  71. const transitionName = defineModel<string>('transitionName');
  72. const transitionLoading = defineModel<boolean>('transitionLoading');
  73. const transitionEnable = defineModel<boolean>('transitionEnable');
  74. const themeColorPrimary = defineModel<string>('themeColorPrimary');
  75. const themeBuiltinType = defineModel<BuiltinThemeType>('themeBuiltinType');
  76. const themeMode = defineModel<ThemeModeType>('themeMode');
  77. const themeRadius = defineModel<string>('themeRadius');
  78. const themeFontSize = defineModel<number>('themeFontSize');
  79. const themeSemiDarkSidebar = defineModel<boolean>('themeSemiDarkSidebar');
  80. const themeSemiDarkHeader = defineModel<boolean>('themeSemiDarkHeader');
  81. const sidebarEnable = defineModel<boolean>('sidebarEnable');
  82. const sidebarWidth = defineModel<number>('sidebarWidth');
  83. const sidebarCollapsed = defineModel<boolean>('sidebarCollapsed');
  84. const sidebarCollapsedShowTitle = defineModel<boolean>(
  85. 'sidebarCollapsedShowTitle',
  86. );
  87. const sidebarAutoActivateChild = defineModel<boolean>(
  88. 'sidebarAutoActivateChild',
  89. );
  90. const sidebarExpandOnHover = defineModel<boolean>('sidebarExpandOnHover');
  91. const sidebarCollapsedButton = defineModel<boolean>('sidebarCollapsedButton');
  92. const sidebarFixedButton = defineModel<boolean>('sidebarFixedButton');
  93. const headerEnable = defineModel<boolean>('headerEnable');
  94. const headerMode = defineModel<LayoutHeaderModeType>('headerMode');
  95. const headerMenuAlign =
  96. defineModel<LayoutHeaderMenuAlignType>('headerMenuAlign');
  97. const breadcrumbEnable = defineModel<boolean>('breadcrumbEnable');
  98. const breadcrumbShowIcon = defineModel<boolean>('breadcrumbShowIcon');
  99. const breadcrumbShowHome = defineModel<boolean>('breadcrumbShowHome');
  100. const breadcrumbStyleType = defineModel<BreadcrumbStyleType>(
  101. 'breadcrumbStyleType',
  102. );
  103. const breadcrumbHideOnlyOne = defineModel<boolean>('breadcrumbHideOnlyOne');
  104. const tabbarEnable = defineModel<boolean>('tabbarEnable');
  105. const tabbarShowIcon = defineModel<boolean>('tabbarShowIcon');
  106. const tabbarShowMore = defineModel<boolean>('tabbarShowMore');
  107. const tabbarShowMaximize = defineModel<boolean>('tabbarShowMaximize');
  108. const tabbarPersist = defineModel<boolean>('tabbarPersist');
  109. const tabbarDraggable = defineModel<boolean>('tabbarDraggable');
  110. const tabbarWheelable = defineModel<boolean>('tabbarWheelable');
  111. const tabbarStyleType = defineModel<string>('tabbarStyleType');
  112. const tabbarMaxCount = defineModel<number>('tabbarMaxCount');
  113. const tabbarMiddleClickToClose = defineModel<boolean>(
  114. 'tabbarMiddleClickToClose',
  115. );
  116. const navigationStyleType = defineModel<NavigationStyleType>(
  117. 'navigationStyleType',
  118. );
  119. const navigationSplit = defineModel<boolean>('navigationSplit');
  120. const navigationAccordion = defineModel<boolean>('navigationAccordion');
  121. // const logoVisible = defineModel<boolean>('logoVisible');
  122. const footerEnable = defineModel<boolean>('footerEnable');
  123. const footerFixed = defineModel<boolean>('footerFixed');
  124. const copyrightSettingShow = defineModel<boolean>('copyrightSettingShow');
  125. const copyrightEnable = defineModel<boolean>('copyrightEnable');
  126. const copyrightCompanyName = defineModel<string>('copyrightCompanyName');
  127. const copyrightCompanySiteLink = defineModel<string>(
  128. 'copyrightCompanySiteLink',
  129. );
  130. const copyrightDate = defineModel<string>('copyrightDate');
  131. const copyrightIcp = defineModel<string>('copyrightIcp');
  132. const copyrightIcpLink = defineModel<string>('copyrightIcpLink');
  133. const shortcutKeysEnable = defineModel<boolean>('shortcutKeysEnable');
  134. const shortcutKeysGlobalSearch = defineModel<boolean>(
  135. 'shortcutKeysGlobalSearch',
  136. );
  137. const shortcutKeysGlobalLogout = defineModel<boolean>(
  138. 'shortcutKeysGlobalLogout',
  139. );
  140. const shortcutKeysGlobalLockScreen = defineModel<boolean>(
  141. 'shortcutKeysGlobalLockScreen',
  142. );
  143. const widgetGlobalSearch = defineModel<boolean>('widgetGlobalSearch');
  144. const widgetFullscreen = defineModel<boolean>('widgetFullscreen');
  145. const widgetLanguageToggle = defineModel<boolean>('widgetLanguageToggle');
  146. const widgetNotification = defineModel<boolean>('widgetNotification');
  147. const widgetThemeToggle = defineModel<boolean>('widgetThemeToggle');
  148. const widgetSidebarToggle = defineModel<boolean>('widgetSidebarToggle');
  149. const widgetLockScreen = defineModel<boolean>('widgetLockScreen');
  150. const widgetRefresh = defineModel<boolean>('widgetRefresh');
  151. const {
  152. diffPreference,
  153. isDark,
  154. isFullContent,
  155. isHeaderNav,
  156. isHeaderSidebarNav,
  157. isMixedNav,
  158. isSideMixedNav,
  159. isSideMode,
  160. isSideNav,
  161. } = usePreferences();
  162. const { copy } = useClipboard({ legacy: true });
  163. const [Drawer] = useVbenDrawer();
  164. const activeTab = ref('appearance');
  165. const tabs = computed((): SegmentedItem[] => {
  166. return [
  167. {
  168. label: $t('preferences.appearance'),
  169. value: 'appearance',
  170. },
  171. {
  172. label: $t('preferences.layout'),
  173. value: 'layout',
  174. },
  175. {
  176. label: $t('preferences.shortcutKeys.title'),
  177. value: 'shortcutKey',
  178. },
  179. {
  180. label: $t('preferences.general'),
  181. value: 'general',
  182. },
  183. ];
  184. });
  185. const showBreadcrumbConfig = computed(() => {
  186. return (
  187. !isFullContent.value &&
  188. !isMixedNav.value &&
  189. !isHeaderNav.value &&
  190. preferences.header.enable
  191. );
  192. });
  193. async function handleCopy() {
  194. await copy(JSON.stringify(diffPreference.value, null, 2));
  195. message.copyPreferencesSuccess?.(
  196. $t('preferences.copyPreferencesSuccessTitle'),
  197. $t('preferences.copyPreferencesSuccess'),
  198. );
  199. }
  200. async function handleClearCache() {
  201. resetPreferences();
  202. clearPreferencesCache();
  203. emit('clearPreferencesAndLogout');
  204. }
  205. async function handleReset() {
  206. if (!diffPreference.value) {
  207. return;
  208. }
  209. resetPreferences();
  210. await loadLocaleMessages(preferences.app.locale);
  211. }
  212. </script>
  213. <template>
  214. <div>
  215. <Drawer
  216. :description="$t('preferences.subtitle')"
  217. :title="$t('preferences.title')"
  218. class="!border-0 sm:max-w-sm"
  219. >
  220. <template #extra>
  221. <div class="flex items-center">
  222. <VbenIconButton
  223. :disabled="!diffPreference"
  224. :tooltip="$t('preferences.resetTip')"
  225. class="relative"
  226. @click="handleReset"
  227. >
  228. <span
  229. v-if="diffPreference"
  230. class="bg-primary absolute right-0.5 top-0.5 h-2 w-2 rounded"
  231. ></span>
  232. <RotateCw class="size-4" />
  233. </VbenIconButton>
  234. <VbenIconButton
  235. :tooltip="
  236. appEnableStickyPreferencesNavigationBar
  237. ? $t('preferences.disableStickyPreferencesNavigationBar')
  238. : $t('preferences.enableStickyPreferencesNavigationBar')
  239. "
  240. class="relative"
  241. @click="
  242. () =>
  243. (appEnableStickyPreferencesNavigationBar =
  244. !appEnableStickyPreferencesNavigationBar)
  245. "
  246. >
  247. <PinOff
  248. v-if="appEnableStickyPreferencesNavigationBar"
  249. class="size-4"
  250. />
  251. <Pin v-else class="size-4" />
  252. </VbenIconButton>
  253. </div>
  254. </template>
  255. <div>
  256. <VbenSegmented
  257. v-model="activeTab"
  258. :tabs="tabs"
  259. :class="{
  260. 'sticky-tabs-header': appEnableStickyPreferencesNavigationBar,
  261. }"
  262. >
  263. <template #general>
  264. <Block :title="$t('preferences.general')">
  265. <General
  266. v-model:app-dynamic-title="appDynamicTitle"
  267. v-model:app-enable-check-updates="appEnableCheckUpdates"
  268. v-model:app-locale="appLocale"
  269. v-model:app-watermark="appWatermark"
  270. v-model:app-watermark-content="appWatermarkContent"
  271. />
  272. </Block>
  273. <Block :title="$t('preferences.animation.title')">
  274. <Animation
  275. v-model:transition-enable="transitionEnable"
  276. v-model:transition-loading="transitionLoading"
  277. v-model:transition-name="transitionName"
  278. v-model:transition-progress="transitionProgress"
  279. />
  280. </Block>
  281. </template>
  282. <template #appearance>
  283. <Block :title="$t('preferences.theme.title')">
  284. <Theme
  285. v-model="themeMode"
  286. v-model:theme-semi-dark-header="themeSemiDarkHeader"
  287. v-model:theme-semi-dark-sidebar="themeSemiDarkSidebar"
  288. />
  289. </Block>
  290. <Block :title="$t('preferences.theme.builtin.title')">
  291. <BuiltinTheme
  292. v-model="themeBuiltinType"
  293. v-model:theme-color-primary="themeColorPrimary"
  294. :is-dark="isDark"
  295. />
  296. </Block>
  297. <Block :title="$t('preferences.theme.radius')">
  298. <Radius v-model="themeRadius" />
  299. </Block>
  300. <Block :title="$t('preferences.theme.fontSize')">
  301. <FontSize v-model="themeFontSize" />
  302. </Block>
  303. <Block :title="$t('preferences.other')">
  304. <ColorMode
  305. v-model:app-color-gray-mode="appColorGrayMode"
  306. v-model:app-color-weak-mode="appColorWeakMode"
  307. />
  308. </Block>
  309. </template>
  310. <template #layout>
  311. <Block :title="$t('preferences.layout')">
  312. <Layout v-model="appLayout" />
  313. </Block>
  314. <Block :title="$t('preferences.content')">
  315. <Content v-model="appContentCompact" />
  316. </Block>
  317. <Block :title="$t('preferences.sidebar.title')">
  318. <Sidebar
  319. v-model:sidebar-auto-activate-child="sidebarAutoActivateChild"
  320. v-model:sidebar-collapsed="sidebarCollapsed"
  321. v-model:sidebar-collapsed-show-title="sidebarCollapsedShowTitle"
  322. v-model:sidebar-enable="sidebarEnable"
  323. v-model:sidebar-expand-on-hover="sidebarExpandOnHover"
  324. v-model:sidebar-width="sidebarWidth"
  325. v-model:sidebar-collapsed-button="sidebarCollapsedButton"
  326. v-model:sidebar-fixed-button="sidebarFixedButton"
  327. :current-layout="appLayout"
  328. :disabled="!isSideMode"
  329. />
  330. </Block>
  331. <Block :title="$t('preferences.header.title')">
  332. <Header
  333. v-model:header-enable="headerEnable"
  334. v-model:header-menu-align="headerMenuAlign"
  335. v-model:header-mode="headerMode"
  336. :disabled="isFullContent"
  337. />
  338. </Block>
  339. <Block :title="$t('preferences.navigationMenu.title')">
  340. <Navigation
  341. v-model:navigation-accordion="navigationAccordion"
  342. v-model:navigation-split="navigationSplit"
  343. v-model:navigation-style-type="navigationStyleType"
  344. :disabled="isFullContent"
  345. :disabled-navigation-split="!isMixedNav"
  346. />
  347. </Block>
  348. <Block :title="$t('preferences.breadcrumb.title')">
  349. <Breadcrumb
  350. v-model:breadcrumb-enable="breadcrumbEnable"
  351. v-model:breadcrumb-hide-only-one="breadcrumbHideOnlyOne"
  352. v-model:breadcrumb-show-home="breadcrumbShowHome"
  353. v-model:breadcrumb-show-icon="breadcrumbShowIcon"
  354. v-model:breadcrumb-style-type="breadcrumbStyleType"
  355. :disabled="
  356. !showBreadcrumbConfig ||
  357. !(isSideNav || isSideMixedNav || isHeaderSidebarNav)
  358. "
  359. />
  360. </Block>
  361. <Block :title="$t('preferences.tabbar.title')">
  362. <Tabbar
  363. v-model:tabbar-draggable="tabbarDraggable"
  364. v-model:tabbar-enable="tabbarEnable"
  365. v-model:tabbar-persist="tabbarPersist"
  366. v-model:tabbar-show-icon="tabbarShowIcon"
  367. v-model:tabbar-show-maximize="tabbarShowMaximize"
  368. v-model:tabbar-show-more="tabbarShowMore"
  369. v-model:tabbar-style-type="tabbarStyleType"
  370. v-model:tabbar-wheelable="tabbarWheelable"
  371. v-model:tabbar-max-count="tabbarMaxCount"
  372. v-model:tabbar-middle-click-to-close="tabbarMiddleClickToClose"
  373. />
  374. </Block>
  375. <Block :title="$t('preferences.widget.title')">
  376. <Widget
  377. v-model:app-preferences-button-position="
  378. appPreferencesButtonPosition
  379. "
  380. v-model:widget-fullscreen="widgetFullscreen"
  381. v-model:widget-global-search="widgetGlobalSearch"
  382. v-model:widget-language-toggle="widgetLanguageToggle"
  383. v-model:widget-lock-screen="widgetLockScreen"
  384. v-model:widget-notification="widgetNotification"
  385. v-model:widget-refresh="widgetRefresh"
  386. v-model:widget-sidebar-toggle="widgetSidebarToggle"
  387. v-model:widget-theme-toggle="widgetThemeToggle"
  388. />
  389. </Block>
  390. <Block :title="$t('preferences.footer.title')">
  391. <Footer
  392. v-model:footer-enable="footerEnable"
  393. v-model:footer-fixed="footerFixed"
  394. />
  395. </Block>
  396. <Block
  397. v-if="copyrightSettingShow"
  398. :title="$t('preferences.copyright.title')"
  399. >
  400. <Copyright
  401. v-model:copyright-company-name="copyrightCompanyName"
  402. v-model:copyright-company-site-link="copyrightCompanySiteLink"
  403. v-model:copyright-date="copyrightDate"
  404. v-model:copyright-enable="copyrightEnable"
  405. v-model:copyright-icp="copyrightIcp"
  406. v-model:copyright-icp-link="copyrightIcpLink"
  407. :disabled="!footerEnable"
  408. />
  409. </Block>
  410. </template>
  411. <template #shortcutKey>
  412. <Block :title="$t('preferences.shortcutKeys.global')">
  413. <GlobalShortcutKeys
  414. v-model:shortcut-keys-enable="shortcutKeysEnable"
  415. v-model:shortcut-keys-global-search="shortcutKeysGlobalSearch"
  416. v-model:shortcut-keys-lock-screen="shortcutKeysGlobalLockScreen"
  417. v-model:shortcut-keys-logout="shortcutKeysGlobalLogout"
  418. />
  419. </Block>
  420. </template>
  421. </VbenSegmented>
  422. </div>
  423. <template #footer>
  424. <VbenButton
  425. :disabled="!diffPreference"
  426. class="mx-4 w-full"
  427. size="sm"
  428. variant="default"
  429. @click="handleCopy"
  430. >
  431. <Copy class="mr-2 size-3" />
  432. {{ $t('preferences.copyPreferences') }}
  433. </VbenButton>
  434. <VbenButton
  435. :disabled="!diffPreference"
  436. class="mr-4 w-full"
  437. size="sm"
  438. variant="ghost"
  439. @click="handleClearCache"
  440. >
  441. {{ $t('preferences.clearAndLogout') }}
  442. </VbenButton>
  443. </template>
  444. </Drawer>
  445. </div>
  446. </template>
  447. <style scoped>
  448. :deep(.sticky-tabs-header [role='tablist']) {
  449. position: sticky;
  450. top: -12px;
  451. z-index: 10;
  452. }
  453. </style>