mirror of
https://github.com/hcengineering/platform.git
synced 2025-04-05 07:19:32 +00:00
Merge remote-tracking branch 'origin/develop' into staging
Signed-off-by: Andrey Sobolev <haiodo@gmail.com>
This commit is contained in:
commit
ac4c5455d3
@ -32,7 +32,7 @@ export async function syncFiles (
|
||||
): Promise<void> {
|
||||
if (exAdapter.adapters === undefined) return
|
||||
|
||||
for (const [name, adapter] of exAdapter.adapters.entries()) {
|
||||
for (const [name, adapter] of [...exAdapter.adapters.entries()].reverse()) {
|
||||
await adapter.make(ctx, workspaceId)
|
||||
|
||||
await retryOnFailure(ctx, 5, async () => {
|
||||
@ -47,7 +47,12 @@ export async function syncFiles (
|
||||
|
||||
for (const data of dataBulk) {
|
||||
const blob = await exAdapter.stat(ctx, workspaceId, data._id)
|
||||
if (blob !== undefined) continue
|
||||
if (blob !== undefined) {
|
||||
if (blob.provider !== name && name === exAdapter.defaultAdapter) {
|
||||
await exAdapter.syncBlobFromStorage(ctx, workspaceId, data._id, exAdapter.defaultAdapter)
|
||||
}
|
||||
continue
|
||||
}
|
||||
|
||||
await exAdapter.syncBlobFromStorage(ctx, workspaceId, data._id, name)
|
||||
|
||||
@ -167,6 +172,13 @@ async function processAdapter (
|
||||
let targetBlob: Blob | ListBlobResult | undefined = targetBlobs.get(data._id)
|
||||
if (targetBlob !== undefined) {
|
||||
console.log('Target blob already exists', targetBlob._id)
|
||||
|
||||
const aggrBlob = await exAdapter.stat(ctx, workspaceId, data._id)
|
||||
if (aggrBlob === undefined || aggrBlob?.provider !== targetBlob.provider) {
|
||||
targetBlob = await exAdapter.syncBlobFromStorage(ctx, workspaceId, targetBlob._id, exAdapter.defaultAdapter)
|
||||
}
|
||||
// We could safely delete source blob
|
||||
toRemove.push(data._id)
|
||||
}
|
||||
|
||||
if (targetBlob === undefined) {
|
||||
@ -176,32 +188,28 @@ async function processAdapter (
|
||||
console.error('blob not found', data._id)
|
||||
continue
|
||||
}
|
||||
await rateLimiter.exec(async () => {
|
||||
targetBlob = await rateLimiter.exec(async () => {
|
||||
try {
|
||||
await retryOnFailure(
|
||||
const result = await retryOnFailure(
|
||||
ctx,
|
||||
5,
|
||||
async () => {
|
||||
await processFile(ctx, source, target, workspaceId, sourceBlob)
|
||||
// We need to sync and update aggregator table for now.
|
||||
targetBlob = await exAdapter.syncBlobFromStorage(
|
||||
ctx,
|
||||
workspaceId,
|
||||
sourceBlob._id,
|
||||
exAdapter.defaultAdapter
|
||||
)
|
||||
return await exAdapter.syncBlobFromStorage(ctx, workspaceId, sourceBlob._id, exAdapter.defaultAdapter)
|
||||
},
|
||||
50
|
||||
)
|
||||
movedCnt += 1
|
||||
movedBytes += sourceBlob.size
|
||||
batchBytes += sourceBlob.size
|
||||
return result
|
||||
} catch (err) {
|
||||
console.error('failed to process blob', data._id, err)
|
||||
}
|
||||
})
|
||||
|
||||
if (targetBlob !== undefined && 'size' in targetBlob && (targetBlob as Blob).size === sourceBlob.size) {
|
||||
if (targetBlob !== undefined) {
|
||||
// We could safely delete source blob
|
||||
toRemove.push(sourceBlob._id)
|
||||
}
|
||||
@ -233,7 +241,10 @@ async function processAdapter (
|
||||
|
||||
await rateLimiter.waitProcessing()
|
||||
if (toRemove.length > 0 && params.move) {
|
||||
await source.remove(ctx, workspaceId, toRemove)
|
||||
while (toRemove.length > 0) {
|
||||
const part = toRemove.splice(0, 500)
|
||||
await source.remove(ctx, workspaceId, part)
|
||||
}
|
||||
}
|
||||
} finally {
|
||||
await iterator.close()
|
||||
|
@ -221,8 +221,12 @@ export function createModel (builder: Builder): void {
|
||||
|
||||
builder.createDoc<ActivityMessageControl<ChunterSpace>>(activity.class.ActivityMessageControl, core.space.Model, {
|
||||
objectClass: chunter.class.DirectMessage,
|
||||
skip: [{ _class: core.class.TxMixin }, { _class: core.class.TxCreateDoc }, { _class: core.class.TxRemoveDoc }],
|
||||
allowedFields: ['members']
|
||||
skip: [
|
||||
{ _class: core.class.TxMixin },
|
||||
{ _class: core.class.TxCreateDoc },
|
||||
{ _class: core.class.TxRemoveDoc },
|
||||
{ _class: core.class.TxUpdateDoc }
|
||||
]
|
||||
})
|
||||
|
||||
builder.createDoc(activity.class.DocUpdateMessageViewlet, core.space.Model, {
|
||||
@ -241,16 +245,6 @@ export function createModel (builder: Builder): void {
|
||||
}
|
||||
})
|
||||
|
||||
builder.createDoc(activity.class.DocUpdateMessageViewlet, core.space.Model, {
|
||||
objectClass: chunter.class.DirectMessage,
|
||||
action: 'update',
|
||||
config: {
|
||||
members: {
|
||||
presenter: chunter.activity.MembersChangedMessage
|
||||
}
|
||||
}
|
||||
})
|
||||
|
||||
builder.mixin(chunter.class.ChatMessage, core.class.Class, activity.mixin.ActivityMessagePreview, {
|
||||
presenter: chunter.component.ChatMessagePreview
|
||||
})
|
||||
|
@ -351,6 +351,17 @@ export const chunterOperation: MigrateOperation = {
|
||||
func: async (client) => {
|
||||
await removeDuplicatedDirects(client)
|
||||
}
|
||||
},
|
||||
{
|
||||
state: 'remove-direct-members-messages',
|
||||
func: async (client) => {
|
||||
await client.deleteMany<DocUpdateMessage>(DOMAIN_ACTIVITY, {
|
||||
_class: activity.class.DocUpdateMessage,
|
||||
attachedToClass: chunter.class.DirectMessage,
|
||||
action: 'update',
|
||||
'attributeUpdates.attrKey': 'members'
|
||||
})
|
||||
}
|
||||
}
|
||||
])
|
||||
},
|
||||
|
@ -105,6 +105,7 @@ export class TTxSidebarEvent extends TTx implements TxSidebarEvent {
|
||||
@UX(workbench.string.Tab)
|
||||
export class TWorkbenchTab extends TPreference implements WorkbenchTab {
|
||||
location!: string
|
||||
name?: string
|
||||
isPinned!: boolean
|
||||
}
|
||||
|
||||
|
@ -20,15 +20,16 @@
|
||||
export let node: MarkupNode
|
||||
export let preview = false
|
||||
|
||||
const is = diffview.component.Highlight
|
||||
|
||||
$: language = node.attrs?.language
|
||||
$: content = node.content ?? []
|
||||
$: value = content.map((node) => node.text).join('/n')
|
||||
$: margin = preview ? '0' : null
|
||||
|
||||
$: props = { value, language }
|
||||
</script>
|
||||
|
||||
{#if node}
|
||||
<pre class="proseCodeBlock" style:margin={preview ? '0' : null}>
|
||||
<code>
|
||||
<Component is={diffview.component.Highlight} props={{ value, language }} />
|
||||
</code>
|
||||
</pre>
|
||||
<pre class="proseCodeBlock" style:margin><code><Component {is} {props} /></code></pre>
|
||||
{/if}
|
||||
|
@ -73,9 +73,10 @@
|
||||
<div class="close-button {orientation}">
|
||||
<ButtonIcon icon={IconClose} size="min" on:click={() => dispatch('close')} />
|
||||
</div>
|
||||
{:else}
|
||||
{:else if $$slots.postfix === undefined}
|
||||
<div />
|
||||
{/if}
|
||||
<slot name="postfix" />
|
||||
</div>
|
||||
|
||||
<style lang="scss">
|
||||
|
@ -98,11 +98,15 @@
|
||||
<svelte:fragment slot="notify">
|
||||
{#if count != null && count > 0}
|
||||
<div class="antiHSpacer" />
|
||||
<NotifyMarker {count} />
|
||||
<div class="notify">
|
||||
<NotifyMarker {count} />
|
||||
</div>
|
||||
<div class="antiHSpacer" />
|
||||
{:else if secondaryNotifyMarker}
|
||||
<div class="antiHSpacer" />
|
||||
<NotifyMarker count={0} kind="with-dot" />
|
||||
<div class="notify">
|
||||
<NotifyMarker count={0} kind="simple" size="xx-small" />
|
||||
</div>
|
||||
<div class="antiHSpacer" />
|
||||
{/if}
|
||||
</svelte:fragment>
|
||||
@ -126,4 +130,12 @@
|
||||
background-color: var(--global-ui-highlight-BackgroundColor);
|
||||
}
|
||||
}
|
||||
|
||||
.notify {
|
||||
display: flex;
|
||||
align-items: center;
|
||||
justify-content: center;
|
||||
width: 1rem;
|
||||
height: 1rem;
|
||||
}
|
||||
</style>
|
||||
|
@ -14,7 +14,7 @@
|
||||
-->
|
||||
<script lang="ts">
|
||||
export let count: number = 0
|
||||
export let kind: 'primary' | 'simple' | 'with-dot' = 'primary'
|
||||
export let kind: 'primary' | 'simple' = 'primary'
|
||||
export let size: 'xx-small' | 'x-small' | 'small' | 'medium' = 'small'
|
||||
|
||||
const maxNumber = 9
|
||||
@ -34,10 +34,6 @@
|
||||
<div class="notifyMarker {size} {kind}" />
|
||||
{/if}
|
||||
|
||||
{#if kind === 'with-dot'}
|
||||
<div class="notifyMarker {size} {kind}">●</div>
|
||||
{/if}
|
||||
|
||||
<style lang="scss">
|
||||
.notifyMarker {
|
||||
display: flex;
|
||||
@ -48,7 +44,6 @@
|
||||
font-weight: 700;
|
||||
|
||||
&.simple,
|
||||
&.with-dot,
|
||||
&.primary {
|
||||
background-color: var(--global-higlight-Color);
|
||||
color: var(--global-on-accent-TextColor);
|
||||
@ -75,10 +70,5 @@
|
||||
height: 1.25rem;
|
||||
font-size: 0.625rem;
|
||||
}
|
||||
|
||||
&.with-dot {
|
||||
font-weight: 400;
|
||||
font-size: 0.25rem;
|
||||
}
|
||||
}
|
||||
</style>
|
||||
|
@ -87,6 +87,9 @@
|
||||
<path fill-rule="evenodd" clip-rule="evenodd"
|
||||
d="M21.7071 2.29289C21.3166 1.90237 20.6834 1.90237 20.2929 2.29289C19.9024 2.68342 19.9024 3.31658 20.2929 3.70711L20.5858 4L14.9628 9.623C14.7214 9.86439 14.394 10 14.0526 10H13.2426C10.2609 10 7.4013 11.1845 5.29289 13.2929C4.90237 13.6834 4.90237 14.3166 5.29289 14.7071L10.5859 20.0001L2 28.5859V30H3.41436L12.0001 21.4143L17.2929 26.7071C17.6834 27.0976 18.3166 27.0976 18.7071 26.7071C20.8155 24.5987 22 21.7391 22 18.7574V17.9474C22 17.606 22.1356 17.2786 22.377 17.0372L28 11.4142L28.2929 11.7071C28.6834 12.0976 29.3166 12.0976 29.7071 11.7071C30.0976 11.3166 30.0976 10.6834 29.7071 10.2929L21.7071 2.29289ZM16.377 11.0372L22 5.41421L26.5858 10L20.9628 15.623C20.3463 16.2395 20 17.0756 20 17.9474V18.7574C20 20.873 19.2746 22.9139 17.9617 24.5474L7.45256 14.0383C9.08614 12.7254 11.127 12 13.2426 12H14.0526C14.9244 12 15.7605 11.6537 16.377 11.0372Z"/>
|
||||
</symbol>
|
||||
<symbol id="pin-tack" viewBox="0 0 18 18">
|
||||
<line x1="9" y1="16.25" x2="9" y2="12.25" stroke-linecap="round" stroke-linejoin="round" stroke="#212121"></line><path d="M14.25,12.25c-.089-.699-.318-1.76-.969-2.875-.335-.574-.703-1.028-1.031-1.375V3.75c0-1.105-.895-2-2-2h-2.5c-1.105,0-2,.895-2,2v4.25c-.329,.347-.697,.801-1.031,1.375-.65,1.115-.88,2.176-.969,2.875H14.25Z" stroke-linecap="round" stroke-linejoin="round"></path>
|
||||
</symbol>
|
||||
<symbol id="model" viewBox="0 0 24 24">
|
||||
<path d="M17.2,7.8h3.6c1.1,0,2-0.9,2-2V4.2c0-1.1-0.9-2-2-2h-3.6c-1.1,0-2,0.9-2,2v0H9.8V4c0-1.5-1.2-2.8-2.8-2.8H4 C2.5,1.2,1.2,2.5,1.2,4v2c0,1.5,1.2,2.8,2.8,2.8h3c1.5,0,2.8-1.2,2.8-2.8V5.8h2V18c0,1.5,1.2,2.8,2.8,2.8h0.8v0c0,1.1,0.9,2,2,2h3.6 c1.1,0,2-0.9,2-2v-1.6c0-1.1-0.9-2-2-2h-3.6c-1.1,0-2,0.9-2,2v0h-0.8c-0.7,0-1.2-0.6-1.2-1.2v-4.8h2v0c0,1.1,0.9,2,2,2h3.6 c1.1,0,2-0.9,2-2v-1.6c0-1.1-0.9-2-2-2h-3.6c-1.1,0-2,0.9-2,2v0h-2v-6h2v0C15.2,6.9,16.1,7.8,17.2,7.8z M8.2,6 c0,0.7-0.6,1.2-1.2,1.2H4C3.3,7.2,2.8,6.7,2.8,6V4c0-0.7,0.6-1.2,1.2-1.2h3c0.7,0,1.2,0.6,1.2,1.2V6z M16.8,19.2 c0-0.2,0.2-0.5,0.5-0.5h3.6c0.2,0,0.5,0.2,0.5,0.5v1.6c0,0.2-0.2,0.5-0.5,0.5h-3.6c-0.2,0-0.5-0.2-0.5-0.5V19.2z M16.8,11.7 c0-0.2,0.2-0.5,0.5-0.5h3.6c0.2,0,0.5,0.2,0.5,0.5v1.6c0,0.2-0.2,0.5-0.5,0.5h-3.6c-0.2,0-0.5-0.2-0.5-0.5V11.7z M16.8,4.2 c0-0.2,0.2-0.5,0.5-0.5h3.6c0.2,0,0.5,0.2,0.5,0.5v1.6c0,0.2-0.2,0.5-0.5,0.5h-3.6c-0.2,0-0.5-0.2-0.5-0.5V4.2z"/>
|
||||
</symbol>
|
||||
|
Before Width: | Height: | Size: 46 KiB After Width: | Height: | Size: 47 KiB |
@ -61,5 +61,6 @@ loadMetadata(view.icon, {
|
||||
Undo: `${icons}#undo`,
|
||||
Video: `${icons}#video`,
|
||||
Audio: `${icons}#audio`,
|
||||
File: `${icons}#file`
|
||||
File: `${icons}#file`,
|
||||
PinTack: `${icons}#pin-tack`
|
||||
})
|
||||
|
@ -257,7 +257,8 @@ const view = plugin(viewId, {
|
||||
Undo: '' as Asset,
|
||||
Video: '' as Asset,
|
||||
Audio: '' as Asset,
|
||||
File: '' as Asset
|
||||
File: '' as Asset,
|
||||
PinTack: '' as Asset
|
||||
},
|
||||
category: {
|
||||
General: '' as Ref<ActionCategory>,
|
||||
|
@ -13,85 +13,51 @@
|
||||
// limitations under the License.
|
||||
-->
|
||||
<script lang="ts">
|
||||
import { AnySvelteComponent, closePanel, getCurrentLocation, Location, ModernTab, navigate } from '@hcengineering/ui'
|
||||
import {
|
||||
AnySvelteComponent,
|
||||
closePanel,
|
||||
getCurrentLocation,
|
||||
Icon,
|
||||
ModernTab,
|
||||
navigate,
|
||||
languageStore,
|
||||
locationToUrl
|
||||
} from '@hcengineering/ui'
|
||||
import { ComponentExtensions, getClient } from '@hcengineering/presentation'
|
||||
import { Asset, getResource, IntlString } from '@hcengineering/platform'
|
||||
import { type Application, WorkbenchTab } from '@hcengineering/workbench'
|
||||
import { Class, Doc, Ref } from '@hcengineering/core'
|
||||
import { Asset, getResource, translate } from '@hcengineering/platform'
|
||||
import { WorkbenchTab } from '@hcengineering/workbench'
|
||||
import view from '@hcengineering/view'
|
||||
import { parseLinkId, showMenu } from '@hcengineering/view-resources'
|
||||
import { Analytics } from '@hcengineering/analytics'
|
||||
import { showMenu } from '@hcengineering/view-resources'
|
||||
|
||||
import { closeTab, getTabLocation, selectTab, tabIdStore, tabsStore } from '../workbench'
|
||||
import { closeTab, getTabDataByLocation, getTabLocation, selectTab, tabIdStore, tabsStore } from '../workbench'
|
||||
import workbench from '../plugin'
|
||||
|
||||
export let tab: WorkbenchTab
|
||||
|
||||
const client = getClient()
|
||||
const linkProviders = client.getModel().findAllSync(view.mixin.LinkIdProvider, {})
|
||||
|
||||
let name: string | undefined = undefined
|
||||
let label: IntlString | undefined = undefined
|
||||
let name: string | undefined = tab.name
|
||||
let icon: Asset | AnySvelteComponent | undefined
|
||||
let iconProps: Record<string, any> | undefined
|
||||
|
||||
async function getResolvedLocation (loc: Location, app?: Application): Promise<Location> {
|
||||
if (app === undefined) return loc
|
||||
if (app.locationResolver) {
|
||||
const resolver = await getResource(app.locationResolver)
|
||||
return (await resolver(loc))?.loc ?? loc
|
||||
}
|
||||
|
||||
return loc
|
||||
}
|
||||
|
||||
async function getTabName (loc: Location): Promise<string | undefined> {
|
||||
if (loc.fragment == null) return
|
||||
const hierarchy = client.getHierarchy()
|
||||
const [, id, _class] = decodeURIComponent(loc.fragment).split('|')
|
||||
if (_class == null) return
|
||||
|
||||
const mixin = hierarchy.classHierarchyMixin(_class as Ref<Class<Doc>>, view.mixin.ObjectTitle)
|
||||
if (mixin === undefined) return
|
||||
const titleProvider = await getResource(mixin.titleProvider)
|
||||
try {
|
||||
const _id = await parseLinkId(linkProviders, id, _class as Ref<Class<Doc>>)
|
||||
return await titleProvider(client, _id)
|
||||
} catch (err: any) {
|
||||
Analytics.handleError(err)
|
||||
console.error(err)
|
||||
}
|
||||
}
|
||||
|
||||
async function updateTabData (tab: WorkbenchTab): Promise<void> {
|
||||
async function updateTabData (tab: WorkbenchTab, lang: string): Promise<void> {
|
||||
const tabLoc = $tabIdStore === tab._id ? getCurrentLocation() : getTabLocation(tab)
|
||||
const alias = tabLoc.path[2]
|
||||
const application = client.getModel().findAllSync<Application>(workbench.class.Application, { alias })[0]
|
||||
const data = await getTabDataByLocation(tabLoc)
|
||||
|
||||
if (application?.locationDataResolver) {
|
||||
const resolver = await getResource(application.locationDataResolver)
|
||||
const data = await resolver(tabLoc)
|
||||
name = data.name
|
||||
label = data.nameIntl ?? application.label ?? workbench.string.Tab
|
||||
|
||||
if (data.iconComponent) {
|
||||
icon = await getResource(data.iconComponent)
|
||||
} else {
|
||||
icon = data.icon ?? application?.icon
|
||||
}
|
||||
iconProps = data.iconProps
|
||||
name = data.name ?? (await translate(data.label, {}, lang))
|
||||
if (data.iconComponent !== undefined) {
|
||||
icon = await getResource(data.iconComponent)
|
||||
} else {
|
||||
const special = tabLoc.path[3]
|
||||
const specialLabel = application?.navigatorModel?.specials?.find((s) => s.id === special)?.label
|
||||
const resolvedLoc = await getResolvedLocation(tabLoc, application)
|
||||
name = await getTabName(resolvedLoc)
|
||||
label = specialLabel ?? application?.label ?? workbench.string.Tab
|
||||
icon = application?.icon
|
||||
iconProps = undefined
|
||||
icon = data.icon
|
||||
}
|
||||
iconProps = data.iconProps
|
||||
|
||||
if (tab.name !== name && tab.location === locationToUrl(tabLoc)) {
|
||||
await client.update(tab, { name })
|
||||
}
|
||||
}
|
||||
|
||||
$: void updateTabData(tab)
|
||||
$: void updateTabData(tab, $languageStore)
|
||||
|
||||
function handleClickTab (): void {
|
||||
selectTab(tab._id)
|
||||
@ -118,7 +84,6 @@
|
||||
</script>
|
||||
|
||||
<ModernTab
|
||||
labelIntl={label}
|
||||
label={name}
|
||||
{icon}
|
||||
{iconProps}
|
||||
@ -132,4 +97,9 @@
|
||||
<svelte:fragment slot="prefix">
|
||||
<ComponentExtensions extension={workbench.extensions.WorkbenchTabExtensions} props={{ tab }} />
|
||||
</svelte:fragment>
|
||||
<svelte:fragment slot="postfix">
|
||||
{#if tab.isPinned}
|
||||
<Icon icon={view.icon.PinTack} size="x-small" />
|
||||
{/if}
|
||||
</svelte:fragment>
|
||||
</ModernTab>
|
||||
|
@ -13,20 +13,26 @@
|
||||
// limitations under the License.
|
||||
//
|
||||
import { derived, get, writable } from 'svelte/store'
|
||||
import core, { concatLink, getCurrentAccount, type Ref } from '@hcengineering/core'
|
||||
import workbench, { type WorkbenchTab } from '@hcengineering/workbench'
|
||||
import core, { type Class, concatLink, type Doc, getCurrentAccount, type Ref } from '@hcengineering/core'
|
||||
import { type Application, workbenchId, type WorkbenchTab } from '@hcengineering/workbench'
|
||||
import {
|
||||
location as locationStore,
|
||||
locationToUrl,
|
||||
parseLocation,
|
||||
type Location,
|
||||
navigate,
|
||||
getCurrentLocation
|
||||
getCurrentLocation,
|
||||
languageStore,
|
||||
type AnyComponent
|
||||
} from '@hcengineering/ui'
|
||||
import presentation, { getClient } from '@hcengineering/presentation'
|
||||
import { getMetadata } from '@hcengineering/platform'
|
||||
import view from '@hcengineering/view'
|
||||
import { type Asset, type IntlString, getMetadata, getResource, translate } from '@hcengineering/platform'
|
||||
import { parseLinkId } from '@hcengineering/view-resources'
|
||||
import { notificationId } from '@hcengineering/notification'
|
||||
|
||||
import { workspaceStore } from './utils'
|
||||
import workbench from './plugin'
|
||||
|
||||
export const tabIdStore = writable<Ref<WorkbenchTab> | undefined>()
|
||||
export const tabsStore = writable<WorkbenchTab[]>([])
|
||||
@ -38,7 +44,14 @@ workspaceStore.subscribe((workspace) => {
|
||||
tabIdStore.set(getTabFromLocalStorage(workspace ?? ''))
|
||||
})
|
||||
|
||||
locationStore.subscribe((loc) => {
|
||||
locationStore.subscribe(() => {
|
||||
void syncTabLoc()
|
||||
})
|
||||
|
||||
tabIdStore.subscribe(saveTabToLocalStorage)
|
||||
|
||||
async function syncTabLoc (): Promise<void> {
|
||||
const loc = getCurrentLocation()
|
||||
const workspace = get(workspaceStore)
|
||||
if (workspace == null || workspace === '') return
|
||||
const tab = get(currentTabStore)
|
||||
@ -51,11 +64,22 @@ locationStore.subscribe((loc) => {
|
||||
return
|
||||
}
|
||||
if (loc.path[2] === '' || loc.path[2] == null) return
|
||||
void getClient().update(tab, { location: locationToUrl(loc) })
|
||||
})
|
||||
|
||||
tabIdStore.subscribe(saveTabToLocalStorage)
|
||||
const data = await getTabDataByLocation(loc)
|
||||
const name = data.name ?? (await translate(data.label, {}, get(languageStore)))
|
||||
|
||||
if (tab.name !== undefined && name !== tab.name && tab.isPinned) {
|
||||
const me = getCurrentAccount()
|
||||
const _id = await getClient().createDoc(workbench.class.WorkbenchTab, core.space.Workspace, {
|
||||
location: locationToUrl(loc),
|
||||
name,
|
||||
attachedTo: me._id,
|
||||
isPinned: false
|
||||
})
|
||||
selectTab(_id)
|
||||
} else {
|
||||
await getClient().update(tab, { location: locationToUrl(loc), name })
|
||||
}
|
||||
}
|
||||
export function syncWorkbenchTab (): void {
|
||||
const workspace = get(workspaceStore)
|
||||
tabIdStore.set(getTabFromLocalStorage(workspace ?? ''))
|
||||
@ -90,7 +114,7 @@ export function selectTab (_id: Ref<WorkbenchTab>): void {
|
||||
tabIdStore.set(_id)
|
||||
}
|
||||
|
||||
export function getTabLocation (tab: WorkbenchTab): Location {
|
||||
export function getTabLocation (tab: Pick<WorkbenchTab, 'location'>): Location {
|
||||
const base = `${window.location.protocol}//${window.location.host}`
|
||||
const front = getMetadata(presentation.metadata.FrontUrl) ?? base
|
||||
const url = new URL(concatLink(front, tab.location))
|
||||
@ -120,13 +144,15 @@ export async function createTab (): Promise<void> {
|
||||
const loc = getCurrentLocation()
|
||||
const client = getClient()
|
||||
const me = getCurrentAccount()
|
||||
const defaultUrl = `${workbenchId}/${loc.path[1]}/${notificationId}`
|
||||
const tab = await client.createDoc(workbench.class.WorkbenchTab, core.space.Workspace, {
|
||||
attachedTo: me._id,
|
||||
location: locationToUrl(loc),
|
||||
location: defaultUrl,
|
||||
isPinned: false
|
||||
})
|
||||
|
||||
selectTab(tab)
|
||||
navigate(getTabLocation({ location: defaultUrl }))
|
||||
}
|
||||
|
||||
export function canCloseTab (tab: WorkbenchTab): boolean {
|
||||
@ -143,3 +169,66 @@ export async function unpinTab (tab: WorkbenchTab): Promise<void> {
|
||||
const client = getClient()
|
||||
await client.update(tab, { isPinned: false })
|
||||
}
|
||||
|
||||
export async function getTabDataByLocation (loc: Location): Promise<{
|
||||
name?: string
|
||||
label: IntlString
|
||||
icon?: Asset
|
||||
iconComponent?: AnyComponent
|
||||
iconProps?: Record<string, any>
|
||||
}> {
|
||||
const client = getClient()
|
||||
const appAlias = loc.path[2]
|
||||
const application = client.getModel().findAllSync<Application>(workbench.class.Application, { alias: appAlias })[0]
|
||||
|
||||
let name: string | undefined
|
||||
let label: IntlString | undefined
|
||||
let icon: Asset | undefined
|
||||
let iconComponent: AnyComponent | undefined
|
||||
let iconProps: Record<string, any> | undefined
|
||||
|
||||
if (application?.locationDataResolver != null) {
|
||||
const resolver = await getResource(application.locationDataResolver)
|
||||
const data = await resolver(loc)
|
||||
name = data.name
|
||||
label = data.nameIntl ?? application.label ?? workbench.string.Tab
|
||||
iconComponent = data.iconComponent
|
||||
icon = data.icon ?? application.icon
|
||||
iconProps = data.iconProps
|
||||
} else {
|
||||
const special = loc.path[3]
|
||||
const specialLabel = application?.navigatorModel?.specials?.find((s) => s.id === special)?.label
|
||||
const resolvedLoc = await getResolvedLocation(loc, application)
|
||||
name = await getDefaultTabName(resolvedLoc)
|
||||
label = specialLabel ?? application?.label ?? workbench.string.Tab
|
||||
icon = application?.icon
|
||||
}
|
||||
|
||||
return { name, label, icon, iconComponent, iconProps }
|
||||
}
|
||||
|
||||
async function getResolvedLocation (loc: Location, app?: Application): Promise<Location> {
|
||||
if (app?.locationResolver === undefined) return loc
|
||||
|
||||
const resolver = await getResource(app.locationResolver)
|
||||
return (await resolver(loc))?.loc ?? loc
|
||||
}
|
||||
|
||||
async function getDefaultTabName (loc: Location): Promise<string | undefined> {
|
||||
if (loc.fragment == null) return
|
||||
const client = getClient()
|
||||
const hierarchy = client.getHierarchy()
|
||||
const [, id, _class] = decodeURIComponent(loc.fragment).split('|')
|
||||
if (_class == null) return
|
||||
|
||||
const mixin = hierarchy.classHierarchyMixin(_class as Ref<Class<Doc>>, view.mixin.ObjectTitle)
|
||||
if (mixin === undefined) return
|
||||
const titleProvider = await getResource(mixin.titleProvider)
|
||||
try {
|
||||
const linkProviders = client.getModel().findAllSync(view.mixin.LinkIdProvider, {})
|
||||
const _id = await parseLinkId(linkProviders, id, _class as Ref<Class<Doc>>)
|
||||
return await titleProvider(client, _id)
|
||||
} catch (err: any) {
|
||||
console.error(err)
|
||||
}
|
||||
}
|
||||
|
@ -113,6 +113,7 @@ export interface TxSidebarEvent<T extends Record<string, any> = Record<string, a
|
||||
export interface WorkbenchTab extends Preference {
|
||||
location: string
|
||||
isPinned: boolean
|
||||
name?: string
|
||||
}
|
||||
|
||||
/**
|
||||
|
@ -62,13 +62,13 @@ export class AggregatorStorageAdapter implements StorageAdapter, StorageAdapterE
|
||||
}
|
||||
}
|
||||
|
||||
const provider = this.adapters.get(current?.provider ?? this.defaultAdapter)
|
||||
const provider = this.adapters.get(providerId ?? current?.provider ?? this.defaultAdapter)
|
||||
if (provider === undefined) {
|
||||
throw new NoSuchKeyError('No such provider found')
|
||||
}
|
||||
const stat = await provider.stat(ctx, workspaceId, objectName)
|
||||
if (stat !== undefined) {
|
||||
stat.provider = current?.provider ?? this.defaultAdapter
|
||||
stat.provider = providerId ?? current?.provider ?? this.defaultAdapter
|
||||
if (current !== undefined) {
|
||||
await this.dbAdapter.clean(ctx, workspaceId, DOMAIN_BLOB, [current._id])
|
||||
}
|
||||
|
@ -628,9 +628,17 @@ export abstract class IssueSyncManagerBase {
|
||||
itemId: target.prjData?.id as string
|
||||
})
|
||||
} catch (err: any) {
|
||||
if (err.errors?.[0]?.type === 'NOT_FOUND') {
|
||||
errors.push({ error: err, response })
|
||||
return errors
|
||||
}
|
||||
Analytics.handleError(err)
|
||||
// Failed to update one particular value, skip it.
|
||||
this.ctx.error('error during field update', { error: err, response })
|
||||
this.ctx.error('error during field update', {
|
||||
error: err,
|
||||
response,
|
||||
workspace: this.provider.getWorkspaceId().name
|
||||
})
|
||||
errors.push({ error: err, response })
|
||||
}
|
||||
}
|
||||
@ -892,10 +900,9 @@ export abstract class IssueSyncManagerBase {
|
||||
}
|
||||
if (fieldsUpdate.length > 0 && syncToProject && target.prjData !== undefined) {
|
||||
const errors = await this.updateIssueValues(target, okit, fieldsUpdate)
|
||||
if (errors.length > 0) {
|
||||
return { externalVersion: '', needUpdate: githubSyncVersion, error: errors }
|
||||
if (errors.length === 0) {
|
||||
needExternalSync = true
|
||||
}
|
||||
needExternalSync = true
|
||||
}
|
||||
// TODO: Add support for labels, milestone, assignees
|
||||
}
|
||||
|
@ -971,6 +971,10 @@ export class PullRequestSyncManager extends IssueSyncManagerBase implements DocS
|
||||
return { needSync: githubSyncVersion }
|
||||
}
|
||||
|
||||
if (info.repository == null) {
|
||||
return { needSync: githubSyncVersion }
|
||||
}
|
||||
|
||||
const pullRequestExternal = info.external as unknown as PullRequestExternalData
|
||||
|
||||
if (info.externalVersion !== githubExternalSyncVersion) {
|
||||
|
@ -265,7 +265,14 @@ export class RepositorySyncMapper implements DocSyncManager {
|
||||
|
||||
let allRepos: GithubIntegrationRepository[] = [...allRepositories]
|
||||
|
||||
const githubRepos:
|
||||
| Repository
|
||||
| Endpoints['GET /installation/repositories']['response']['data']['repositories'][0][] = []
|
||||
for await (const { repository } of iterable) {
|
||||
githubRepos.push(repository)
|
||||
}
|
||||
|
||||
for (const repository of githubRepos) {
|
||||
const integrationRepo: GithubIntegrationRepository | undefined = allRepos.find(
|
||||
(it) => it.repositoryId === repository.id
|
||||
)
|
||||
@ -325,13 +332,8 @@ export class RepositorySyncMapper implements DocSyncManager {
|
||||
|
||||
// Ok we have repos removed from integration, we need to delete them.
|
||||
for (const repo of allRepos) {
|
||||
await this.client.remove(repo)
|
||||
const prj = projects.find((it) => it._id === repo.githubProject)
|
||||
if (prj !== undefined) {
|
||||
await this.client.update(prj, {
|
||||
$pull: { repositories: repo._id }
|
||||
})
|
||||
}
|
||||
// Mark as archived
|
||||
await this.client.update(repo, { archived: true })
|
||||
}
|
||||
|
||||
// We need to delete and disconnect missing repositories.
|
||||
|
@ -395,9 +395,6 @@ export class GithubWorker implements IntegrationManager {
|
||||
periodicSyncPromise: Promise<void> | undefined
|
||||
async performPeriodicSync (): Promise<void> {
|
||||
try {
|
||||
for (const inst of this.integrations.values()) {
|
||||
await this.repositoryManager.reloadRepositories(inst)
|
||||
}
|
||||
this.triggerUpdate()
|
||||
} catch (err: any) {
|
||||
Analytics.handleError(err)
|
||||
|
Loading…
Reference in New Issue
Block a user