EQMS-1474: Folder deletion fixes (#8214)

Signed-off-by: Victor Ilyushchenko <alt13ri@gmail.com>
This commit is contained in:
Victor Ilyushchenko 2025-03-14 12:26:48 +03:00 committed by GitHub
parent f48db496ae
commit 5f17580acb
No known key found for this signature in database
GPG Key ID: B5690EEEBB952194
10 changed files with 134 additions and 63 deletions

View File

@ -302,7 +302,10 @@
"MakeDocumentObsoleteDialog": "Označit {count, plural, one {dokument jako zastaralý} other {dokumenty jako zastaralé}}",
"MakeDocumentObsoleteConfirm": "Opravdu chcete označit následující dokumenty jako zastaralé: {titles}?",
"LatestVersionHint": "nejnovější"
"LatestVersionHint": "nejnovější",
"CannotDeleteFolder": "Složku nelze smazat",
"CannotDeleteFolderHint": "Před odstraněním složky prosím přesuňte všechny podřízené dokumenty na jiné místo."
},
"controlledDocStates": {
"Empty": "",

View File

@ -309,7 +309,10 @@
"MakeDocumentObsoleteDialog": "{count, plural, one {Dokument als veraltet markieren} other {Dokumente als veraltet markieren}}",
"MakeDocumentObsoleteConfirm": "Möchten Sie die folgenden Dokumente wirklich als veraltet markieren: {titles}?",
"LatestVersionHint": "neueste"
"LatestVersionHint": "neueste",
"CannotDeleteFolder": "Der Ordner kann nicht gelöscht werden",
"CannotDeleteFolderHint": "Bitte verschieben Sie alle untergeordneten Dokumente an einen anderen Ort, bevor Sie den Ordner löschen."
},
"controlledDocStates": {
"Empty": "",

View File

@ -311,7 +311,10 @@
"RenameFolder": "Rename folder",
"CreateChildFolder": "Create child folder",
"LatestVersionHint": "latest"
"LatestVersionHint": "latest",
"CannotDeleteFolder": "The folder cannot be deleted",
"CannotDeleteFolderHint": "Please move all child documents to another location before deleting the folder."
},
"controlledDocStates": {
"Empty": "",

View File

@ -269,7 +269,10 @@
"MakeDocumentObsoleteDialog": "Marquer {count, plural, one {le document comme obsolète} other {les documents comme obsolètes}}",
"MakeDocumentObsoleteConfirm": "Voulez-vous vraiment marquer les documents suivants comme obsolètes : {titles} ?",
"LatestVersionHint": "dernier"
"LatestVersionHint": "dernier",
"CannotDeleteFolder": "Le dossier ne peut pas être supprimé",
"CannotDeleteFolderHint": "Veuillez déplacer tous les documents enfants vers un autre emplacement avant de supprimer le dossier."
},
"controlledDocStates": {
"Empty": "",

View File

@ -267,7 +267,10 @@
"MakeDocumentObsoleteDialog": "Segna {count, plural, one {il documento come obsoleto} other {i documenti come obsoleti}}",
"MakeDocumentObsoleteConfirm": "Vuoi davvero segnare i seguenti documenti come obsoleti: {titles}?",
"LatestVersionHint": "ultimo"
"LatestVersionHint": "ultimo",
"CannotDeleteFolder": "Impossibile eliminare la cartella",
"CannotDeleteFolderHint": "Sposta tutti i documenti figli in un'altra posizione prima di eliminare la cartella."
},
"controlledDocStates": {
"Empty": "",

View File

@ -311,7 +311,10 @@
"MakeDocumentObsoleteDialog": "Пометить {count, plural, one {документ как устаревший} other {документы как устаревшие}}",
"MakeDocumentObsoleteConfirm": "Вы действительно хотите пометить следующие документы как устаревшие: {titles}?",
"LatestVersionHint": "последняя"
"LatestVersionHint": "последняя",
"CannotDeleteFolder": "Папка не может быть удалена",
"CannotDeleteFolderHint": "Пожалуйста, переместите все дочерние документы в другое место перед удалением папки."
},
"controlledDocStates": {
"Empty": "",

View File

@ -308,7 +308,10 @@
"MakeDocumentObsoleteDialog": "标记 {count, plural, one {文档为过时} other {文档为过时}}",
"MakeDocumentObsoleteConfirm": "您确定要将以下文档标记为过时吗:{titles}",
"LatestVersionHint": "最新"
"LatestVersionHint": "最新",
"CannotDeleteFolder": "无法删除文件夹",
"CannotDeleteFolderHint": "请在删除文件夹之前将所有子文档移动到其他位置。"
},
"controlledDocStates": {
"Empty": "",

View File

@ -214,7 +214,10 @@ export default mergeIds(documentsId, documents, {
CreateDocumentTemplateFailed: '' as IntlString,
TryAgain: '' as IntlString,
LatestVersionHint: '' as IntlString
LatestVersionHint: '' as IntlString,
CannotDeleteFolder: '' as IntlString,
CannotDeleteFolderHint: '' as IntlString
},
controlledDocStates: {
Empty: '' as IntlString,

View File

@ -33,7 +33,8 @@ import documents, {
compareDocumentVersions,
emptyBundle,
getDocumentName,
getFirstRank
getFirstRank,
transferDocuments
} from '@hcengineering/controlled-documents'
import core, {
type Class,
@ -53,7 +54,7 @@ import core, {
getCurrentAccount
} from '@hcengineering/core'
import { type IntlString, translate } from '@hcengineering/platform'
import { createQuery, getClient } from '@hcengineering/presentation'
import { createQuery, getClient, MessageBox } from '@hcengineering/presentation'
import request, { type Request, RequestStatus } from '@hcengineering/request'
import { isEmptyMarkup } from '@hcengineering/text'
import { type Location, getUserTimezone, showPopup } from '@hcengineering/ui'
@ -551,46 +552,22 @@ export async function canRenameFolder (
return await isEditableProject(doc.project)
}
export async function canDeleteFolder (obj?: Doc | Doc[]): Promise<boolean> {
if (obj == null) {
return false
}
export async function canDeleteFolder (doc: ProjectDocument): Promise<boolean> {
if (doc?._class === undefined) return false
const client = getClient()
const hierarchy = client.getHierarchy()
const objs = (Array.isArray(obj) ? obj : [obj]) as Document[]
const isFolders = objs.every((doc) => isFolder(hierarchy, doc))
if (!isFolders) {
return false
}
const folders = objs as unknown as ProjectDocument[]
const pjMeta = await client.findAll(documents.class.ProjectMeta, { _id: { $in: folders.map((f) => f.attachedTo) } })
const directChildren = await client.findAll(documents.class.ProjectMeta, {
parent: { $in: pjMeta.map((p) => p.meta) }
})
if (directChildren.length > 0) {
if (!isFolder(hierarchy, doc)) {
return false
}
const currentUser = getCurrentAccount() as PersonAccount
const isOwner = objs.every((doc) => doc.owner === currentUser.person)
if (isOwner) {
if (doc.createdBy === currentUser._id) {
return true
}
const spaces = new Set(objs.map((doc) => doc.space))
return await Promise.all(
Array.from(spaces).map(
async (space) => await checkPermission(getClient(), documents.permission.ArchiveDocument, space)
)
).then((res) => res.every((r) => r))
return await checkPermission(getClient(), documents.permission.ArchiveDocument, doc.space)
}
export async function canDeleteDocumentCategory (doc?: Doc | Doc[]): Promise<boolean> {
@ -835,25 +812,70 @@ export async function renameFolder (doc: ProjectDocument): Promise<void> {
showPopup(documents.component.CreateFolder, props)
}
export async function deleteFolder (obj: ProjectDocument | ProjectDocument[]): Promise<void> {
export async function deleteFolder (obj: ProjectDocument): Promise<void> {
const success = await _deleteFolder(obj)
if (!success) {
showPopup(MessageBox, {
label: documentsResources.string.CannotDeleteFolder,
message: documentsResources.string.CannotDeleteFolderHint,
canSubmit: false
})
}
}
async function _deleteFolder (obj: ProjectDocument): Promise<boolean> {
const client = getClient()
if (!(await canDeleteFolder(obj))) {
return
return false
}
const objs = Array.isArray(obj) ? obj : [obj]
const space = obj.space
const project = obj.project
const pjmeta = await client.findAll(documents.class.ProjectMeta, { _id: { $in: objs.map((p) => p.attachedTo) } })
const meta = await client.findAll(documents.class.DocumentMeta, { _id: { $in: pjmeta.map((p) => p.meta) } })
const bundle: DocumentBundle = {
...emptyBundle(),
ProjectMeta: await client.findAll(documents.class.ProjectMeta, { space, project }),
ProjectDocument: await client.findAll(documents.class.ProjectDocument, { space, project }),
DocumentMeta: await client.findAll(documents.class.DocumentMeta, { space }),
ControlledDocument: await client.findAll(documents.class.ControlledDocument, { space })
}
const prjMeta = bundle.ProjectMeta.find((m) => m._id === obj.attachedTo)
if (prjMeta === undefined) return false
const tree = new ProjectDocumentTree(bundle, { keepRemoved: true })
const movableStates = [DocumentState.Deleted, DocumentState.Obsolete]
const descendants = tree.descendantsOf(prjMeta.meta)
for (const meta of descendants) {
const bundle = tree.bundleOf(meta)
const docs = bundle?.ControlledDocument ?? []
const movable = docs.every((d) => movableStates.includes(d.state))
if (!movable) return false
}
const children = tree.childrenOf(prjMeta.meta)
if (children.length > 0) {
await transferDocuments(client, {
sourceDocumentIds: children,
sourceSpaceId: obj.space,
sourceProjectId: obj.project,
targetSpaceId: obj.space,
targetProjectId: obj.project
})
}
const toRemoval = [obj, prjMeta]
const docsToRemove = [...objs, ...pjmeta, ...meta]
const ops = client.apply()
for (const doc of docsToRemove) {
for (const doc of toRemoval) {
await ops.remove(doc)
}
await ops.commit()
return true
}
export async function createDocument (space: DocumentSpace): Promise<void> {

View File

@ -197,13 +197,17 @@ function extractPresentableStateFromDocumentBundle (bundle: DocumentBundle, prjm
return bundle
}
export interface ProjectDocumentTreeOptions {
keepRemoved?: boolean
}
export class ProjectDocumentTree {
parents: Map<Ref<DocumentMeta>, Ref<DocumentMeta>>
nodesChildren: Map<Ref<DocumentMeta>, DocumentBundle[]>
nodes: Map<Ref<DocumentMeta>, DocumentBundle>
links: Map<Ref<Doc>, Ref<DocumentMeta>>
constructor (bundle?: DocumentBundle) {
constructor (bundle?: DocumentBundle, options?: ProjectDocumentTreeOptions) {
bundle = { ...emptyBundle(), ...bundle }
const { bundles, links } = compileBundles(bundle)
this.links = links
@ -211,6 +215,8 @@ export class ProjectDocumentTree {
this.nodesChildren = new Map()
this.parents = new Map()
const keepRemoved = options?.keepRemoved ?? false
bundles.sort((a, b) => {
const rankA = a.ProjectMeta[0]?.rank ?? ''
const rankB = b.ProjectMeta[0]?.rank ?? ''
@ -224,7 +230,7 @@ export class ProjectDocumentTree {
const presentable = extractPresentableStateFromDocumentBundle(bundle, prjmeta)
this.nodes.set(prjmeta.meta, presentable)
const parent = prjmeta.path[0] ?? documents.ids.NoParent
const parent = prjmeta.parent ?? documents.ids.NoParent
this.parents.set(prjmeta.meta, parent)
if (!this.nodesChildren.has(parent)) {
@ -234,10 +240,12 @@ export class ProjectDocumentTree {
}
const nodesForRemoval = new Set<Ref<DocumentMeta>>()
for (const [id, node] of this.nodes) {
const state = node.ControlledDocument[0]?.state
const isRemoved = state === DocumentState.Obsolete || state === DocumentState.Deleted
if (isRemoved) nodesForRemoval.add(id)
if (!keepRemoved) {
for (const [id, node] of this.nodes) {
const state = node.ControlledDocument[0]?.state
const isRemoved = state === DocumentState.Obsolete || state === DocumentState.Deleted
if (isRemoved) nodesForRemoval.add(id)
}
}
for (const id of this.nodes.keys()) {
@ -320,8 +328,12 @@ export async function findProjectDocsHierarchy (
space: Ref<DocumentSpace>,
project?: Ref<Project<DocumentSpace>>
): Promise<ProjectDocumentTree> {
const ProjectMeta = await client.findAll(documents.class.ProjectMeta, { space, project })
return new ProjectDocumentTree({ ...emptyBundle(), ProjectMeta })
const bundle: DocumentBundle = {
...emptyBundle(),
DocumentMeta: await client.findAll(documents.class.DocumentMeta, { space }),
ProjectMeta: await client.findAll(documents.class.ProjectMeta, { space, project })
}
return new ProjectDocumentTree(bundle, { keepRemoved: true })
}
export interface DocumentBundle {
@ -507,20 +519,30 @@ async function _buildDocumentTransferContext (
client: TxOperations,
request: DocumentTransferRequest
): Promise<DocumentTransferContext | undefined> {
const isSameSpace = request.sourceSpaceId === request.targetSpaceId
const sourceTree = await findProjectDocsHierarchy(client, request.sourceSpaceId, request.sourceProjectId)
const targetTree = await findProjectDocsHierarchy(client, request.targetSpaceId, request.targetProjectId)
const targetTree = isSameSpace
? sourceTree
: await findProjectDocsHierarchy(client, request.targetSpaceId, request.targetProjectId)
const docIds = new Set<Ref<DocumentMeta>>(request.sourceDocumentIds)
for (const id of request.sourceDocumentIds) {
sourceTree.descendantsOf(id).forEach((d) => docIds.add(d))
}
if (request.targetParentId !== undefined && docIds.has(request.targetParentId)) {
return
}
const bundles = await findAllDocumentBundles(client, Array.from(docIds))
const targetParentBundle =
request.targetParentId !== undefined ? await findOneDocumentBundle(client, request.targetParentId) : undefined
const sourceSpace = await client.findOne(documents.class.DocumentSpace, { _id: request.sourceSpaceId })
const targetSpace = await client.findOne(documents.class.DocumentSpace, { _id: request.targetSpaceId })
const targetSpace = isSameSpace
? sourceSpace
: await client.findOne(documents.class.DocumentSpace, { _id: request.targetSpaceId })
if (sourceSpace === undefined || targetSpace === undefined) return
@ -565,7 +587,6 @@ async function _transferDocuments (
mode: 'default' | 'check' = 'default'
): Promise<boolean> {
if (cx.bundles.length < 1) return false
if (cx.targetSpace._id === cx.sourceSpace._id) return false
const hierarchy = client.getHierarchy()
@ -621,13 +642,17 @@ async function _transferDocuments (
let key: keyof DocumentBundle
for (key in bundle) {
bundle[key].forEach((doc) => {
update(doc, { space: cx.targetSpace._id })
})
for (const doc of bundle[key]) {
const space = cx.targetSpace._id
if (doc.space !== space) update(doc, { space })
}
}
for (const m of bundle.ProjectMeta) {
if (m.project !== project) update(m, { project })
}
for (const m of bundle.ProjectDocument) {
if (m.project !== project) update(m, { project })
}
for (const m of bundle.ProjectMeta) update(m, { project })
for (const m of bundle.ProjectDocument) update(m, { project })
}
if (mode === 'check') return true