123456789101112131415161718192021222324252627282930313233343536373839404142434445464748495051525354555657585960616263646566676869707172737475767778798081828384858687888990919293949596979899100101102103104105106107108109110111112113114115116117118119120121122123124125126127128129130131132133134135136137138139140141 |
- // Copyright (C) 2012-2025 Zammad Foundation, https://zammad-foundation.org/
- import { keyBy } from 'lodash-es'
- import { computed } from 'vue'
- import { useTicketView } from '#shared/entities/ticket/composables/useTicketView.ts'
- import { useMentionSubscribeMutation } from '#shared/entities/ticket/graphql/mutations/subscribe.api.ts'
- import { useMentionUnsubscribeMutation } from '#shared/entities/ticket/graphql/mutations/unsubscribe.api.ts'
- import type { TicketById } from '#shared/entities/ticket/types.ts'
- import type { TicketQuery } from '#shared/graphql/types.ts'
- import { MutationHandler } from '#shared/server/apollo/handler/index.ts'
- import { useSessionStore } from '#shared/stores/session.ts'
- import type { Ref } from 'vue'
- export const useTicketSubscribe = (ticket: Ref<TicketById | undefined>) => {
- const { isTicketAgent } = useTicketView(ticket)
- const canManageSubscription = computed(() => isTicketAgent.value)
- const session = useSessionStore()
- const createTicketCacheUpdater = (subscribed: boolean) => {
- return (previousQuery: Record<string, unknown>) => {
- const prev = previousQuery as TicketQuery
- if (!ticket.value || !prev || prev.ticket?.id !== ticket.value.id) {
- return prev
- }
- return {
- ticket: {
- ...prev.ticket,
- subscribed,
- },
- }
- }
- }
- const subscribeHandler = new MutationHandler(
- useMentionSubscribeMutation({
- updateQueries: {
- ticket: createTicketCacheUpdater(true),
- },
- }),
- )
- const unsubscribeMutation = new MutationHandler(
- useMentionUnsubscribeMutation({
- updateQueries: {
- ticket: createTicketCacheUpdater(false),
- },
- }),
- )
- const isSubscriptionLoading = computed(() => {
- return (
- subscribeHandler.loading().value || unsubscribeMutation.loading().value
- )
- })
- const subscribe = async (ticketId: string) => {
- const result = await subscribeHandler.send({ ticketId })
- return !!result?.mentionSubscribe?.success
- }
- const unsubscribe = async (ticketId: string) => {
- const result = await unsubscribeMutation.send({ ticketId })
- return !!result?.mentionUnsubscribe?.success
- }
- const toggleSubscribe = async () => {
- if (!ticket.value || isSubscriptionLoading.value) return false
- const { id, subscribed } = ticket.value
- if (!subscribed) {
- return subscribe(id)
- }
- return unsubscribe(id)
- }
- const isSubscribed = computed(() => !!ticket.value?.subscribed)
- const subscribers = computed(
- () =>
- ticket.value?.mentions?.edges
- ?.filter(({ node }) => node.user.active)
- .map(({ node }) => ({
- user: node.user,
- access: node.userTicketAccess,
- })) || [],
- )
- const subscribersWithoutMe = computed(
- () =>
- ticket.value?.mentions?.edges
- ?.filter(({ node }) => node.user.id !== session.userId)
- .map(({ node }) => node.user) || [],
- )
- const subscribersAccessLookup = computed(() =>
- keyBy(
- ticket.value?.mentions?.edges
- ?.filter(({ node }) => node.user.id !== session.userId)
- .map(({ node }) => ({
- userId: node.user.id,
- access: node.userTicketAccess,
- })) || [],
- 'userId',
- ),
- )
- const hasMe = computed(() => {
- if (!ticket.value?.mentions) return false
- return ticket.value.mentions.edges.some(
- ({ node }) => node.user.id === session.userId,
- )
- })
- const totalSubscribers = computed(() => {
- if (!ticket.value?.mentions) return 0
- return ticket.value.mentions.totalCount
- })
- const totalSubscribersWithoutMe = computed(() => {
- if (!ticket.value?.mentions) return 0
- // -1 for current user, who is shown as toggler
- return ticket.value.mentions.totalCount - (hasMe.value ? 1 : 0)
- })
- return {
- isSubscriptionLoading,
- isSubscribed,
- toggleSubscribe,
- canManageSubscription,
- subscribers,
- totalSubscribers,
- subscribersWithoutMe,
- subscribersAccessLookup,
- totalSubscribersWithoutMe,
- hasMe,
- }
- }
|