mirror of
https://github.com/hcengineering/platform.git
synced 2025-04-05 15:31:37 +00:00
295 lines
9.2 KiB
Svelte
295 lines
9.2 KiB
Svelte
<!--
|
|
// Copyright © 2023 Hardcore Engineering Inc.
|
|
//
|
|
// Licensed under the Eclipse Public License, Version 2.0 (the "License");
|
|
// you may not use this file except in compliance with the License. You may
|
|
// obtain a copy of the License at https://www.eclipse.org/legal/epl-2.0
|
|
//
|
|
// Unless required by applicable law or agreed to in writing, software
|
|
// distributed under the License is distributed on an "AS IS" BASIS,
|
|
// WITHOUT WARRANTIES OR CONDITIONS OF ANY KIND, either express or implied.
|
|
//
|
|
// See the License for the specific language governing permissions and
|
|
// limitations under the License.
|
|
-->
|
|
<script lang="ts">
|
|
import notification, {
|
|
ActivityNotificationViewlet,
|
|
DisplayInboxNotification,
|
|
DocNotifyContext
|
|
} from '@hcengineering/notification'
|
|
import { ActionContext, getClient } from '@hcengineering/presentation'
|
|
import view, { Viewlet } from '@hcengineering/view'
|
|
import {
|
|
AnyComponent,
|
|
Component,
|
|
defineSeparators,
|
|
getLocation,
|
|
Label,
|
|
Loading,
|
|
location as locationStore,
|
|
navigate,
|
|
Scroller,
|
|
Separator,
|
|
TabItem,
|
|
TabList,
|
|
Location
|
|
} from '@hcengineering/ui'
|
|
import chunter from '@hcengineering/chunter'
|
|
import { Ref, WithLookup } from '@hcengineering/core'
|
|
import { ViewletSelector } from '@hcengineering/view-resources'
|
|
import activity from '@hcengineering/activity'
|
|
|
|
import { InboxNotificationsClientImpl } from '../../inboxNotificationsClient'
|
|
import Filter from '../Filter.svelte'
|
|
import { getDisplayInboxNotifications, resolveLocation } from '../../utils'
|
|
import { InboxNotificationsFilter } from '../../types'
|
|
import { onDestroy } from 'svelte'
|
|
|
|
export let visibleNav: boolean = true
|
|
export let navFloat: boolean = false
|
|
export let appsDirection: 'vertical' | 'horizontal' = 'horizontal'
|
|
|
|
const client = getClient()
|
|
const hierarchy = client.getHierarchy()
|
|
const inboxClient = InboxNotificationsClientImpl.getClient()
|
|
const notificationsByContextStore = inboxClient.inboxNotificationsByContext
|
|
const notifyContextsStore = inboxClient.docNotifyContexts
|
|
|
|
const allTab: TabItem = {
|
|
id: 'all',
|
|
labelIntl: notification.string.All
|
|
}
|
|
const channelTab: TabItem = {
|
|
id: chunter.class.Channel,
|
|
labelIntl: chunter.string.Channels
|
|
}
|
|
const directTab: TabItem = {
|
|
id: chunter.class.DirectMessage,
|
|
labelIntl: chunter.string.Direct
|
|
}
|
|
|
|
let displayNotifications: DisplayInboxNotification[] = []
|
|
let displayContextsIds = new Set<Ref<DocNotifyContext>>()
|
|
|
|
let filteredNotifications: DisplayInboxNotification[] = []
|
|
let filter: InboxNotificationsFilter = 'all'
|
|
|
|
let tabItems: TabItem[] = []
|
|
let selectedTabId: string = allTab.id
|
|
|
|
let selectedContextId: Ref<DocNotifyContext> | undefined = undefined
|
|
let selectedContext: DocNotifyContext | undefined = undefined
|
|
let selectedComponent: AnyComponent | undefined = undefined
|
|
|
|
let viewlets: ActivityNotificationViewlet[] = []
|
|
|
|
let viewlet: WithLookup<Viewlet> | undefined
|
|
let loading = true
|
|
|
|
client.findAll(notification.class.ActivityNotificationViewlet, {}).then((res) => {
|
|
viewlets = res
|
|
})
|
|
|
|
$: displayNotifications = getDisplayInboxNotifications($notificationsByContextStore, filter)
|
|
$: displayContextsIds = new Set(displayNotifications.map(({ docNotifyContext }) => docNotifyContext))
|
|
|
|
$: filteredNotifications = filterNotifications(selectedTabId, displayNotifications, $notifyContextsStore)
|
|
|
|
locationStore.subscribe((newLocation) => {
|
|
syncLocation(newLocation)
|
|
})
|
|
|
|
async function syncLocation (newLocation: Location) {
|
|
const loc = await resolveLocation(newLocation)
|
|
|
|
selectedContextId = loc?.loc.fragment as Ref<DocNotifyContext> | undefined
|
|
|
|
if (selectedContextId !== selectedContext?._id) {
|
|
selectedContext = undefined
|
|
}
|
|
}
|
|
|
|
$: selectedContext = selectedContextId
|
|
? selectedContext ?? $notifyContextsStore.find(({ _id }) => _id === selectedContextId)
|
|
: undefined
|
|
|
|
$: updateSelectedPanel(selectedContext)
|
|
$: updateTabItems(displayContextsIds, $notifyContextsStore)
|
|
|
|
function updateTabItems (displayContextsIds: Set<Ref<DocNotifyContext>>, notifyContexts: DocNotifyContext[]): void {
|
|
const displayClasses = new Set(
|
|
notifyContexts
|
|
.filter(
|
|
({ _id, attachedToClass }) =>
|
|
displayContextsIds.has(_id) && !hierarchy.isDerived(attachedToClass, activity.class.ActivityMessage)
|
|
)
|
|
.map(({ attachedToClass }) => attachedToClass)
|
|
)
|
|
|
|
const fixedTabs = [
|
|
allTab,
|
|
displayClasses.has(chunter.class.Channel) ? channelTab : undefined,
|
|
displayClasses.has(chunter.class.DirectMessage) ? directTab : undefined
|
|
].filter((tab): tab is TabItem => tab !== undefined)
|
|
|
|
tabItems = fixedTabs.concat(
|
|
Array.from(displayClasses.values())
|
|
.filter((_class) => ![chunter.class.Channel, chunter.class.DirectMessage].includes(_class))
|
|
.map((_class) => ({
|
|
id: _class,
|
|
// TODO: need to get plural form
|
|
labelIntl: hierarchy.getClass(_class).label
|
|
}))
|
|
)
|
|
}
|
|
|
|
function selectTab (event: CustomEvent) {
|
|
if (event.detail !== undefined) {
|
|
selectedTabId = event.detail.id
|
|
}
|
|
}
|
|
|
|
async function selectContext (event?: CustomEvent) {
|
|
selectedContext = event?.detail?.context
|
|
selectedContextId = selectedContext?._id
|
|
|
|
const loc = getLocation()
|
|
|
|
if (selectedContext !== undefined) {
|
|
loc.fragment = selectedContext._id
|
|
loc.query = { message: event?.detail?.notification?.attachedTo ?? null }
|
|
} else {
|
|
loc.fragment = undefined
|
|
loc.query = undefined
|
|
}
|
|
|
|
navigate(loc)
|
|
}
|
|
|
|
async function updateSelectedPanel (selectedContext?: DocNotifyContext) {
|
|
if (selectedContext === undefined) {
|
|
selectedComponent = undefined
|
|
return
|
|
}
|
|
|
|
const isChunterChannel = hierarchy.isDerived(selectedContext.attachedToClass, chunter.class.ChunterSpace)
|
|
const panelComponent = hierarchy.classHierarchyMixin(selectedContext.attachedToClass, view.mixin.ObjectPanel)
|
|
|
|
selectedComponent = panelComponent?.component ?? view.component.EditDoc
|
|
|
|
const contextNotifications = $notificationsByContextStore.get(selectedContext._id) ?? []
|
|
|
|
await inboxClient.readNotifications(
|
|
contextNotifications
|
|
.filter(({ _class, isViewed }) =>
|
|
isChunterChannel ? _class === notification.class.CommonInboxNotification : !isViewed
|
|
)
|
|
.map(({ _id }) => _id)
|
|
)
|
|
}
|
|
|
|
function filterNotifications (
|
|
selectedTabId: string,
|
|
displayNotifications: DisplayInboxNotification[],
|
|
notifyContexts: DocNotifyContext[]
|
|
) {
|
|
if (selectedTabId === allTab.id) {
|
|
return displayNotifications
|
|
}
|
|
|
|
return displayNotifications.filter(({ docNotifyContext }) => {
|
|
const context = notifyContexts.find(({ _id }) => _id === docNotifyContext)
|
|
|
|
return context !== undefined && context.attachedToClass === selectedTabId
|
|
})
|
|
}
|
|
|
|
defineSeparators('inbox', [
|
|
{ minSize: 30, maxSize: 50, size: 40, float: 'navigator' },
|
|
{ size: 'auto', minSize: 30, maxSize: 'auto', float: undefined }
|
|
])
|
|
</script>
|
|
|
|
<ActionContext
|
|
context={{
|
|
mode: 'browser'
|
|
}}
|
|
/>
|
|
|
|
<div class="flex-row-top h-full">
|
|
{#if visibleNav}
|
|
<div
|
|
class="antiPanel-navigator {appsDirection === 'horizontal'
|
|
? 'portrait'
|
|
: 'landscape'} background-comp-header-color"
|
|
>
|
|
<div class="antiPanel-wrap__content">
|
|
<div class="ac-header full divide caption-height">
|
|
<div class="ac-header__wrap-title mr-3">
|
|
<span class="ac-header__title"><Label label={notification.string.Inbox} /></span>
|
|
</div>
|
|
<ViewletSelector bind:viewlet bind:loading viewletQuery={{ attachTo: notification.class.DocNotifyContext }} />
|
|
<div class="flex flex-gap-2">
|
|
<Filter bind:filter />
|
|
</div>
|
|
</div>
|
|
|
|
<div class="tabs">
|
|
<TabList items={tabItems} selected={selectedTabId} on:select={selectTab} />
|
|
</div>
|
|
|
|
{#if loading || !viewlet?.$lookup?.descriptor}
|
|
<Loading />
|
|
{:else if viewlet}
|
|
<Scroller>
|
|
<div class="notifications">
|
|
<Component
|
|
is={viewlet.$lookup.descriptor.component}
|
|
props={{
|
|
notifications: filteredNotifications,
|
|
viewlets,
|
|
selectedContext
|
|
}}
|
|
on:click={selectContext}
|
|
/>
|
|
</div>
|
|
</Scroller>
|
|
{/if}
|
|
</div>
|
|
<Separator name="inbox" float={navFloat ? 'navigator' : true} index={0} />
|
|
</div>
|
|
<Separator name="inbox" float={navFloat} index={0} />
|
|
{/if}
|
|
<div class="antiPanel-component filled w-full">
|
|
{#if selectedContext && selectedComponent}
|
|
<Component
|
|
is={selectedComponent}
|
|
props={{
|
|
_id: selectedContext.attachedTo,
|
|
_class: selectedContext.attachedToClass,
|
|
embedded: true,
|
|
context: selectedContext,
|
|
props: { context: selectedContext }
|
|
}}
|
|
on:close={() => selectContext(undefined)}
|
|
/>
|
|
{/if}
|
|
</div>
|
|
</div>
|
|
|
|
<style lang="scss">
|
|
.tabs {
|
|
display: flex;
|
|
margin: 0.5rem;
|
|
padding-bottom: 0.5rem;
|
|
border-bottom: 1px solid var(--theme-navpanel-border);
|
|
}
|
|
|
|
.notifications {
|
|
margin: 0.5rem;
|
|
padding: 0.5rem;
|
|
height: 100%;
|
|
}
|
|
</style>
|