123456789101112131415161718192021222324252627282930313233343536373839404142434445464748495051525354555657585960616263646566676869707172737475767778798081828384858687888990919293949596979899100101102103104105106107108109110111112113114115116117118119120121122123124125126127128129130131132133134135136137138139140141142143144145146147148149150151152153154155156157158159160161162163 |
- // Copyright (C) 2012-2025 Zammad Foundation, https://zammad-foundation.org/
- import { defineStore } from 'pinia'
- import { computed, effectScope, ref, type Ref } from 'vue'
- import { useNotifications } from '#shared/components/CommonNotifications/index.ts'
- import { useApplicationLoaded } from '#shared/composables/useApplicationLoaded.ts'
- import { useApplicationConfigQuery } from '#shared/graphql/queries/applicationConfig.api.ts'
- import { useConfigUpdatesSubscription } from '#shared/graphql/subscriptions/configUpdates.api.ts'
- import type {
- ApplicationConfigQuery,
- ApplicationConfigQueryVariables,
- } from '#shared/graphql/types.ts'
- import {
- QueryHandler,
- SubscriptionHandler,
- } from '#shared/server/apollo/handler/index.ts'
- import type { ConfigList } from '#shared/types/store.ts'
- import testFlags from '#shared/utils/testFlags.ts'
- let configUpdatesSubscriptionInitialized = false
- let applicationConfigQuery: QueryHandler<
- ApplicationConfigQuery,
- ApplicationConfigQueryVariables
- >
- const getApplicationConfigQuery = () => {
- if (applicationConfigQuery) return applicationConfigQuery
- const scope = effectScope()
- scope.run(() => {
- applicationConfigQuery = new QueryHandler(
- useApplicationConfigQuery({ fetchPolicy: 'no-cache' }),
- )
- })
- return applicationConfigQuery
- }
- // TODO: consider switching from notification to a modal dialog, and improving the message
- const notifications = useNotifications()
- const { loaded } = useApplicationLoaded()
- export const useApplicationStore = defineStore(
- 'application',
- () => {
- const loading = computed(() => !loaded.value)
- const initialized = ref(false)
- const setLoaded = (): void => {
- if (notifications.hasErrors()) {
- const loadingAppElement: Maybe<HTMLElement> =
- document.getElementById('loading-app')
- loadingAppElement
- ?.getElementsByClassName('loading-animation')
- .item(0)
- ?.classList.add('error')
- loadingAppElement
- ?.getElementsByClassName('loading-sr-text')
- .item(0)
- ?.setAttribute('aria-hidden', 'true')
- const loadingFailedElement = loadingAppElement
- ?.getElementsByClassName('loading-failed')
- .item(0)
- loadingFailedElement?.classList.add('active')
- loadingFailedElement?.setAttribute('aria-hidden', 'false')
- return
- }
- loaded.value = true
- testFlags.set('applicationLoaded.loaded')
- }
- const config = ref<Record<string, unknown>>({})
- const initializeConfigUpdateSubscription = (): void => {
- const scope = effectScope()
- scope.run(() => {
- const configUpdatesSubscription = new SubscriptionHandler(
- useConfigUpdatesSubscription(),
- )
- configUpdatesSubscription.onResult((result) => {
- const updatedSetting = result.data?.configUpdates.setting
- if (updatedSetting) {
- config.value[updatedSetting.key] = updatedSetting.value
- } else {
- testFlags.set('useConfigUpdatesSubscription.subscribed')
- }
- })
- configUpdatesSubscriptionInitialized = true
- })
- }
- const getConfig = async (): Promise<void> => {
- const configQuery = getApplicationConfigQuery()
- const { data: result } = await configQuery.refetch()
- if (result?.applicationConfig) {
- result.applicationConfig.forEach((item) => {
- config.value[item.key] = item.value
- })
- // app/assets/javascripts/app/config.coffee
- config.value.api_path = '/api/v1'
- }
- if (!configUpdatesSubscriptionInitialized) {
- initializeConfigUpdateSubscription()
- }
- }
- const resetAndGetConfig = async (): Promise<void> => {
- config.value = {}
- await getConfig()
- }
- const hasCustomProductBranding = computed(() =>
- Boolean(
- config.value.product_logo && config.value.product_logo !== 'logo.svg',
- ),
- )
- // this is called right before router renders a page, - we remove "loading" element here instead of "setLoaded"
- // so we don't have a short period when App route is not fetched yet and we see a black screen
- const setInitialized = () => {
- initialized.value = true
- const appElement = document.getElementById('app')
- if (appElement) {
- appElement.dataset.loaded = 'true'
- }
- const loadingAppElement = document.getElementById('loading-app')
- loadingAppElement?.remove()
- }
- return {
- loaded,
- loading,
- setLoaded,
- config: config as unknown as Ref<ConfigList>,
- initializeConfigUpdateSubscription,
- getConfig,
- resetAndGetConfig,
- hasCustomProductBranding,
- initialized,
- setInitialized,
- }
- },
- {
- requiresAuth: false,
- },
- )
|