123456789101112131415161718192021222324252627282930313233343536373839404142434445464748495051525354555657585960616263646566676869707172737475767778798081828384858687888990919293949596979899100101102103104105106107108109110111112113114115116117118119120121122123124125126127128129130131132133134135136137138139140141142143144145146147148149150151152153154155156157158159160161162163164165166167168169170171172173174175176177178179180181182183184185186187188189190191192193194195196197198199200201 |
- <!-- Copyright (C) 2012-2025 Zammad Foundation, https://zammad-foundation.org/ -->
- <script setup lang="ts">
- /* eslint-disable vue/no-v-html */
- import { storeToRefs } from 'pinia'
- import { computed, ref } from 'vue'
- import { useRouter } from 'vue-router'
- import { useRawHTMLIcon } from '#shared/components/CommonIcon/useRawHTMLIcon.ts'
- import CommonUserAvatar from '#shared/components/CommonUserAvatar/CommonUserAvatar.vue'
- import FormGroup from '#shared/components/Form/FormGroup.vue'
- import { useForceDesktop } from '#shared/composables/useForceDesktop.ts'
- import { useLocaleUpdate } from '#shared/composables/useLocaleUpdate.ts'
- import { useProductAboutQuery } from '#shared/graphql/queries/about.api.ts'
- import { i18n } from '#shared/i18n.ts'
- import { QueryHandler } from '#shared/server/apollo/handler/index.ts'
- import { useSessionStore } from '#shared/stores/session.ts'
- import { browser, os } from '#shared/utils/browser.ts'
- import { usePWASupport, isStandalone } from '#shared/utils/pwa.ts'
- import CommonSectionMenu from '#mobile/components/CommonSectionMenu/CommonSectionMenu.vue'
- import CommonSectionMenuItem from '#mobile/components/CommonSectionMenu/CommonSectionMenuItem.vue'
- import CommonSectionMenuLink from '#mobile/components/CommonSectionMenu/CommonSectionMenuLink.vue'
- import CommonSectionPopup from '#mobile/components/CommonSectionPopup/CommonSectionPopup.vue'
- import { useHeader } from '#mobile/composables/useHeader.ts'
- const router = useRouter()
- const logout = () => {
- router.push('/logout')
- }
- useHeader({
- title: __('Account'),
- backUrl: '/',
- backAvoidHomeButton: true,
- })
- const session = useSessionStore()
- const { user } = storeToRefs(session)
- const { modelCurrentLocale, localeOptions, isSavingLocale, translation } =
- useLocaleUpdate()
- const hasVersionPermission = session.hasPermission('admin')
- const productAboutQuery = new QueryHandler(
- useProductAboutQuery({
- enabled: hasVersionPermission,
- fetchPolicy: 'cache-first',
- }),
- { errorNotificationMessage: __('The product version could not be fetched.') },
- )
- const productAbout = productAboutQuery.result()
- const isMobileIOS = browser.name?.includes('Safari') && os.name?.includes('iOS')
- const { canInstallPWA, installPWA } = usePWASupport()
- const showInstallButton = computed(
- () => !isStandalone() && (canInstallPWA.value || isMobileIOS),
- )
- const showInstallIOSPopup = ref(false)
- const installPWAMessage = computed(() => {
- const iconShare = useRawHTMLIcon({
- class: 'inline-flex text-blue',
- decorative: true,
- size: 'small',
- name: 'ios-share',
- })
- const iconAdd = useRawHTMLIcon({
- class: 'inline-flex',
- decorative: true,
- size: 'small',
- name: 'add-square',
- })
- return i18n.t(
- __(
- 'To install %s as an app, press the %s "Share" button and then the %s "Add to Home Screen" button.',
- ),
- __('Zammad'),
- iconShare,
- iconAdd,
- )
- })
- const installZammadPWA = () => {
- if (isStandalone()) return
- // on chromium this will show a chrome popup with native "install" button
- if (canInstallPWA.value) {
- installPWA()
- return
- }
- // on iOS we cannot install it with native functionality, so we show
- // instructions on how to install it
- // let's pray Apple will add native functionality in the future
- if (isMobileIOS) {
- showInstallIOSPopup.value = true
- }
- }
- const { forceDesktop } = useForceDesktop()
- </script>
- <template>
- <div class="px-4">
- <div v-if="user" class="flex flex-col items-center justify-center py-6">
- <div>
- <CommonUserAvatar :entity="user" size="xl" personal />
- </div>
- <div class="mt-2 text-xl font-bold">
- {{ user.firstname }} {{ user.lastname }}
- </div>
- </div>
- <CommonSectionMenu>
- <CommonSectionMenuLink
- v-if="session.hasPermission('user_preferences.avatar')"
- :icon="{ name: 'person', size: 'base' }"
- icon-bg="bg-pink"
- link="/account/avatar"
- >
- {{ $t('Avatar') }}
- </CommonSectionMenuLink>
- <CommonSectionMenuLink
- v-if="showInstallButton"
- :icon="{ name: 'install', size: 'small' }"
- icon-bg="bg-blue"
- @click="installZammadPWA"
- >
- {{ $t('Install App') }}
- </CommonSectionMenuLink>
- <CommonSectionMenuLink
- link="/"
- link-external
- :icon="{ name: 'desktop', size: 'base' }"
- icon-bg="bg-orange"
- @click="forceDesktop"
- >
- {{ $t('Continue to desktop') }}
- </CommonSectionMenuLink>
- </CommonSectionMenu>
- <!--
- The shorthand "no-options-label-translation" is not working currently because of a FormKit limitation,
- therefore we need to explicitly set it to true.
- -->
- <FormGroup v-if="session.hasPermission('user_preferences.language')">
- <FormKit
- v-model="modelCurrentLocale"
- type="treeselect"
- :label="__('Language')"
- :options="localeOptions"
- :disabled="isSavingLocale"
- :no-options-label-translation="true"
- sorting="label"
- />
- <template #help>
- {{ $t('Did you know?') }}
- <CommonLink class="text-blue" target="_blank" :link="translation.link">
- {{ $t('You can help translating Zammad.') }}
- </CommonLink>
- </template>
- </FormGroup>
- <CommonSectionMenu v-if="hasVersionPermission">
- <CommonSectionMenuItem :label="__('Version')">
- {{
- $t('This is Zammad version %s', productAbout?.productAbout as string)
- }}
- </CommonSectionMenuItem>
- </CommonSectionMenu>
- <div class="mb-4">
- <FormKit
- variant="danger"
- wrapper-class="mt-4 text-base flex grow justify-center items-center"
- input-class="py-2 px-4 w-full h-14 rounded-xl select-none"
- type="submit"
- name="signout"
- @click="logout"
- >
- {{ $t('Sign out') }}
- </FormKit>
- </div>
- <CommonSectionPopup v-model:state="showInstallIOSPopup" :messages="[]">
- <template #header>
- <section class="inline-flex min-h-[54px] items-center p-3">
- <span v-html="installPWAMessage" />
- </section>
- </template>
- </CommonSectionPopup>
- </div>
- </template>
|