preferences-sheet.vue 14 KB

123456789101112131415161718192021222324252627282930313233343536373839404142434445464748495051525354555657585960616263646566676869707172737475767778798081828384858687888990919293949596979899100101102103104105106107108109110111112113114115116117118119120121122123124125126127128129130131132133134135136137138139140141142143144145146147148149150151152153154155156157158159160161162163164165166167168169170171172173174175176177178179180181182183184185186187188189190191192193194195196197198199200201202203204205206207208209210211212213214215216217218219220221222223224225226227228229230231232233234235236237238239240241242243244245246247248249250251252253254255256257258259260261262263264265266267268269270271272273274275276277278279280281282283284285286287288289290291292293294295296297298299300301302303304305306307308309310311312313314315316317318319320321322323324325326327328329330331332333334335336337338339340341342343344345346347348349350351352353354355356357358359360361362363364365366367368369370371372373374375376377378379380381382383384385386387388389390391392393394395396397398399400401402403404405406407408409410411412413414415416417418419420421422423424425426427428429
  1. <script setup lang="ts">
  2. import type { SegmentedItem } from '@vben-core/shadcn-ui';
  3. import type {
  4. BreadcrumbStyleType,
  5. BuiltinThemeType,
  6. ContentCompactType,
  7. LayoutHeaderModeType,
  8. LayoutType,
  9. NavigationStyleType,
  10. SupportedLanguagesType,
  11. ThemeModeType,
  12. } from '@vben-core/typings';
  13. import { computed, ref } from 'vue';
  14. import { IcRoundFolderCopy, IcRoundRestartAlt } from '@vben-core/icons';
  15. import { $t, loadLocaleMessages } from '@vben-core/locales';
  16. import {
  17. clearPreferencesCache,
  18. preferences,
  19. resetPreferences,
  20. usePreferences,
  21. } from '@vben-core/preferences';
  22. import {
  23. VbenButton,
  24. VbenIconButton,
  25. VbenSegmented,
  26. VbenSheet,
  27. useToast,
  28. } from '@vben-core/shadcn-ui';
  29. import { useClipboard } from '@vueuse/core';
  30. import {
  31. Animation,
  32. Block,
  33. Breadcrumb,
  34. BuiltinTheme,
  35. ColorMode,
  36. Content,
  37. Copyright,
  38. Footer,
  39. General,
  40. GlobalShortcutKeys,
  41. Header,
  42. Layout,
  43. Navigation,
  44. Radius,
  45. Sidebar,
  46. Tabbar,
  47. Theme,
  48. Widget,
  49. } from './blocks';
  50. import IconSetting from './icons/setting.vue';
  51. import { useOpenPreferences } from './use-open-preferences';
  52. const emit = defineEmits<{ clearPreferencesAndLogout: [] }>();
  53. const { toast } = useToast();
  54. const appLocale = defineModel<SupportedLanguagesType>('appLocale');
  55. const appDynamicTitle = defineModel<boolean>('appDynamicTitle');
  56. const appLayout = defineModel<LayoutType>('appLayout');
  57. const appColorGrayMode = defineModel<boolean>('appColorGrayMode');
  58. const appColorWeakMode = defineModel<boolean>('appColorWeakMode');
  59. const appContentCompact = defineModel<ContentCompactType>('appContentCompact');
  60. const transitionProgress = defineModel<boolean>('transitionProgress');
  61. const transitionName = defineModel<string>('transitionName');
  62. const transitionLoading = defineModel<boolean>('transitionLoading');
  63. const transitionEnable = defineModel<boolean>('transitionEnable');
  64. const themeColorPrimary = defineModel<string>('themeColorPrimary');
  65. const themeBuiltinType = defineModel<BuiltinThemeType>('themeBuiltinType');
  66. const themeMode = defineModel<ThemeModeType>('themeMode');
  67. const themeRadius = defineModel<string>('themeRadius');
  68. const themeSemiDarkMenu = defineModel<boolean>('themeSemiDarkMenu');
  69. const sidebarEnable = defineModel<boolean>('sidebarEnable');
  70. const sidebarWidth = defineModel<number>('sidebarWidth');
  71. const sidebarCollapsed = defineModel<boolean>('sidebarCollapsed');
  72. const sidebarCollapsedShowTitle = defineModel<boolean>(
  73. 'sidebarCollapsedShowTitle',
  74. );
  75. const headerEnable = defineModel<boolean>('headerEnable');
  76. const headerMode = defineModel<LayoutHeaderModeType>('headerMode');
  77. const breadcrumbEnable = defineModel<boolean>('breadcrumbEnable');
  78. const breadcrumbShowIcon = defineModel<boolean>('breadcrumbShowIcon');
  79. const breadcrumbShowHome = defineModel<boolean>('breadcrumbShowHome');
  80. const breadcrumbStyleType = defineModel<BreadcrumbStyleType>(
  81. 'breadcrumbStyleType',
  82. );
  83. const breadcrumbHideOnlyOne = defineModel<boolean>('breadcrumbHideOnlyOne');
  84. const tabbarEnable = defineModel<boolean>('tabbarEnable');
  85. const tabbarShowIcon = defineModel<boolean>('tabbarShowIcon');
  86. const tabbarPersist = defineModel<boolean>('tabbarPersist');
  87. const tabbarDragable = defineModel<boolean>('tabbarDragable');
  88. const tabbarStyleType = defineModel<string>('tabbarStyleType');
  89. const navigationStyleType = defineModel<NavigationStyleType>(
  90. 'navigationStyleType',
  91. );
  92. const navigationSplit = defineModel<boolean>('navigationSplit');
  93. const navigationAccordion = defineModel<boolean>('navigationAccordion');
  94. // const logoVisible = defineModel<boolean>('logoVisible');
  95. const footerEnable = defineModel<boolean>('footerEnable');
  96. const footerFixed = defineModel<boolean>('footerFixed');
  97. const copyrightEnable = defineModel<boolean>('copyrightEnable');
  98. const copyrightCompanyName = defineModel<string>('copyrightCompanyName');
  99. const copyrightCompanySiteLink = defineModel<string>(
  100. 'copyrightCompanySiteLink',
  101. );
  102. const copyrightDate = defineModel<string>('copyrightDate');
  103. const copyrightIcp = defineModel<string>('copyrightIcp');
  104. const copyrightIcpLink = defineModel<string>('copyrightIcpLink');
  105. const shortcutKeysEnable = defineModel<boolean>('shortcutKeysEnable');
  106. const shortcutKeysGlobalSearch = defineModel<boolean>(
  107. 'shortcutKeysGlobalSearch',
  108. );
  109. const shortcutKeysGlobalLogout = defineModel<boolean>(
  110. 'shortcutKeysGlobalLogout',
  111. );
  112. const shortcutKeysGlobalPreferences = defineModel<boolean>(
  113. 'shortcutKeysGlobalPreferences',
  114. );
  115. const shortcutKeysGlobalLockScreen = defineModel<boolean>(
  116. 'shortcutKeysGlobalLockScreen',
  117. );
  118. const widgetGlobalSearch = defineModel<boolean>('widgetGlobalSearch');
  119. const widgetFullscreen = defineModel<boolean>('widgetFullscreen');
  120. const widgetLanguageToggle = defineModel<boolean>('widgetLanguageToggle');
  121. const widgetNotification = defineModel<boolean>('widgetNotification');
  122. const widgetThemeToggle = defineModel<boolean>('widgetThemeToggle');
  123. const widgetAiAssistant = defineModel<boolean>('widgetAiAssistant');
  124. const widgetSidebarToggle = defineModel<boolean>('widgetSidebarToggle');
  125. const widgetLockScreen = defineModel<boolean>('widgetLockScreen');
  126. const {
  127. diffPreference,
  128. isDark,
  129. isFullContent,
  130. isHeaderNav,
  131. isMixedNav,
  132. isSideMixedNav,
  133. isSideMode,
  134. isSideNav,
  135. } = usePreferences();
  136. const { copy } = useClipboard();
  137. const activeTab = ref('appearance');
  138. const tabs = computed((): SegmentedItem[] => {
  139. return [
  140. {
  141. label: $t('preferences.appearance'),
  142. value: 'appearance',
  143. },
  144. {
  145. label: $t('preferences.layout'),
  146. value: 'layout',
  147. },
  148. {
  149. label: $t('preferences.shortcutKeys.title'),
  150. value: 'shortcutKey',
  151. },
  152. {
  153. label: $t('preferences.general'),
  154. value: 'general',
  155. },
  156. ];
  157. });
  158. const showBreadcrumbConfig = computed(() => {
  159. return (
  160. !isFullContent.value &&
  161. !isMixedNav.value &&
  162. !isHeaderNav.value &&
  163. preferences.header.enable
  164. );
  165. });
  166. const { openPreferences } = useOpenPreferences();
  167. async function handleCopy() {
  168. await copy(JSON.stringify(diffPreference.value, null, 2));
  169. toast({
  170. description: $t('preferences.copyPreferences'),
  171. title: $t('preferences.copyPreferencesSuccess'),
  172. });
  173. }
  174. async function handleClearCache() {
  175. resetPreferences();
  176. clearPreferencesCache();
  177. emit('clearPreferencesAndLogout');
  178. }
  179. async function handleReset() {
  180. if (!diffPreference.value) {
  181. return;
  182. }
  183. resetPreferences();
  184. await loadLocaleMessages(preferences.app.locale);
  185. toast({
  186. description: $t('preferences.resetTitle'),
  187. title: $t('preferences.resetSuccess'),
  188. });
  189. }
  190. </script>
  191. <template>
  192. <div class="z-100 fixed right-0 top-1/2">
  193. <VbenSheet
  194. v-model:open="openPreferences"
  195. :description="$t('preferences.subtitle')"
  196. :title="$t('preferences.title')"
  197. >
  198. <template #trigger>
  199. <VbenButton
  200. :title="$t('preferences.title')"
  201. class="bg-primary flex-col-center h-12 w-12 cursor-pointer rounded-l-lg rounded-r-none border-none"
  202. >
  203. <IconSetting
  204. class="duration-3000 fill-primary-foreground animate-spin text-2xl"
  205. />
  206. </VbenButton>
  207. </template>
  208. <template #extra>
  209. <div class="flex items-center">
  210. <VbenIconButton
  211. :disabled="!diffPreference"
  212. :tooltip="$t('preferences.resetTip')"
  213. class="relative"
  214. >
  215. <span
  216. v-if="diffPreference"
  217. class="bg-primary absolute right-0.5 top-0.5 h-2 w-2 rounded"
  218. ></span>
  219. <IcRoundRestartAlt class="size-5" @click="handleReset" />
  220. </VbenIconButton>
  221. </div>
  222. </template>
  223. <div class="p-4 pt-4">
  224. <VbenSegmented v-model="activeTab" :tabs="tabs">
  225. <template #general>
  226. <Block :title="$t('preferences.general')">
  227. <General
  228. v-model:app-dynamic-title="appDynamicTitle"
  229. v-model:app-locale="appLocale"
  230. />
  231. </Block>
  232. <Block :title="$t('preferences.animation.title')">
  233. <Animation
  234. v-model:transition-enable="transitionEnable"
  235. v-model:transition-loading="transitionLoading"
  236. v-model:transition-name="transitionName"
  237. v-model:transition-progress="transitionProgress"
  238. />
  239. </Block>
  240. </template>
  241. <template #appearance>
  242. <Block :title="$t('preferences.theme.title')">
  243. <Theme
  244. v-model="themeMode"
  245. v-model:theme-semi-dark-menu="themeSemiDarkMenu"
  246. />
  247. </Block>
  248. <!-- <Block :title="$t('preferences.theme-color')">
  249. <ThemeColor
  250. v-model="themeColorPrimary"
  251. :color-primary-presets="colorPrimaryPresets"
  252. />
  253. </Block> -->
  254. <Block :title="$t('preferences.theme.builtin.title')">
  255. <BuiltinTheme
  256. v-model="themeBuiltinType"
  257. v-model:theme-color-primary="themeColorPrimary"
  258. :is-dark="isDark"
  259. />
  260. </Block>
  261. <Block :title="$t('preferences.theme.radius')">
  262. <Radius v-model="themeRadius" />
  263. </Block>
  264. <Block :title="$t('preferences.other')">
  265. <ColorMode
  266. v-model:app-color-gray-mode="appColorGrayMode"
  267. v-model:app-color-weak-mode="appColorWeakMode"
  268. />
  269. </Block>
  270. </template>
  271. <template #layout>
  272. <Block :title="$t('preferences.layout')">
  273. <Layout v-model="appLayout" />
  274. </Block>
  275. <Block :title="$t('preferences.content')">
  276. <Content v-model="appContentCompact" />
  277. </Block>
  278. <Block :title="$t('preferences.sidebar.title')">
  279. <Sidebar
  280. v-model:sidebar-collapsed="sidebarCollapsed"
  281. v-model:sidebar-collapsed-show-title="sidebarCollapsedShowTitle"
  282. v-model:sidebar-enable="sidebarEnable"
  283. v-model:sidebar-width="sidebarWidth"
  284. :disabled="!isSideMode"
  285. />
  286. </Block>
  287. <Block :title="$t('preferences.header.title')">
  288. <Header
  289. v-model:headerEnable="headerEnable"
  290. v-model:headerMode="headerMode"
  291. :disabled="isFullContent"
  292. />
  293. </Block>
  294. <Block :title="$t('preferences.navigationMenu.title')">
  295. <Navigation
  296. v-model:navigation-accordion="navigationAccordion"
  297. v-model:navigation-split="navigationSplit"
  298. v-model:navigation-style-type="navigationStyleType"
  299. :disabled="isFullContent"
  300. :disabled-navigation-split="!isMixedNav"
  301. />
  302. </Block>
  303. <Block :title="$t('preferences.breadcrumb.title')">
  304. <Breadcrumb
  305. v-model:breadcrumb-enable="breadcrumbEnable"
  306. v-model:breadcrumb-hide-only-one="breadcrumbHideOnlyOne"
  307. v-model:breadcrumb-show-home="breadcrumbShowHome"
  308. v-model:breadcrumb-show-icon="breadcrumbShowIcon"
  309. v-model:breadcrumb-style-type="breadcrumbStyleType"
  310. :disabled="
  311. !showBreadcrumbConfig || !(isSideNav || isSideMixedNav)
  312. "
  313. />
  314. </Block>
  315. <Block :title="$t('preferences.tabbar.title')">
  316. <Tabbar
  317. v-model:tabbar-dragable="tabbarDragable"
  318. v-model:tabbar-enable="tabbarEnable"
  319. v-model:tabbar-persist="tabbarPersist"
  320. v-model:tabbar-show-icon="tabbarShowIcon"
  321. v-model:tabbar-style-type="tabbarStyleType"
  322. />
  323. </Block>
  324. <Block :title="$t('preferences.widget.title')">
  325. <Widget
  326. v-model:widget-ai-assistant="widgetAiAssistant"
  327. v-model:widget-fullscreen="widgetFullscreen"
  328. v-model:widget-global-search="widgetGlobalSearch"
  329. v-model:widget-language-toggle="widgetLanguageToggle"
  330. v-model:widget-lock-screen="widgetLockScreen"
  331. v-model:widget-notification="widgetNotification"
  332. v-model:widget-sidebar-toggle="widgetSidebarToggle"
  333. v-model:widget-theme-toggle="widgetThemeToggle"
  334. />
  335. </Block>
  336. <Block :title="$t('preferences.footer.title')">
  337. <Footer
  338. v-model:footer-enable="footerEnable"
  339. v-model:footer-fixed="footerFixed"
  340. />
  341. </Block>
  342. <Block :title="$t('preferences.copyright.title')">
  343. <Copyright
  344. v-model:copyright-company-name="copyrightCompanyName"
  345. v-model:copyright-company-site-link="copyrightCompanySiteLink"
  346. v-model:copyright-date="copyrightDate"
  347. v-model:copyright-enable="copyrightEnable"
  348. v-model:copyright-icp="copyrightIcp"
  349. v-model:copyright-icp-link="copyrightIcpLink"
  350. :disabled="!footerEnable"
  351. />
  352. </Block>
  353. </template>
  354. <template #shortcutKey>
  355. <Block :title="$t('preferences.shortcutKeys.global')">
  356. <GlobalShortcutKeys
  357. v-model:shortcut-keys-enable="shortcutKeysEnable"
  358. v-model:shortcut-keys-global-search="shortcutKeysGlobalSearch"
  359. v-model:shortcut-keys-lock-screen="shortcutKeysGlobalLockScreen"
  360. v-model:shortcut-keys-logout="shortcutKeysGlobalLogout"
  361. v-model:shortcut-keys-preferences="
  362. shortcutKeysGlobalPreferences
  363. "
  364. />
  365. </Block>
  366. </template>
  367. </VbenSegmented>
  368. </div>
  369. <template #footer>
  370. <VbenButton
  371. :disabled="!diffPreference"
  372. class="mx-4 w-full"
  373. size="sm"
  374. variant="default"
  375. @click="handleCopy"
  376. >
  377. <IcRoundFolderCopy class="mr-2 size-3" />
  378. {{ $t('preferences.copyPreferences') }}
  379. </VbenButton>
  380. <VbenButton
  381. :disabled="!diffPreference"
  382. class="mr-4 w-full"
  383. size="sm"
  384. variant="ghost"
  385. @click="handleClearCache"
  386. >
  387. <!-- <IcRoundRestartAlt class="mr-2 size-4" /> -->
  388. {{ $t('preferences.clearAndLogout') }}
  389. </VbenButton>
  390. </template>
  391. </VbenSheet>
  392. </div>
  393. </template>