preferences-sheet.vue 14 KB

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