From b4b598f542d0cefc5f2d5d6c7286f0a312cf6a55 Mon Sep 17 00:00:00 2001 From: Lauri Ojansivu Date: Thu, 23 Oct 2025 04:03:52 +0300 Subject: [PATCH 01/84] Fix board reloading page every second. Thanks to xet7 ! --- client/components/boards/boardBody.js | 81 ++++++++++++++++++--------- 1 file changed, 53 insertions(+), 28 deletions(-) diff --git a/client/components/boards/boardBody.js b/client/components/boards/boardBody.js index f1f6b9a5a..5eb317b9b 100644 --- a/client/components/boards/boardBody.js +++ b/client/components/boards/boardBody.js @@ -17,6 +17,8 @@ BlazeComponent.extendComponent({ this.isConverting = new ReactiveVar(false); this.isMigrating = new ReactiveVar(false); this._swimlaneCreated = new Set(); // Track boards where we've created swimlanes + this._boardProcessed = false; // Track if board has been processed + this._lastProcessedBoardId = null; // Track last processed board ID // The pattern we use to manually handle data loading is described here: // https://kadira.io/academy/meteor-routing-guide/content/subscriptions-and-data-management/using-subs-manager @@ -28,21 +30,33 @@ BlazeComponent.extendComponent({ const handle = subManager.subscribe('board', currentBoardId, false); - Tracker.nonreactive(() => { - Tracker.autorun(() => { - if (handle.ready()) { + // Use a separate autorun for subscription ready state to avoid reactive loops + this.subscriptionReadyAutorun = Tracker.autorun(() => { + if (handle.ready()) { + // Only run conversion/migration logic once per board + if (!this._boardProcessed || this._lastProcessedBoardId !== currentBoardId) { + this._boardProcessed = true; + this._lastProcessedBoardId = currentBoardId; + // Ensure default swimlane exists (only once per board) this.ensureDefaultSwimlane(currentBoardId); // Check if board needs conversion this.checkAndConvertBoard(currentBoardId); - } else { - this.isBoardReady.set(false); } - }); + } else { + this.isBoardReady.set(false); + } }); }); }, + onDestroyed() { + // Clean up the subscription ready autorun to prevent memory leaks + if (this.subscriptionReadyAutorun) { + this.subscriptionReadyAutorun.stop(); + } + }, + ensureDefaultSwimlane(boardId) { // Only create swimlane once per board if (this._swimlaneCreated.has(boardId)) { @@ -441,39 +455,50 @@ BlazeComponent.extendComponent({ this._isDragging = false; // Used to set the overlay this.mouseHasEnterCardDetails = false; + this._sortFieldsFixed = new Set(); // Track which boards have had sort fields fixed // fix swimlanes sort field if there are null values const currentBoardData = Utils.getCurrentBoard(); if (currentBoardData && Swimlanes) { - const nullSortSwimlanes = currentBoardData.nullSortSwimlanes(); - if (nullSortSwimlanes.length > 0) { - const swimlanes = currentBoardData.swimlanes(); - let count = 0; - swimlanes.forEach(s => { - Swimlanes.update(s._id, { - $set: { - sort: count, - }, + const boardId = currentBoardData._id; + // Only fix sort fields once per board to prevent reactive loops + if (!this._sortFieldsFixed.has(`swimlanes-${boardId}`)) { + const nullSortSwimlanes = currentBoardData.nullSortSwimlanes(); + if (nullSortSwimlanes.length > 0) { + const swimlanes = currentBoardData.swimlanes(); + let count = 0; + swimlanes.forEach(s => { + Swimlanes.update(s._id, { + $set: { + sort: count, + }, + }); + count += 1; }); - count += 1; - }); + } + this._sortFieldsFixed.add(`swimlanes-${boardId}`); } } // fix lists sort field if there are null values if (currentBoardData && Lists) { - const nullSortLists = currentBoardData.nullSortLists(); - if (nullSortLists.length > 0) { - const lists = currentBoardData.lists(); - let count = 0; - lists.forEach(l => { - Lists.update(l._id, { - $set: { - sort: count, - }, + const boardId = currentBoardData._id; + // Only fix sort fields once per board to prevent reactive loops + if (!this._sortFieldsFixed.has(`lists-${boardId}`)) { + const nullSortLists = currentBoardData.nullSortLists(); + if (nullSortLists.length > 0) { + const lists = currentBoardData.lists(); + let count = 0; + lists.forEach(l => { + Lists.update(l._id, { + $set: { + sort: count, + }, + }); + count += 1; }); - count += 1; - }); + } + this._sortFieldsFixed.add(`lists-${boardId}`); } } }, From 940df024561114a31f4c23e811f29053a1e2ee2d Mon Sep 17 00:00:00 2001 From: Lauri Ojansivu Date: Thu, 23 Oct 2025 04:08:49 +0300 Subject: [PATCH 02/84] Updated translations. --- imports/i18n/data/ms-MY.i18n.json | 6 +++--- 1 file changed, 3 insertions(+), 3 deletions(-) diff --git a/imports/i18n/data/ms-MY.i18n.json b/imports/i18n/data/ms-MY.i18n.json index ede03686c..2112ab947 100644 --- a/imports/i18n/data/ms-MY.i18n.json +++ b/imports/i18n/data/ms-MY.i18n.json @@ -89,9 +89,9 @@ "setListWidthPopup-title": "Tetapkan Lebar", "set-list-width": "Tetapkan Lebar", "set-list-width-value": "Tetapkan lebar minimum dan maksimum (piksel)", - "list-width-error-message": "List widths must be integers greater than 100", - "keyboard-shortcuts-enabled": "Keyboard shortcuts enabled. Click to disable.", - "keyboard-shortcuts-disabled": "Keyboard shortcuts disabled. Click to enable.", + "list-width-error-message": "Lebar senarai mestilah integer lebih besar dari 100", + "keyboard-shortcuts-enabled": "Pintasan papan kekunci didayakan. Klik untuk batal.", + "keyboard-shortcuts-disabled": "Pintasan papan kekunci didayakan. Klik untuk batal.", "setSwimlaneHeightPopup-title": "Set Swimlane Height", "set-swimlane-height": "Set Swimlane Height", "set-swimlane-height-value": "Swimlane Height (pixels)", From 0cbc9402f3648c9a33423febcae07b6dca94aab8 Mon Sep 17 00:00:00 2001 From: Lauri Ojansivu Date: Thu, 23 Oct 2025 04:09:14 +0300 Subject: [PATCH 03/84] v8.14 --- CHANGELOG.md | 9 +++++++++ Dockerfile | 6 +++--- Stackerfile.yml | 2 +- docs/Platforms/Propietary/Windows/Offline.md | 4 ++-- package-lock.json | 2 +- package.json | 2 +- sandstorm-pkgdef.capnp | 4 ++-- snapcraft.yaml | 8 ++++---- 8 files changed, 23 insertions(+), 14 deletions(-) diff --git a/CHANGELOG.md b/CHANGELOG.md index 30a838e0e..cb1bae471 100644 --- a/CHANGELOG.md +++ b/CHANGELOG.md @@ -22,6 +22,15 @@ Fixing other platforms In Progress. WeKan 8.00-8.06 had wrong raw database directory setting /var/snap/wekan/common/wekan and some cards were not visible. Those are fixed at WeKan 8.07 where database directory is back to /var/snap/wekan/common and all cards are visible. +# v8.14 2025-10-23 WeKan ® release + +This release fixes the following bugs: + +- [Fix board reloading page every second](https://github.com/wekan/wekan/commit/b4b598f542d0cefc5f2d5d6c7286f0a312cf6a55). + Thanks to xet7. + +Thanks to above GitHub users for their contributions and translators for their translations. + # v8.12 2025-10-23 WeKan ® release This release fixes the following bugs: diff --git a/Dockerfile b/Dockerfile index b53398c74..83582c102 100644 --- a/Dockerfile +++ b/Dockerfile @@ -249,9 +249,9 @@ cd /home/wekan/app # Remove legacy webbroser bundle, so that Wekan works also at Android Firefox, iOS Safari, etc. #rm -rf /home/wekan/app_build/bundle/programs/web.browser.legacy #mv /home/wekan/app_build/bundle /build -wget "https://github.com/wekan/wekan/releases/download/v8.12/wekan-8.12-amd64.zip" -unzip wekan-8.12-amd64.zip -rm wekan-8.12-amd64.zip +wget "https://github.com/wekan/wekan/releases/download/v8.14/wekan-8.14-amd64.zip" +unzip wekan-8.14-amd64.zip +rm wekan-8.14-amd64.zip mv /home/wekan/app/bundle /build # Put back the original tar diff --git a/Stackerfile.yml b/Stackerfile.yml index e472459bc..567bcc521 100644 --- a/Stackerfile.yml +++ b/Stackerfile.yml @@ -1,5 +1,5 @@ appId: wekan-public/apps/77b94f60-dec9-0136-304e-16ff53095928 -appVersion: "v8.12.0" +appVersion: "v8.14.0" files: userUploads: - README.md diff --git a/docs/Platforms/Propietary/Windows/Offline.md b/docs/Platforms/Propietary/Windows/Offline.md index e4a9865e3..44a35e12a 100644 --- a/docs/Platforms/Propietary/Windows/Offline.md +++ b/docs/Platforms/Propietary/Windows/Offline.md @@ -10,7 +10,7 @@ This is without container (without Docker or Snap). Right click and download files 1-4: -1. [wekan-8.12-amd64-windows.zip](https://github.com/wekan/wekan/releases/download/v8.12/wekan-8.12-amd64-windows.zip) +1. [wekan-8.14-amd64-windows.zip](https://github.com/wekan/wekan/releases/download/v8.14/wekan-8.14-amd64-windows.zip) 2. [node.exe](https://nodejs.org/dist/latest-v14.x/win-x64/node.exe) @@ -22,7 +22,7 @@ Right click and download files 1-4: 6. Double click `mongodb-windows-x86_64-7.0.25-signed.msi` . In installer, uncheck downloading MongoDB compass. -7. Unzip `wekan-8.12-amd64-windows.zip` , inside it is directory `bundle`, to it copy other files: +7. Unzip `wekan-8.14-amd64-windows.zip` , inside it is directory `bundle`, to it copy other files: ``` bundle (directory) diff --git a/package-lock.json b/package-lock.json index 9d5478751..445be6069 100644 --- a/package-lock.json +++ b/package-lock.json @@ -1,6 +1,6 @@ { "name": "wekan", - "version": "v8.12.0", + "version": "v8.14.0", "lockfileVersion": 1, "requires": true, "dependencies": { diff --git a/package.json b/package.json index 5d2d1a430..f52f12815 100644 --- a/package.json +++ b/package.json @@ -1,6 +1,6 @@ { "name": "wekan", - "version": "v8.12.0", + "version": "v8.14.0", "description": "Open-Source kanban", "private": true, "repository": { diff --git a/sandstorm-pkgdef.capnp b/sandstorm-pkgdef.capnp index cf9bae85d..9683133b9 100644 --- a/sandstorm-pkgdef.capnp +++ b/sandstorm-pkgdef.capnp @@ -22,10 +22,10 @@ const pkgdef :Spk.PackageDefinition = ( appTitle = (defaultText = "Wekan"), # The name of the app as it is displayed to the user. - appVersion = 812, + appVersion = 814, # Increment this for every release. - appMarketingVersion = (defaultText = "8.12.0~2025-10-23"), + appMarketingVersion = (defaultText = "8.14.0~2025-10-23"), # Human-readable presentation of the app version. minUpgradableAppVersion = 0, diff --git a/snapcraft.yaml b/snapcraft.yaml index c91a57b7f..a7e4f76f6 100644 --- a/snapcraft.yaml +++ b/snapcraft.yaml @@ -1,5 +1,5 @@ name: wekan -version: '8.12' +version: '8.14' base: core24 summary: Open Source kanban description: | @@ -166,9 +166,9 @@ parts: # Cleanup mkdir .build cd .build - wget https://github.com/wekan/wekan/releases/download/v8.12/wekan-8.12-amd64.zip - unzip wekan-8.12-amd64.zip - rm wekan-8.12-amd64.zip + wget https://github.com/wekan/wekan/releases/download/v8.14/wekan-8.14-amd64.zip + unzip wekan-8.14-amd64.zip + rm wekan-8.14-amd64.zip cd .. ##cd .build/bundle ##find . -type d -name '*-garbage*' | xargs rm -rf From 8662c96d1c8d4fa76ce7b31eb06678ad59c3ebe1 Mon Sep 17 00:00:00 2001 From: Lauri Ojansivu Date: Thu, 23 Oct 2025 04:33:34 +0300 Subject: [PATCH 04/84] Fix drag lists did not work. Thanks to xet7 ! --- client/components/swimlanes/swimlanes.js | 204 +++++++++++++++++++++++ 1 file changed, 204 insertions(+) diff --git a/client/components/swimlanes/swimlanes.js b/client/components/swimlanes/swimlanes.js index cb0eb4c9d..c1ddcd369 100644 --- a/client/components/swimlanes/swimlanes.js +++ b/client/components/swimlanes/swimlanes.js @@ -722,6 +722,96 @@ setTimeout(() => { } }, stop(evt, ui) { + // To attribute the new index number, we need to get the DOM element + // of the previous and the following list -- if any. + const prevListDom = ui.item.prev('.js-list').get(0); + const nextListDom = ui.item.next('.js-list').get(0); + const sortIndex = calculateIndex(prevListDom, nextListDom, 1); + + const listDomElement = ui.item.get(0); + if (!listDomElement) { + console.error('List DOM element not found during drag stop'); + return; + } + + let list; + try { + list = Blaze.getData(listDomElement); + } catch (error) { + console.error('Error getting list data:', error); + return; + } + + if (!list) { + console.error('List data not found for element:', listDomElement); + return; + } + + // Detect if the list was dropped in a different swimlane + const targetSwimlaneDom = ui.item.closest('.js-swimlane'); + let targetSwimlaneId = null; + + if (targetSwimlaneDom.length > 0) { + // List was dropped in a swimlane + try { + targetSwimlaneId = targetSwimlaneDom.attr('id').replace('swimlane-', ''); + } catch (error) { + console.error('Error getting target swimlane ID:', error); + return; + } + } else { + // List was dropped in lists view (not swimlanes view) + // In this case, assign to the default swimlane + const currentBoard = ReactiveCache.getBoard(Session.get('currentBoard')); + if (currentBoard) { + const defaultSwimlane = currentBoard.getDefaultSwimline(); + if (defaultSwimlane) { + targetSwimlaneId = defaultSwimlane._id; + } + } + } + + // Get the original swimlane ID of the list (handle backward compatibility) + const originalSwimlaneId = list.getEffectiveSwimlaneId ? list.getEffectiveSwimlaneId() : (list.swimlaneId || null); + + // Prepare update object + const updateData = { + sort: sortIndex.base, + }; + + // Check if the list was dropped in a different swimlane + const isDifferentSwimlane = targetSwimlaneId && targetSwimlaneId !== originalSwimlaneId; + + // If the list was dropped in a different swimlane, update the swimlaneId + if (isDifferentSwimlane) { + updateData.swimlaneId = targetSwimlaneId; + + // Move all cards in the list to the new swimlane + const cardsInList = ReactiveCache.getCards({ + listId: list._id, + archived: false + }); + + cardsInList.forEach(card => { + card.move(list.boardId, targetSwimlaneId, list._id); + }); + + // Don't cancel the sortable when moving to a different swimlane + // The DOM move should be allowed to complete + } else { + // If staying in the same swimlane, cancel the sortable to prevent DOM manipulation issues + $swimlane.sortable('cancel'); + } + + try { + Lists.update(list._id, { + $set: updateData, + }); + } catch (error) { + console.error('Error updating list:', error); + return; + } + // Try to get board component try { const boardComponent = BlazeComponent.getComponentForElement(ui.item[0]); @@ -731,6 +821,18 @@ setTimeout(() => { } catch (e) { // Silent fail } + + // Re-enable dragscroll after list dragging is complete + try { + dragscroll.reset(); + } catch (e) { + // Silent fail + } + + // Re-enable dragscroll on all swimlanes + $('.js-swimlane').each(function() { + $(this).addClass('dragscroll'); + }); } }); } @@ -769,6 +871,96 @@ setTimeout(() => { } }, stop(evt, ui) { + // To attribute the new index number, we need to get the DOM element + // of the previous and the following list -- if any. + const prevListDom = ui.item.prev('.js-list').get(0); + const nextListDom = ui.item.next('.js-list').get(0); + const sortIndex = calculateIndex(prevListDom, nextListDom, 1); + + const listDomElement = ui.item.get(0); + if (!listDomElement) { + console.error('List DOM element not found during drag stop'); + return; + } + + let list; + try { + list = Blaze.getData(listDomElement); + } catch (error) { + console.error('Error getting list data:', error); + return; + } + + if (!list) { + console.error('List data not found for element:', listDomElement); + return; + } + + // Detect if the list was dropped in a different swimlane + const targetSwimlaneDom = ui.item.closest('.js-swimlane'); + let targetSwimlaneId = null; + + if (targetSwimlaneDom.length > 0) { + // List was dropped in a swimlane + try { + targetSwimlaneId = targetSwimlaneDom.attr('id').replace('swimlane-', ''); + } catch (error) { + console.error('Error getting target swimlane ID:', error); + return; + } + } else { + // List was dropped in lists view (not swimlanes view) + // In this case, assign to the default swimlane + const currentBoard = ReactiveCache.getBoard(Session.get('currentBoard')); + if (currentBoard) { + const defaultSwimlane = currentBoard.getDefaultSwimline(); + if (defaultSwimlane) { + targetSwimlaneId = defaultSwimlane._id; + } + } + } + + // Get the original swimlane ID of the list (handle backward compatibility) + const originalSwimlaneId = list.getEffectiveSwimlaneId ? list.getEffectiveSwimlaneId() : (list.swimlaneId || null); + + // Prepare update object + const updateData = { + sort: sortIndex.base, + }; + + // Check if the list was dropped in a different swimlane + const isDifferentSwimlane = targetSwimlaneId && targetSwimlaneId !== originalSwimlaneId; + + // If the list was dropped in a different swimlane, update the swimlaneId + if (isDifferentSwimlane) { + updateData.swimlaneId = targetSwimlaneId; + + // Move all cards in the list to the new swimlane + const cardsInList = ReactiveCache.getCards({ + listId: list._id, + archived: false + }); + + cardsInList.forEach(card => { + card.move(list.boardId, targetSwimlaneId, list._id); + }); + + // Don't cancel the sortable when moving to a different swimlane + // The DOM move should be allowed to complete + } else { + // If staying in the same swimlane, cancel the sortable to prevent DOM manipulation issues + $listsGroup.sortable('cancel'); + } + + try { + Lists.update(list._id, { + $set: updateData, + }); + } catch (error) { + console.error('Error updating list:', error); + return; + } + // Try to get board component try { const boardComponent = BlazeComponent.getComponentForElement(ui.item[0]); @@ -778,6 +970,18 @@ setTimeout(() => { } catch (e) { // Silent fail } + + // Re-enable dragscroll after list dragging is complete + try { + dragscroll.reset(); + } catch (e) { + // Silent fail + } + + // Re-enable dragscroll on all swimlanes + $('.js-swimlane').each(function() { + $(this).addClass('dragscroll'); + }); } }); } From 0cebd8aa4dbe0bf2418b814716744ab806b671c2 Mon Sep 17 00:00:00 2001 From: Lauri Ojansivu Date: Thu, 23 Oct 2025 04:35:33 +0300 Subject: [PATCH 05/84] Fix drag lists did not work. Part 2. Thanks to xet7 ! --- client/components/swimlanes/swimlanes.js | 10 ---------- 1 file changed, 10 deletions(-) diff --git a/client/components/swimlanes/swimlanes.js b/client/components/swimlanes/swimlanes.js index c1ddcd369..47b9c5843 100644 --- a/client/components/swimlanes/swimlanes.js +++ b/client/components/swimlanes/swimlanes.js @@ -730,7 +730,6 @@ setTimeout(() => { const listDomElement = ui.item.get(0); if (!listDomElement) { - console.error('List DOM element not found during drag stop'); return; } @@ -738,12 +737,10 @@ setTimeout(() => { try { list = Blaze.getData(listDomElement); } catch (error) { - console.error('Error getting list data:', error); return; } if (!list) { - console.error('List data not found for element:', listDomElement); return; } @@ -756,7 +753,6 @@ setTimeout(() => { try { targetSwimlaneId = targetSwimlaneDom.attr('id').replace('swimlane-', ''); } catch (error) { - console.error('Error getting target swimlane ID:', error); return; } } else { @@ -808,7 +804,6 @@ setTimeout(() => { $set: updateData, }); } catch (error) { - console.error('Error updating list:', error); return; } @@ -879,7 +874,6 @@ setTimeout(() => { const listDomElement = ui.item.get(0); if (!listDomElement) { - console.error('List DOM element not found during drag stop'); return; } @@ -887,12 +881,10 @@ setTimeout(() => { try { list = Blaze.getData(listDomElement); } catch (error) { - console.error('Error getting list data:', error); return; } if (!list) { - console.error('List data not found for element:', listDomElement); return; } @@ -905,7 +897,6 @@ setTimeout(() => { try { targetSwimlaneId = targetSwimlaneDom.attr('id').replace('swimlane-', ''); } catch (error) { - console.error('Error getting target swimlane ID:', error); return; } } else { @@ -957,7 +948,6 @@ setTimeout(() => { $set: updateData, }); } catch (error) { - console.error('Error updating list:', error); return; } From 7fe7fb4c1597585f01b75355b0e691ac09063018 Mon Sep 17 00:00:00 2001 From: Lauri Ojansivu Date: Thu, 23 Oct 2025 04:41:34 +0300 Subject: [PATCH 06/84] v8.15 --- CHANGELOG.md | 11 +++++++++++ Dockerfile | 6 +++--- Stackerfile.yml | 2 +- docs/Platforms/Propietary/Windows/Offline.md | 4 ++-- package-lock.json | 2 +- package.json | 2 +- sandstorm-pkgdef.capnp | 4 ++-- snapcraft.yaml | 8 ++++---- 8 files changed, 25 insertions(+), 14 deletions(-) diff --git a/CHANGELOG.md b/CHANGELOG.md index cb1bae471..2ba71ed71 100644 --- a/CHANGELOG.md +++ b/CHANGELOG.md @@ -22,6 +22,17 @@ Fixing other platforms In Progress. WeKan 8.00-8.06 had wrong raw database directory setting /var/snap/wekan/common/wekan and some cards were not visible. Those are fixed at WeKan 8.07 where database directory is back to /var/snap/wekan/common and all cards are visible. +# v8.15 2025-10-23 WeKan ® release + +This release fixes the following bugs: + +- Fix drag lists did not work + [Part 1](https://github.com/wekan/wekan/commit/8662c96d1c8d4fa76ce7b31eb06678ad59c3ebe1), + [Part 2](https://github.com/wekan/wekan/commit/0cebd8aa4dbe0bf2418b814716744ab806b671c2). + Thanks to xet7. + +Thanks to above GitHub users for their contributions and translators for their translations. + # v8.14 2025-10-23 WeKan ® release This release fixes the following bugs: diff --git a/Dockerfile b/Dockerfile index 83582c102..149a46db4 100644 --- a/Dockerfile +++ b/Dockerfile @@ -249,9 +249,9 @@ cd /home/wekan/app # Remove legacy webbroser bundle, so that Wekan works also at Android Firefox, iOS Safari, etc. #rm -rf /home/wekan/app_build/bundle/programs/web.browser.legacy #mv /home/wekan/app_build/bundle /build -wget "https://github.com/wekan/wekan/releases/download/v8.14/wekan-8.14-amd64.zip" -unzip wekan-8.14-amd64.zip -rm wekan-8.14-amd64.zip +wget "https://github.com/wekan/wekan/releases/download/v8.15/wekan-8.15-amd64.zip" +unzip wekan-8.15-amd64.zip +rm wekan-8.15-amd64.zip mv /home/wekan/app/bundle /build # Put back the original tar diff --git a/Stackerfile.yml b/Stackerfile.yml index 567bcc521..c17665e86 100644 --- a/Stackerfile.yml +++ b/Stackerfile.yml @@ -1,5 +1,5 @@ appId: wekan-public/apps/77b94f60-dec9-0136-304e-16ff53095928 -appVersion: "v8.14.0" +appVersion: "v8.15.0" files: userUploads: - README.md diff --git a/docs/Platforms/Propietary/Windows/Offline.md b/docs/Platforms/Propietary/Windows/Offline.md index 44a35e12a..8b94bb464 100644 --- a/docs/Platforms/Propietary/Windows/Offline.md +++ b/docs/Platforms/Propietary/Windows/Offline.md @@ -10,7 +10,7 @@ This is without container (without Docker or Snap). Right click and download files 1-4: -1. [wekan-8.14-amd64-windows.zip](https://github.com/wekan/wekan/releases/download/v8.14/wekan-8.14-amd64-windows.zip) +1. [wekan-8.15-amd64-windows.zip](https://github.com/wekan/wekan/releases/download/v8.15/wekan-8.15-amd64-windows.zip) 2. [node.exe](https://nodejs.org/dist/latest-v14.x/win-x64/node.exe) @@ -22,7 +22,7 @@ Right click and download files 1-4: 6. Double click `mongodb-windows-x86_64-7.0.25-signed.msi` . In installer, uncheck downloading MongoDB compass. -7. Unzip `wekan-8.14-amd64-windows.zip` , inside it is directory `bundle`, to it copy other files: +7. Unzip `wekan-8.15-amd64-windows.zip` , inside it is directory `bundle`, to it copy other files: ``` bundle (directory) diff --git a/package-lock.json b/package-lock.json index 445be6069..3fb6043e9 100644 --- a/package-lock.json +++ b/package-lock.json @@ -1,6 +1,6 @@ { "name": "wekan", - "version": "v8.14.0", + "version": "v8.15.0", "lockfileVersion": 1, "requires": true, "dependencies": { diff --git a/package.json b/package.json index f52f12815..a4dcfe949 100644 --- a/package.json +++ b/package.json @@ -1,6 +1,6 @@ { "name": "wekan", - "version": "v8.14.0", + "version": "v8.15.0", "description": "Open-Source kanban", "private": true, "repository": { diff --git a/sandstorm-pkgdef.capnp b/sandstorm-pkgdef.capnp index 9683133b9..edc3008e1 100644 --- a/sandstorm-pkgdef.capnp +++ b/sandstorm-pkgdef.capnp @@ -22,10 +22,10 @@ const pkgdef :Spk.PackageDefinition = ( appTitle = (defaultText = "Wekan"), # The name of the app as it is displayed to the user. - appVersion = 814, + appVersion = 815, # Increment this for every release. - appMarketingVersion = (defaultText = "8.14.0~2025-10-23"), + appMarketingVersion = (defaultText = "8.15.0~2025-10-23"), # Human-readable presentation of the app version. minUpgradableAppVersion = 0, diff --git a/snapcraft.yaml b/snapcraft.yaml index a7e4f76f6..dab9beb63 100644 --- a/snapcraft.yaml +++ b/snapcraft.yaml @@ -1,5 +1,5 @@ name: wekan -version: '8.14' +version: '8.15' base: core24 summary: Open Source kanban description: | @@ -166,9 +166,9 @@ parts: # Cleanup mkdir .build cd .build - wget https://github.com/wekan/wekan/releases/download/v8.14/wekan-8.14-amd64.zip - unzip wekan-8.14-amd64.zip - rm wekan-8.14-amd64.zip + wget https://github.com/wekan/wekan/releases/download/v8.15/wekan-8.15-amd64.zip + unzip wekan-8.15-amd64.zip + rm wekan-8.15-amd64.zip cd .. ##cd .build/bundle ##find . -type d -name '*-garbage*' | xargs rm -rf From 91b846e2cdee9154b045d11b4b4c1a7ae1d79016 Mon Sep 17 00:00:00 2001 From: Lauri Ojansivu Date: Thu, 23 Oct 2025 05:50:43 +0300 Subject: [PATCH 07/84] List menu / More / Delete duplicate lists that do not have any cards. Thanks to xet7 ! --- client/components/sidebar/sidebar.jade | 10 +++++++ client/components/sidebar/sidebar.js | 39 ++++++++++++++++++++++++++ imports/i18n/data/en.i18n.json | 2 ++ 3 files changed, 51 insertions(+) diff --git a/client/components/sidebar/sidebar.jade b/client/components/sidebar/sidebar.jade index fdc4e6b07..662c84ad8 100644 --- a/client/components/sidebar/sidebar.jade +++ b/client/components/sidebar/sidebar.jade @@ -537,6 +537,12 @@ template(name="archiveBoardPopup") | 📦 | {{_ 'archive'}} +template(name="deleteDuplicateListsPopup") + p {{_ 'delete-duplicate-lists-confirm'}} + button.js-confirm.negate.full(type="submit") + | 🗑️ + | {{_ 'delete'}} + template(name="outgoingWebhooksPopup") each integrations form.integration-form @@ -621,6 +627,10 @@ template(name="boardMenuPopup") if currentUser.isBoardAdmin hr ul.pop-over-list + li + a.js-delete-duplicate-lists + | 🗑️ + | {{_ 'delete-duplicate-lists'}} li a.js-archive-board | ➡️📦 diff --git a/client/components/sidebar/sidebar.js b/client/components/sidebar/sidebar.js index 3d507ff8c..18f271691 100644 --- a/client/components/sidebar/sidebar.js +++ b/client/components/sidebar/sidebar.js @@ -275,6 +275,45 @@ Template.boardMenuPopup.events({ 'click .js-change-background-image': Popup.open('boardChangeBackgroundImage'), 'click .js-board-info-on-my-boards': Popup.open('boardInfoOnMyBoards'), 'click .js-change-language': Popup.open('changeLanguage'), + 'click .js-delete-duplicate-lists': Popup.afterConfirm('deleteDuplicateLists', function() { + const currentBoard = Utils.getCurrentBoard(); + if (!currentBoard) return; + + // Get all lists in the current board + const allLists = ReactiveCache.getLists({ boardId: currentBoard._id, archived: false }); + + // Group lists by title to find duplicates + const listsByTitle = {}; + allLists.forEach(list => { + if (!listsByTitle[list.title]) { + listsByTitle[list.title] = []; + } + listsByTitle[list.title].push(list); + }); + + // Find and delete duplicate lists that have no cards + let deletedCount = 0; + Object.keys(listsByTitle).forEach(title => { + const listsWithSameTitle = listsByTitle[title]; + if (listsWithSameTitle.length > 1) { + // Keep the first list, delete the rest if they have no cards + for (let i = 1; i < listsWithSameTitle.length; i++) { + const list = listsWithSameTitle[i]; + const cardsInList = ReactiveCache.getCards({ listId: list._id, archived: false }); + + if (cardsInList.length === 0) { + Lists.remove(list._id); + deletedCount++; + } + } + } + }); + + // Show notification + if (deletedCount > 0) { + // You could add a toast notification here if available + } + }), 'click .js-archive-board ': Popup.afterConfirm('archiveBoard', function() { const currentBoard = Utils.getCurrentBoard(); currentBoard.archive(); diff --git a/imports/i18n/data/en.i18n.json b/imports/i18n/data/en.i18n.json index bcdd00061..eef27e1fe 100644 --- a/imports/i18n/data/en.i18n.json +++ b/imports/i18n/data/en.i18n.json @@ -755,6 +755,8 @@ "delete-board-confirm-popup": "All lists, cards, labels, and activities will be deleted and you won't be able to recover the board contents. There is no undo.", "boardDeletePopup-title": "Delete Board?", "delete-board": "Delete Board", + "delete-duplicate-lists": "Delete Duplicate Lists", + "delete-duplicate-lists-confirm": "Are you sure? This will delete all duplicate lists that have the same name and contain no cards.", "default-subtasks-board": "Subtasks for __board__ board", "default": "Default", "defaultdefault": "Default", From 92bfbb2d0ce808620be436d8ada2bb7d81487074 Mon Sep 17 00:00:00 2001 From: Lauri Ojansivu Date: Thu, 23 Oct 2025 05:54:25 +0300 Subject: [PATCH 08/84] Updated ChangeLog. --- CHANGELOG.md | 9 +++++++++ 1 file changed, 9 insertions(+) diff --git a/CHANGELOG.md b/CHANGELOG.md index 2ba71ed71..ffef70a57 100644 --- a/CHANGELOG.md +++ b/CHANGELOG.md @@ -22,6 +22,15 @@ Fixing other platforms In Progress. WeKan 8.00-8.06 had wrong raw database directory setting /var/snap/wekan/common/wekan and some cards were not visible. Those are fixed at WeKan 8.07 where database directory is back to /var/snap/wekan/common and all cards are visible. +# Upcoming WeKan ® release + +This release adds the following new features: + +- [List menu / More / Delete duplicate lists that do not have any cards](https://github.com/wekan/wekan/commit/91b846e2cdee9154b045d11b4b4c1a7ae1d79016). + Thanks to xet7. + +Thanks to above GitHub users for their contributions and translators for their translations. + # v8.15 2025-10-23 WeKan ® release This release fixes the following bugs: From d1a51b42f6910375ac0e08e92e29dc1e26c689eb Mon Sep 17 00:00:00 2001 From: Lauri Ojansivu Date: Fri, 24 Oct 2025 18:43:21 +0300 Subject: [PATCH 09/84] Updated translations. --- imports/i18n/data/af.i18n.json | 2 + imports/i18n/data/af_ZA.i18n.json | 2 + imports/i18n/data/ar-DZ.i18n.json | 2 + imports/i18n/data/ar-EG.i18n.json | 2 + imports/i18n/data/ar.i18n.json | 2 + imports/i18n/data/ary.i18n.json | 2 + imports/i18n/data/ast-ES.i18n.json | 2 + imports/i18n/data/az-AZ.i18n.json | 2 + imports/i18n/data/az-LA.i18n.json | 2 + imports/i18n/data/az.i18n.json | 2 + imports/i18n/data/bg.i18n.json | 2 + imports/i18n/data/br.i18n.json | 2 + imports/i18n/data/ca.i18n.json | 2 + imports/i18n/data/ca@valencia.i18n.json | 2 + imports/i18n/data/ca_ES.i18n.json | 2 + imports/i18n/data/cmn.i18n.json | 2 + imports/i18n/data/cs-CZ.i18n.json | 2 + imports/i18n/data/cs.i18n.json | 2 + imports/i18n/data/cy-GB.i18n.json | 2 + imports/i18n/data/cy.i18n.json | 2 + imports/i18n/data/da.i18n.json | 2 + imports/i18n/data/de-AT.i18n.json | 2 + imports/i18n/data/de-CH.i18n.json | 2 + imports/i18n/data/de.i18n.json | 2 + imports/i18n/data/de_DE.i18n.json | 2 + imports/i18n/data/el-GR.i18n.json | 2 + imports/i18n/data/el.i18n.json | 2 + imports/i18n/data/en-BR.i18n.json | 2 + imports/i18n/data/en-DE.i18n.json | 2 + imports/i18n/data/en-GB.i18n.json | 2 + imports/i18n/data/en-IT.i18n.json | 2 + imports/i18n/data/en-MY.i18n.json | 2 + imports/i18n/data/en-YS.i18n.json | 2 + imports/i18n/data/en_AU.i18n.json | 2 + imports/i18n/data/en_ID.i18n.json | 2 + imports/i18n/data/en_SG.i18n.json | 2 + imports/i18n/data/en_TR.i18n.json | 2 + imports/i18n/data/en_ZA.i18n.json | 2 + imports/i18n/data/eo.i18n.json | 2 + imports/i18n/data/es-AR.i18n.json | 2 + imports/i18n/data/es-CL.i18n.json | 2 + imports/i18n/data/es-LA.i18n.json | 2 + imports/i18n/data/es-MX.i18n.json | 2 + imports/i18n/data/es-PE.i18n.json | 2 + imports/i18n/data/es-PY.i18n.json | 2 + imports/i18n/data/es.i18n.json | 2 + imports/i18n/data/es_CO.i18n.json | 2 + imports/i18n/data/et-EE.i18n.json | 2 + imports/i18n/data/eu.i18n.json | 2 + imports/i18n/data/fa-IR.i18n.json | 2 + imports/i18n/data/fa.i18n.json | 2 + imports/i18n/data/fi.i18n.json | 2 + imports/i18n/data/fr-CH.i18n.json | 2 + imports/i18n/data/fr-FR.i18n.json | 2 + imports/i18n/data/fr.i18n.json | 124 ++++++++++++------------ imports/i18n/data/fy-NL.i18n.json | 2 + imports/i18n/data/fy.i18n.json | 2 + imports/i18n/data/gl-ES.i18n.json | 2 + imports/i18n/data/gl.i18n.json | 2 + imports/i18n/data/gu-IN.i18n.json | 2 + imports/i18n/data/he-IL.i18n.json | 2 + imports/i18n/data/he.i18n.json | 2 + imports/i18n/data/hi-IN.i18n.json | 2 + imports/i18n/data/hi.i18n.json | 2 + imports/i18n/data/hr.i18n.json | 2 + imports/i18n/data/hu.i18n.json | 2 + imports/i18n/data/hy.i18n.json | 2 + imports/i18n/data/id.i18n.json | 2 + imports/i18n/data/ig.i18n.json | 2 + imports/i18n/data/it.i18n.json | 2 + imports/i18n/data/ja-HI.i18n.json | 2 + imports/i18n/data/ja.i18n.json | 2 + imports/i18n/data/ka.i18n.json | 2 + imports/i18n/data/km.i18n.json | 2 + imports/i18n/data/ko-KR.i18n.json | 2 + imports/i18n/data/ko.i18n.json | 2 + imports/i18n/data/lt.i18n.json | 2 + imports/i18n/data/lv.i18n.json | 2 + imports/i18n/data/mk.i18n.json | 2 + imports/i18n/data/mn.i18n.json | 2 + imports/i18n/data/ms-MY.i18n.json | 110 ++++++++++----------- imports/i18n/data/ms.i18n.json | 2 + imports/i18n/data/nb.i18n.json | 2 + imports/i18n/data/nl-NL.i18n.json | 2 + imports/i18n/data/nl.i18n.json | 14 +-- imports/i18n/data/oc.i18n.json | 2 + imports/i18n/data/or_IN.i18n.json | 2 + imports/i18n/data/pa.i18n.json | 2 + imports/i18n/data/pl-PL.i18n.json | 2 + imports/i18n/data/pl.i18n.json | 2 + imports/i18n/data/pt-BR.i18n.json | 14 +-- imports/i18n/data/pt.i18n.json | 2 + imports/i18n/data/pt_PT.i18n.json | 2 + imports/i18n/data/ro-RO.i18n.json | 2 + imports/i18n/data/ro.i18n.json | 2 + imports/i18n/data/ru-UA.i18n.json | 2 + imports/i18n/data/ru.i18n.json | 2 + imports/i18n/data/sk.i18n.json | 2 + imports/i18n/data/sl.i18n.json | 2 + imports/i18n/data/sr.i18n.json | 2 + imports/i18n/data/sv.i18n.json | 22 +++-- imports/i18n/data/sw.i18n.json | 2 + imports/i18n/data/ta.i18n.json | 2 + imports/i18n/data/te-IN.i18n.json | 2 + imports/i18n/data/th.i18n.json | 2 + imports/i18n/data/tk_TM.i18n.json | 2 + imports/i18n/data/tlh.i18n.json | 2 + imports/i18n/data/tr.i18n.json | 2 + imports/i18n/data/ug.i18n.json | 2 + imports/i18n/data/uk-UA.i18n.json | 2 + imports/i18n/data/uk.i18n.json | 2 + imports/i18n/data/uz-AR.i18n.json | 2 + imports/i18n/data/uz-LA.i18n.json | 2 + imports/i18n/data/uz-UZ.i18n.json | 2 + imports/i18n/data/uz.i18n.json | 2 + imports/i18n/data/ve-CC.i18n.json | 2 + imports/i18n/data/ve-PP.i18n.json | 2 + imports/i18n/data/ve.i18n.json | 2 + imports/i18n/data/vi-VN.i18n.json | 2 + imports/i18n/data/vi.i18n.json | 2 + imports/i18n/data/vl-SS.i18n.json | 2 + imports/i18n/data/vo.i18n.json | 2 + imports/i18n/data/wa-RR.i18n.json | 2 + imports/i18n/data/wa.i18n.json | 2 + imports/i18n/data/wo.i18n.json | 2 + imports/i18n/data/wuu-Hans.i18n.json | 2 + imports/i18n/data/xh.i18n.json | 2 + imports/i18n/data/yi.i18n.json | 2 + imports/i18n/data/yo.i18n.json | 2 + imports/i18n/data/yue_CN.i18n.json | 2 + imports/i18n/data/zgh.i18n.json | 2 + imports/i18n/data/zh-CN.i18n.json | 2 + imports/i18n/data/zh-GB.i18n.json | 2 + imports/i18n/data/zh-HK.i18n.json | 2 + imports/i18n/data/zh-Hans.i18n.json | 2 + imports/i18n/data/zh-Hant.i18n.json | 8 +- imports/i18n/data/zh-TW.i18n.json | 8 +- imports/i18n/data/zh.i18n.json | 2 + imports/i18n/data/zu-ZA.i18n.json | 2 + imports/i18n/data/zu.i18n.json | 2 + 140 files changed, 423 insertions(+), 143 deletions(-) diff --git a/imports/i18n/data/af.i18n.json b/imports/i18n/data/af.i18n.json index bcdd00061..eef27e1fe 100644 --- a/imports/i18n/data/af.i18n.json +++ b/imports/i18n/data/af.i18n.json @@ -755,6 +755,8 @@ "delete-board-confirm-popup": "All lists, cards, labels, and activities will be deleted and you won't be able to recover the board contents. There is no undo.", "boardDeletePopup-title": "Delete Board?", "delete-board": "Delete Board", + "delete-duplicate-lists": "Delete Duplicate Lists", + "delete-duplicate-lists-confirm": "Are you sure? This will delete all duplicate lists that have the same name and contain no cards.", "default-subtasks-board": "Subtasks for __board__ board", "default": "Default", "defaultdefault": "Default", diff --git a/imports/i18n/data/af_ZA.i18n.json b/imports/i18n/data/af_ZA.i18n.json index 471a35a37..ecc4080e2 100644 --- a/imports/i18n/data/af_ZA.i18n.json +++ b/imports/i18n/data/af_ZA.i18n.json @@ -755,6 +755,8 @@ "delete-board-confirm-popup": "All lists, cards, labels, and activities will be deleted and you won't be able to recover the board contents. There is no undo.", "boardDeletePopup-title": "Delete Board?", "delete-board": "Delete Board", + "delete-duplicate-lists": "Delete Duplicate Lists", + "delete-duplicate-lists-confirm": "Are you sure? This will delete all duplicate lists that have the same name and contain no cards.", "default-subtasks-board": "Subtasks for __board__ board", "default": "Default", "defaultdefault": "Default", diff --git a/imports/i18n/data/ar-DZ.i18n.json b/imports/i18n/data/ar-DZ.i18n.json index 510b05b6b..1ab8eeaa8 100644 --- a/imports/i18n/data/ar-DZ.i18n.json +++ b/imports/i18n/data/ar-DZ.i18n.json @@ -755,6 +755,8 @@ "delete-board-confirm-popup": "All lists, cards, labels, and activities will be deleted and you won't be able to recover the board contents. There is no undo.", "boardDeletePopup-title": "Delete Board?", "delete-board": "Delete Board", + "delete-duplicate-lists": "Delete Duplicate Lists", + "delete-duplicate-lists-confirm": "Are you sure? This will delete all duplicate lists that have the same name and contain no cards.", "default-subtasks-board": "Subtasks for __board__ board", "default": "Default", "defaultdefault": "Default", diff --git a/imports/i18n/data/ar-EG.i18n.json b/imports/i18n/data/ar-EG.i18n.json index 510b05b6b..1ab8eeaa8 100644 --- a/imports/i18n/data/ar-EG.i18n.json +++ b/imports/i18n/data/ar-EG.i18n.json @@ -755,6 +755,8 @@ "delete-board-confirm-popup": "All lists, cards, labels, and activities will be deleted and you won't be able to recover the board contents. There is no undo.", "boardDeletePopup-title": "Delete Board?", "delete-board": "Delete Board", + "delete-duplicate-lists": "Delete Duplicate Lists", + "delete-duplicate-lists-confirm": "Are you sure? This will delete all duplicate lists that have the same name and contain no cards.", "default-subtasks-board": "Subtasks for __board__ board", "default": "Default", "defaultdefault": "Default", diff --git a/imports/i18n/data/ar.i18n.json b/imports/i18n/data/ar.i18n.json index 5c478b0b1..b4628a45d 100644 --- a/imports/i18n/data/ar.i18n.json +++ b/imports/i18n/data/ar.i18n.json @@ -755,6 +755,8 @@ "delete-board-confirm-popup": "All lists, cards, labels, and activities will be deleted and you won't be able to recover the board contents. There is no undo.", "boardDeletePopup-title": "Delete Board?", "delete-board": "Delete Board", + "delete-duplicate-lists": "Delete Duplicate Lists", + "delete-duplicate-lists-confirm": "Are you sure? This will delete all duplicate lists that have the same name and contain no cards.", "default-subtasks-board": "Subtasks for __board__ board", "default": "Default", "defaultdefault": "Default", diff --git a/imports/i18n/data/ary.i18n.json b/imports/i18n/data/ary.i18n.json index 510b05b6b..1ab8eeaa8 100644 --- a/imports/i18n/data/ary.i18n.json +++ b/imports/i18n/data/ary.i18n.json @@ -755,6 +755,8 @@ "delete-board-confirm-popup": "All lists, cards, labels, and activities will be deleted and you won't be able to recover the board contents. There is no undo.", "boardDeletePopup-title": "Delete Board?", "delete-board": "Delete Board", + "delete-duplicate-lists": "Delete Duplicate Lists", + "delete-duplicate-lists-confirm": "Are you sure? This will delete all duplicate lists that have the same name and contain no cards.", "default-subtasks-board": "Subtasks for __board__ board", "default": "Default", "defaultdefault": "Default", diff --git a/imports/i18n/data/ast-ES.i18n.json b/imports/i18n/data/ast-ES.i18n.json index 510b05b6b..1ab8eeaa8 100644 --- a/imports/i18n/data/ast-ES.i18n.json +++ b/imports/i18n/data/ast-ES.i18n.json @@ -755,6 +755,8 @@ "delete-board-confirm-popup": "All lists, cards, labels, and activities will be deleted and you won't be able to recover the board contents. There is no undo.", "boardDeletePopup-title": "Delete Board?", "delete-board": "Delete Board", + "delete-duplicate-lists": "Delete Duplicate Lists", + "delete-duplicate-lists-confirm": "Are you sure? This will delete all duplicate lists that have the same name and contain no cards.", "default-subtasks-board": "Subtasks for __board__ board", "default": "Default", "defaultdefault": "Default", diff --git a/imports/i18n/data/az-AZ.i18n.json b/imports/i18n/data/az-AZ.i18n.json index 510b05b6b..1ab8eeaa8 100644 --- a/imports/i18n/data/az-AZ.i18n.json +++ b/imports/i18n/data/az-AZ.i18n.json @@ -755,6 +755,8 @@ "delete-board-confirm-popup": "All lists, cards, labels, and activities will be deleted and you won't be able to recover the board contents. There is no undo.", "boardDeletePopup-title": "Delete Board?", "delete-board": "Delete Board", + "delete-duplicate-lists": "Delete Duplicate Lists", + "delete-duplicate-lists-confirm": "Are you sure? This will delete all duplicate lists that have the same name and contain no cards.", "default-subtasks-board": "Subtasks for __board__ board", "default": "Default", "defaultdefault": "Default", diff --git a/imports/i18n/data/az-LA.i18n.json b/imports/i18n/data/az-LA.i18n.json index 510b05b6b..1ab8eeaa8 100644 --- a/imports/i18n/data/az-LA.i18n.json +++ b/imports/i18n/data/az-LA.i18n.json @@ -755,6 +755,8 @@ "delete-board-confirm-popup": "All lists, cards, labels, and activities will be deleted and you won't be able to recover the board contents. There is no undo.", "boardDeletePopup-title": "Delete Board?", "delete-board": "Delete Board", + "delete-duplicate-lists": "Delete Duplicate Lists", + "delete-duplicate-lists-confirm": "Are you sure? This will delete all duplicate lists that have the same name and contain no cards.", "default-subtasks-board": "Subtasks for __board__ board", "default": "Default", "defaultdefault": "Default", diff --git a/imports/i18n/data/az.i18n.json b/imports/i18n/data/az.i18n.json index 510b05b6b..1ab8eeaa8 100644 --- a/imports/i18n/data/az.i18n.json +++ b/imports/i18n/data/az.i18n.json @@ -755,6 +755,8 @@ "delete-board-confirm-popup": "All lists, cards, labels, and activities will be deleted and you won't be able to recover the board contents. There is no undo.", "boardDeletePopup-title": "Delete Board?", "delete-board": "Delete Board", + "delete-duplicate-lists": "Delete Duplicate Lists", + "delete-duplicate-lists-confirm": "Are you sure? This will delete all duplicate lists that have the same name and contain no cards.", "default-subtasks-board": "Subtasks for __board__ board", "default": "Default", "defaultdefault": "Default", diff --git a/imports/i18n/data/bg.i18n.json b/imports/i18n/data/bg.i18n.json index 4297d1cd7..043015731 100644 --- a/imports/i18n/data/bg.i18n.json +++ b/imports/i18n/data/bg.i18n.json @@ -755,6 +755,8 @@ "delete-board-confirm-popup": "Всички списъци, карти, имена и действия ще бъдат изтрити и няма да можете да възстановите съдържанието на дъската. Няма връщане назад.", "boardDeletePopup-title": "Изтриване на Таблото?", "delete-board": "Изтрий таблото", + "delete-duplicate-lists": "Delete Duplicate Lists", + "delete-duplicate-lists-confirm": "Are you sure? This will delete all duplicate lists that have the same name and contain no cards.", "default-subtasks-board": "Подзадачи за табло __board__", "default": "по подразбиране", "defaultdefault": "по подразбиране", diff --git a/imports/i18n/data/br.i18n.json b/imports/i18n/data/br.i18n.json index b9e016668..899f9dbb2 100644 --- a/imports/i18n/data/br.i18n.json +++ b/imports/i18n/data/br.i18n.json @@ -755,6 +755,8 @@ "delete-board-confirm-popup": "All lists, cards, labels, and activities will be deleted and you won't be able to recover the board contents. There is no undo.", "boardDeletePopup-title": "Delete Board?", "delete-board": "Delete Board", + "delete-duplicate-lists": "Delete Duplicate Lists", + "delete-duplicate-lists-confirm": "Are you sure? This will delete all duplicate lists that have the same name and contain no cards.", "default-subtasks-board": "Subtasks for __board__ board", "default": "Default", "defaultdefault": "Default", diff --git a/imports/i18n/data/ca.i18n.json b/imports/i18n/data/ca.i18n.json index 28660dddf..019a5eacd 100644 --- a/imports/i18n/data/ca.i18n.json +++ b/imports/i18n/data/ca.i18n.json @@ -755,6 +755,8 @@ "delete-board-confirm-popup": "Se suprimiran totes les llistes, fitxes, etiquetes i activitats i no podreu recuperar el contingut del tauler. No hi ha cap desfer.", "boardDeletePopup-title": "Vols suprimir el tauler?", "delete-board": "Suprimeix el tauler", + "delete-duplicate-lists": "Delete Duplicate Lists", + "delete-duplicate-lists-confirm": "Are you sure? This will delete all duplicate lists that have the same name and contain no cards.", "default-subtasks-board": "Subtasques per al tauler __board__", "default": "Per defecte", "defaultdefault": "Per defecte", diff --git a/imports/i18n/data/ca@valencia.i18n.json b/imports/i18n/data/ca@valencia.i18n.json index 6455ca3ed..ab9a378b1 100644 --- a/imports/i18n/data/ca@valencia.i18n.json +++ b/imports/i18n/data/ca@valencia.i18n.json @@ -755,6 +755,8 @@ "delete-board-confirm-popup": "All lists, cards, labels, and activities will be deleted and you won't be able to recover the board contents. There is no undo.", "boardDeletePopup-title": "Delete Board?", "delete-board": "Delete Board", + "delete-duplicate-lists": "Delete Duplicate Lists", + "delete-duplicate-lists-confirm": "Are you sure? This will delete all duplicate lists that have the same name and contain no cards.", "default-subtasks-board": "Subtasks for __board__ board", "default": "Default", "defaultdefault": "Default", diff --git a/imports/i18n/data/ca_ES.i18n.json b/imports/i18n/data/ca_ES.i18n.json index 06d043b5f..7e2ba5ba7 100644 --- a/imports/i18n/data/ca_ES.i18n.json +++ b/imports/i18n/data/ca_ES.i18n.json @@ -755,6 +755,8 @@ "delete-board-confirm-popup": "All lists, cards, labels, and activities will be deleted and you won't be able to recover the board contents. There is no undo.", "boardDeletePopup-title": "Delete Board?", "delete-board": "Delete Board", + "delete-duplicate-lists": "Delete Duplicate Lists", + "delete-duplicate-lists-confirm": "Are you sure? This will delete all duplicate lists that have the same name and contain no cards.", "default-subtasks-board": "Subtasks for __board__ board", "default": "Default", "defaultdefault": "Default", diff --git a/imports/i18n/data/cmn.i18n.json b/imports/i18n/data/cmn.i18n.json index e2718dafa..f99bd548e 100644 --- a/imports/i18n/data/cmn.i18n.json +++ b/imports/i18n/data/cmn.i18n.json @@ -755,6 +755,8 @@ "delete-board-confirm-popup": "All lists, cards, labels, and activities will be deleted and you won't be able to recover the board contents. There is no undo.", "boardDeletePopup-title": "Delete Board?", "delete-board": "Delete Board", + "delete-duplicate-lists": "Delete Duplicate Lists", + "delete-duplicate-lists-confirm": "Are you sure? This will delete all duplicate lists that have the same name and contain no cards.", "default-subtasks-board": "Subtasks for __board__ board", "default": "Default", "defaultdefault": "Default", diff --git a/imports/i18n/data/cs-CZ.i18n.json b/imports/i18n/data/cs-CZ.i18n.json index f7b2fe0b3..0e915cedf 100644 --- a/imports/i18n/data/cs-CZ.i18n.json +++ b/imports/i18n/data/cs-CZ.i18n.json @@ -755,6 +755,8 @@ "delete-board-confirm-popup": "Všechny sloupce, štítky a aktivity budou smazány a obsah tabla nebude možné obnovit. Toto nelze vrátit zpět.", "boardDeletePopup-title": "Smazat tablo?", "delete-board": "Smazat tablo", + "delete-duplicate-lists": "Delete Duplicate Lists", + "delete-duplicate-lists-confirm": "Are you sure? This will delete all duplicate lists that have the same name and contain no cards.", "default-subtasks-board": "Podúkoly pro tablo __board__", "default": "Výchozí", "defaultdefault": "Výchozí", diff --git a/imports/i18n/data/cs.i18n.json b/imports/i18n/data/cs.i18n.json index a41e04773..58abbc60d 100644 --- a/imports/i18n/data/cs.i18n.json +++ b/imports/i18n/data/cs.i18n.json @@ -755,6 +755,8 @@ "delete-board-confirm-popup": "Všechny sloupce, štítky a aktivity budou smazány a obsah tabla nebude možné obnovit. Toto nelze vrátit zpět.", "boardDeletePopup-title": "Smazat tablo?", "delete-board": "Smazat tablo", + "delete-duplicate-lists": "Delete Duplicate Lists", + "delete-duplicate-lists-confirm": "Are you sure? This will delete all duplicate lists that have the same name and contain no cards.", "default-subtasks-board": "Podúkoly pro tablo __board__", "default": "Výchozí", "defaultdefault": "Výchozí", diff --git a/imports/i18n/data/cy-GB.i18n.json b/imports/i18n/data/cy-GB.i18n.json index 510b05b6b..1ab8eeaa8 100644 --- a/imports/i18n/data/cy-GB.i18n.json +++ b/imports/i18n/data/cy-GB.i18n.json @@ -755,6 +755,8 @@ "delete-board-confirm-popup": "All lists, cards, labels, and activities will be deleted and you won't be able to recover the board contents. There is no undo.", "boardDeletePopup-title": "Delete Board?", "delete-board": "Delete Board", + "delete-duplicate-lists": "Delete Duplicate Lists", + "delete-duplicate-lists-confirm": "Are you sure? This will delete all duplicate lists that have the same name and contain no cards.", "default-subtasks-board": "Subtasks for __board__ board", "default": "Default", "defaultdefault": "Default", diff --git a/imports/i18n/data/cy.i18n.json b/imports/i18n/data/cy.i18n.json index 510b05b6b..1ab8eeaa8 100644 --- a/imports/i18n/data/cy.i18n.json +++ b/imports/i18n/data/cy.i18n.json @@ -755,6 +755,8 @@ "delete-board-confirm-popup": "All lists, cards, labels, and activities will be deleted and you won't be able to recover the board contents. There is no undo.", "boardDeletePopup-title": "Delete Board?", "delete-board": "Delete Board", + "delete-duplicate-lists": "Delete Duplicate Lists", + "delete-duplicate-lists-confirm": "Are you sure? This will delete all duplicate lists that have the same name and contain no cards.", "default-subtasks-board": "Subtasks for __board__ board", "default": "Default", "defaultdefault": "Default", diff --git a/imports/i18n/data/da.i18n.json b/imports/i18n/data/da.i18n.json index 79bdfb7aa..bfe07883d 100644 --- a/imports/i18n/data/da.i18n.json +++ b/imports/i18n/data/da.i18n.json @@ -755,6 +755,8 @@ "delete-board-confirm-popup": "Alle lister, kort, etiketter og aktiviteter vil blive slettet og du får ikke mulighed for at genskabe tavlens indhold. Dette kan ikke fortrydes.", "boardDeletePopup-title": "Slet tavle?", "delete-board": "Slet tavle", + "delete-duplicate-lists": "Delete Duplicate Lists", + "delete-duplicate-lists-confirm": "Are you sure? This will delete all duplicate lists that have the same name and contain no cards.", "default-subtasks-board": "Delopgaver for tavlen __board__", "default": "Standard", "defaultdefault": "Standard", diff --git a/imports/i18n/data/de-AT.i18n.json b/imports/i18n/data/de-AT.i18n.json index 4f5633d2e..625ad5655 100644 --- a/imports/i18n/data/de-AT.i18n.json +++ b/imports/i18n/data/de-AT.i18n.json @@ -755,6 +755,8 @@ "delete-board-confirm-popup": "Alle Listen, Karten, Labels und Akivitäten werden gelöscht und Sie können die Inhalte des Boards nicht wiederherstellen! Die Aktion kann nicht rückgängig gemacht werden.", "boardDeletePopup-title": "Board löschen?", "delete-board": "Board löschen", + "delete-duplicate-lists": "Delete Duplicate Lists", + "delete-duplicate-lists-confirm": "Are you sure? This will delete all duplicate lists that have the same name and contain no cards.", "default-subtasks-board": "Teilaufgabe für __board__ Board", "default": "Standard", "defaultdefault": "Standard", diff --git a/imports/i18n/data/de-CH.i18n.json b/imports/i18n/data/de-CH.i18n.json index 77c5ba55e..4c9682382 100644 --- a/imports/i18n/data/de-CH.i18n.json +++ b/imports/i18n/data/de-CH.i18n.json @@ -755,6 +755,8 @@ "delete-board-confirm-popup": "Alle Listen, Karten, Labels und Akivitäten werden gelöscht und Sie können die Inhalte des Boards nicht wiederherstellen! Die Aktion kann nicht rückgängig gemacht werden.", "boardDeletePopup-title": "Board löschen?", "delete-board": "Board löschen", + "delete-duplicate-lists": "Delete Duplicate Lists", + "delete-duplicate-lists-confirm": "Are you sure? This will delete all duplicate lists that have the same name and contain no cards.", "default-subtasks-board": "Teilaufgabe für __board__ Board", "default": "Standard", "defaultdefault": "Standard", diff --git a/imports/i18n/data/de.i18n.json b/imports/i18n/data/de.i18n.json index 586c2ab97..f138b8d6c 100644 --- a/imports/i18n/data/de.i18n.json +++ b/imports/i18n/data/de.i18n.json @@ -755,6 +755,8 @@ "delete-board-confirm-popup": "Alle Listen, Karten, Labels und Akivitäten werden gelöscht und Sie können die Inhalte des Boards nicht wiederherstellen! Die Aktion kann nicht rückgängig gemacht werden.", "boardDeletePopup-title": "Board löschen?", "delete-board": "Board löschen", + "delete-duplicate-lists": "Delete Duplicate Lists", + "delete-duplicate-lists-confirm": "Are you sure? This will delete all duplicate lists that have the same name and contain no cards.", "default-subtasks-board": "Teilaufgabe für __board__ Board", "default": "Standard", "defaultdefault": "Standard", diff --git a/imports/i18n/data/de_DE.i18n.json b/imports/i18n/data/de_DE.i18n.json index 44f91e057..c995377fb 100644 --- a/imports/i18n/data/de_DE.i18n.json +++ b/imports/i18n/data/de_DE.i18n.json @@ -755,6 +755,8 @@ "delete-board-confirm-popup": "Alle Listen, Karten, Labels und Akivitäten werden gelöscht und Sie können die Inhalte des Boards nicht wiederherstellen! Die Aktion kann nicht rückgängig gemacht werden.", "boardDeletePopup-title": "Board löschen?", "delete-board": "Board löschen", + "delete-duplicate-lists": "Delete Duplicate Lists", + "delete-duplicate-lists-confirm": "Are you sure? This will delete all duplicate lists that have the same name and contain no cards.", "default-subtasks-board": "Teilaufgabe für __board__ Board", "default": "Standard", "defaultdefault": "Standard", diff --git a/imports/i18n/data/el-GR.i18n.json b/imports/i18n/data/el-GR.i18n.json index d1030cd9a..48fcea53a 100644 --- a/imports/i18n/data/el-GR.i18n.json +++ b/imports/i18n/data/el-GR.i18n.json @@ -755,6 +755,8 @@ "delete-board-confirm-popup": "All lists, cards, labels, and activities will be deleted and you won't be able to recover the board contents. There is no undo.", "boardDeletePopup-title": "Διαγραφή Πίνακα;", "delete-board": "Διαγραφή Πίνακα", + "delete-duplicate-lists": "Delete Duplicate Lists", + "delete-duplicate-lists-confirm": "Are you sure? This will delete all duplicate lists that have the same name and contain no cards.", "default-subtasks-board": "Subtasks for __board__ board", "default": "Προεπιλογή", "defaultdefault": "Προεπιλογή", diff --git a/imports/i18n/data/el.i18n.json b/imports/i18n/data/el.i18n.json index 096701fd0..c4da6359d 100644 --- a/imports/i18n/data/el.i18n.json +++ b/imports/i18n/data/el.i18n.json @@ -755,6 +755,8 @@ "delete-board-confirm-popup": "All lists, cards, labels, and activities will be deleted and you won't be able to recover the board contents. There is no undo.", "boardDeletePopup-title": "Διαγραφή Πίνακα;", "delete-board": "Διαγραφή Πίνακα", + "delete-duplicate-lists": "Delete Duplicate Lists", + "delete-duplicate-lists-confirm": "Are you sure? This will delete all duplicate lists that have the same name and contain no cards.", "default-subtasks-board": "Subtasks for __board__ board", "default": "Προεπιλογή", "defaultdefault": "Προεπιλογή", diff --git a/imports/i18n/data/en-BR.i18n.json b/imports/i18n/data/en-BR.i18n.json index 510b05b6b..1ab8eeaa8 100644 --- a/imports/i18n/data/en-BR.i18n.json +++ b/imports/i18n/data/en-BR.i18n.json @@ -755,6 +755,8 @@ "delete-board-confirm-popup": "All lists, cards, labels, and activities will be deleted and you won't be able to recover the board contents. There is no undo.", "boardDeletePopup-title": "Delete Board?", "delete-board": "Delete Board", + "delete-duplicate-lists": "Delete Duplicate Lists", + "delete-duplicate-lists-confirm": "Are you sure? This will delete all duplicate lists that have the same name and contain no cards.", "default-subtasks-board": "Subtasks for __board__ board", "default": "Default", "defaultdefault": "Default", diff --git a/imports/i18n/data/en-DE.i18n.json b/imports/i18n/data/en-DE.i18n.json index 6895e31db..50ddae1d2 100644 --- a/imports/i18n/data/en-DE.i18n.json +++ b/imports/i18n/data/en-DE.i18n.json @@ -755,6 +755,8 @@ "delete-board-confirm-popup": "All lists, cards, labels, and activities will be deleted and you won't be able to recover the board contents. There is no undo.", "boardDeletePopup-title": "Delete Board?", "delete-board": "Delete Board", + "delete-duplicate-lists": "Delete Duplicate Lists", + "delete-duplicate-lists-confirm": "Are you sure? This will delete all duplicate lists that have the same name and contain no cards.", "default-subtasks-board": "Subtasks for __board__ board", "default": "Default", "defaultdefault": "Default", diff --git a/imports/i18n/data/en-GB.i18n.json b/imports/i18n/data/en-GB.i18n.json index bcdd00061..eef27e1fe 100644 --- a/imports/i18n/data/en-GB.i18n.json +++ b/imports/i18n/data/en-GB.i18n.json @@ -755,6 +755,8 @@ "delete-board-confirm-popup": "All lists, cards, labels, and activities will be deleted and you won't be able to recover the board contents. There is no undo.", "boardDeletePopup-title": "Delete Board?", "delete-board": "Delete Board", + "delete-duplicate-lists": "Delete Duplicate Lists", + "delete-duplicate-lists-confirm": "Are you sure? This will delete all duplicate lists that have the same name and contain no cards.", "default-subtasks-board": "Subtasks for __board__ board", "default": "Default", "defaultdefault": "Default", diff --git a/imports/i18n/data/en-IT.i18n.json b/imports/i18n/data/en-IT.i18n.json index 510b05b6b..1ab8eeaa8 100644 --- a/imports/i18n/data/en-IT.i18n.json +++ b/imports/i18n/data/en-IT.i18n.json @@ -755,6 +755,8 @@ "delete-board-confirm-popup": "All lists, cards, labels, and activities will be deleted and you won't be able to recover the board contents. There is no undo.", "boardDeletePopup-title": "Delete Board?", "delete-board": "Delete Board", + "delete-duplicate-lists": "Delete Duplicate Lists", + "delete-duplicate-lists-confirm": "Are you sure? This will delete all duplicate lists that have the same name and contain no cards.", "default-subtasks-board": "Subtasks for __board__ board", "default": "Default", "defaultdefault": "Default", diff --git a/imports/i18n/data/en-MY.i18n.json b/imports/i18n/data/en-MY.i18n.json index 510b05b6b..1ab8eeaa8 100644 --- a/imports/i18n/data/en-MY.i18n.json +++ b/imports/i18n/data/en-MY.i18n.json @@ -755,6 +755,8 @@ "delete-board-confirm-popup": "All lists, cards, labels, and activities will be deleted and you won't be able to recover the board contents. There is no undo.", "boardDeletePopup-title": "Delete Board?", "delete-board": "Delete Board", + "delete-duplicate-lists": "Delete Duplicate Lists", + "delete-duplicate-lists-confirm": "Are you sure? This will delete all duplicate lists that have the same name and contain no cards.", "default-subtasks-board": "Subtasks for __board__ board", "default": "Default", "defaultdefault": "Default", diff --git a/imports/i18n/data/en-YS.i18n.json b/imports/i18n/data/en-YS.i18n.json index 510b05b6b..1ab8eeaa8 100644 --- a/imports/i18n/data/en-YS.i18n.json +++ b/imports/i18n/data/en-YS.i18n.json @@ -755,6 +755,8 @@ "delete-board-confirm-popup": "All lists, cards, labels, and activities will be deleted and you won't be able to recover the board contents. There is no undo.", "boardDeletePopup-title": "Delete Board?", "delete-board": "Delete Board", + "delete-duplicate-lists": "Delete Duplicate Lists", + "delete-duplicate-lists-confirm": "Are you sure? This will delete all duplicate lists that have the same name and contain no cards.", "default-subtasks-board": "Subtasks for __board__ board", "default": "Default", "defaultdefault": "Default", diff --git a/imports/i18n/data/en_AU.i18n.json b/imports/i18n/data/en_AU.i18n.json index bcdd00061..eef27e1fe 100644 --- a/imports/i18n/data/en_AU.i18n.json +++ b/imports/i18n/data/en_AU.i18n.json @@ -755,6 +755,8 @@ "delete-board-confirm-popup": "All lists, cards, labels, and activities will be deleted and you won't be able to recover the board contents. There is no undo.", "boardDeletePopup-title": "Delete Board?", "delete-board": "Delete Board", + "delete-duplicate-lists": "Delete Duplicate Lists", + "delete-duplicate-lists-confirm": "Are you sure? This will delete all duplicate lists that have the same name and contain no cards.", "default-subtasks-board": "Subtasks for __board__ board", "default": "Default", "defaultdefault": "Default", diff --git a/imports/i18n/data/en_ID.i18n.json b/imports/i18n/data/en_ID.i18n.json index bcdd00061..eef27e1fe 100644 --- a/imports/i18n/data/en_ID.i18n.json +++ b/imports/i18n/data/en_ID.i18n.json @@ -755,6 +755,8 @@ "delete-board-confirm-popup": "All lists, cards, labels, and activities will be deleted and you won't be able to recover the board contents. There is no undo.", "boardDeletePopup-title": "Delete Board?", "delete-board": "Delete Board", + "delete-duplicate-lists": "Delete Duplicate Lists", + "delete-duplicate-lists-confirm": "Are you sure? This will delete all duplicate lists that have the same name and contain no cards.", "default-subtasks-board": "Subtasks for __board__ board", "default": "Default", "defaultdefault": "Default", diff --git a/imports/i18n/data/en_SG.i18n.json b/imports/i18n/data/en_SG.i18n.json index bcdd00061..eef27e1fe 100644 --- a/imports/i18n/data/en_SG.i18n.json +++ b/imports/i18n/data/en_SG.i18n.json @@ -755,6 +755,8 @@ "delete-board-confirm-popup": "All lists, cards, labels, and activities will be deleted and you won't be able to recover the board contents. There is no undo.", "boardDeletePopup-title": "Delete Board?", "delete-board": "Delete Board", + "delete-duplicate-lists": "Delete Duplicate Lists", + "delete-duplicate-lists-confirm": "Are you sure? This will delete all duplicate lists that have the same name and contain no cards.", "default-subtasks-board": "Subtasks for __board__ board", "default": "Default", "defaultdefault": "Default", diff --git a/imports/i18n/data/en_TR.i18n.json b/imports/i18n/data/en_TR.i18n.json index bcdd00061..eef27e1fe 100644 --- a/imports/i18n/data/en_TR.i18n.json +++ b/imports/i18n/data/en_TR.i18n.json @@ -755,6 +755,8 @@ "delete-board-confirm-popup": "All lists, cards, labels, and activities will be deleted and you won't be able to recover the board contents. There is no undo.", "boardDeletePopup-title": "Delete Board?", "delete-board": "Delete Board", + "delete-duplicate-lists": "Delete Duplicate Lists", + "delete-duplicate-lists-confirm": "Are you sure? This will delete all duplicate lists that have the same name and contain no cards.", "default-subtasks-board": "Subtasks for __board__ board", "default": "Default", "defaultdefault": "Default", diff --git a/imports/i18n/data/en_ZA.i18n.json b/imports/i18n/data/en_ZA.i18n.json index bcdd00061..eef27e1fe 100644 --- a/imports/i18n/data/en_ZA.i18n.json +++ b/imports/i18n/data/en_ZA.i18n.json @@ -755,6 +755,8 @@ "delete-board-confirm-popup": "All lists, cards, labels, and activities will be deleted and you won't be able to recover the board contents. There is no undo.", "boardDeletePopup-title": "Delete Board?", "delete-board": "Delete Board", + "delete-duplicate-lists": "Delete Duplicate Lists", + "delete-duplicate-lists-confirm": "Are you sure? This will delete all duplicate lists that have the same name and contain no cards.", "default-subtasks-board": "Subtasks for __board__ board", "default": "Default", "defaultdefault": "Default", diff --git a/imports/i18n/data/eo.i18n.json b/imports/i18n/data/eo.i18n.json index 66aaa66fa..20a85f9f6 100644 --- a/imports/i18n/data/eo.i18n.json +++ b/imports/i18n/data/eo.i18n.json @@ -755,6 +755,8 @@ "delete-board-confirm-popup": "All lists, cards, labels, and activities will be deleted and you won't be able to recover the board contents. There is no undo.", "boardDeletePopup-title": "Delete Board?", "delete-board": "Delete Board", + "delete-duplicate-lists": "Delete Duplicate Lists", + "delete-duplicate-lists-confirm": "Are you sure? This will delete all duplicate lists that have the same name and contain no cards.", "default-subtasks-board": "Subtasks for __board__ board", "default": "Defaŭlto", "defaultdefault": "Defaŭlto", diff --git a/imports/i18n/data/es-AR.i18n.json b/imports/i18n/data/es-AR.i18n.json index 6be715c61..874d60c15 100644 --- a/imports/i18n/data/es-AR.i18n.json +++ b/imports/i18n/data/es-AR.i18n.json @@ -755,6 +755,8 @@ "delete-board-confirm-popup": "All lists, cards, labels, and activities will be deleted and you won't be able to recover the board contents. There is no undo.", "boardDeletePopup-title": "Delete Board?", "delete-board": "Delete Board", + "delete-duplicate-lists": "Delete Duplicate Lists", + "delete-duplicate-lists-confirm": "Are you sure? This will delete all duplicate lists that have the same name and contain no cards.", "default-subtasks-board": "Subtasks for __board__ board", "default": "Default", "defaultdefault": "Default", diff --git a/imports/i18n/data/es-CL.i18n.json b/imports/i18n/data/es-CL.i18n.json index d3c77ecd4..6b167dc52 100644 --- a/imports/i18n/data/es-CL.i18n.json +++ b/imports/i18n/data/es-CL.i18n.json @@ -755,6 +755,8 @@ "delete-board-confirm-popup": "Se eliminarán todas las listas, tarjetas, etiquetas y actividades, y no podrás recuperar los contenidos del tablero. Esta acción no puede deshacerse.", "boardDeletePopup-title": "¿Eliminar el tablero?", "delete-board": "Eliminar el tablero", + "delete-duplicate-lists": "Delete Duplicate Lists", + "delete-duplicate-lists-confirm": "Are you sure? This will delete all duplicate lists that have the same name and contain no cards.", "default-subtasks-board": "Subtareas para el tablero __board__", "default": "Por defecto", "defaultdefault": "Por defecto", diff --git a/imports/i18n/data/es-LA.i18n.json b/imports/i18n/data/es-LA.i18n.json index bcdd00061..eef27e1fe 100644 --- a/imports/i18n/data/es-LA.i18n.json +++ b/imports/i18n/data/es-LA.i18n.json @@ -755,6 +755,8 @@ "delete-board-confirm-popup": "All lists, cards, labels, and activities will be deleted and you won't be able to recover the board contents. There is no undo.", "boardDeletePopup-title": "Delete Board?", "delete-board": "Delete Board", + "delete-duplicate-lists": "Delete Duplicate Lists", + "delete-duplicate-lists-confirm": "Are you sure? This will delete all duplicate lists that have the same name and contain no cards.", "default-subtasks-board": "Subtasks for __board__ board", "default": "Default", "defaultdefault": "Default", diff --git a/imports/i18n/data/es-MX.i18n.json b/imports/i18n/data/es-MX.i18n.json index 29cf721b7..858a623c9 100644 --- a/imports/i18n/data/es-MX.i18n.json +++ b/imports/i18n/data/es-MX.i18n.json @@ -755,6 +755,8 @@ "delete-board-confirm-popup": "All lists, cards, labels, and activities will be deleted and you won't be able to recover the board contents. There is no undo.", "boardDeletePopup-title": "Delete Board?", "delete-board": "Delete Board", + "delete-duplicate-lists": "Delete Duplicate Lists", + "delete-duplicate-lists-confirm": "Are you sure? This will delete all duplicate lists that have the same name and contain no cards.", "default-subtasks-board": "Subtasks for __board__ board", "default": "Default", "defaultdefault": "Default", diff --git a/imports/i18n/data/es-PE.i18n.json b/imports/i18n/data/es-PE.i18n.json index 6a26d0f5e..276b1d368 100644 --- a/imports/i18n/data/es-PE.i18n.json +++ b/imports/i18n/data/es-PE.i18n.json @@ -755,6 +755,8 @@ "delete-board-confirm-popup": "Se eliminarán todas las listas, tarjetas, etiquetas y actividades, y no podrás recuperar los contenidos del tablero. Esta acción no puede deshacerse.", "boardDeletePopup-title": "¿Eliminar el tablero?", "delete-board": "Eliminar el tablero", + "delete-duplicate-lists": "Delete Duplicate Lists", + "delete-duplicate-lists-confirm": "Are you sure? This will delete all duplicate lists that have the same name and contain no cards.", "default-subtasks-board": "Subtareas para el tablero __board__", "default": "Por defecto", "defaultdefault": "Por defecto", diff --git a/imports/i18n/data/es-PY.i18n.json b/imports/i18n/data/es-PY.i18n.json index bcdd00061..eef27e1fe 100644 --- a/imports/i18n/data/es-PY.i18n.json +++ b/imports/i18n/data/es-PY.i18n.json @@ -755,6 +755,8 @@ "delete-board-confirm-popup": "All lists, cards, labels, and activities will be deleted and you won't be able to recover the board contents. There is no undo.", "boardDeletePopup-title": "Delete Board?", "delete-board": "Delete Board", + "delete-duplicate-lists": "Delete Duplicate Lists", + "delete-duplicate-lists-confirm": "Are you sure? This will delete all duplicate lists that have the same name and contain no cards.", "default-subtasks-board": "Subtasks for __board__ board", "default": "Default", "defaultdefault": "Default", diff --git a/imports/i18n/data/es.i18n.json b/imports/i18n/data/es.i18n.json index 31713206d..c0ea65cd5 100644 --- a/imports/i18n/data/es.i18n.json +++ b/imports/i18n/data/es.i18n.json @@ -755,6 +755,8 @@ "delete-board-confirm-popup": "Se eliminarán todas las listas, tarjetas, etiquetas y actividades, y no podrás recuperar los contenidos del tablero. Esta acción no puede deshacerse.", "boardDeletePopup-title": "¿Eliminar el tablero?", "delete-board": "Eliminar el tablero", + "delete-duplicate-lists": "Delete Duplicate Lists", + "delete-duplicate-lists-confirm": "Are you sure? This will delete all duplicate lists that have the same name and contain no cards.", "default-subtasks-board": "Subtareas para el tablero __board__", "default": "Por defecto", "defaultdefault": "Por defecto", diff --git a/imports/i18n/data/es_CO.i18n.json b/imports/i18n/data/es_CO.i18n.json index bcdd00061..eef27e1fe 100644 --- a/imports/i18n/data/es_CO.i18n.json +++ b/imports/i18n/data/es_CO.i18n.json @@ -755,6 +755,8 @@ "delete-board-confirm-popup": "All lists, cards, labels, and activities will be deleted and you won't be able to recover the board contents. There is no undo.", "boardDeletePopup-title": "Delete Board?", "delete-board": "Delete Board", + "delete-duplicate-lists": "Delete Duplicate Lists", + "delete-duplicate-lists-confirm": "Are you sure? This will delete all duplicate lists that have the same name and contain no cards.", "default-subtasks-board": "Subtasks for __board__ board", "default": "Default", "defaultdefault": "Default", diff --git a/imports/i18n/data/et-EE.i18n.json b/imports/i18n/data/et-EE.i18n.json index e53458d17..40bf924c4 100644 --- a/imports/i18n/data/et-EE.i18n.json +++ b/imports/i18n/data/et-EE.i18n.json @@ -755,6 +755,8 @@ "delete-board-confirm-popup": "Kõik nimekirjad, kaardid, sildid ja tegevused kustutatakse ja te ei saa tahvli sisu taastada. Tühistamist ei ole võimalik teha.", "boardDeletePopup-title": "Kustuta juhatus?", "delete-board": "Kustuta juhatus", + "delete-duplicate-lists": "Delete Duplicate Lists", + "delete-duplicate-lists-confirm": "Are you sure? This will delete all duplicate lists that have the same name and contain no cards.", "default-subtasks-board": "Juhatuse __board__ alamülesanded", "default": "Vaikimisi", "defaultdefault": "Vaikimisi", diff --git a/imports/i18n/data/eu.i18n.json b/imports/i18n/data/eu.i18n.json index 4c84d0baf..84dd9440d 100644 --- a/imports/i18n/data/eu.i18n.json +++ b/imports/i18n/data/eu.i18n.json @@ -755,6 +755,8 @@ "delete-board-confirm-popup": "Zerrenda, txartel eta aktibitate guztiak ezabatuko dira eta ezingo dituzu berreskuratu arbelaren edukiak. Atzera bueltarik ez du.", "boardDeletePopup-title": "Delete Board?", "delete-board": "Delete Board", + "delete-duplicate-lists": "Delete Duplicate Lists", + "delete-duplicate-lists-confirm": "Are you sure? This will delete all duplicate lists that have the same name and contain no cards.", "default-subtasks-board": "Subtasks for __board__ board", "default": "Lehenetsia", "defaultdefault": "Lehenetsia", diff --git a/imports/i18n/data/fa-IR.i18n.json b/imports/i18n/data/fa-IR.i18n.json index 82f4b17c3..ddfb6a36d 100644 --- a/imports/i18n/data/fa-IR.i18n.json +++ b/imports/i18n/data/fa-IR.i18n.json @@ -755,6 +755,8 @@ "delete-board-confirm-popup": "تمام لیست ها، کارت ها، لیبل ها و فعالیت ها حذف خواهند شد و شما نمی توانید محتوای برد را بازیابی کنید. هیچ واکنشی وجود ندارد", "boardDeletePopup-title": "حذف برد؟", "delete-board": "حذف برد", + "delete-duplicate-lists": "Delete Duplicate Lists", + "delete-duplicate-lists-confirm": "Are you sure? This will delete all duplicate lists that have the same name and contain no cards.", "default-subtasks-board": "زیروظایفِ برد __board__", "default": "پیش‌فرض", "defaultdefault": "پیش‌فرض", diff --git a/imports/i18n/data/fa.i18n.json b/imports/i18n/data/fa.i18n.json index 7b18e0445..38d280360 100644 --- a/imports/i18n/data/fa.i18n.json +++ b/imports/i18n/data/fa.i18n.json @@ -755,6 +755,8 @@ "delete-board-confirm-popup": "تمام لیست ها، کارت ها، لیبل ها و فعالیت ها حذف خواهند شد و شما نمی توانید محتوای برد را بازیابی کنید. هیچ واکنشی وجود ندارد", "boardDeletePopup-title": "حذف برد؟", "delete-board": "حذف برد", + "delete-duplicate-lists": "Delete Duplicate Lists", + "delete-duplicate-lists-confirm": "Are you sure? This will delete all duplicate lists that have the same name and contain no cards.", "default-subtasks-board": "زیروظایفِ برد __board__", "default": "پیش‌فرض", "defaultdefault": "پیش‌فرض", diff --git a/imports/i18n/data/fi.i18n.json b/imports/i18n/data/fi.i18n.json index b9c5fc9db..cdec49369 100644 --- a/imports/i18n/data/fi.i18n.json +++ b/imports/i18n/data/fi.i18n.json @@ -755,6 +755,8 @@ "delete-board-confirm-popup": "Kaikki listat, kortit, nimilaput ja toimet poistetaan ja et pysty palauttamaan taulun sisältöä. Tätä ei voi peruuttaa.", "boardDeletePopup-title": "Poista taulu?", "delete-board": "Poista taulu", + "delete-duplicate-lists": "Poista ylimääräiset lista kopiot", + "delete-duplicate-lists-confirm": "Oletko varma? Tämä poistaa kaikki ylimääräiset lista kopiot, joilla on sama nimi ja jotka eivät sisällä kortteja.", "default-subtasks-board": "Alitehtävät taululle __board__", "default": "Oletus", "defaultdefault": "Oletus", diff --git a/imports/i18n/data/fr-CH.i18n.json b/imports/i18n/data/fr-CH.i18n.json index ffc428279..ab5df8545 100644 --- a/imports/i18n/data/fr-CH.i18n.json +++ b/imports/i18n/data/fr-CH.i18n.json @@ -755,6 +755,8 @@ "delete-board-confirm-popup": "All lists, cards, labels, and activities will be deleted and you won't be able to recover the board contents. There is no undo.", "boardDeletePopup-title": "Delete Board?", "delete-board": "Delete Board", + "delete-duplicate-lists": "Delete Duplicate Lists", + "delete-duplicate-lists-confirm": "Are you sure? This will delete all duplicate lists that have the same name and contain no cards.", "default-subtasks-board": "Subtasks for __board__ board", "default": "Default", "defaultdefault": "Default", diff --git a/imports/i18n/data/fr-FR.i18n.json b/imports/i18n/data/fr-FR.i18n.json index feb923960..643f299b7 100644 --- a/imports/i18n/data/fr-FR.i18n.json +++ b/imports/i18n/data/fr-FR.i18n.json @@ -755,6 +755,8 @@ "delete-board-confirm-popup": "Toutes les listes, cartes, étiquettes et activités seront supprimées et vous ne pourrez pas retrouver le contenu du tableau. Cela est irréversible.", "boardDeletePopup-title": "Supprimer le tableau ?", "delete-board": "Supprimer le tableau", + "delete-duplicate-lists": "Delete Duplicate Lists", + "delete-duplicate-lists-confirm": "Are you sure? This will delete all duplicate lists that have the same name and contain no cards.", "default-subtasks-board": "Sous-tâches du tableau __board__", "default": "Défaut", "defaultdefault": "Défaut", diff --git a/imports/i18n/data/fr.i18n.json b/imports/i18n/data/fr.i18n.json index a12c9a2e9..042abb941 100644 --- a/imports/i18n/data/fr.i18n.json +++ b/imports/i18n/data/fr.i18n.json @@ -357,9 +357,9 @@ "custom-fields": "Champs personnalisés", "date": "Date", "date-format": "Format de la date", - "date-format-yyyy-mm-dd": "YYYY-MM-DD", - "date-format-dd-mm-yyyy": "DD-MM-YYYY", - "date-format-mm-dd-yyyy": "MM-DD-YYYY", + "date-format-yyyy-mm-dd": "AAAA-MM-JJ", + "date-format-dd-mm-yyyy": "JJ-MM-AAAA", + "date-format-mm-dd-yyyy": "MM-JJ-AAAA", "decline": "Refuser", "default-avatar": "Avatar par défaut", "delete": "Supprimer", @@ -384,21 +384,21 @@ "editLabelPopup-title": "Modifier l'étiquette", "editNotificationPopup-title": "Modifier la notification", "editProfilePopup-title": "Modifier le profil", - "email": "E-mail", - "email-address": "Email Address", + "email": "Courriel", + "email-address": "Adresse de courriel", "email-enrollAccount-subject": "Un compte a été créé pour vous sur __siteName__", "email-enrollAccount-text": "Bonjour __user__,\n\nPour commencer à utiliser ce service, il suffit de cliquer sur le lien ci-dessous.\n\n__url__\n\nMerci.", "email-fail": "Échec de l'envoi du courriel.", - "email-fail-text": "Une erreur est survenue en tentant d'envoyer l'email", - "email-invalid": "Adresse e-mail incorrecte.", - "email-invite": "Inviter par e-mail", + "email-fail-text": "Une erreur est survenue en tentant d'envoyer le courriel", + "email-invalid": "Adresse de courriel incorrecte.", + "email-invite": "Inviter par courriel", "email-invite-subject": "__inviter__ vous a envoyé une invitation", "email-invite-text": "Cher/Chère __user__,\n\n__inviter__ vous invite à rejoindre le tableau \"__board__\" pour collaborer.\n\nVeuillez suivre le lien ci-dessous :\n\n__url__\n\nMerci.", "email-resetPassword-subject": "Réinitialiser votre mot de passe sur __siteName__", "email-resetPassword-text": "Bonjour __user__,\n\nPour réinitialiser votre mot de passe, cliquez sur le lien ci-dessous.\n\n__url__\n\nMerci.", "email-sent": "Courriel envoyé", "email-verifyEmail-subject": "Vérifier votre adresse de courriel sur __siteName__", - "email-verifyEmail-text": "Bonjour __user__,\n\nPour vérifier votre compte courriel, il suffit de cliquer sur le lien ci-dessous.\n\n__url__\n\nMerci.", + "email-verifyEmail-text": "Bonjour __user__,\n\nPour vérifier le courriel de votre compte, il suffit de cliquer sur le lien ci-dessous.\n\n__url__\n\nMerci.", "enable-vertical-scrollbars": "Activer les barres de défilement verticales", "enable-wip-limit": "Activer la limite WIP", "error-board-doesNotExist": "Ce tableau n'existe pas", @@ -414,7 +414,7 @@ "error-username-taken": "Ce nom d'utilisateur est déjà utilisé", "error-orgname-taken": "Ce nom d'organisation est déjà utilisé", "error-teamname-taken": "Ce nom d'équipe est déjà utilisé", - "error-email-taken": "Cette adresse mail est déjà utilisée", + "error-email-taken": "Cette adresse de courriel est déjà utilisée", "export-board": "Exporter le tableau", "export-board-json": "Exporter le tableau en JSON", "export-board-csv": "Exporter le tableau en CSV", @@ -638,9 +638,9 @@ "upload": "Télécharger", "upload-avatar": "Télécharger un avatar", "uploaded-avatar": "Avatar téléchargé", - "uploading-files": "Uploading files", - "upload-failed": "Upload failed", - "upload-completed": "Upload completed", + "uploading-files": "Chargement des fichiers", + "upload-failed": "Le chargement a échoué", + "upload-completed": "Le chargement est terminé", "custom-top-left-corner-logo-image-url": "URL de l'Image du logo personnalisé dans le coin supérieur gauche", "custom-top-left-corner-logo-link-url": "Lien URL du logo personnalisé dans le coin supérieur gauche", "custom-top-left-corner-logo-height": "Hauteur du logo personnalisé dans le coin supérieur gauche. Défaut : 27", @@ -676,9 +676,9 @@ "invite": "Inviter", "invite-people": "Inviter une personne", "to-boards": "Au(x) tableau(x)", - "email-addresses": "Adresses mail", - "smtp-host-description": "L'adresse du serveur SMTP qui gère vos mails.", - "smtp-port-description": "Le port des mails sortants du serveur SMTP.", + "email-addresses": "Adresses de courriel", + "smtp-host-description": "L'adresse du serveur SMTP qui gère vos courriels.", + "smtp-port-description": "Le port du serveur SMTP utilisé pour les courriels sortants.", "smtp-tls-description": "Activer la gestion de TLS sur le serveur SMTP", "smtp-host": "Hôte SMTP", "smtp-port": "Port SMTP", @@ -686,12 +686,12 @@ "smtp-password": "Mot de passe", "smtp-tls": "Prise en charge de TLS", "send-from": "De", - "send-smtp-test": "Envoyer un mail de test à vous-même", + "send-smtp-test": "Envoyer un courriel de test à vous-même", "invitation-code": "Code d'invitation", "email-invite-register-subject": "__inviter__ vous a envoyé une invitation", "email-invite-register-text": "Cher/Chère __user__,\n\n__inviter__ vous invite à le rejoindre sur le tableau kanban pour collaborer.\n\nVeuillez suivre le lien ci-dessous :\n__url__\n\nVotre code d'invitation est : __icode__\n\nMerci.", - "email-smtp-test-subject": "E-mail de test SMTP", - "email-smtp-test-text": "Vous avez envoyé un mail avec succès", + "email-smtp-test-subject": "Courriel de test SMTP", + "email-smtp-test-text": "Vous avez envoyé un courriel avec succès", "error-invitation-code-not-exist": "Ce code d'invitation n'existe pas.", "error-notAuthorized": "Vous n'êtes pas autorisé à accéder à cette page.", "webhook-title": "Nom du webhook", @@ -730,7 +730,7 @@ "yes": "Oui", "no": "Non", "accounts": "Comptes", - "accounts-allowEmailChange": "Autoriser le changement d'adresse mail", + "accounts-allowEmailChange": "Autoriser le changement d'adresse de courriel", "accounts-allowUserNameChange": "Autoriser le changement d'identifiant", "tableVisibilityMode-allowPrivateOnly": "Visibilité des tableaux: N'autoriser que des tableaux privés", "tableVisibilityMode" : "Visibilité des tableaux", @@ -755,6 +755,8 @@ "delete-board-confirm-popup": "Toutes les listes, cartes, étiquettes et activités seront supprimées et vous ne pourrez pas retrouver le contenu du tableau. Cela est irréversible.", "boardDeletePopup-title": "Supprimer le tableau ?", "delete-board": "Supprimer le tableau", + "delete-duplicate-lists": "Supprimer les listes en doublon ? ", + "delete-duplicate-lists-confirm": "Êtes-vous sûr ? Cela supprimera toutes les listes en doublon qui ont le même nom et qui ne contiennent aucune carte.", "default-subtasks-board": "Sous-tâches du tableau __board__", "default": "Défaut", "defaultdefault": "Défaut", @@ -848,7 +850,7 @@ "r-uncheck": "Décocher", "r-item": "élément", "r-of-checklist": "de la check-list", - "r-send-email": "Envoyer un email", + "r-send-email": "Envoyer un courriel", "r-to": "à", "r-of": "sur", "r-subject": "sujet", @@ -857,7 +859,7 @@ "r-d-move-to-top-spec": "Déplacer la carte en haut de la liste", "r-d-move-to-bottom-gen": "Déplacer la carte en bas de sa liste", "r-d-move-to-bottom-spec": "Déplacer la carte en bas de la liste", - "r-d-send-email": "Envoyer un email", + "r-d-send-email": "Envoyer le courriel", "r-d-send-email-to": "à", "r-d-send-email-subject": "sujet", "r-d-send-email-message": "message", @@ -1016,8 +1018,8 @@ "dueCardsViewChange-choice-me": "Moi", "dueCardsViewChange-choice-all": "Tous les utilisateurs", "dueCardsViewChange-choice-all-description": "Visualise toutes les cartes incomplètes avec une date *échue* pour lesquelles l'utilisateur possède les droits", - "dueCards-noResults-title": "No Due Cards Found", - "dueCards-noResults-description": "You don't have any cards with due dates at the moment.", + "dueCards-noResults-title": "Aucune carte avec échéance trouvée", + "dueCards-noResults-description": "Vous n'avez aucune carte avec échéance en ce moment.", "broken-cards": "Cartes en erreur", "board-title-not-found": "Tableau '%s' non trouvé.", "swimlane-title-not-found": "Couloir '%s' non trouvé.", @@ -1323,8 +1325,8 @@ "admin-people-user-inactive": "L'utilisateur est désactivé - Cliquer pour l'activer", "accounts-lockout-all-users-unlocked": "Tous les utilisateurs bloqués ont été déverrouillés", "accounts-lockout-unlock-all": "Tout déverrouiller", - "active-cron-jobs": "Active Scheduled Jobs", - "add-cron-job": "Add Scheduled Job", + "active-cron-jobs": "Travaux actifs planifiés", + "add-cron-job": "Ajouter un travail planifié", "add-cron-job-placeholder": "Add Scheduled Job functionality coming soon", "attachment-storage-configuration": "Attachment Storage Configuration", "attachments-path": "Attachments Path", @@ -1338,8 +1340,8 @@ "board-cleanup-failed": "Failed to schedule board cleanup", "board-cleanup-scheduled": "Board cleanup scheduled successfully", "board-operations": "Board Operations", - "cron-jobs": "Scheduled Jobs", - "cron-migrations": "Scheduled Migrations", + "cron-jobs": "Travaux planifiés", + "cron-migrations": "Migrations planifiées", "cron-job-delete-confirm": "Are you sure you want to delete this scheduled job?", "cron-job-delete-failed": "Failed to delete scheduled job", "cron-job-deleted": "Scheduled job deleted successfully", @@ -1398,34 +1400,34 @@ "attachment-monitoring": "Attachment Monitoring", "attachment-settings": "Attachment Settings", "attachment-storage-settings": "Storage Settings", - "automatic-migration": "Automatic Migration", - "back-to-settings": "Back to Settings", - "board-id": "Board ID", + "automatic-migration": "Migration automatique", + "back-to-settings": "Retour aux paramètres", + "board-id": "ID du tableau", "board-migration": "Board Migration", - "card-show-lists-on-minicard": "Show Lists on Minicard", + "card-show-lists-on-minicard": "Afficher les listes sur la mini-carte", "cleanup": "Cleanup", "cleanup-old-jobs": "Cleanup Old Jobs", "completed": "Terminé", "conversion-info-text": "This conversion is performed once per board and improves performance. You can continue using the board normally.", "converting-board": "Converting Board", "converting-board-description": "Converting board structure for improved functionality. This may take a few moments.", - "cpu-cores": "CPU Cores", - "cpu-usage": "CPU Usage", + "cpu-cores": "Cœurs du CPU ", + "cpu-usage": "Utilisation du CPU", "current-action": "Current Action", - "database-migration": "Database Migration", + "database-migration": "Migration base de données", "database-migration-description": "Updating database structure for improved functionality and performance. This process may take several minutes.", - "database-migrations": "Database Migrations", - "days-old": "Days Old", - "duration": "Duration", - "errors": "Errors", - "estimated-time-remaining": "Estimated time remaining", - "every-1-day": "Every 1 day", - "every-1-hour": "Every 1 hour", - "every-1-minute": "Every 1 minute", - "every-10-minutes": "Every 10 minutes", - "every-30-minutes": "Every 30 minutes", - "every-5-minutes": "Every 5 minutes", - "every-6-hours": "Every 6 hours", + "database-migrations": "Migrations base de données", + "days-old": "Jours d'ancienneté", + "duration": "Durée", + "errors": "Erreurs", + "estimated-time-remaining": "Temps restant estimé", + "every-1-day": "Tous les jours", + "every-1-hour": "Toutes les heures", + "every-1-minute": "Toutes les minutes", + "every-10-minutes": "Toutes les 10 minutes", + "every-30-minutes": "Toutes les 30 minutes", + "every-5-minutes": "Toutes les 5 minutes", + "every-6-hours": "Toutes les 6 heures", "export-monitoring": "Export Monitoring", "filesystem-attachments": "Filesystem Attachments", "filesystem-size": "Filesystem Size", @@ -1434,22 +1436,22 @@ "gridfs-attachments": "GridFS Attachments", "gridfs-size": "GridFS Size", "gridfs-storage": "GridFS", - "hide-list-on-minicard": "Hide List on Minicard", + "hide-list-on-minicard": "Masquer la liste sur la mini-carte", "idle-migration": "Idle Migration", "job-description": "Job Description", "job-details": "Job Details", "job-name": "Job Name", "job-queue": "Job Queue", - "last-run": "Last Run", + "last-run": "Dernière exécution", "max-concurrent": "Max Concurrent", - "memory-usage": "Memory Usage", + "memory-usage": "Utilisation de la mémoire", "migrate-all-to-filesystem": "Migrate All to Filesystem", "migrate-all-to-gridfs": "Migrate All to GridFS", "migrate-all-to-s3": "Migrate All to S3", "migrated-attachments": "Migrated Attachments", "migration-batch-size": "Batch Size", "migration-batch-size-description": "Number of attachments to process in each batch (1-100)", - "migration-cpu-threshold": "CPU Threshold (%)", + "migration-cpu-threshold": "Seuil CPU (%)", "migration-cpu-threshold-description": "Pause migration when CPU usage exceeds this percentage (10-90)", "migration-delay-ms": "Delay (ms)", "migration-delay-ms-description": "Delay between batches in milliseconds (100-10000)", @@ -1463,26 +1465,26 @@ "migration-warning-text": "Please do not close your browser during migration. The process will continue in the background but may take longer to complete.", "monitoring-export-failed": "Failed to export monitoring data", "monitoring-refresh-failed": "Failed to refresh monitoring data", - "next": "Next", - "next-run": "Next Run", + "next": "Suivant", + "next-run": "Prochaine exécution", "of": "sur", - "operation-type": "Operation Type", + "operation-type": "Type d'opération", "overall-progress": "Overall Progress", "page": "Page", "pause-migration": "Pause Migration", - "previous": "Previous", + "previous": "Précédent", "refresh": "Refresh", "refresh-monitoring": "Refresh Monitoring", "remaining-attachments": "Remaining Attachments", "resume-migration": "Resume Migration", - "run-once": "Run once", - "s3-attachments": "S3 Attachments", - "s3-size": "S3 Size", + "run-once": "Exécuter une fois", + "s3-attachments": "Pièces jointes S3", + "s3-size": "Taille S3", "s3-storage": "S3", "scanning-status": "Scanning Status", "schedule": "Schedule", "search-boards-or-operations": "Search boards or operations...", - "show-list-on-minicard": "Show List on Minicard", + "show-list-on-minicard": "Afficher la liste sur la mini-carte", "showing": "Showing", "start-test-operation": "Start Test Operation", "start-time": "Start Time", @@ -1492,8 +1494,8 @@ "system-resources": "System Resources", "total-attachments": "Total Attachments", "total-operations": "Total Operations", - "total-size": "Total Size", - "unmigrated-boards": "Unmigrated Boards", + "total-size": "Taille totale", + "unmigrated-boards": "Tableaux non migrés", "weight": "Poids", "idle": "Inactif", "complete": "Terminé", diff --git a/imports/i18n/data/fy-NL.i18n.json b/imports/i18n/data/fy-NL.i18n.json index bcdd00061..eef27e1fe 100644 --- a/imports/i18n/data/fy-NL.i18n.json +++ b/imports/i18n/data/fy-NL.i18n.json @@ -755,6 +755,8 @@ "delete-board-confirm-popup": "All lists, cards, labels, and activities will be deleted and you won't be able to recover the board contents. There is no undo.", "boardDeletePopup-title": "Delete Board?", "delete-board": "Delete Board", + "delete-duplicate-lists": "Delete Duplicate Lists", + "delete-duplicate-lists-confirm": "Are you sure? This will delete all duplicate lists that have the same name and contain no cards.", "default-subtasks-board": "Subtasks for __board__ board", "default": "Default", "defaultdefault": "Default", diff --git a/imports/i18n/data/fy.i18n.json b/imports/i18n/data/fy.i18n.json index bcdd00061..eef27e1fe 100644 --- a/imports/i18n/data/fy.i18n.json +++ b/imports/i18n/data/fy.i18n.json @@ -755,6 +755,8 @@ "delete-board-confirm-popup": "All lists, cards, labels, and activities will be deleted and you won't be able to recover the board contents. There is no undo.", "boardDeletePopup-title": "Delete Board?", "delete-board": "Delete Board", + "delete-duplicate-lists": "Delete Duplicate Lists", + "delete-duplicate-lists-confirm": "Are you sure? This will delete all duplicate lists that have the same name and contain no cards.", "default-subtasks-board": "Subtasks for __board__ board", "default": "Default", "defaultdefault": "Default", diff --git a/imports/i18n/data/gl-ES.i18n.json b/imports/i18n/data/gl-ES.i18n.json index 6e381a12a..8c523be4e 100644 --- a/imports/i18n/data/gl-ES.i18n.json +++ b/imports/i18n/data/gl-ES.i18n.json @@ -755,6 +755,8 @@ "delete-board-confirm-popup": "All lists, cards, labels, and activities will be deleted and you won't be able to recover the board contents. There is no undo.", "boardDeletePopup-title": "Delete Board?", "delete-board": "Delete Board", + "delete-duplicate-lists": "Delete Duplicate Lists", + "delete-duplicate-lists-confirm": "Are you sure? This will delete all duplicate lists that have the same name and contain no cards.", "default-subtasks-board": "Subtasks for __board__ board", "default": "Default", "defaultdefault": "Default", diff --git a/imports/i18n/data/gl.i18n.json b/imports/i18n/data/gl.i18n.json index 9c24ed023..9b9438f28 100644 --- a/imports/i18n/data/gl.i18n.json +++ b/imports/i18n/data/gl.i18n.json @@ -755,6 +755,8 @@ "delete-board-confirm-popup": "All lists, cards, labels, and activities will be deleted and you won't be able to recover the board contents. There is no undo.", "boardDeletePopup-title": "Delete Board?", "delete-board": "Delete Board", + "delete-duplicate-lists": "Delete Duplicate Lists", + "delete-duplicate-lists-confirm": "Are you sure? This will delete all duplicate lists that have the same name and contain no cards.", "default-subtasks-board": "Subtasks for __board__ board", "default": "Default", "defaultdefault": "Default", diff --git a/imports/i18n/data/gu-IN.i18n.json b/imports/i18n/data/gu-IN.i18n.json index 510b05b6b..1ab8eeaa8 100644 --- a/imports/i18n/data/gu-IN.i18n.json +++ b/imports/i18n/data/gu-IN.i18n.json @@ -755,6 +755,8 @@ "delete-board-confirm-popup": "All lists, cards, labels, and activities will be deleted and you won't be able to recover the board contents. There is no undo.", "boardDeletePopup-title": "Delete Board?", "delete-board": "Delete Board", + "delete-duplicate-lists": "Delete Duplicate Lists", + "delete-duplicate-lists-confirm": "Are you sure? This will delete all duplicate lists that have the same name and contain no cards.", "default-subtasks-board": "Subtasks for __board__ board", "default": "Default", "defaultdefault": "Default", diff --git a/imports/i18n/data/he-IL.i18n.json b/imports/i18n/data/he-IL.i18n.json index d8cd290c4..b34277bac 100644 --- a/imports/i18n/data/he-IL.i18n.json +++ b/imports/i18n/data/he-IL.i18n.json @@ -755,6 +755,8 @@ "delete-board-confirm-popup": "All lists, cards, labels, and activities will be deleted and you won't be able to recover the board contents. There is no undo.", "boardDeletePopup-title": "Delete Board?", "delete-board": "Delete Board", + "delete-duplicate-lists": "Delete Duplicate Lists", + "delete-duplicate-lists-confirm": "Are you sure? This will delete all duplicate lists that have the same name and contain no cards.", "default-subtasks-board": "Subtasks for __board__ board", "default": "Default", "defaultdefault": "Default", diff --git a/imports/i18n/data/he.i18n.json b/imports/i18n/data/he.i18n.json index ad49be122..2b6d8f5ef 100644 --- a/imports/i18n/data/he.i18n.json +++ b/imports/i18n/data/he.i18n.json @@ -755,6 +755,8 @@ "delete-board-confirm-popup": "כל הרשימות, הכרטיסים, התווית והפעולות יימחקו ולא תהיה לך דרך לשחזר את תכני הלוח. אין אפשרות לבטל.", "boardDeletePopup-title": "למחוק את הלוח?", "delete-board": "מחיקת לוח", + "delete-duplicate-lists": "Delete Duplicate Lists", + "delete-duplicate-lists-confirm": "Are you sure? This will delete all duplicate lists that have the same name and contain no cards.", "default-subtasks-board": "תת־משימות עבור הלוח __board__", "default": "בררת מחדל", "defaultdefault": "בררת מחדל", diff --git a/imports/i18n/data/hi-IN.i18n.json b/imports/i18n/data/hi-IN.i18n.json index fc15c79d7..e65f94bf7 100644 --- a/imports/i18n/data/hi-IN.i18n.json +++ b/imports/i18n/data/hi-IN.i18n.json @@ -755,6 +755,8 @@ "delete-board-confirm-popup": "All lists, कार्ड,नामपत्र , और activities हो जाएगा deleted और you won't be able तक recover the बोर्ड contents. There is no undo.", "boardDeletePopup-title": "मिटाएँ बोर्ड?", "delete-board": "मिटाएँ बोर्ड", + "delete-duplicate-lists": "Delete Duplicate Lists", + "delete-duplicate-lists-confirm": "Are you sure? This will delete all duplicate lists that have the same name and contain no cards.", "default-subtasks-board": "Subtasks for __board__ बोर्ड", "default": "Default", "defaultdefault": "Default", diff --git a/imports/i18n/data/hi.i18n.json b/imports/i18n/data/hi.i18n.json index 34d873c70..8febb7ce7 100644 --- a/imports/i18n/data/hi.i18n.json +++ b/imports/i18n/data/hi.i18n.json @@ -755,6 +755,8 @@ "delete-board-confirm-popup": "All lists, कार्ड,नामपत्र , और activities हो जाएगा deleted और you won't be able तक recover the बोर्ड contents. There is no undo.", "boardDeletePopup-title": "मिटाएँ बोर्ड?", "delete-board": "मिटाएँ बोर्ड", + "delete-duplicate-lists": "Delete Duplicate Lists", + "delete-duplicate-lists-confirm": "Are you sure? This will delete all duplicate lists that have the same name and contain no cards.", "default-subtasks-board": "Subtasks for __board__ बोर्ड", "default": "Default", "defaultdefault": "Default", diff --git a/imports/i18n/data/hr.i18n.json b/imports/i18n/data/hr.i18n.json index e50f9b494..5b66d06dd 100644 --- a/imports/i18n/data/hr.i18n.json +++ b/imports/i18n/data/hr.i18n.json @@ -755,6 +755,8 @@ "delete-board-confirm-popup": "All lists, cards, labels, and activities will be deleted and you won't be able to recover the board contents. There is no undo.", "boardDeletePopup-title": "Obrisati ploču?", "delete-board": "Obriši ploču", + "delete-duplicate-lists": "Delete Duplicate Lists", + "delete-duplicate-lists-confirm": "Are you sure? This will delete all duplicate lists that have the same name and contain no cards.", "default-subtasks-board": "Subtasks for __board__ board", "default": "Zadano", "defaultdefault": "Zadano", diff --git a/imports/i18n/data/hu.i18n.json b/imports/i18n/data/hu.i18n.json index e0dc13889..82d25500a 100644 --- a/imports/i18n/data/hu.i18n.json +++ b/imports/i18n/data/hu.i18n.json @@ -755,6 +755,8 @@ "delete-board-confirm-popup": "Minden Lista, Kártya, Címke és Esemény véglegesen törlésre kerül és nincs rá mód, hogy visszanyerd a Tábla tartalmát. Nincs visszavonási lehetőség sem.", "boardDeletePopup-title": "TÖRLÖD a Táblát?", "delete-board": "Tábla törlése", + "delete-duplicate-lists": "Delete Duplicate Lists", + "delete-duplicate-lists-confirm": "Are you sure? This will delete all duplicate lists that have the same name and contain no cards.", "default-subtasks-board": "Rész-feladatok ehhez a Táblához: __board__", "default": "Alapértelmezett", "defaultdefault": "Alapértelmezett", diff --git a/imports/i18n/data/hy.i18n.json b/imports/i18n/data/hy.i18n.json index 5f4d671f2..78f216ac6 100644 --- a/imports/i18n/data/hy.i18n.json +++ b/imports/i18n/data/hy.i18n.json @@ -755,6 +755,8 @@ "delete-board-confirm-popup": "All lists, cards, labels, and activities will be deleted and you won't be able to recover the board contents. There is no undo.", "boardDeletePopup-title": "Delete Board?", "delete-board": "Delete Board", + "delete-duplicate-lists": "Delete Duplicate Lists", + "delete-duplicate-lists-confirm": "Are you sure? This will delete all duplicate lists that have the same name and contain no cards.", "default-subtasks-board": "Subtasks for __board__ board", "default": "Default", "defaultdefault": "Default", diff --git a/imports/i18n/data/id.i18n.json b/imports/i18n/data/id.i18n.json index 7ee0e51d8..e5646003c 100644 --- a/imports/i18n/data/id.i18n.json +++ b/imports/i18n/data/id.i18n.json @@ -755,6 +755,8 @@ "delete-board-confirm-popup": "All lists, cards, labels, and activities will be deleted and you won't be able to recover the board contents. There is no undo.", "boardDeletePopup-title": "Hapus Papan?", "delete-board": "Hapus Papan", + "delete-duplicate-lists": "Delete Duplicate Lists", + "delete-duplicate-lists-confirm": "Are you sure? This will delete all duplicate lists that have the same name and contain no cards.", "default-subtasks-board": "Subtasks for __board__ board", "default": "Standar", "defaultdefault": "Standar", diff --git a/imports/i18n/data/ig.i18n.json b/imports/i18n/data/ig.i18n.json index 09e6ad143..38e4d0620 100644 --- a/imports/i18n/data/ig.i18n.json +++ b/imports/i18n/data/ig.i18n.json @@ -755,6 +755,8 @@ "delete-board-confirm-popup": "All lists, cards, labels, and activities will be deleted and you won't be able to recover the board contents. There is no undo.", "boardDeletePopup-title": "Delete Board?", "delete-board": "Delete Board", + "delete-duplicate-lists": "Delete Duplicate Lists", + "delete-duplicate-lists-confirm": "Are you sure? This will delete all duplicate lists that have the same name and contain no cards.", "default-subtasks-board": "Subtasks for __board__ board", "default": "Default", "defaultdefault": "Default", diff --git a/imports/i18n/data/it.i18n.json b/imports/i18n/data/it.i18n.json index 00abbe4c5..a6046dd35 100644 --- a/imports/i18n/data/it.i18n.json +++ b/imports/i18n/data/it.i18n.json @@ -755,6 +755,8 @@ "delete-board-confirm-popup": "Tutte le liste, schede, etichette e azioni saranno rimosse e non sarai più in grado di recuperare il contenuto della bacheca. L'azione non è annullabile.", "boardDeletePopup-title": "Eliminare la bacheca?", "delete-board": "Elimina bacheca", + "delete-duplicate-lists": "Delete Duplicate Lists", + "delete-duplicate-lists-confirm": "Are you sure? This will delete all duplicate lists that have the same name and contain no cards.", "default-subtasks-board": "Sottocompiti per la bacheca __board__", "default": "Predefinito", "defaultdefault": "Predefinito", diff --git a/imports/i18n/data/ja-HI.i18n.json b/imports/i18n/data/ja-HI.i18n.json index 1f3b6d7b1..db428e93a 100644 --- a/imports/i18n/data/ja-HI.i18n.json +++ b/imports/i18n/data/ja-HI.i18n.json @@ -755,6 +755,8 @@ "delete-board-confirm-popup": "All lists, cards, labels, and activities will be deleted and you won't be able to recover the board contents. There is no undo.", "boardDeletePopup-title": "Delete Board?", "delete-board": "Delete Board", + "delete-duplicate-lists": "Delete Duplicate Lists", + "delete-duplicate-lists-confirm": "Are you sure? This will delete all duplicate lists that have the same name and contain no cards.", "default-subtasks-board": "Subtasks for __board__ board", "default": "Default", "defaultdefault": "Default", diff --git a/imports/i18n/data/ja.i18n.json b/imports/i18n/data/ja.i18n.json index d69fc4d27..3f6a29f13 100644 --- a/imports/i18n/data/ja.i18n.json +++ b/imports/i18n/data/ja.i18n.json @@ -755,6 +755,8 @@ "delete-board-confirm-popup": "すべてのリスト、カード、ラベル、アクティビティは削除され、ボードの内容を元に戻すことができません。", "boardDeletePopup-title": "ボードを削除しますか?", "delete-board": "ボードを削除", + "delete-duplicate-lists": "Delete Duplicate Lists", + "delete-duplicate-lists-confirm": "Are you sure? This will delete all duplicate lists that have the same name and contain no cards.", "default-subtasks-board": "__board__ ボードのサブタスク", "default": "デフォルト", "defaultdefault": "デフォルト", diff --git a/imports/i18n/data/ka.i18n.json b/imports/i18n/data/ka.i18n.json index 001b2ff8c..0e41f2671 100644 --- a/imports/i18n/data/ka.i18n.json +++ b/imports/i18n/data/ka.i18n.json @@ -755,6 +755,8 @@ "delete-board-confirm-popup": "ყველა ჩამონათვალი, ბარათი, ნიშანი და აქტივობა წაიშლება და თქვენ ვეღარ შეძლებთ მის აღდგენას.", "boardDeletePopup-title": "წავშალოთ დაფა?", "delete-board": "დაფის წაშლა", + "delete-duplicate-lists": "Delete Duplicate Lists", + "delete-duplicate-lists-confirm": "Are you sure? This will delete all duplicate lists that have the same name and contain no cards.", "default-subtasks-board": "ქვესაქმიანობა __board__ დაფისთვის", "default": "Default", "defaultdefault": "Default", diff --git a/imports/i18n/data/km.i18n.json b/imports/i18n/data/km.i18n.json index 04a544ca7..99d757f8b 100644 --- a/imports/i18n/data/km.i18n.json +++ b/imports/i18n/data/km.i18n.json @@ -755,6 +755,8 @@ "delete-board-confirm-popup": "All lists, cards, labels, and activities will be deleted and you won't be able to recover the board contents. There is no undo.", "boardDeletePopup-title": "Delete Board?", "delete-board": "Delete Board", + "delete-duplicate-lists": "Delete Duplicate Lists", + "delete-duplicate-lists-confirm": "Are you sure? This will delete all duplicate lists that have the same name and contain no cards.", "default-subtasks-board": "Subtasks for __board__ board", "default": "Default", "defaultdefault": "Default", diff --git a/imports/i18n/data/ko-KR.i18n.json b/imports/i18n/data/ko-KR.i18n.json index c447ad681..284d8e20b 100644 --- a/imports/i18n/data/ko-KR.i18n.json +++ b/imports/i18n/data/ko-KR.i18n.json @@ -755,6 +755,8 @@ "delete-board-confirm-popup": "All lists, cards, labels, and activities will be deleted and you won't be able to recover the board contents. There is no undo.", "boardDeletePopup-title": "Delete Board?", "delete-board": "Delete Board", + "delete-duplicate-lists": "Delete Duplicate Lists", + "delete-duplicate-lists-confirm": "Are you sure? This will delete all duplicate lists that have the same name and contain no cards.", "default-subtasks-board": "Subtasks for __board__ board", "default": "Default", "defaultdefault": "Default", diff --git a/imports/i18n/data/ko.i18n.json b/imports/i18n/data/ko.i18n.json index c423e1b7a..62f508b82 100644 --- a/imports/i18n/data/ko.i18n.json +++ b/imports/i18n/data/ko.i18n.json @@ -755,6 +755,8 @@ "delete-board-confirm-popup": "모든 목록, 카드, 레이블 및 활동이 삭제되고 보드 내용을 복구할 수 없습니다. 실행 취소는 불가능합니다.", "boardDeletePopup-title": "보드 삭제?", "delete-board": "보드 삭제", + "delete-duplicate-lists": "Delete Duplicate Lists", + "delete-duplicate-lists-confirm": "Are you sure? This will delete all duplicate lists that have the same name and contain no cards.", "default-subtasks-board": "Subtasks for __board__ board", "default": "기본", "defaultdefault": "기본", diff --git a/imports/i18n/data/lt.i18n.json b/imports/i18n/data/lt.i18n.json index 510b05b6b..1ab8eeaa8 100644 --- a/imports/i18n/data/lt.i18n.json +++ b/imports/i18n/data/lt.i18n.json @@ -755,6 +755,8 @@ "delete-board-confirm-popup": "All lists, cards, labels, and activities will be deleted and you won't be able to recover the board contents. There is no undo.", "boardDeletePopup-title": "Delete Board?", "delete-board": "Delete Board", + "delete-duplicate-lists": "Delete Duplicate Lists", + "delete-duplicate-lists-confirm": "Are you sure? This will delete all duplicate lists that have the same name and contain no cards.", "default-subtasks-board": "Subtasks for __board__ board", "default": "Default", "defaultdefault": "Default", diff --git a/imports/i18n/data/lv.i18n.json b/imports/i18n/data/lv.i18n.json index deaddc65b..d051fb8a5 100644 --- a/imports/i18n/data/lv.i18n.json +++ b/imports/i18n/data/lv.i18n.json @@ -755,6 +755,8 @@ "delete-board-confirm-popup": "Visi saraksti, kartiņas, birkas un darbības tiks dzēstas un dēli nevarēs atgūt. Darbība nav atsaucama.", "boardDeletePopup-title": "Dzēst dēli?", "delete-board": "Dzēst dēli", + "delete-duplicate-lists": "Delete Duplicate Lists", + "delete-duplicate-lists-confirm": "Are you sure? This will delete all duplicate lists that have the same name and contain no cards.", "default-subtasks-board": "Apakšuzdevumi priekš __board__ dēļa", "default": "Noklusēts", "defaultdefault": "Noklusēts", diff --git a/imports/i18n/data/mk.i18n.json b/imports/i18n/data/mk.i18n.json index 14eb56f73..491feeb6e 100644 --- a/imports/i18n/data/mk.i18n.json +++ b/imports/i18n/data/mk.i18n.json @@ -755,6 +755,8 @@ "delete-board-confirm-popup": "All lists, cards, labels, and activities will be deleted and you won't be able to recover the board contents. There is no undo.", "boardDeletePopup-title": "Изтриване на Таблото?", "delete-board": "Изтрий таблото", + "delete-duplicate-lists": "Delete Duplicate Lists", + "delete-duplicate-lists-confirm": "Are you sure? This will delete all duplicate lists that have the same name and contain no cards.", "default-subtasks-board": "Подзадачи за табло __board__", "default": "по подразбиране", "defaultdefault": "по подразбиране", diff --git a/imports/i18n/data/mn.i18n.json b/imports/i18n/data/mn.i18n.json index 7ef8ae586..de4627964 100644 --- a/imports/i18n/data/mn.i18n.json +++ b/imports/i18n/data/mn.i18n.json @@ -755,6 +755,8 @@ "delete-board-confirm-popup": "All lists, cards, labels, and activities will be deleted and you won't be able to recover the board contents. There is no undo.", "boardDeletePopup-title": "Delete Board?", "delete-board": "Delete Board", + "delete-duplicate-lists": "Delete Duplicate Lists", + "delete-duplicate-lists-confirm": "Are you sure? This will delete all duplicate lists that have the same name and contain no cards.", "default-subtasks-board": "Subtasks for __board__ board", "default": "Default", "defaultdefault": "Default", diff --git a/imports/i18n/data/ms-MY.i18n.json b/imports/i18n/data/ms-MY.i18n.json index 2112ab947..e1ea29b8a 100644 --- a/imports/i18n/data/ms-MY.i18n.json +++ b/imports/i18n/data/ms-MY.i18n.json @@ -92,10 +92,10 @@ "list-width-error-message": "Lebar senarai mestilah integer lebih besar dari 100", "keyboard-shortcuts-enabled": "Pintasan papan kekunci didayakan. Klik untuk batal.", "keyboard-shortcuts-disabled": "Pintasan papan kekunci didayakan. Klik untuk batal.", - "setSwimlaneHeightPopup-title": "Set Swimlane Height", - "set-swimlane-height": "Set Swimlane Height", - "set-swimlane-height-value": "Swimlane Height (pixels)", - "swimlane-height-error-message": "Swimlane height must be a positive integer", + "setSwimlaneHeightPopup-title": "Tetapkan ketinggian Swimlane", + "set-swimlane-height": "Tetapkan ketinggian Swimlane", + "set-swimlane-height-value": "Ketinggian Swimlane (piksel)", + "swimlane-height-error-message": "Ketinggian Swimlane mestilah integer positif", "add-swimlane": "Add Swimlane", "add-subtask": "Add Subtask", "add-checklist": "Add Checklist", @@ -103,10 +103,10 @@ "close-add-checklist-item": "Close add an item to checklist form", "close-edit-checklist-item": "Close edit an item to checklist form", "convertChecklistItemToCardPopup-title": "Convert to Card", - "add-cover": "Add cover image to minicard", + "add-cover": "Tambah imej depan pada kad mini", "add-label": "Add Label", "add-list": "Tambah Senarai", - "add-after-list": "Add After List", + "add-after-list": "Tambah selepas senarai", "add-members": "Tambah Ahli", "added": "Ditambah", "addMemberPopup-title": "Ahli-ahli", @@ -125,7 +125,7 @@ "archive": "Move to Archive", "archive-all": "Move All to Archive", "archive-board": "Move Board to Archive", - "archive-board-confirm": "Are you sure you want to archive this board?", + "archive-board-confirm": "Anda pasti untuk mengarkibkan papan ini?", "archive-card": "Move Card to Archive", "archive-list": "Move List to Archive", "archive-swimlane": "Move Swimlane to Archive", @@ -147,13 +147,13 @@ "attachmentDeletePopup-title": "Delete Attachment?", "attachments": "Attachments", "auto-watch": "Automatically watch boards when they are created", - "avatar-too-big": "The avatar is too large (__size__ max)", + "avatar-too-big": "Saiz Avatar terlalu besar (__size__max)", "back": "Back", "board-change-color": "Change color", - "board-change-background-image": "Change Background Image", - "board-background-image-url": "Background Image URL", - "add-background-image": "Add Background Image", - "remove-background-image": "Remove Background Image", + "board-change-background-image": "Ubah imej latar", + "board-background-image-url": "URL imej latar", + "add-background-image": "Tambah imej latar", + "remove-background-image": "Hapus imej latar", "show-at-all-boards-page" : "Show at All Boards page", "board-info-on-my-boards" : "All Boards Settings", "boardInfoOnMyBoardsPopup-title" : "All Boards Settings", @@ -166,9 +166,9 @@ "board-public-info": "This board will be public.", "board-drag-drop-reorder-or-click-open": "Drag and drop to reorder board icons. Click board icon to open board.", "boardChangeColorPopup-title": "Change Board Background", - "boardChangeBackgroundImagePopup-title": "Change Background Image", + "boardChangeBackgroundImagePopup-title": "Ubah imej latar", "allBoardsChangeColorPopup-title": "Change color", - "allBoardsChangeBackgroundImagePopup-title": "Change Background Image", + "allBoardsChangeBackgroundImagePopup-title": "Ubah imej latar", "boardChangeTitlePopup-title": "Rename Board", "boardChangeVisibilityPopup-title": "Change Visibility", "boardChangeWatchPopup-title": "Change Watch", @@ -177,22 +177,22 @@ "boardChangeViewPopup-title": "Board View", "boards": "Boards", "board-view": "Board View", - "desktop-mode": "Desktop Mode", - "mobile-mode": "Mobile Mode", - "mobile-desktop-toggle": "Toggle between Mobile and Desktop Mode", - "zoom-in": "Zoom In", - "zoom-out": "Zoom Out", - "click-to-change-zoom": "Click to change zoom level", - "zoom-level": "Zoom Level", - "enter-zoom-level": "Enter zoom level (50-300%):", + "desktop-mode": "Mod Desktop", + "mobile-mode": "Mod Mudah Alih", + "mobile-desktop-toggle": "Tukar antara mod mudah alih dan desktop", + "zoom-in": "Zum masuk", + "zoom-out": "Zum keluar", + "click-to-change-zoom": "Klik untuk ubah aras zum", + "zoom-level": "Aras zum", + "enter-zoom-level": "Masukkan aras zum (50-300%)", "board-view-cal": "Calendar", "board-view-swimlanes": "Swimlanes", "board-view-collapse": "Collapse", "board-view-gantt": "Gantt", "board-view-lists": "Lists", - "bucket-example": "Like \"Bucket List\" for example", - "calendar-previous-month-label": "Previous Month", - "calendar-next-month-label": "Next Month", + "bucket-example": "Contohnya \"Bucket List\"", + "calendar-previous-month-label": "Bulan Sebelum", + "calendar-next-month-label": "Bulan Depan", "cancel": "Cancel", "card-archived": "This card is moved to Archive.", "board-archived": "This board is moved to Archive.", @@ -277,9 +277,9 @@ "checklists": "Checklists", "click-to-star": "Click to star this board.", "click-to-unstar": "Click to unstar this board.", - "click-to-enable-auto-width": "Auto list width disabled. Click to enable.", - "click-to-disable-auto-width": "Auto list width enabled. Click to disable.", - "auto-list-width": "Auto list width", + "click-to-enable-auto-width": "Lebar senarai lalai tidak didayakan. Klik untuk dayakan.", + "click-to-disable-auto-width": "Lebar senarai lalai didayakan. Klik untuk tidak didayakan.", + "auto-list-width": "Lebar senarai lalai.", "clipboard": "Clipboard or drag & drop", "close": "Close", "close-board": "Close Board", @@ -311,7 +311,7 @@ "color-white": "white", "color-yellow": "yellow", "unset-color": "Unset", - "comments": "Comments", + "comments": "Komen", "comment": "Comment", "comment-placeholder": "Write Comment", "comment-only": "Comment only", @@ -356,10 +356,10 @@ "custom-field-text": "Text", "custom-fields": "Custom Fields", "date": "Date", - "date-format": "Date Format", - "date-format-yyyy-mm-dd": "YYYY-MM-DD", - "date-format-dd-mm-yyyy": "DD-MM-YYYY", - "date-format-mm-dd-yyyy": "MM-DD-YYYY", + "date-format": "Format Tarikh", + "date-format-yyyy-mm-dd": "TTTT-BB-HH", + "date-format-dd-mm-yyyy": "HH-BB-TTTT", + "date-format-mm-dd-yyyy": "BB-HH-TTTT", "decline": "Decline", "default-avatar": "Default avatar", "delete": "Delete", @@ -385,7 +385,7 @@ "editNotificationPopup-title": "Edit Notification", "editProfilePopup-title": "Edit Profile", "email": "Email", - "email-address": "Email Address", + "email-address": "Alamat Email", "email-enrollAccount-subject": "An account created for you on __siteName__", "email-enrollAccount-text": "Hello __user__,\n\nTo start using the service, simply click the link below.\n\n__url__\n\nThanks.", "email-fail": "Sending email failed", @@ -399,14 +399,14 @@ "email-sent": "Email sent", "email-verifyEmail-subject": "Verify your email address on __siteName__", "email-verifyEmail-text": "Hello __user__,\n\nTo verify your account email, simply click the link below.\n\n__url__\n\nThanks.", - "enable-vertical-scrollbars": "Enable vertical scrollbars", + "enable-vertical-scrollbars": "Dayakan bar skrol melintang", "enable-wip-limit": "Enable WIP Limit", "error-board-doesNotExist": "This board does not exist", "error-board-notAdmin": "You need to be admin of this board to do that", "error-board-notAMember": "You need to be a member of this board to do that", "error-json-malformed": "Your text is not valid JSON", "error-json-schema": "Your JSON data does not include the proper information in the correct format", - "error-csv-schema": "Your CSV(Comma Separated Values)/TSV (Tab Separated Values) does not include the proper information in the correct format ", + "error-csv-schema": "CSV(Comma Separated Values)/TSV (Tab Separated Values) anda tidak mengikut format yang betul dengan maklumat yang bersesuaian.", "error-list-doesNotExist": "This list does not exist", "error-user-doesNotExist": "This user does not exist", "error-user-notAllowSelf": "You can not invite yourself", @@ -445,7 +445,7 @@ "filter-overdue": "Overdue", "filter-due-today": "Due today", "filter-due-this-week": "Due this week", - "filter-due-next-week": "Due next week", + "filter-due-next-week": "Tamat minggu depan", "filter-due-tomorrow": "Due tomorrow", "list-filter-label": "Filter List by Title", "filter-clear": "Clear filter", @@ -574,7 +574,7 @@ "public": "Public", "public-desc": "This board is public. It's visible to anyone with the link and will show up in search engines like Google. Only people added to the board can edit.", "quick-access-description": "Star a board to add a shortcut in this bar.", - "remove-cover": "Remove cover image from minicard", + "remove-cover": "Hapus imej depan daripada kad mini", "remove-from-board": "Remove from Board", "remove-label": "Remove Label", "listDeletePopup-title": "Delete List ?", @@ -596,14 +596,14 @@ "select-board": "Select Board", "set-wip-limit-value": "Set a limit for the maximum number of tasks in this list", "setWipLimitPopup-title": "Set WIP Limit", - "shortcut-add-self": "Add yourself to current card", + "shortcut-add-self": "Tambah diri sendiri kepada kad ini", "shortcut-assign-self": "Assign yourself to current card", "shortcut-autocomplete-emoji": "Autocomplete emoji", "shortcut-autocomplete-members": "Autocomplete members", "shortcut-clear-filters": "Clear all filters", "shortcut-close-dialog": "Close Dialog", "shortcut-filter-my-cards": "Filter my cards", - "shortcut-filter-my-assigned-cards": "Filter my assigned cards", + "shortcut-filter-my-assigned-cards": "Tapis kad sendiri", "shortcut-show-shortcuts": "Bring up this shortcuts list", "shortcut-toggle-filterbar": "Toggle Filter Sidebar", "shortcut-toggle-searchbar": "Toggle Search Sidebar", @@ -638,9 +638,9 @@ "upload": "Upload", "upload-avatar": "Upload an avatar", "uploaded-avatar": "Uploaded an avatar", - "uploading-files": "Uploading files", - "upload-failed": "Upload failed", - "upload-completed": "Upload completed", + "uploading-files": "Muat naik fail", + "upload-failed": "Muat naik gagal", + "upload-completed": "Muat naik selesai", "custom-top-left-corner-logo-image-url": "Custom Top Left Corner Logo Image URL", "custom-top-left-corner-logo-link-url": "Custom Top Left Corner Logo Link URL", "custom-top-left-corner-logo-height": "Custom Top Left Corner Logo Height. Default: 27", @@ -755,6 +755,8 @@ "delete-board-confirm-popup": "All lists, cards, labels, and activities will be deleted and you won't be able to recover the board contents. There is no undo.", "boardDeletePopup-title": "Delete Board?", "delete-board": "Delete Board", + "delete-duplicate-lists": "Delete Duplicate Lists", + "delete-duplicate-lists-confirm": "Are you sure? This will delete all duplicate lists that have the same name and contain no cards.", "default-subtasks-board": "Subtasks for __board__ board", "default": "Default", "defaultdefault": "Default", @@ -769,9 +771,9 @@ "deposit-subtasks-list": "Landing list for subtasks deposited here:", "show-parent-in-minicard": "Show parent in minicard:", "description-on-minicard": "Description on minicard", - "cover-attachment-on-minicard": "Cover image on minicard", - "badge-attachment-on-minicard": "Count of attachments on minicard", - "card-sorting-by-number-on-minicard": "Card sorting by number on minicard", + "cover-attachment-on-minicard": "Imej depan kad mini", + "badge-attachment-on-minicard": "Bilangan lampiran pada kad mini", + "card-sorting-by-number-on-minicard": "Kad disusun mengikut nombor pada kad mini", "prefix-with-full-path": "Prefix with full path", "prefix-with-parent": "Prefix with parent", "subtext-with-full-path": "Subtext with full path", @@ -884,7 +886,7 @@ "r-items-list": "item1,item2,item3", "r-add-swimlane": "Add swimlane", "r-swimlane-name": "swimlane name", - "r-board-note": "Note: leave a field empty to match every possible value. ", + "r-board-note": "Nota: Tinggalkan ruang kosong untuk padankan setiap nilai yang mungkin.", "r-checklist-note": "Note: checklist's items have to be written as comma separated values.", "r-when-a-card-is-moved": "When a card is moved to another list", "r-set": "Set", @@ -915,10 +917,10 @@ "oidc-button-text": "Customize the OIDC button text", "default-authentication-method": "Default Authentication Method", "duplicate-board": "Duplicate Board", - "duplicate-board-confirm": "Are you sure you want to duplicate this board?", - "org-number": "The number of organizations is: ", - "team-number": "The number of teams is: ", - "people-number": "The number of people is: ", + "duplicate-board-confirm": "Anda pasti untuk menduplikasi papan ini?", + "org-number": "Jumlah Organisasi ialah:", + "team-number": "Jumlah kumpulan ialah:", + "people-number": "Jumlah individu ialah:", "swimlaneDeletePopup-title": "Delete Swimlane ?", "swimlane-delete-pop": "All actions will be removed from the activity feed and you won't be able to recover the swimlane. There is no undo.", "restore-all": "Restore all", @@ -986,7 +988,7 @@ "delete-linked-card-before-this-card": "You can not delete this card before first deleting linked card that has", "delete-linked-cards-before-this-list": "You can not delete this list before first deleting linked cards that are pointing to cards in this list", "hide-checked-items": "Hide checked items", - "hide-finished-checklist": "Hide finished checklist", + "hide-finished-checklist": "Sembunyikan senarai semak yang telah selesai", "task": "Task", "create-task": "Create Task", "ok": "OK", @@ -994,7 +996,7 @@ "teams": "Teams", "displayName": "Nama Paparan", "shortName": "Nama Ringkas", - "autoAddUsersWithDomainName": "Automatically add users with the domain name", + "autoAddUsersWithDomainName": "Tambah pengguna dengan nama domain ini secara lalai", "website": "Laman Sesawang", "person": "Person", "my-cards": "Kad Saya", @@ -1016,7 +1018,7 @@ "dueCardsViewChange-choice-me": "Saya", "dueCardsViewChange-choice-all": "Semua Pengguna", "dueCardsViewChange-choice-all-description": "Shows all incomplete cards with a *Due* date from boards for which the user has permission.", - "dueCards-noResults-title": "No Due Cards Found", + "dueCards-noResults-title": "Tiada Kad Tamat Ditemui", "dueCards-noResults-description": "You don't have any cards with due dates at the moment.", "broken-cards": "Broken Cards", "board-title-not-found": "Board '%s' not found.", diff --git a/imports/i18n/data/ms.i18n.json b/imports/i18n/data/ms.i18n.json index 4f030d480..ad26e0f97 100644 --- a/imports/i18n/data/ms.i18n.json +++ b/imports/i18n/data/ms.i18n.json @@ -755,6 +755,8 @@ "delete-board-confirm-popup": "Semua senarai, kad, label dan aktiviti akan dihapus dan anda tidak akan dapat pulihkan semula kandungan papan. Tiada undur semula", "boardDeletePopup-title": "Hapus Papan?", "delete-board": "Hapus papan", + "delete-duplicate-lists": "Delete Duplicate Lists", + "delete-duplicate-lists-confirm": "Are you sure? This will delete all duplicate lists that have the same name and contain no cards.", "default-subtasks-board": "subtugas untuk __board__ papan", "default": "Lalai", "defaultdefault": "Lalai", diff --git a/imports/i18n/data/nb.i18n.json b/imports/i18n/data/nb.i18n.json index 58958c9f4..f3fb0f0ed 100644 --- a/imports/i18n/data/nb.i18n.json +++ b/imports/i18n/data/nb.i18n.json @@ -755,6 +755,8 @@ "delete-board-confirm-popup": "Alle lister, kort, etiketter og aktiviteter vil bli slettet og du vil ikke kunne gjenopprette innholdet på tavlen. Det er ikke mulig å angre.", "boardDeletePopup-title": "Slett Tavle?", "delete-board": "Slett Tavle", + "delete-duplicate-lists": "Delete Duplicate Lists", + "delete-duplicate-lists-confirm": "Are you sure? This will delete all duplicate lists that have the same name and contain no cards.", "default-subtasks-board": "Underoppgave for __board__ tavle", "default": "Standard", "defaultdefault": "Standard", diff --git a/imports/i18n/data/nl-NL.i18n.json b/imports/i18n/data/nl-NL.i18n.json index a5d645f17..e1e932403 100644 --- a/imports/i18n/data/nl-NL.i18n.json +++ b/imports/i18n/data/nl-NL.i18n.json @@ -755,6 +755,8 @@ "delete-board-confirm-popup": "Alle lijsten, kaarten, labels en activiteiten zullen worden verwijderd en je kunt de bordinhoud niet terughalen. Er is geen herstelmogelijkheid.", "boardDeletePopup-title": "Bord verwijderen?", "delete-board": "Verwijder bord", + "delete-duplicate-lists": "Delete Duplicate Lists", + "delete-duplicate-lists-confirm": "Are you sure? This will delete all duplicate lists that have the same name and contain no cards.", "default-subtasks-board": "Subtaken voor __board__ bord", "default": "Standaard", "defaultdefault": "Standaard", diff --git a/imports/i18n/data/nl.i18n.json b/imports/i18n/data/nl.i18n.json index 4dd27955d..e854597fd 100644 --- a/imports/i18n/data/nl.i18n.json +++ b/imports/i18n/data/nl.i18n.json @@ -357,9 +357,9 @@ "custom-fields": "Maatwerkvelden", "date": "Datum", "date-format": "Datumformaat", - "date-format-yyyy-mm-dd": "YYYY-MM-DD", - "date-format-dd-mm-yyyy": "DD-MM-YYYY", - "date-format-mm-dd-yyyy": "MM-DD-YYYY", + "date-format-yyyy-mm-dd": "JJJJ-MM-DD", + "date-format-dd-mm-yyyy": "DD-MM-JJJJ", + "date-format-mm-dd-yyyy": "MM-DD-JJJJ", "decline": "Weigeren", "default-avatar": "Standaard avatar", "delete": "Verwijderen", @@ -385,7 +385,7 @@ "editNotificationPopup-title": "Wijzig notificatie", "editProfilePopup-title": "Wijzig profiel", "email": "E-mail", - "email-address": "Email Address", + "email-address": "Emailadres", "email-enrollAccount-subject": "Er is een account voor je aangemaakt op __siteName__", "email-enrollAccount-text": "Hallo __user__,\n\nOm gebruik te maken van de online dienst, kan je op de volgende link klikken.\n\n__url__\n\nBedankt.", "email-fail": "E-mail verzenden is mislukt", @@ -755,6 +755,8 @@ "delete-board-confirm-popup": "Alle lijsten, kaarten, labels en activiteiten zullen worden verwijderd en je kunt de bordinhoud niet terughalen. Er is geen herstelmogelijkheid.", "boardDeletePopup-title": "Bord verwijderen?", "delete-board": "Verwijder bord", + "delete-duplicate-lists": "Verwijder Dubbele Lijsten", + "delete-duplicate-lists-confirm": "Weet je het zeker? Alle dubbele lijsten die dezelfde naam hebben en geen kaarten bevatten worden verwijderd.", "default-subtasks-board": "Subtaken voor __board__ bord", "default": "Standaard", "defaultdefault": "Standaard", @@ -1016,8 +1018,8 @@ "dueCardsViewChange-choice-me": "Mij", "dueCardsViewChange-choice-all": "Alle gebruikers", "dueCardsViewChange-choice-all-description": "Toon incomplete kaarten met een *achterstallige* datum van borden waarvoor de gebruiker toegang heeft.", - "dueCards-noResults-title": "No Due Cards Found", - "dueCards-noResults-description": "You don't have any cards with due dates at the moment.", + "dueCards-noResults-title": "Geen Achterstallige Kaarten Gevonden", + "dueCards-noResults-description": "Je hebt nu geen kaarten met achterstallige datums.", "broken-cards": "Defecte kaarten", "board-title-not-found": "Bord '%s' niet gevonden.", "swimlane-title-not-found": "Swimlane '%s' niet gevonden.", diff --git a/imports/i18n/data/oc.i18n.json b/imports/i18n/data/oc.i18n.json index 2b85e2c67..5da5ae2bc 100644 --- a/imports/i18n/data/oc.i18n.json +++ b/imports/i18n/data/oc.i18n.json @@ -755,6 +755,8 @@ "delete-board-confirm-popup": "All lists, cards, labels, and activities will be deleted and you won't be able to recover the board contents. There is no undo.", "boardDeletePopup-title": "Suprimir lo tablèu ?", "delete-board": "Tablèu suprimit", + "delete-duplicate-lists": "Delete Duplicate Lists", + "delete-duplicate-lists-confirm": "Are you sure? This will delete all duplicate lists that have the same name and contain no cards.", "default-subtasks-board": "Subtasks for __board__ board", "default": "Default", "defaultdefault": "Default", diff --git a/imports/i18n/data/or_IN.i18n.json b/imports/i18n/data/or_IN.i18n.json index bcdd00061..eef27e1fe 100644 --- a/imports/i18n/data/or_IN.i18n.json +++ b/imports/i18n/data/or_IN.i18n.json @@ -755,6 +755,8 @@ "delete-board-confirm-popup": "All lists, cards, labels, and activities will be deleted and you won't be able to recover the board contents. There is no undo.", "boardDeletePopup-title": "Delete Board?", "delete-board": "Delete Board", + "delete-duplicate-lists": "Delete Duplicate Lists", + "delete-duplicate-lists-confirm": "Are you sure? This will delete all duplicate lists that have the same name and contain no cards.", "default-subtasks-board": "Subtasks for __board__ board", "default": "Default", "defaultdefault": "Default", diff --git a/imports/i18n/data/pa.i18n.json b/imports/i18n/data/pa.i18n.json index 510b05b6b..1ab8eeaa8 100644 --- a/imports/i18n/data/pa.i18n.json +++ b/imports/i18n/data/pa.i18n.json @@ -755,6 +755,8 @@ "delete-board-confirm-popup": "All lists, cards, labels, and activities will be deleted and you won't be able to recover the board contents. There is no undo.", "boardDeletePopup-title": "Delete Board?", "delete-board": "Delete Board", + "delete-duplicate-lists": "Delete Duplicate Lists", + "delete-duplicate-lists-confirm": "Are you sure? This will delete all duplicate lists that have the same name and contain no cards.", "default-subtasks-board": "Subtasks for __board__ board", "default": "Default", "defaultdefault": "Default", diff --git a/imports/i18n/data/pl-PL.i18n.json b/imports/i18n/data/pl-PL.i18n.json index 555e519a0..641328e6d 100644 --- a/imports/i18n/data/pl-PL.i18n.json +++ b/imports/i18n/data/pl-PL.i18n.json @@ -755,6 +755,8 @@ "delete-board-confirm-popup": "Wszystkie listy, etykiety oraz aktywności zostaną usunięte i nie będziesz w stanie przywrócić zawartości tablicy. Tego nie da się cofnąć.", "boardDeletePopup-title": "Usunąć tablicę?", "delete-board": "Usuń tablicę", + "delete-duplicate-lists": "Delete Duplicate Lists", + "delete-duplicate-lists-confirm": "Are you sure? This will delete all duplicate lists that have the same name and contain no cards.", "default-subtasks-board": "Podzadania dla tablicy __board__", "default": "Domyślny", "defaultdefault": "Domyślny", diff --git a/imports/i18n/data/pl.i18n.json b/imports/i18n/data/pl.i18n.json index 79250ae10..7ae12a506 100644 --- a/imports/i18n/data/pl.i18n.json +++ b/imports/i18n/data/pl.i18n.json @@ -755,6 +755,8 @@ "delete-board-confirm-popup": "Wszystkie listy, etykiety oraz aktywności zostaną usunięte i nie będziesz w stanie przywrócić zawartości tablicy. Tego nie da się cofnąć.", "boardDeletePopup-title": "Usunąć tablicę?", "delete-board": "Usuń tablicę", + "delete-duplicate-lists": "Delete Duplicate Lists", + "delete-duplicate-lists-confirm": "Are you sure? This will delete all duplicate lists that have the same name and contain no cards.", "default-subtasks-board": "Podzadania dla tablicy __board__", "default": "Domyślny", "defaultdefault": "Domyślny", diff --git a/imports/i18n/data/pt-BR.i18n.json b/imports/i18n/data/pt-BR.i18n.json index 6b8cbd923..5a135f543 100644 --- a/imports/i18n/data/pt-BR.i18n.json +++ b/imports/i18n/data/pt-BR.i18n.json @@ -357,9 +357,9 @@ "custom-fields": "Campos customizados", "date": "Data", "date-format": "Formato da Data", - "date-format-yyyy-mm-dd": "YYYY-MM-DD", - "date-format-dd-mm-yyyy": "DD-MM-YYYY", - "date-format-mm-dd-yyyy": "MM-DD-YYYY", + "date-format-yyyy-mm-dd": "AAAA-MM-DD", + "date-format-dd-mm-yyyy": "DD-MM-AAAA", + "date-format-mm-dd-yyyy": "MM-DD-AAAA", "decline": "Rejeitar", "default-avatar": "Avatar padrão", "delete": "Excluir", @@ -385,7 +385,7 @@ "editNotificationPopup-title": "Editar Notificações", "editProfilePopup-title": "Editar Perfil", "email": "E-mail", - "email-address": "Email Address", + "email-address": "Endereço de e-mail", "email-enrollAccount-subject": "Uma conta foi criada para você em __siteName__", "email-enrollAccount-text": "Olá __user__\npara iniciar utilizando o serviço basta clicar no link abaixo.\n__url__\nMuito Obrigado.", "email-fail": "Falhou ao enviar e-mail", @@ -755,6 +755,8 @@ "delete-board-confirm-popup": "Todas as listas, cartões, etiquetas e atividades serão excluídas e você não poderá recuperar o conteúdo do quadro. Não há como desfazer.", "boardDeletePopup-title": "Excluir quadro?", "delete-board": "Excluir quadro", + "delete-duplicate-lists": "Excluir Listas Duplicadas", + "delete-duplicate-lists-confirm": "Você tem certeza? Isso vai apagar todas as litas duplicadas que possuem o mesmo nome e que não possuem cartões", "default-subtasks-board": "Subtarefas para quadro __board__", "default": "Padrão", "defaultdefault": "Padrão", @@ -1016,8 +1018,8 @@ "dueCardsViewChange-choice-me": "Eu", "dueCardsViewChange-choice-all": "Todos os usuários", "dueCardsViewChange-choice-all-description": "Mostrar todos os cartões incompletos com *Prazo Final* nos quadros em que o usuário tem permissão", - "dueCards-noResults-title": "No Due Cards Found", - "dueCards-noResults-description": "You don't have any cards with due dates at the moment.", + "dueCards-noResults-title": "Sem Cartões com prazo final", + "dueCards-noResults-description": "Você não possui cartões com prazo final neste momento", "broken-cards": "Cartões quebrados", "board-title-not-found": "Quadro '%s' não encontrado.", "swimlane-title-not-found": "Raia '%s' não encontrada.", diff --git a/imports/i18n/data/pt.i18n.json b/imports/i18n/data/pt.i18n.json index 9cd9017c4..1f228c838 100644 --- a/imports/i18n/data/pt.i18n.json +++ b/imports/i18n/data/pt.i18n.json @@ -755,6 +755,8 @@ "delete-board-confirm-popup": "Todas as listas, cartões, etiquetas e atividades serão apagadas e não poderá recuperar o conteúdo do quadro. Não é reversível.", "boardDeletePopup-title": "Apagar Quadro?", "delete-board": "Apagar Quadro", + "delete-duplicate-lists": "Delete Duplicate Lists", + "delete-duplicate-lists-confirm": "Are you sure? This will delete all duplicate lists that have the same name and contain no cards.", "default-subtasks-board": "Sub-tarefas para o quadro __board__", "default": "Omissão", "defaultdefault": "Omissão", diff --git a/imports/i18n/data/pt_PT.i18n.json b/imports/i18n/data/pt_PT.i18n.json index 1ce34dfd8..fa36c3204 100644 --- a/imports/i18n/data/pt_PT.i18n.json +++ b/imports/i18n/data/pt_PT.i18n.json @@ -755,6 +755,8 @@ "delete-board-confirm-popup": "Todas as listas, cartões, etiquetas e atividades serão apagadas e não poderá recuperar o conteúdo do quadro. Não é reversível.", "boardDeletePopup-title": "Apagar Quadro?", "delete-board": "Apagar Quadro", + "delete-duplicate-lists": "Delete Duplicate Lists", + "delete-duplicate-lists-confirm": "Are you sure? This will delete all duplicate lists that have the same name and contain no cards.", "default-subtasks-board": "Sub-tarefas para o quadro __board__", "default": "Omissão", "defaultdefault": "Omissão", diff --git a/imports/i18n/data/ro-RO.i18n.json b/imports/i18n/data/ro-RO.i18n.json index 6d125e0e0..0adfeaf0e 100644 --- a/imports/i18n/data/ro-RO.i18n.json +++ b/imports/i18n/data/ro-RO.i18n.json @@ -755,6 +755,8 @@ "delete-board-confirm-popup": "All lists, cards, labels, and activities will be deleted and you won't be able to recover the board contents. There is no undo.", "boardDeletePopup-title": "Delete Board?", "delete-board": "Delete Board", + "delete-duplicate-lists": "Delete Duplicate Lists", + "delete-duplicate-lists-confirm": "Are you sure? This will delete all duplicate lists that have the same name and contain no cards.", "default-subtasks-board": "Subtasks for __board__ board", "default": "Default", "defaultdefault": "Default", diff --git a/imports/i18n/data/ro.i18n.json b/imports/i18n/data/ro.i18n.json index d2952f106..5fbd58448 100644 --- a/imports/i18n/data/ro.i18n.json +++ b/imports/i18n/data/ro.i18n.json @@ -755,6 +755,8 @@ "delete-board-confirm-popup": "All lists, cards, labels, and activities will be deleted and you won't be able to recover the board contents. There is no undo.", "boardDeletePopup-title": "Delete Board?", "delete-board": "Delete Board", + "delete-duplicate-lists": "Delete Duplicate Lists", + "delete-duplicate-lists-confirm": "Are you sure? This will delete all duplicate lists that have the same name and contain no cards.", "default-subtasks-board": "Subtasks for __board__ board", "default": "Default", "defaultdefault": "Default", diff --git a/imports/i18n/data/ru-UA.i18n.json b/imports/i18n/data/ru-UA.i18n.json index b1ca45606..055c85cdc 100644 --- a/imports/i18n/data/ru-UA.i18n.json +++ b/imports/i18n/data/ru-UA.i18n.json @@ -755,6 +755,8 @@ "delete-board-confirm-popup": "All lists, cards, labels, and activities will be deleted and you won't be able to recover the board contents. There is no undo.", "boardDeletePopup-title": "Delete Board?", "delete-board": "Delete Board", + "delete-duplicate-lists": "Delete Duplicate Lists", + "delete-duplicate-lists-confirm": "Are you sure? This will delete all duplicate lists that have the same name and contain no cards.", "default-subtasks-board": "Subtasks for __board__ board", "default": "Default", "defaultdefault": "Default", diff --git a/imports/i18n/data/ru.i18n.json b/imports/i18n/data/ru.i18n.json index a80e4f894..354f21a23 100644 --- a/imports/i18n/data/ru.i18n.json +++ b/imports/i18n/data/ru.i18n.json @@ -755,6 +755,8 @@ "delete-board-confirm-popup": "Все списки, карточки, метки и действия будут удалены, и вы не сможете восстановить содержимое доски. Отменить нельзя.", "boardDeletePopup-title": "Удалить доску?", "delete-board": "Удалить доску", + "delete-duplicate-lists": "Delete Duplicate Lists", + "delete-duplicate-lists-confirm": "Are you sure? This will delete all duplicate lists that have the same name and contain no cards.", "default-subtasks-board": "Подзадача для доски __board__", "default": "По умолчанию", "defaultdefault": "По умолчанию", diff --git a/imports/i18n/data/sk.i18n.json b/imports/i18n/data/sk.i18n.json index bd84a2e77..4eff2776c 100644 --- a/imports/i18n/data/sk.i18n.json +++ b/imports/i18n/data/sk.i18n.json @@ -755,6 +755,8 @@ "delete-board-confirm-popup": "All lists, cards, labels, and activities will be deleted and you won't be able to recover the board contents. There is no undo.", "boardDeletePopup-title": "Zmazať nástenku?", "delete-board": "Zmazať nástenku", + "delete-duplicate-lists": "Delete Duplicate Lists", + "delete-duplicate-lists-confirm": "Are you sure? This will delete all duplicate lists that have the same name and contain no cards.", "default-subtasks-board": "Subtasks for __board__ board", "default": "Default", "defaultdefault": "Default", diff --git a/imports/i18n/data/sl.i18n.json b/imports/i18n/data/sl.i18n.json index 84c16407f..7c7f0d684 100644 --- a/imports/i18n/data/sl.i18n.json +++ b/imports/i18n/data/sl.i18n.json @@ -755,6 +755,8 @@ "delete-board-confirm-popup": "Vsi seznami, kartice, oznake in dejavnosti bodo izbrisani in vsebine table ne boste mogli obnoviti. Razveljavitve ni.", "boardDeletePopup-title": "Izbriši tablo?", "delete-board": "Izbriši tablo", + "delete-duplicate-lists": "Delete Duplicate Lists", + "delete-duplicate-lists-confirm": "Are you sure? This will delete all duplicate lists that have the same name and contain no cards.", "default-subtasks-board": "Podopravila za tablo", "default": "Privzeto", "defaultdefault": "Privzeto", diff --git a/imports/i18n/data/sr.i18n.json b/imports/i18n/data/sr.i18n.json index 1bf947d88..2022488c1 100644 --- a/imports/i18n/data/sr.i18n.json +++ b/imports/i18n/data/sr.i18n.json @@ -755,6 +755,8 @@ "delete-board-confirm-popup": "Све деонице, картице са задацима, налепнице и радње биће избачене и нећете моћи да повратите садржај књиге пословања. Опозив ове радње неће бити могућ.", "boardDeletePopup-title": "Избацићете пословну књигу?", "delete-board": "Избаци пословну књигу", + "delete-duplicate-lists": "Delete Duplicate Lists", + "delete-duplicate-lists-confirm": "Are you sure? This will delete all duplicate lists that have the same name and contain no cards.", "default-subtasks-board": "Под задаци за __board__ књигу пословања", "default": "Подразумевано", "defaultdefault": "Подразумевано", diff --git a/imports/i18n/data/sv.i18n.json b/imports/i18n/data/sv.i18n.json index 12289c879..6a4e1616d 100644 --- a/imports/i18n/data/sv.i18n.json +++ b/imports/i18n/data/sv.i18n.json @@ -357,9 +357,9 @@ "custom-fields": "Anpassade fält", "date": "Datum", "date-format": "Datumformat", - "date-format-yyyy-mm-dd": "YYYY-MM-DD", - "date-format-dd-mm-yyyy": "DD-MM-YYYY", - "date-format-mm-dd-yyyy": "MM-DD-YYYY", + "date-format-yyyy-mm-dd": "ÅÅÅÅ-MM-DD", + "date-format-dd-mm-yyyy": "DD-MM-ÅÅÅÅ", + "date-format-mm-dd-yyyy": "MM-DD-ÅÅÅÅ", "decline": "Neka", "default-avatar": "Standard avatar", "delete": "Ta bort", @@ -385,7 +385,7 @@ "editNotificationPopup-title": "Redigera notis", "editProfilePopup-title": "Redigera profil", "email": "E-post", - "email-address": "Email Address", + "email-address": "E-postadress", "email-enrollAccount-subject": "Ett konto skapat för dig på __siteName__", "email-enrollAccount-text": "Hej __user__,\n\nFör att börja använda tjänsten, klicka på länken nedan.\n\n__url__\n\nTack!", "email-fail": "Sändning av e-post misslyckades", @@ -755,6 +755,8 @@ "delete-board-confirm-popup": "Alla listor, kort, etiketter och aktiviteter kommer tas bort och du kommer inte kunna återställa tavlans innehåll. Det går inte att ångra.", "boardDeletePopup-title": "Ta bort tavla?", "delete-board": "Ta bort tavla", + "delete-duplicate-lists": "Ta bort dubblettlistor", + "delete-duplicate-lists-confirm": "Är du säker? Detta kommer att ta bort alla dubblettlistor som har samma namn och inte innehåller några kort.", "default-subtasks-board": "Deluppgifter för __board__ board", "default": "Standard", "defaultdefault": "Standard", @@ -1016,8 +1018,8 @@ "dueCardsViewChange-choice-me": "Jag", "dueCardsViewChange-choice-all": "Alla användare", "dueCardsViewChange-choice-all-description": "Visar alla oklara kort med *förfallo* datum från tavlor som användaren har tillgång till.", - "dueCards-noResults-title": "No Due Cards Found", - "dueCards-noResults-description": "You don't have any cards with due dates at the moment.", + "dueCards-noResults-title": " Inga kort med förfallodatum hittades", + "dueCards-noResults-description": "Du har inga kort med förfallodatum just nu.", "broken-cards": "Trasiga kort", "board-title-not-found": "Tavla '%s' hittades inte.", "swimlane-title-not-found": "Simbana '%s' hittades inte.", @@ -1328,10 +1330,10 @@ "add-cron-job-placeholder": "Funktionen för att lägga till schemalagda jobb kommer snart", "attachment-storage-configuration": "Konfiguration av fillagringsplats", "attachments-path": " Sökväg för bilagor", - "attachments-path-description": "Path where attachment files are stored", - "avatars-path": "Avatars Path", - "avatars-path-description": "Path where avatar files are stored", - "board-archive-failed": "Failed to schedule board archive", + "attachments-path-description": "Sökväg där bifogade filer lagras", + "avatars-path": "Sökväg för avatarer", + "avatars-path-description": "Sökväg där avatarfiler lagras", + "board-archive-failed": "Misslyckades att schemalägga arkivering av tavla", "board-archive-scheduled": "Board archive scheduled successfully", "board-backup-failed": "Failed to schedule board backup", "board-backup-scheduled": "Board backup scheduled successfully", diff --git a/imports/i18n/data/sw.i18n.json b/imports/i18n/data/sw.i18n.json index 565cf03f9..ae5c41a0d 100644 --- a/imports/i18n/data/sw.i18n.json +++ b/imports/i18n/data/sw.i18n.json @@ -755,6 +755,8 @@ "delete-board-confirm-popup": "All lists, cards, labels, and activities will be deleted and you won't be able to recover the board contents. There is no undo.", "boardDeletePopup-title": "Delete Board?", "delete-board": "Delete Board", + "delete-duplicate-lists": "Delete Duplicate Lists", + "delete-duplicate-lists-confirm": "Are you sure? This will delete all duplicate lists that have the same name and contain no cards.", "default-subtasks-board": "Subtasks for __board__ board", "default": "Default", "defaultdefault": "Default", diff --git a/imports/i18n/data/ta.i18n.json b/imports/i18n/data/ta.i18n.json index 285d23de6..c2e38c9eb 100644 --- a/imports/i18n/data/ta.i18n.json +++ b/imports/i18n/data/ta.i18n.json @@ -755,6 +755,8 @@ "delete-board-confirm-popup": "All lists, cards, labels, and activities will be deleted and you won't be able to recover the board contents. There is no undo.", "boardDeletePopup-title": "Delete Board?", "delete-board": "Delete Board", + "delete-duplicate-lists": "Delete Duplicate Lists", + "delete-duplicate-lists-confirm": "Are you sure? This will delete all duplicate lists that have the same name and contain no cards.", "default-subtasks-board": "Subtasks for __board__ board", "default": "Default", "defaultdefault": "Default", diff --git a/imports/i18n/data/te-IN.i18n.json b/imports/i18n/data/te-IN.i18n.json index bcdd00061..eef27e1fe 100644 --- a/imports/i18n/data/te-IN.i18n.json +++ b/imports/i18n/data/te-IN.i18n.json @@ -755,6 +755,8 @@ "delete-board-confirm-popup": "All lists, cards, labels, and activities will be deleted and you won't be able to recover the board contents. There is no undo.", "boardDeletePopup-title": "Delete Board?", "delete-board": "Delete Board", + "delete-duplicate-lists": "Delete Duplicate Lists", + "delete-duplicate-lists-confirm": "Are you sure? This will delete all duplicate lists that have the same name and contain no cards.", "default-subtasks-board": "Subtasks for __board__ board", "default": "Default", "defaultdefault": "Default", diff --git a/imports/i18n/data/th.i18n.json b/imports/i18n/data/th.i18n.json index 963f73cca..fc992cb5d 100644 --- a/imports/i18n/data/th.i18n.json +++ b/imports/i18n/data/th.i18n.json @@ -755,6 +755,8 @@ "delete-board-confirm-popup": "All lists, cards, labels, and activities will be deleted and you won't be able to recover the board contents. There is no undo.", "boardDeletePopup-title": "Delete Board?", "delete-board": "Delete Board", + "delete-duplicate-lists": "Delete Duplicate Lists", + "delete-duplicate-lists-confirm": "Are you sure? This will delete all duplicate lists that have the same name and contain no cards.", "default-subtasks-board": "Subtasks for __board__ board", "default": "Default", "defaultdefault": "Default", diff --git a/imports/i18n/data/tk_TM.i18n.json b/imports/i18n/data/tk_TM.i18n.json index bcdd00061..eef27e1fe 100644 --- a/imports/i18n/data/tk_TM.i18n.json +++ b/imports/i18n/data/tk_TM.i18n.json @@ -755,6 +755,8 @@ "delete-board-confirm-popup": "All lists, cards, labels, and activities will be deleted and you won't be able to recover the board contents. There is no undo.", "boardDeletePopup-title": "Delete Board?", "delete-board": "Delete Board", + "delete-duplicate-lists": "Delete Duplicate Lists", + "delete-duplicate-lists-confirm": "Are you sure? This will delete all duplicate lists that have the same name and contain no cards.", "default-subtasks-board": "Subtasks for __board__ board", "default": "Default", "defaultdefault": "Default", diff --git a/imports/i18n/data/tlh.i18n.json b/imports/i18n/data/tlh.i18n.json index bcdd00061..eef27e1fe 100644 --- a/imports/i18n/data/tlh.i18n.json +++ b/imports/i18n/data/tlh.i18n.json @@ -755,6 +755,8 @@ "delete-board-confirm-popup": "All lists, cards, labels, and activities will be deleted and you won't be able to recover the board contents. There is no undo.", "boardDeletePopup-title": "Delete Board?", "delete-board": "Delete Board", + "delete-duplicate-lists": "Delete Duplicate Lists", + "delete-duplicate-lists-confirm": "Are you sure? This will delete all duplicate lists that have the same name and contain no cards.", "default-subtasks-board": "Subtasks for __board__ board", "default": "Default", "defaultdefault": "Default", diff --git a/imports/i18n/data/tr.i18n.json b/imports/i18n/data/tr.i18n.json index 418fc1917..a4d61f39c 100644 --- a/imports/i18n/data/tr.i18n.json +++ b/imports/i18n/data/tr.i18n.json @@ -755,6 +755,8 @@ "delete-board-confirm-popup": "Tüm listeler, kartlar, etiketler ve etkinlikler silinecek ve pano içeriğini kurtaramayacaksınız. Geri dönüş yok.", "boardDeletePopup-title": "Panoyu Sil?", "delete-board": "Panoyu Sil", + "delete-duplicate-lists": "Delete Duplicate Lists", + "delete-duplicate-lists-confirm": "Are you sure? This will delete all duplicate lists that have the same name and contain no cards.", "default-subtasks-board": "__board__ panosu için alt görevler", "default": "Varsayılan", "defaultdefault": "Varsayılan", diff --git a/imports/i18n/data/ug.i18n.json b/imports/i18n/data/ug.i18n.json index bcdd00061..eef27e1fe 100644 --- a/imports/i18n/data/ug.i18n.json +++ b/imports/i18n/data/ug.i18n.json @@ -755,6 +755,8 @@ "delete-board-confirm-popup": "All lists, cards, labels, and activities will be deleted and you won't be able to recover the board contents. There is no undo.", "boardDeletePopup-title": "Delete Board?", "delete-board": "Delete Board", + "delete-duplicate-lists": "Delete Duplicate Lists", + "delete-duplicate-lists-confirm": "Are you sure? This will delete all duplicate lists that have the same name and contain no cards.", "default-subtasks-board": "Subtasks for __board__ board", "default": "Default", "defaultdefault": "Default", diff --git a/imports/i18n/data/uk-UA.i18n.json b/imports/i18n/data/uk-UA.i18n.json index 4e7e492d8..037a7a3e5 100644 --- a/imports/i18n/data/uk-UA.i18n.json +++ b/imports/i18n/data/uk-UA.i18n.json @@ -755,6 +755,8 @@ "delete-board-confirm-popup": "Усі списки, картки, мітки та діяльність будуть видалені, і ви не зможете відновити вміст дошки. Немає відкату.", "boardDeletePopup-title": "Видалити дошку?", "delete-board": "Видалити дошку", + "delete-duplicate-lists": "Delete Duplicate Lists", + "delete-duplicate-lists-confirm": "Are you sure? This will delete all duplicate lists that have the same name and contain no cards.", "default-subtasks-board": "Підзадачі для дошки __board__", "default": "За замовчуванням", "defaultdefault": "За замовчуванням", diff --git a/imports/i18n/data/uk.i18n.json b/imports/i18n/data/uk.i18n.json index 222a2e60f..e1d01b105 100644 --- a/imports/i18n/data/uk.i18n.json +++ b/imports/i18n/data/uk.i18n.json @@ -755,6 +755,8 @@ "delete-board-confirm-popup": "Усі списки, картки, мітки та діяльність будуть видалені, і ви не зможете відновити вміст дошки. Немає відкату.", "boardDeletePopup-title": "Видалити дошку?", "delete-board": "Видалити дошку", + "delete-duplicate-lists": "Delete Duplicate Lists", + "delete-duplicate-lists-confirm": "Are you sure? This will delete all duplicate lists that have the same name and contain no cards.", "default-subtasks-board": "Підзадачі для дошки __board__", "default": "За замовчуванням", "defaultdefault": "За замовчуванням", diff --git a/imports/i18n/data/uz-AR.i18n.json b/imports/i18n/data/uz-AR.i18n.json index bcdd00061..eef27e1fe 100644 --- a/imports/i18n/data/uz-AR.i18n.json +++ b/imports/i18n/data/uz-AR.i18n.json @@ -755,6 +755,8 @@ "delete-board-confirm-popup": "All lists, cards, labels, and activities will be deleted and you won't be able to recover the board contents. There is no undo.", "boardDeletePopup-title": "Delete Board?", "delete-board": "Delete Board", + "delete-duplicate-lists": "Delete Duplicate Lists", + "delete-duplicate-lists-confirm": "Are you sure? This will delete all duplicate lists that have the same name and contain no cards.", "default-subtasks-board": "Subtasks for __board__ board", "default": "Default", "defaultdefault": "Default", diff --git a/imports/i18n/data/uz-LA.i18n.json b/imports/i18n/data/uz-LA.i18n.json index bcdd00061..eef27e1fe 100644 --- a/imports/i18n/data/uz-LA.i18n.json +++ b/imports/i18n/data/uz-LA.i18n.json @@ -755,6 +755,8 @@ "delete-board-confirm-popup": "All lists, cards, labels, and activities will be deleted and you won't be able to recover the board contents. There is no undo.", "boardDeletePopup-title": "Delete Board?", "delete-board": "Delete Board", + "delete-duplicate-lists": "Delete Duplicate Lists", + "delete-duplicate-lists-confirm": "Are you sure? This will delete all duplicate lists that have the same name and contain no cards.", "default-subtasks-board": "Subtasks for __board__ board", "default": "Default", "defaultdefault": "Default", diff --git a/imports/i18n/data/uz-UZ.i18n.json b/imports/i18n/data/uz-UZ.i18n.json index bcdd00061..eef27e1fe 100644 --- a/imports/i18n/data/uz-UZ.i18n.json +++ b/imports/i18n/data/uz-UZ.i18n.json @@ -755,6 +755,8 @@ "delete-board-confirm-popup": "All lists, cards, labels, and activities will be deleted and you won't be able to recover the board contents. There is no undo.", "boardDeletePopup-title": "Delete Board?", "delete-board": "Delete Board", + "delete-duplicate-lists": "Delete Duplicate Lists", + "delete-duplicate-lists-confirm": "Are you sure? This will delete all duplicate lists that have the same name and contain no cards.", "default-subtasks-board": "Subtasks for __board__ board", "default": "Default", "defaultdefault": "Default", diff --git a/imports/i18n/data/uz.i18n.json b/imports/i18n/data/uz.i18n.json index bcdd00061..eef27e1fe 100644 --- a/imports/i18n/data/uz.i18n.json +++ b/imports/i18n/data/uz.i18n.json @@ -755,6 +755,8 @@ "delete-board-confirm-popup": "All lists, cards, labels, and activities will be deleted and you won't be able to recover the board contents. There is no undo.", "boardDeletePopup-title": "Delete Board?", "delete-board": "Delete Board", + "delete-duplicate-lists": "Delete Duplicate Lists", + "delete-duplicate-lists-confirm": "Are you sure? This will delete all duplicate lists that have the same name and contain no cards.", "default-subtasks-board": "Subtasks for __board__ board", "default": "Default", "defaultdefault": "Default", diff --git a/imports/i18n/data/ve-CC.i18n.json b/imports/i18n/data/ve-CC.i18n.json index bcdd00061..eef27e1fe 100644 --- a/imports/i18n/data/ve-CC.i18n.json +++ b/imports/i18n/data/ve-CC.i18n.json @@ -755,6 +755,8 @@ "delete-board-confirm-popup": "All lists, cards, labels, and activities will be deleted and you won't be able to recover the board contents. There is no undo.", "boardDeletePopup-title": "Delete Board?", "delete-board": "Delete Board", + "delete-duplicate-lists": "Delete Duplicate Lists", + "delete-duplicate-lists-confirm": "Are you sure? This will delete all duplicate lists that have the same name and contain no cards.", "default-subtasks-board": "Subtasks for __board__ board", "default": "Default", "defaultdefault": "Default", diff --git a/imports/i18n/data/ve-PP.i18n.json b/imports/i18n/data/ve-PP.i18n.json index bcdd00061..eef27e1fe 100644 --- a/imports/i18n/data/ve-PP.i18n.json +++ b/imports/i18n/data/ve-PP.i18n.json @@ -755,6 +755,8 @@ "delete-board-confirm-popup": "All lists, cards, labels, and activities will be deleted and you won't be able to recover the board contents. There is no undo.", "boardDeletePopup-title": "Delete Board?", "delete-board": "Delete Board", + "delete-duplicate-lists": "Delete Duplicate Lists", + "delete-duplicate-lists-confirm": "Are you sure? This will delete all duplicate lists that have the same name and contain no cards.", "default-subtasks-board": "Subtasks for __board__ board", "default": "Default", "defaultdefault": "Default", diff --git a/imports/i18n/data/ve.i18n.json b/imports/i18n/data/ve.i18n.json index bcdd00061..eef27e1fe 100644 --- a/imports/i18n/data/ve.i18n.json +++ b/imports/i18n/data/ve.i18n.json @@ -755,6 +755,8 @@ "delete-board-confirm-popup": "All lists, cards, labels, and activities will be deleted and you won't be able to recover the board contents. There is no undo.", "boardDeletePopup-title": "Delete Board?", "delete-board": "Delete Board", + "delete-duplicate-lists": "Delete Duplicate Lists", + "delete-duplicate-lists-confirm": "Are you sure? This will delete all duplicate lists that have the same name and contain no cards.", "default-subtasks-board": "Subtasks for __board__ board", "default": "Default", "defaultdefault": "Default", diff --git a/imports/i18n/data/vi-VN.i18n.json b/imports/i18n/data/vi-VN.i18n.json index bef9464f7..9dbb74fa4 100644 --- a/imports/i18n/data/vi-VN.i18n.json +++ b/imports/i18n/data/vi-VN.i18n.json @@ -755,6 +755,8 @@ "delete-board-confirm-popup": "All lists, cards, labels, and activities will be deleted and you won't be able to recover the board contents. There is no undo.", "boardDeletePopup-title": "Delete Board?", "delete-board": "Delete Board", + "delete-duplicate-lists": "Delete Duplicate Lists", + "delete-duplicate-lists-confirm": "Are you sure? This will delete all duplicate lists that have the same name and contain no cards.", "default-subtasks-board": "Subtasks for __board__ board", "default": "Default", "defaultdefault": "Default", diff --git a/imports/i18n/data/vi.i18n.json b/imports/i18n/data/vi.i18n.json index 702b0e2c5..8843c5833 100644 --- a/imports/i18n/data/vi.i18n.json +++ b/imports/i18n/data/vi.i18n.json @@ -755,6 +755,8 @@ "delete-board-confirm-popup": "Tất cả danh sách, thẻ, nhãn và hoạt động sẽ bị xóa và bạn sẽ không thể khôi phục nội dung bảng. Không thể hoàn tác.", "boardDeletePopup-title": "Xoá Bảng?", "delete-board": "Xoá Bảng", + "delete-duplicate-lists": "Delete Duplicate Lists", + "delete-duplicate-lists-confirm": "Are you sure? This will delete all duplicate lists that have the same name and contain no cards.", "default-subtasks-board": "Nhiệm vụ phụ cho __board__ bảng", "default": "Mặc định", "defaultdefault": "Mặc định", diff --git a/imports/i18n/data/vl-SS.i18n.json b/imports/i18n/data/vl-SS.i18n.json index bcdd00061..eef27e1fe 100644 --- a/imports/i18n/data/vl-SS.i18n.json +++ b/imports/i18n/data/vl-SS.i18n.json @@ -755,6 +755,8 @@ "delete-board-confirm-popup": "All lists, cards, labels, and activities will be deleted and you won't be able to recover the board contents. There is no undo.", "boardDeletePopup-title": "Delete Board?", "delete-board": "Delete Board", + "delete-duplicate-lists": "Delete Duplicate Lists", + "delete-duplicate-lists-confirm": "Are you sure? This will delete all duplicate lists that have the same name and contain no cards.", "default-subtasks-board": "Subtasks for __board__ board", "default": "Default", "defaultdefault": "Default", diff --git a/imports/i18n/data/vo.i18n.json b/imports/i18n/data/vo.i18n.json index bcdd00061..eef27e1fe 100644 --- a/imports/i18n/data/vo.i18n.json +++ b/imports/i18n/data/vo.i18n.json @@ -755,6 +755,8 @@ "delete-board-confirm-popup": "All lists, cards, labels, and activities will be deleted and you won't be able to recover the board contents. There is no undo.", "boardDeletePopup-title": "Delete Board?", "delete-board": "Delete Board", + "delete-duplicate-lists": "Delete Duplicate Lists", + "delete-duplicate-lists-confirm": "Are you sure? This will delete all duplicate lists that have the same name and contain no cards.", "default-subtasks-board": "Subtasks for __board__ board", "default": "Default", "defaultdefault": "Default", diff --git a/imports/i18n/data/wa-RR.i18n.json b/imports/i18n/data/wa-RR.i18n.json index bcdd00061..eef27e1fe 100644 --- a/imports/i18n/data/wa-RR.i18n.json +++ b/imports/i18n/data/wa-RR.i18n.json @@ -755,6 +755,8 @@ "delete-board-confirm-popup": "All lists, cards, labels, and activities will be deleted and you won't be able to recover the board contents. There is no undo.", "boardDeletePopup-title": "Delete Board?", "delete-board": "Delete Board", + "delete-duplicate-lists": "Delete Duplicate Lists", + "delete-duplicate-lists-confirm": "Are you sure? This will delete all duplicate lists that have the same name and contain no cards.", "default-subtasks-board": "Subtasks for __board__ board", "default": "Default", "defaultdefault": "Default", diff --git a/imports/i18n/data/wa.i18n.json b/imports/i18n/data/wa.i18n.json index bcdd00061..eef27e1fe 100644 --- a/imports/i18n/data/wa.i18n.json +++ b/imports/i18n/data/wa.i18n.json @@ -755,6 +755,8 @@ "delete-board-confirm-popup": "All lists, cards, labels, and activities will be deleted and you won't be able to recover the board contents. There is no undo.", "boardDeletePopup-title": "Delete Board?", "delete-board": "Delete Board", + "delete-duplicate-lists": "Delete Duplicate Lists", + "delete-duplicate-lists-confirm": "Are you sure? This will delete all duplicate lists that have the same name and contain no cards.", "default-subtasks-board": "Subtasks for __board__ board", "default": "Default", "defaultdefault": "Default", diff --git a/imports/i18n/data/wo.i18n.json b/imports/i18n/data/wo.i18n.json index bcdd00061..eef27e1fe 100644 --- a/imports/i18n/data/wo.i18n.json +++ b/imports/i18n/data/wo.i18n.json @@ -755,6 +755,8 @@ "delete-board-confirm-popup": "All lists, cards, labels, and activities will be deleted and you won't be able to recover the board contents. There is no undo.", "boardDeletePopup-title": "Delete Board?", "delete-board": "Delete Board", + "delete-duplicate-lists": "Delete Duplicate Lists", + "delete-duplicate-lists-confirm": "Are you sure? This will delete all duplicate lists that have the same name and contain no cards.", "default-subtasks-board": "Subtasks for __board__ board", "default": "Default", "defaultdefault": "Default", diff --git a/imports/i18n/data/wuu-Hans.i18n.json b/imports/i18n/data/wuu-Hans.i18n.json index bcdd00061..eef27e1fe 100644 --- a/imports/i18n/data/wuu-Hans.i18n.json +++ b/imports/i18n/data/wuu-Hans.i18n.json @@ -755,6 +755,8 @@ "delete-board-confirm-popup": "All lists, cards, labels, and activities will be deleted and you won't be able to recover the board contents. There is no undo.", "boardDeletePopup-title": "Delete Board?", "delete-board": "Delete Board", + "delete-duplicate-lists": "Delete Duplicate Lists", + "delete-duplicate-lists-confirm": "Are you sure? This will delete all duplicate lists that have the same name and contain no cards.", "default-subtasks-board": "Subtasks for __board__ board", "default": "Default", "defaultdefault": "Default", diff --git a/imports/i18n/data/xh.i18n.json b/imports/i18n/data/xh.i18n.json index bcdd00061..eef27e1fe 100644 --- a/imports/i18n/data/xh.i18n.json +++ b/imports/i18n/data/xh.i18n.json @@ -755,6 +755,8 @@ "delete-board-confirm-popup": "All lists, cards, labels, and activities will be deleted and you won't be able to recover the board contents. There is no undo.", "boardDeletePopup-title": "Delete Board?", "delete-board": "Delete Board", + "delete-duplicate-lists": "Delete Duplicate Lists", + "delete-duplicate-lists-confirm": "Are you sure? This will delete all duplicate lists that have the same name and contain no cards.", "default-subtasks-board": "Subtasks for __board__ board", "default": "Default", "defaultdefault": "Default", diff --git a/imports/i18n/data/yi.i18n.json b/imports/i18n/data/yi.i18n.json index bcdd00061..eef27e1fe 100644 --- a/imports/i18n/data/yi.i18n.json +++ b/imports/i18n/data/yi.i18n.json @@ -755,6 +755,8 @@ "delete-board-confirm-popup": "All lists, cards, labels, and activities will be deleted and you won't be able to recover the board contents. There is no undo.", "boardDeletePopup-title": "Delete Board?", "delete-board": "Delete Board", + "delete-duplicate-lists": "Delete Duplicate Lists", + "delete-duplicate-lists-confirm": "Are you sure? This will delete all duplicate lists that have the same name and contain no cards.", "default-subtasks-board": "Subtasks for __board__ board", "default": "Default", "defaultdefault": "Default", diff --git a/imports/i18n/data/yo.i18n.json b/imports/i18n/data/yo.i18n.json index bcdd00061..eef27e1fe 100644 --- a/imports/i18n/data/yo.i18n.json +++ b/imports/i18n/data/yo.i18n.json @@ -755,6 +755,8 @@ "delete-board-confirm-popup": "All lists, cards, labels, and activities will be deleted and you won't be able to recover the board contents. There is no undo.", "boardDeletePopup-title": "Delete Board?", "delete-board": "Delete Board", + "delete-duplicate-lists": "Delete Duplicate Lists", + "delete-duplicate-lists-confirm": "Are you sure? This will delete all duplicate lists that have the same name and contain no cards.", "default-subtasks-board": "Subtasks for __board__ board", "default": "Default", "defaultdefault": "Default", diff --git a/imports/i18n/data/yue_CN.i18n.json b/imports/i18n/data/yue_CN.i18n.json index bcdd00061..eef27e1fe 100644 --- a/imports/i18n/data/yue_CN.i18n.json +++ b/imports/i18n/data/yue_CN.i18n.json @@ -755,6 +755,8 @@ "delete-board-confirm-popup": "All lists, cards, labels, and activities will be deleted and you won't be able to recover the board contents. There is no undo.", "boardDeletePopup-title": "Delete Board?", "delete-board": "Delete Board", + "delete-duplicate-lists": "Delete Duplicate Lists", + "delete-duplicate-lists-confirm": "Are you sure? This will delete all duplicate lists that have the same name and contain no cards.", "default-subtasks-board": "Subtasks for __board__ board", "default": "Default", "defaultdefault": "Default", diff --git a/imports/i18n/data/zgh.i18n.json b/imports/i18n/data/zgh.i18n.json index bcdd00061..eef27e1fe 100644 --- a/imports/i18n/data/zgh.i18n.json +++ b/imports/i18n/data/zgh.i18n.json @@ -755,6 +755,8 @@ "delete-board-confirm-popup": "All lists, cards, labels, and activities will be deleted and you won't be able to recover the board contents. There is no undo.", "boardDeletePopup-title": "Delete Board?", "delete-board": "Delete Board", + "delete-duplicate-lists": "Delete Duplicate Lists", + "delete-duplicate-lists-confirm": "Are you sure? This will delete all duplicate lists that have the same name and contain no cards.", "default-subtasks-board": "Subtasks for __board__ board", "default": "Default", "defaultdefault": "Default", diff --git a/imports/i18n/data/zh-CN.i18n.json b/imports/i18n/data/zh-CN.i18n.json index a52be2803..2f3dba633 100644 --- a/imports/i18n/data/zh-CN.i18n.json +++ b/imports/i18n/data/zh-CN.i18n.json @@ -755,6 +755,8 @@ "delete-board-confirm-popup": "所有列表、卡片、标签和活动都回被删除,将无法恢复看板内容。不支持撤销。", "boardDeletePopup-title": "删除看板?", "delete-board": "删除看板", + "delete-duplicate-lists": "Delete Duplicate Lists", + "delete-duplicate-lists-confirm": "Are you sure? This will delete all duplicate lists that have the same name and contain no cards.", "default-subtasks-board": "__board__ 看板的子任务", "default": "默认", "defaultdefault": "默认", diff --git a/imports/i18n/data/zh-GB.i18n.json b/imports/i18n/data/zh-GB.i18n.json index 0c4f978f7..151bcf1fd 100644 --- a/imports/i18n/data/zh-GB.i18n.json +++ b/imports/i18n/data/zh-GB.i18n.json @@ -755,6 +755,8 @@ "delete-board-confirm-popup": "All lists, cards, labels, and activities will be deleted and you won't be able to recover the board contents. There is no undo.", "boardDeletePopup-title": "Delete Board?", "delete-board": "Delete Board", + "delete-duplicate-lists": "Delete Duplicate Lists", + "delete-duplicate-lists-confirm": "Are you sure? This will delete all duplicate lists that have the same name and contain no cards.", "default-subtasks-board": "Subtasks for __board__ board", "default": "Default", "defaultdefault": "Default", diff --git a/imports/i18n/data/zh-HK.i18n.json b/imports/i18n/data/zh-HK.i18n.json index 1e13b42da..5c71d764d 100644 --- a/imports/i18n/data/zh-HK.i18n.json +++ b/imports/i18n/data/zh-HK.i18n.json @@ -755,6 +755,8 @@ "delete-board-confirm-popup": "All lists, cards, labels, and activities will be deleted and you won't be able to recover the board contents. There is no undo.", "boardDeletePopup-title": "Delete Board?", "delete-board": "Delete Board", + "delete-duplicate-lists": "Delete Duplicate Lists", + "delete-duplicate-lists-confirm": "Are you sure? This will delete all duplicate lists that have the same name and contain no cards.", "default-subtasks-board": "Subtasks for __board__ board", "default": "Default", "defaultdefault": "Default", diff --git a/imports/i18n/data/zh-Hans.i18n.json b/imports/i18n/data/zh-Hans.i18n.json index 3cb94e011..1938e0590 100644 --- a/imports/i18n/data/zh-Hans.i18n.json +++ b/imports/i18n/data/zh-Hans.i18n.json @@ -755,6 +755,8 @@ "delete-board-confirm-popup": "All lists, cards, labels, and activities will be deleted and you won't be able to recover the board contents. There is no undo.", "boardDeletePopup-title": "Delete Board?", "delete-board": "Delete Board", + "delete-duplicate-lists": "Delete Duplicate Lists", + "delete-duplicate-lists-confirm": "Are you sure? This will delete all duplicate lists that have the same name and contain no cards.", "default-subtasks-board": "Subtasks for __board__ board", "default": "Default", "defaultdefault": "Default", diff --git a/imports/i18n/data/zh-Hant.i18n.json b/imports/i18n/data/zh-Hant.i18n.json index bcdd00061..c1d5ed8b7 100644 --- a/imports/i18n/data/zh-Hant.i18n.json +++ b/imports/i18n/data/zh-Hant.i18n.json @@ -1,6 +1,6 @@ { - "accept": "Accept", - "act-activity-notify": "Activity Notification", + "accept": "接受", + "act-activity-notify": "動態通知", "act-addAttachment": "added attachment __attachment__ to card __card__ at list __list__ at swimlane __swimlane__ at board __board__", "act-deleteAttachment": "deleted attachment __attachment__ at card __card__ at list __list__ at swimlane __swimlane__ at board __board__", "act-addSubtask": "added subtask __subtask__ to card __card__ at list __list__ at swimlane __swimlane__ at board __board__", @@ -48,7 +48,7 @@ "activity-added": "added %s to %s", "activity-archived": "%s moved to Archive", "activity-attached": "attached %s to %s", - "activity-created": "created %s", + "activity-created": "已建立", "activity-changedListTitle": "renamed list to %s", "activity-customfield-created": "created custom field %s", "activity-excluded": "excluded %s from %s", @@ -755,6 +755,8 @@ "delete-board-confirm-popup": "All lists, cards, labels, and activities will be deleted and you won't be able to recover the board contents. There is no undo.", "boardDeletePopup-title": "Delete Board?", "delete-board": "Delete Board", + "delete-duplicate-lists": "Delete Duplicate Lists", + "delete-duplicate-lists-confirm": "Are you sure? This will delete all duplicate lists that have the same name and contain no cards.", "default-subtasks-board": "Subtasks for __board__ board", "default": "Default", "defaultdefault": "Default", diff --git a/imports/i18n/data/zh-TW.i18n.json b/imports/i18n/data/zh-TW.i18n.json index 383163ae0..754785cd2 100644 --- a/imports/i18n/data/zh-TW.i18n.json +++ b/imports/i18n/data/zh-TW.i18n.json @@ -385,7 +385,7 @@ "editNotificationPopup-title": "更改通知", "editProfilePopup-title": "編輯個人資料", "email": "電子郵件", - "email-address": "Email Address", + "email-address": "電子郵件地址", "email-enrollAccount-subject": "您在 __siteName__ 的帳號已經建立", "email-enrollAccount-text": "親愛的 __user__,\n\n點選下面的連結,即刻開始使用這項服務。\n\n__url__\n\n謝謝。", "email-fail": "郵件寄送失敗", @@ -755,6 +755,8 @@ "delete-board-confirm-popup": "所有清單、卡片、標籤和活動都會被刪除,將無法恢覆看板內容。不支援撤銷。", "boardDeletePopup-title": "刪除看板?", "delete-board": "刪除看板", + "delete-duplicate-lists": "Delete Duplicate Lists", + "delete-duplicate-lists-confirm": "Are you sure? This will delete all duplicate lists that have the same name and contain no cards.", "default-subtasks-board": "__board__ 看板的子任務", "default": "預設值", "defaultdefault": "預設值", @@ -1016,8 +1018,8 @@ "dueCardsViewChange-choice-me": "我", "dueCardsViewChange-choice-all": "全部使用者", "dueCardsViewChange-choice-all-description": "顯示看板內所有已設定到期日,且使用者有權限的未完成卡片", - "dueCards-noResults-title": "No Due Cards Found", - "dueCards-noResults-description": "You don't have any cards with due dates at the moment.", + "dueCards-noResults-title": "找不到到期卡片", + "dueCards-noResults-description": "您目前沒有任何有到期日的卡片。", "broken-cards": "損毀卡片", "board-title-not-found": "看板%s不存在", "swimlane-title-not-found": "泳道流程圖%s不存在", diff --git a/imports/i18n/data/zh.i18n.json b/imports/i18n/data/zh.i18n.json index 39eb3f485..78fd22be4 100644 --- a/imports/i18n/data/zh.i18n.json +++ b/imports/i18n/data/zh.i18n.json @@ -755,6 +755,8 @@ "delete-board-confirm-popup": "All lists, cards, labels, and activities will be deleted and you won't be able to recover the board contents. There is no undo.", "boardDeletePopup-title": "Delete Board?", "delete-board": "Delete Board", + "delete-duplicate-lists": "Delete Duplicate Lists", + "delete-duplicate-lists-confirm": "Are you sure? This will delete all duplicate lists that have the same name and contain no cards.", "default-subtasks-board": "Subtasks for __board__ board", "default": "Default", "defaultdefault": "Default", diff --git a/imports/i18n/data/zu-ZA.i18n.json b/imports/i18n/data/zu-ZA.i18n.json index bcdd00061..eef27e1fe 100644 --- a/imports/i18n/data/zu-ZA.i18n.json +++ b/imports/i18n/data/zu-ZA.i18n.json @@ -755,6 +755,8 @@ "delete-board-confirm-popup": "All lists, cards, labels, and activities will be deleted and you won't be able to recover the board contents. There is no undo.", "boardDeletePopup-title": "Delete Board?", "delete-board": "Delete Board", + "delete-duplicate-lists": "Delete Duplicate Lists", + "delete-duplicate-lists-confirm": "Are you sure? This will delete all duplicate lists that have the same name and contain no cards.", "default-subtasks-board": "Subtasks for __board__ board", "default": "Default", "defaultdefault": "Default", diff --git a/imports/i18n/data/zu.i18n.json b/imports/i18n/data/zu.i18n.json index bcdd00061..eef27e1fe 100644 --- a/imports/i18n/data/zu.i18n.json +++ b/imports/i18n/data/zu.i18n.json @@ -755,6 +755,8 @@ "delete-board-confirm-popup": "All lists, cards, labels, and activities will be deleted and you won't be able to recover the board contents. There is no undo.", "boardDeletePopup-title": "Delete Board?", "delete-board": "Delete Board", + "delete-duplicate-lists": "Delete Duplicate Lists", + "delete-duplicate-lists-confirm": "Are you sure? This will delete all duplicate lists that have the same name and contain no cards.", "default-subtasks-board": "Subtasks for __board__ board", "default": "Default", "defaultdefault": "Default", From 034dc08269520ca31c780cce64e0150969e9228e Mon Sep 17 00:00:00 2001 From: Lauri Ojansivu Date: Sat, 25 Oct 2025 19:17:09 +0300 Subject: [PATCH 10/84] Disabled migrations that happen when opening board. Defaulting to per-swimlane lists and drag drop list to same or different swimlane. Thanks to xet7 ! --- client/components/boards/boardBody.js | 77 +++++++++++++----------- client/components/lists/list.css | 3 - client/components/swimlanes/swimlanes.js | 17 +++--- models/swimlanes.js | 8 ++- 4 files changed, 57 insertions(+), 48 deletions(-) diff --git a/client/components/boards/boardBody.js b/client/components/boards/boardBody.js index 5eb317b9b..a9b04cddb 100644 --- a/client/components/boards/boardBody.js +++ b/client/components/boards/boardBody.js @@ -99,51 +99,60 @@ BlazeComponent.extendComponent({ } // Check if board needs migration based on migration version - const needsMigration = !board.migrationVersion || board.migrationVersion < 1; + // DISABLED: Migration check and execution + // const needsMigration = !board.migrationVersion || board.migrationVersion < 1; - if (needsMigration) { - // Start background migration for old boards - this.isMigrating.set(true); - await this.startBackgroundMigration(boardId); - this.isMigrating.set(false); - } + // if (needsMigration) { + // // Start background migration for old boards + // this.isMigrating.set(true); + // await this.startBackgroundMigration(boardId); + // this.isMigrating.set(false); + // } // Check if board needs conversion (for old structure) - if (boardConverter.isBoardConverted(boardId)) { - if (process.env.DEBUG === 'true') { - console.log(`Board ${boardId} has already been converted, skipping conversion`); - } - this.isBoardReady.set(true); - } else { - const needsConversion = boardConverter.needsConversion(boardId); - - if (needsConversion) { - this.isConverting.set(true); - const success = await boardConverter.convertBoard(boardId); - this.isConverting.set(false); - - if (success) { - this.isBoardReady.set(true); - } else { - console.error('Board conversion failed, setting ready to true anyway'); - this.isBoardReady.set(true); // Still show board even if conversion failed - } - } else { - this.isBoardReady.set(true); - } - } + // DISABLED: Board conversion logic + // if (boardConverter.isBoardConverted(boardId)) { + // if (process.env.DEBUG === 'true') { + // console.log(`Board ${boardId} has already been converted, skipping conversion`); + // } + // this.isBoardReady.set(true); + // } else { + // const needsConversion = boardConverter.needsConversion(boardId); + // + // if (needsConversion) { + // this.isConverting.set(true); + // const success = await boardConverter.convertBoard(boardId); + // this.isConverting.set(false); + // + // if (success) { + // this.isBoardReady.set(true); + // } else { + // console.error('Board conversion failed, setting ready to true anyway'); + // this.isBoardReady.set(true); // Still show board even if conversion failed + // } + // } else { + // this.isBoardReady.set(true); + // } + // } + + // Set board ready immediately since conversions are disabled + this.isBoardReady.set(true); // Convert shared lists to per-swimlane lists if needed - await this.convertSharedListsToPerSwimlane(boardId); + // DISABLED: Shared lists conversion + // await this.convertSharedListsToPerSwimlane(boardId); // Fix missing lists migration (for cards with wrong listId references) - await this.fixMissingLists(boardId); + // DISABLED: Missing lists fix + // await this.fixMissingLists(boardId); // Fix duplicate lists created by WeKan 8.10 - await this.fixDuplicateLists(boardId); + // DISABLED: Duplicate lists fix + // await this.fixDuplicateLists(boardId); // Start attachment migration in background if needed - this.startAttachmentMigrationIfNeeded(boardId); + // DISABLED: Attachment migration + // this.startAttachmentMigrationIfNeeded(boardId); } catch (error) { console.error('Error during board conversion check:', error); this.isConverting.set(false); diff --git a/client/components/lists/list.css b/client/components/lists/list.css index 7c238efcd..53426199b 100644 --- a/client/components/lists/list.css +++ b/client/components/lists/list.css @@ -378,9 +378,6 @@ body.list-resizing-active * { position: relative; text-overflow: ellipsis; white-space: nowrap; -} -.list-header .list-rotated { - } .list-header .list-header-watch-icon { padding-left: 10px; diff --git a/client/components/swimlanes/swimlanes.js b/client/components/swimlanes/swimlanes.js index 47b9c5843..e0dd896d5 100644 --- a/client/components/swimlanes/swimlanes.js +++ b/client/components/swimlanes/swimlanes.js @@ -228,10 +228,8 @@ function initSortable(boardComponent, $listsDom) { // Don't cancel the sortable when moving to a different swimlane // The DOM move should be allowed to complete - } else { - // If staying in the same swimlane, cancel the sortable to prevent DOM manipulation issues - $listsDom.sortable('cancel'); } + // Allow reordering within the same swimlane by not canceling the sortable try { Lists.update(list._id, { @@ -682,6 +680,11 @@ Template.swimlane.helpers({ canSeeAddList() { return ReactiveCache.getCurrentUser().isBoardAdmin(); }, + + lists() { + // Return per-swimlane lists for this swimlane + return this.myLists(); + } }); // Initialize sortable on DOM elements @@ -794,10 +797,8 @@ setTimeout(() => { // Don't cancel the sortable when moving to a different swimlane // The DOM move should be allowed to complete - } else { - // If staying in the same swimlane, cancel the sortable to prevent DOM manipulation issues - $swimlane.sortable('cancel'); } + // Allow reordering within the same swimlane by not canceling the sortable try { Lists.update(list._id, { @@ -938,10 +939,8 @@ setTimeout(() => { // Don't cancel the sortable when moving to a different swimlane // The DOM move should be allowed to complete - } else { - // If staying in the same swimlane, cancel the sortable to prevent DOM manipulation issues - $listsGroup.sortable('cancel'); } + // Allow reordering within the same swimlane by not canceling the sortable try { Lists.update(list._id, { diff --git a/models/swimlanes.js b/models/swimlanes.js index 012e56f21..659111d04 100644 --- a/models/swimlanes.js +++ b/models/swimlanes.js @@ -232,8 +232,12 @@ Swimlanes.helpers({ }, myLists() { - // Revert to shared lists: provide lists by board for this swimlane's board - return ReactiveCache.getLists({ boardId: this.boardId }); + // Return per-swimlane lists: provide lists specific to this swimlane + return ReactiveCache.getLists({ + boardId: this.boardId, + swimlaneId: this._id, + archived: false + }); }, allCards() { From 0c99cb3103fec61cedde25b41a4a5ce30777f9dc Mon Sep 17 00:00:00 2001 From: Lauri Ojansivu Date: Sat, 25 Oct 2025 19:19:35 +0300 Subject: [PATCH 11/84] Updated ChangeLog. --- CHANGELOG.md | 2 ++ 1 file changed, 2 insertions(+) diff --git a/CHANGELOG.md b/CHANGELOG.md index ffef70a57..8c590aacc 100644 --- a/CHANGELOG.md +++ b/CHANGELOG.md @@ -28,6 +28,8 @@ This release adds the following new features: - [List menu / More / Delete duplicate lists that do not have any cards](https://github.com/wekan/wekan/commit/91b846e2cdee9154b045d11b4b4c1a7ae1d79016). Thanks to xet7. +- [Disabled migrations that happen when opening board. Defaulting to per-swimlane lists and drag drop list to same or different swimlane](https://github.com/wekan/wekan/commit/034dc08269520ca31c780cce64e0150969e9228e). + Thanks to xet7. Thanks to above GitHub users for their contributions and translators for their translations. From ecf2418347cae4329deb292b534f68eb099d3f90 Mon Sep 17 00:00:00 2001 From: Lauri Ojansivu Date: Sat, 25 Oct 2025 19:23:35 +0300 Subject: [PATCH 12/84] Fix changing swimlane color to not reload webpage. Thanks to xet7 ! --- client/components/swimlanes/swimlaneHeader.js | 5 +++++ 1 file changed, 5 insertions(+) diff --git a/client/components/swimlanes/swimlaneHeader.js b/client/components/swimlanes/swimlaneHeader.js index 17988f454..c0ef35453 100644 --- a/client/components/swimlanes/swimlaneHeader.js +++ b/client/components/swimlanes/swimlaneHeader.js @@ -178,6 +178,11 @@ BlazeComponent.extendComponent({ events() { return [ { + 'submit form'(event) { + event.preventDefault(); + this.currentSwimlane.setColor(this.currentColor.get()); + Popup.back(); + }, 'click .js-palette-color'() { this.currentColor.set(this.currentData().color); }, From bccc22c5fedd5374f8fe78195c289c958e4a6171 Mon Sep 17 00:00:00 2001 From: Lauri Ojansivu Date: Sat, 25 Oct 2025 19:25:04 +0300 Subject: [PATCH 13/84] Updated ChangeLog. --- CHANGELOG.md | 5 +++++ 1 file changed, 5 insertions(+) diff --git a/CHANGELOG.md b/CHANGELOG.md index 8c590aacc..a8fe3555f 100644 --- a/CHANGELOG.md +++ b/CHANGELOG.md @@ -31,6 +31,11 @@ This release adds the following new features: - [Disabled migrations that happen when opening board. Defaulting to per-swimlane lists and drag drop list to same or different swimlane](https://github.com/wekan/wekan/commit/034dc08269520ca31c780cce64e0150969e9228e). Thanks to xet7. +and fixes the following bugs: + +- [Fix changing swimlane color to not reload webpage](https://github.com/wekan/wekan/commit/ecf2418347cae4329deb292b534f68eb099d3f90). + Thanks to xet7. + Thanks to above GitHub users for their contributions and translators for their translations. # v8.15 2025-10-23 WeKan ® release From 30620d0ca4b750582429cba18d8c676b2191e57a Mon Sep 17 00:00:00 2001 From: Lauri Ojansivu Date: Sat, 25 Oct 2025 21:09:07 +0300 Subject: [PATCH 14/84] Some migrations and mobile fixes. Thanks to xet7 ! --- client/components/boards/boardBody.css | 46 +- client/components/boards/boardBody.js | 202 +++-- client/components/boards/boardHeader.css | 126 +-- client/components/lists/list.css | 66 +- client/components/migrationProgress.css | 467 +++++------ client/components/migrationProgress.jade | 100 +-- client/components/migrationProgress.js | 234 +++++- client/components/swimlanes/swimlanes.css | 2 +- client/components/users/userAvatar.jade | 2 +- client/components/users/userAvatar.js | 4 +- models/attachments.js | 12 + models/avatars.js | 16 +- models/lib/universalUrlGenerator.js | 194 +++++ server/00checkStartup.js | 6 + server/cors.js | 15 + .../migrations/comprehensiveBoardMigration.js | 767 ++++++++++++++++++ server/migrations/fixAllFileUrls.js | 277 +++++++ server/migrations/fixAvatarUrls.js | 128 +++ server/routes/avatarServer.js | 123 +++ server/routes/universalFileServer.js | 393 +++++++++ 20 files changed, 2638 insertions(+), 542 deletions(-) create mode 100644 models/lib/universalUrlGenerator.js create mode 100644 server/migrations/comprehensiveBoardMigration.js create mode 100644 server/migrations/fixAllFileUrls.js create mode 100644 server/migrations/fixAvatarUrls.js create mode 100644 server/routes/avatarServer.js create mode 100644 server/routes/universalFileServer.js diff --git a/client/components/boards/boardBody.css b/client/components/boards/boardBody.css index 05fa8fc58..f65cbaffc 100644 --- a/client/components/boards/boardBody.css +++ b/client/components/boards/boardBody.css @@ -269,57 +269,71 @@ } /* Mobile view styles - applied when isMiniScreen is true (iPhone, etc.) */ .board-wrapper.mobile-view { - width: 100% !important; - min-width: 100% !important; + width: 100vw !important; + max-width: 100vw !important; + min-width: 100vw !important; left: 0 !important; right: 0 !important; + overflow-x: hidden !important; + overflow-y: auto !important; } .board-wrapper.mobile-view .board-canvas { - width: 100% !important; - min-width: 100% !important; + width: 100vw !important; + max-width: 100vw !important; + min-width: 100vw !important; left: 0 !important; right: 0 !important; + overflow-x: hidden !important; + overflow-y: auto !important; } .board-wrapper.mobile-view .board-canvas.mobile-view .swimlane { border-bottom: 1px solid #ccc; - display: flex; + display: block !important; flex-direction: column; margin: 0; padding: 0; - overflow-x: hidden; + overflow-x: hidden !important; overflow-y: auto; - width: 100%; - min-width: 100%; + width: 100vw !important; + max-width: 100vw !important; + min-width: 100vw !important; } @media screen and (max-width: 800px), screen and (max-device-width: 932px) and (-webkit-min-device-pixel-ratio: 3) { .board-wrapper { - width: 100% !important; - min-width: 100% !important; + width: 100vw !important; + max-width: 100vw !important; + min-width: 100vw !important; left: 0 !important; right: 0 !important; + overflow-x: hidden !important; + overflow-y: auto !important; } .board-wrapper .board-canvas { - width: 100% !important; - min-width: 100% !important; + width: 100vw !important; + max-width: 100vw !important; + min-width: 100vw !important; left: 0 !important; right: 0 !important; + overflow-x: hidden !important; + overflow-y: auto !important; } .board-wrapper .board-canvas .swimlane { border-bottom: 1px solid #ccc; - display: flex; + display: block !important; flex-direction: column; margin: 0; padding: 0; - overflow-x: hidden; + overflow-x: hidden !important; overflow-y: auto; - width: 100%; - min-width: 100%; + width: 100vw !important; + max-width: 100vw !important; + min-width: 100vw !important; } } .calendar-event-green { diff --git a/client/components/boards/boardBody.js b/client/components/boards/boardBody.js index a9b04cddb..e8e83a134 100644 --- a/client/components/boards/boardBody.js +++ b/client/components/boards/boardBody.js @@ -4,6 +4,7 @@ import dragscroll from '@wekanteam/dragscroll'; import { boardConverter } from '/client/lib/boardConverter'; import { migrationManager } from '/client/lib/migrationManager'; import { attachmentMigrationManager } from '/client/lib/attachmentMigrationManager'; +import { migrationProgressManager } from '/client/components/migrationProgress'; import Swimlanes from '/models/swimlanes'; import Lists from '/models/lists'; @@ -98,61 +99,25 @@ BlazeComponent.extendComponent({ return; } - // Check if board needs migration based on migration version - // DISABLED: Migration check and execution - // const needsMigration = !board.migrationVersion || board.migrationVersion < 1; + // Check if board needs comprehensive migration + const needsMigration = await this.checkComprehensiveMigration(boardId); - // if (needsMigration) { - // // Start background migration for old boards - // this.isMigrating.set(true); - // await this.startBackgroundMigration(boardId); - // this.isMigrating.set(false); - // } + if (needsMigration) { + // Start comprehensive migration + this.isMigrating.set(true); + const success = await this.executeComprehensiveMigration(boardId); + this.isMigrating.set(false); + + if (success) { + this.isBoardReady.set(true); + } else { + console.error('Comprehensive migration failed, setting ready to true anyway'); + this.isBoardReady.set(true); // Still show board even if migration failed + } + } else { + this.isBoardReady.set(true); + } - // Check if board needs conversion (for old structure) - // DISABLED: Board conversion logic - // if (boardConverter.isBoardConverted(boardId)) { - // if (process.env.DEBUG === 'true') { - // console.log(`Board ${boardId} has already been converted, skipping conversion`); - // } - // this.isBoardReady.set(true); - // } else { - // const needsConversion = boardConverter.needsConversion(boardId); - // - // if (needsConversion) { - // this.isConverting.set(true); - // const success = await boardConverter.convertBoard(boardId); - // this.isConverting.set(false); - // - // if (success) { - // this.isBoardReady.set(true); - // } else { - // console.error('Board conversion failed, setting ready to true anyway'); - // this.isBoardReady.set(true); // Still show board even if conversion failed - // } - // } else { - // this.isBoardReady.set(true); - // } - // } - - // Set board ready immediately since conversions are disabled - this.isBoardReady.set(true); - - // Convert shared lists to per-swimlane lists if needed - // DISABLED: Shared lists conversion - // await this.convertSharedListsToPerSwimlane(boardId); - - // Fix missing lists migration (for cards with wrong listId references) - // DISABLED: Missing lists fix - // await this.fixMissingLists(boardId); - - // Fix duplicate lists created by WeKan 8.10 - // DISABLED: Duplicate lists fix - // await this.fixDuplicateLists(boardId); - - // Start attachment migration in background if needed - // DISABLED: Attachment migration - // this.startAttachmentMigrationIfNeeded(boardId); } catch (error) { console.error('Error during board conversion check:', error); this.isConverting.set(false); @@ -161,6 +126,137 @@ BlazeComponent.extendComponent({ } }, + /** + * Check if board needs comprehensive migration + */ + async checkComprehensiveMigration(boardId) { + try { + return new Promise((resolve, reject) => { + Meteor.call('comprehensiveBoardMigration.needsMigration', boardId, (error, result) => { + if (error) { + console.error('Error checking comprehensive migration:', error); + reject(error); + } else { + resolve(result); + } + }); + }); + } catch (error) { + console.error('Error checking comprehensive migration:', error); + return false; + } + }, + + /** + * Execute comprehensive migration for a board + */ + async executeComprehensiveMigration(boardId) { + try { + // Start progress tracking + migrationProgressManager.startMigration(); + + // Simulate progress updates since we can't easily pass callbacks through Meteor methods + const progressSteps = [ + { step: 'analyze_board_structure', name: 'Analyze Board Structure', duration: 1000 }, + { step: 'fix_orphaned_cards', name: 'Fix Orphaned Cards', duration: 2000 }, + { step: 'convert_shared_lists', name: 'Convert Shared Lists', duration: 3000 }, + { step: 'ensure_per_swimlane_lists', name: 'Ensure Per-Swimlane Lists', duration: 1500 }, + { step: 'cleanup_empty_lists', name: 'Cleanup Empty Lists', duration: 1000 }, + { step: 'validate_migration', name: 'Validate Migration', duration: 1000 }, + { step: 'fix_avatar_urls', name: 'Fix Avatar URLs', duration: 1000 }, + { step: 'fix_attachment_urls', name: 'Fix Attachment URLs', duration: 1000 } + ]; + + // Start the actual migration + const migrationPromise = new Promise((resolve, reject) => { + Meteor.call('comprehensiveBoardMigration.execute', boardId, (error, result) => { + if (error) { + console.error('Error executing comprehensive migration:', error); + migrationProgressManager.failMigration(error); + reject(error); + } else { + if (process.env.DEBUG === 'true') { + console.log('Comprehensive migration completed for board:', boardId, result); + } + resolve(result.success); + } + }); + }); + + // Simulate progress updates + const progressPromise = this.simulateMigrationProgress(progressSteps); + + // Wait for both to complete + const [migrationResult] = await Promise.all([migrationPromise, progressPromise]); + + migrationProgressManager.completeMigration(); + return migrationResult; + + } catch (error) { + console.error('Error executing comprehensive migration:', error); + migrationProgressManager.failMigration(error); + return false; + } + }, + + /** + * Simulate migration progress updates + */ + async simulateMigrationProgress(progressSteps) { + const totalSteps = progressSteps.length; + + for (let i = 0; i < progressSteps.length; i++) { + const step = progressSteps[i]; + const stepProgress = Math.round(((i + 1) / totalSteps) * 100); + + // Update progress for this step + migrationProgressManager.updateProgress({ + overallProgress: stepProgress, + currentStep: i + 1, + totalSteps, + stepName: step.step, + stepProgress: 0, + stepStatus: `Starting ${step.name}...`, + stepDetails: null, + boardId: Session.get('currentBoard') + }); + + // Simulate step progress + const stepDuration = step.duration; + const updateInterval = 100; // Update every 100ms + const totalUpdates = stepDuration / updateInterval; + + for (let j = 0; j < totalUpdates; j++) { + const stepStepProgress = Math.round(((j + 1) / totalUpdates) * 100); + + migrationProgressManager.updateProgress({ + overallProgress: stepProgress, + currentStep: i + 1, + totalSteps, + stepName: step.step, + stepProgress: stepStepProgress, + stepStatus: `Processing ${step.name}...`, + stepDetails: { progress: `${stepStepProgress}%` }, + boardId: Session.get('currentBoard') + }); + + await new Promise(resolve => setTimeout(resolve, updateInterval)); + } + + // Complete the step + migrationProgressManager.updateProgress({ + overallProgress: stepProgress, + currentStep: i + 1, + totalSteps, + stepName: step.step, + stepProgress: 100, + stepStatus: `${step.name} completed`, + stepDetails: { status: 'completed' }, + boardId: Session.get('currentBoard') + }); + } + }, + async startBackgroundMigration(boardId) { try { // Start background migration using the cron system diff --git a/client/components/boards/boardHeader.css b/client/components/boards/boardHeader.css index f3cb652e7..faf20e2f5 100644 --- a/client/components/boards/boardHeader.css +++ b/client/components/boards/boardHeader.css @@ -505,73 +505,73 @@ flex-wrap: nowrap !important; align-items: stretch !important; justify-content: flex-start !important; - width: 100% !important; - max-width: 100% !important; - min-width: 100% !important; + width: 100vw !important; + max-width: 100vw !important; + min-width: 100vw !important; overflow-x: hidden !important; overflow-y: auto !important; } -.mobile-mode .swimlane { - display: block !important; - width: 100% !important; - max-width: 100% !important; - min-width: 100% !important; - margin: 0 0 2rem 0 !important; - padding: 0 !important; - float: none !important; - clear: both !important; -} + .mobile-mode .swimlane { + display: block !important; + width: 100vw !important; + max-width: 100vw !important; + min-width: 100vw !important; + margin: 0 0 2rem 0 !important; + padding: 0 !important; + float: none !important; + clear: both !important; + } -.mobile-mode .swimlane .swimlane-header { - display: block !important; - width: 100% !important; - max-width: 100% !important; - min-width: 100% !important; - margin: 0 0 1rem 0 !important; - padding: 1rem !important; - font-size: clamp(18px, 2.5vw, 32px) !important; - font-weight: bold !important; - border-bottom: 2px solid #ccc !important; -} + .mobile-mode .swimlane .swimlane-header { + display: block !important; + width: 100vw !important; + max-width: 100vw !important; + min-width: 100vw !important; + margin: 0 0 1rem 0 !important; + padding: 1rem !important; + font-size: clamp(18px, 2.5vw, 32px) !important; + font-weight: bold !important; + border-bottom: 2px solid #ccc !important; + } -.mobile-mode .swimlane .lists { - display: block !important; - width: 100% !important; - max-width: 100% !important; - min-width: 100% !important; - margin: 0 !important; - padding: 0 !important; - flex-direction: column !important; - flex-wrap: nowrap !important; - align-items: stretch !important; - justify-content: flex-start !important; -} + .mobile-mode .swimlane .lists { + display: block !important; + width: 100vw !important; + max-width: 100vw !important; + min-width: 100vw !important; + margin: 0 !important; + padding: 0 !important; + flex-direction: column !important; + flex-wrap: nowrap !important; + align-items: stretch !important; + justify-content: flex-start !important; + } -.mobile-mode .list { - display: block !important; - width: 100% !important; - max-width: 100% !important; - min-width: 100% !important; - margin: 0 0 2rem 0 !important; - padding: 0 !important; - float: none !important; - clear: both !important; - border-left: none !important; - border-right: none !important; - border-top: none !important; - border-bottom: 2px solid #ccc !important; - flex: none !important; - flex-basis: auto !important; - flex-grow: 0 !important; - flex-shrink: 0 !important; - position: static !important; - left: auto !important; - right: auto !important; - top: auto !important; - bottom: auto !important; - transform: none !important; -} + .mobile-mode .list { + display: block !important; + width: 100vw !important; + max-width: 100vw !important; + min-width: 100vw !important; + margin: 0 0 2rem 0 !important; + padding: 0 !important; + float: none !important; + clear: both !important; + border-left: none !important; + border-right: none !important; + border-top: none !important; + border-bottom: 2px solid #ccc !important; + flex: none !important; + flex-basis: auto !important; + flex-grow: 0 !important; + flex-shrink: 0 !important; + position: static !important; + left: auto !important; + right: auto !important; + top: auto !important; + bottom: auto !important; + transform: none !important; + } .mobile-mode .list:first-child { margin-left: 0 !important; @@ -667,9 +667,9 @@ flex-wrap: nowrap !important; align-items: stretch !important; justify-content: flex-start !important; - width: 100% !important; - max-width: 100% !important; - min-width: 100% !important; + width: 100vw !important; + max-width: 100vw !important; + min-width: 100vw !important; overflow-x: hidden !important; overflow-y: auto !important; } diff --git a/client/components/lists/list.css b/client/components/lists/list.css index 53426199b..77e78de29 100644 --- a/client/components/lists/list.css +++ b/client/components/lists/list.css @@ -641,17 +641,22 @@ body.list-resizing-active * { .mini-list.mobile-view { flex: 0 0 60px; height: auto; - width: 100%; - min-width: 100%; + width: 100vw; + max-width: 100vw; + min-width: 100vw; border-left: 0px !important; border-bottom: 1px solid #ccc; + display: block !important; } .list.mobile-view { - display: contents; + display: block !important; flex-basis: auto; - width: 100%; - min-width: 100%; + width: 100vw; + max-width: 100vw; + min-width: 100vw; border-left: 0px !important; + margin: 0 !important; + padding: 0 !important; } .list.mobile-view:first-child { margin-left: 0px; @@ -659,9 +664,11 @@ body.list-resizing-active * { .list.mobile-view.ui-sortable-helper { flex: 0 0 60px; height: 60px; - width: 100%; + width: 100vw; + max-width: 100vw; border-left: 0px !important; border-bottom: 1px solid #ccc; + display: block !important; } .list.mobile-view.ui-sortable-helper .list-header.ui-sortable-handle { cursor: grabbing; @@ -669,14 +676,17 @@ body.list-resizing-active * { .list.mobile-view.placeholder { flex: 0 0 60px; height: 60px; - width: 100%; + width: 100vw; + max-width: 100vw; border-left: 0px !important; border-bottom: 1px solid #ccc; + display: block !important; } .list.mobile-view .list-body { padding: 15px 19px; - width: 100%; - min-width: 100%; + width: 100vw; + max-width: 100vw; + min-width: 100vw; } .list.mobile-view .list-header { /*Updated padding values for mobile devices, this should fix text grouping issue*/ @@ -685,8 +695,9 @@ body.list-resizing-active * { min-height: 30px; margin-top: 10px; align-items: center; - width: 100%; - min-width: 100%; + width: 100vw; + max-width: 100vw; + min-width: 100vw; /* Force grid layout for iPhone */ display: grid !important; grid-template-columns: 30px 1fr auto auto !important; @@ -767,17 +778,22 @@ body.list-resizing-active * { .mini-list { flex: 0 0 60px; height: auto; - width: 100%; - min-width: 100%; + width: 100vw; + max-width: 100vw; + min-width: 100vw; border-left: 0px !important; border-bottom: 1px solid #ccc; + display: block !important; } .list { - display: contents; + display: block !important; flex-basis: auto; - width: 100%; - min-width: 100%; + width: 100vw; + max-width: 100vw; + min-width: 100vw; border-left: 0px !important; + margin: 0 !important; + padding: 0 !important; } .list:first-child { margin-left: 0px; @@ -785,9 +801,11 @@ body.list-resizing-active * { .list.ui-sortable-helper { flex: 0 0 60px; height: 60px; - width: 100%; + width: 100vw; + max-width: 100vw; border-left: 0px !important; border-bottom: 1px solid #ccc; + display: block !important; } .list.ui-sortable-helper .list-header.ui-sortable-handle { cursor: grabbing; @@ -795,14 +813,17 @@ body.list-resizing-active * { .list.placeholder { flex: 0 0 60px; height: 60px; - width: 100%; + width: 100vw; + max-width: 100vw; border-left: 0px !important; border-bottom: 1px solid #ccc; + display: block !important; } .list-body { padding: 15px 19px; - width: 100%; - min-width: 100%; + width: 100vw; + max-width: 100vw; + min-width: 100vw; } .list-header { /*Updated padding values for mobile devices, this should fix text grouping issue*/ @@ -811,8 +832,9 @@ body.list-resizing-active * { min-height: 30px; margin-top: 10px; align-items: center; - width: 100%; - min-width: 100%; + width: 100vw; + max-width: 100vw; + min-width: 100vw; } .list-header .list-header-left-icon { padding: 7px; diff --git a/client/components/migrationProgress.css b/client/components/migrationProgress.css index d44f4eda8..f3b9a45d4 100644 --- a/client/components/migrationProgress.css +++ b/client/components/migrationProgress.css @@ -1,38 +1,33 @@ /* Migration Progress Styles */ -.migration-overlay { +.migration-progress-overlay { position: fixed; top: 0; left: 0; - width: 100%; - height: 100%; - background-color: rgba(0, 0, 0, 0.8); - z-index: 10000; - display: none; + right: 0; + bottom: 0; + background: rgba(0, 0, 0, 0.7); + z-index: 9999; + display: flex; align-items: center; justify-content: center; - overflow-y: auto; + backdrop-filter: blur(2px); } -.migration-overlay.active { - display: flex; -} - -.migration-modal { +.migration-progress-modal { background: white; - border-radius: 12px; - box-shadow: 0 20px 60px rgba(0, 0, 0, 0.4); - max-width: 800px; - width: 95%; - max-height: 90vh; + border-radius: 8px; + box-shadow: 0 10px 30px rgba(0, 0, 0, 0.3); + max-width: 500px; + width: 90%; + max-height: 80vh; overflow: hidden; - animation: slideInScale 0.4s ease-out; - margin: 20px; + animation: migrationModalSlideIn 0.3s ease-out; } -@keyframes slideInScale { +@keyframes migrationModalSlideIn { from { opacity: 0; - transform: translateY(-30px) scale(0.95); + transform: translateY(-20px) scale(0.95); } to { opacity: 1; @@ -40,333 +35,235 @@ } } -.migration-header { - padding: 24px 32px 20px; - border-bottom: 2px solid #e0e0e0; - text-align: center; +.migration-progress-header { background: linear-gradient(135deg, #667eea 0%, #764ba2 100%); color: white; + padding: 20px; + display: flex; + justify-content: space-between; + align-items: center; } -.migration-header h3 { - margin: 0 0 8px 0; - font-size: 24px; +.migration-progress-title { + margin: 0; + font-size: 18px; font-weight: 600; } -.migration-header h3 i { - margin-right: 12px; - color: #FFD700; -} - -.migration-header p { - margin: 0; +.migration-progress-close { + cursor: pointer; font-size: 16px; - opacity: 0.9; + opacity: 0.8; + transition: opacity 0.2s ease; } -.migration-content { - padding: 24px 32px; - max-height: 60vh; - overflow-y: auto; +.migration-progress-close:hover { + opacity: 1; } -.migration-overview { - margin-bottom: 32px; - padding: 20px; - background: #f8f9fa; - border-radius: 8px; - border-left: 4px solid #667eea; +.migration-progress-content { + padding: 30px; } -.overall-progress { - margin-bottom: 20px; +.migration-progress-overall { + margin-bottom: 25px; } -.progress-bar { - width: 100%; - height: 12px; - background-color: #e0e0e0; - border-radius: 6px; - overflow: hidden; +.migration-progress-overall-label { + font-weight: 600; + color: #333; margin-bottom: 8px; - position: relative; + font-size: 14px; } -.progress-fill { +.migration-progress-overall-bar { + background: #e9ecef; + border-radius: 10px; + height: 12px; + overflow: hidden; + margin-bottom: 5px; +} + +.migration-progress-overall-fill { + background: linear-gradient(90deg, #28a745, #20c997); height: 100%; - background: linear-gradient(90deg, #667eea, #764ba2); - border-radius: 6px; + border-radius: 10px; transition: width 0.3s ease; position: relative; } -.progress-fill::after { +.migration-progress-overall-fill::after { content: ''; position: absolute; top: 0; left: 0; right: 0; bottom: 0; - background: linear-gradient( - 90deg, - transparent, - rgba(255, 255, 255, 0.4), - transparent - ); - animation: shimmer 2s infinite; + background: linear-gradient(90deg, transparent, rgba(255,255,255,0.3), transparent); + animation: migrationProgressShimmer 2s infinite; } -@keyframes shimmer { - 0% { - transform: translateX(-100%); - } - 100% { - transform: translateX(100%); - } +@keyframes migrationProgressShimmer { + 0% { transform: translateX(-100%); } + 100% { transform: translateX(100%); } } -.progress-text { - text-align: center; - font-weight: 700; - color: #667eea; - font-size: 18px; -} - -.progress-label { - text-align: center; - color: #666; - font-size: 14px; - margin-top: 4px; -} - -.current-step { - text-align: center; - color: #333; - font-size: 16px; - font-weight: 500; - margin-bottom: 16px; -} - -.current-step i { - margin-right: 8px; - color: #667eea; -} - -.estimated-time { - text-align: center; - color: #666; - font-size: 14px; - background-color: #fff3cd; - padding: 8px 12px; - border-radius: 4px; - border: 1px solid #ffeaa7; -} - -.estimated-time i { - margin-right: 6px; - color: #f39c12; -} - -.migration-steps { - margin-bottom: 24px; -} - -.migration-steps h4 { - margin: 0 0 16px 0; - color: #333; - font-size: 18px; - font-weight: 600; -} - -.steps-list { - max-height: 300px; - overflow-y: auto; - border: 1px solid #e0e0e0; - border-radius: 8px; -} - -.migration-step { - padding: 16px 20px; - border-bottom: 1px solid #f0f0f0; - transition: all 0.3s ease; -} - -.migration-step:last-child { - border-bottom: none; -} - -.migration-step.completed { - background-color: #d4edda; - border-left: 4px solid #28a745; -} - -.migration-step.current { - background-color: #cce7ff; - border-left: 4px solid #667eea; - animation: pulse 2s infinite; -} - -@keyframes pulse { - 0% { - box-shadow: 0 0 0 0 rgba(102, 126, 234, 0.4); - } - 70% { - box-shadow: 0 0 0 10px rgba(102, 126, 234, 0); - } - 100% { - box-shadow: 0 0 0 0 rgba(102, 126, 234, 0); - } -} - -.step-header { - display: flex; - align-items: center; - margin-bottom: 8px; -} - -.step-icon { - margin-right: 12px; - font-size: 18px; - width: 24px; - text-align: center; -} - -.step-icon i.fa-check-circle { - color: #28a745; -} - -.step-icon i.fa-cog.fa-spin { - color: #667eea; -} - -.step-icon i.fa-circle-o { - color: #ccc; -} - -.step-info { - flex: 1; -} - -.step-name { - font-weight: 600; - color: #333; - font-size: 14px; - margin-bottom: 2px; -} - -.step-description { - color: #666; - font-size: 12px; - line-height: 1.3; -} - -.step-progress { +.migration-progress-overall-percentage { text-align: right; - min-width: 40px; -} - -.step-progress .progress-text { font-size: 12px; + color: #666; font-weight: 600; } -.step-progress-bar { - width: 100%; - height: 4px; - background-color: #e0e0e0; - border-radius: 2px; - overflow: hidden; - margin-top: 8px; +.migration-progress-current-step { + margin-bottom: 25px; } -.step-progress-bar .progress-fill { +.migration-progress-step-label { + font-weight: 600; + color: #333; + margin-bottom: 8px; + font-size: 14px; +} + +.migration-progress-step-bar { + background: #e9ecef; + border-radius: 8px; + height: 8px; + overflow: hidden; + margin-bottom: 5px; +} + +.migration-progress-step-fill { + background: linear-gradient(90deg, #007bff, #0056b3); height: 100%; - background: linear-gradient(90deg, #667eea, #764ba2); - border-radius: 2px; + border-radius: 8px; transition: width 0.3s ease; } -.migration-status { - text-align: center; - color: #333; - font-size: 16px; - background-color: #e3f2fd; - padding: 12px 16px; +.migration-progress-step-percentage { + text-align: right; + font-size: 12px; + color: #666; + font-weight: 600; +} + +.migration-progress-status { + margin-bottom: 20px; + padding: 15px; + background: #f8f9fa; border-radius: 6px; - border: 1px solid #bbdefb; - margin-bottom: 16px; + border-left: 4px solid #007bff; } -.migration-status i { - margin-right: 8px; - color: #2196f3; +.migration-progress-status-label { + font-weight: 600; + color: #333; + margin-bottom: 5px; + font-size: 13px; } -.migration-footer { - padding: 16px 32px 24px; - border-top: 1px solid #e0e0e0; - background-color: #f8f9fa; +.migration-progress-status-text { + color: #555; + font-size: 14px; + line-height: 1.4; } -.migration-info { +.migration-progress-details { + margin-bottom: 20px; + padding: 12px; + background: #e3f2fd; + border-radius: 6px; + border-left: 4px solid #2196f3; +} + +.migration-progress-details-label { + font-weight: 600; + color: #1976d2; + margin-bottom: 5px; + font-size: 13px; +} + +.migration-progress-details-text { + color: #1565c0; + font-size: 13px; + line-height: 1.4; +} + +.migration-progress-footer { + padding: 20px 30px; + background: #f8f9fa; + border-top: 1px solid #e9ecef; +} + +.migration-progress-note { text-align: center; color: #666; font-size: 13px; - line-height: 1.4; - margin-bottom: 8px; -} - -.migration-info i { - margin-right: 6px; - color: #667eea; -} - -.migration-warning { - text-align: center; - color: #856404; - font-size: 12px; - line-height: 1.3; - background-color: #fff3cd; - padding: 8px 12px; - border-radius: 4px; - border: 1px solid #ffeaa7; -} - -.migration-warning i { - margin-right: 6px; - color: #f39c12; + font-style: italic; } /* Responsive design */ -@media (max-width: 768px) { - .migration-modal { - width: 98%; - margin: 10px; +@media (max-width: 600px) { + .migration-progress-modal { + width: 95%; + margin: 20px; } - .migration-header, - .migration-content, - .migration-footer { - padding-left: 16px; - padding-right: 16px; + .migration-progress-content { + padding: 20px; } - .migration-header h3 { - font-size: 20px; + .migration-progress-header { + padding: 15px; } - .step-header { - flex-direction: column; - align-items: flex-start; - } - - .step-progress { - text-align: left; - margin-top: 8px; - } - - .steps-list { - max-height: 200px; + .migration-progress-title { + font-size: 16px; } } + +/* Dark mode support */ +@media (prefers-color-scheme: dark) { + .migration-progress-modal { + background: #2d3748; + color: #e2e8f0; + } + + .migration-progress-overall-label, + .migration-progress-step-label, + .migration-progress-status-label { + color: #e2e8f0; + } + + .migration-progress-status { + background: #4a5568; + border-left-color: #63b3ed; + } + + .migration-progress-status-text { + color: #cbd5e0; + } + + .migration-progress-details { + background: #2b6cb0; + border-left-color: #4299e1; + } + + .migration-progress-details-label { + color: #bee3f8; + } + + .migration-progress-details-text { + color: #90cdf4; + } + + .migration-progress-footer { + background: #4a5568; + border-top-color: #718096; + } + + .migration-progress-note { + color: #a0aec0; + } +} \ No newline at end of file diff --git a/client/components/migrationProgress.jade b/client/components/migrationProgress.jade index 274ea4621..250e20920 100644 --- a/client/components/migrationProgress.jade +++ b/client/components/migrationProgress.jade @@ -1,63 +1,43 @@ template(name="migrationProgress") - .migration-overlay(class="{{#if isMigrating}}active{{/if}}") - .migration-modal - .migration-header - h3 - | 🗄️ - | {{_ 'database-migration'}} - p {{_ 'database-migration-description'}} - - .migration-content - .migration-overview - .overall-progress - .progress-bar - .progress-fill(style="width: {{migrationProgress}}%") - .progress-text {{migrationProgress}}% - .progress-label {{_ 'overall-progress'}} - - .current-step - | ⚙️ - | {{migrationCurrentStep}} - - .estimated-time(style="{{#unless migrationEstimatedTime}}display: none;{{/unless}}") - | ⏰ - | {{_ 'estimated-time-remaining'}}: {{migrationEstimatedTime}} + if isMigrating + .migration-progress-overlay + .migration-progress-modal + .migration-progress-header + h3.migration-progress-title + | 🔄 Board Migration in Progress + .migration-progress-close.js-close-migration-progress + | ❌ - .migration-steps - h4 {{_ 'migration-steps'}} - .steps-list - each migrationSteps - .migration-step(class="{{#if completed}}completed{{/if}}" class="{{#if isCurrentStep}}current{{/if}}") - .step-header - .step-icon - if completed - | ✅ - else if isCurrentStep - | ⚙️ - else - | ⭕ - .step-info - .step-name {{name}} - .step-description {{description}} - .step-progress - if completed - .progress-text 100% - else if isCurrentStep - .progress-text {{progress}}% - else - .progress-text 0% - if isCurrentStep - .step-progress-bar - .progress-fill(style="width: {{progress}}%") + .migration-progress-content + .migration-progress-overall + .migration-progress-overall-label + | Overall Progress: {{currentStep}} of {{totalSteps}} steps + .migration-progress-overall-bar + .migration-progress-overall-fill(style="{{progressBarStyle}}") + .migration-progress-overall-percentage + | {{overallProgress}}% + + .migration-progress-current-step + .migration-progress-step-label + | Current Step: {{stepNameFormatted}} + .migration-progress-step-bar + .migration-progress-step-fill(style="{{stepProgressBarStyle}}") + .migration-progress-step-percentage + | {{stepProgress}}% + + .migration-progress-status + .migration-progress-status-label + | Status: + .migration-progress-status-text + | {{stepStatus}} + + if stepDetailsFormatted + .migration-progress-details + .migration-progress-details-label + | Details: + .migration-progress-details-text + | {{stepDetailsFormatted}} - .migration-status - | ℹ️ - | {{migrationStatus}} - - .migration-footer - .migration-info - | 💡 - | {{_ 'migration-info-text'}} - .migration-warning - | ⚠️ - | {{_ 'migration-warning-text'}} + .migration-progress-footer + .migration-progress-note + | Please wait while we migrate your board to the latest structure... \ No newline at end of file diff --git a/client/components/migrationProgress.js b/client/components/migrationProgress.js index 83a05ea36..7c4064d39 100644 --- a/client/components/migrationProgress.js +++ b/client/components/migrationProgress.js @@ -1,54 +1,212 @@ -import { Template } from 'meteor/templating'; -import { - migrationManager, - isMigrating, - migrationProgress, - migrationStatus, - migrationCurrentStep, - migrationEstimatedTime, - migrationSteps -} from '/client/lib/migrationManager'; +/** + * Migration Progress Component + * Displays detailed progress for comprehensive board migration + */ +import { ReactiveVar } from 'meteor/reactive-var'; +import { ReactiveCache } from '/imports/reactiveCache'; + +// Reactive variables for migration progress +export const migrationProgress = new ReactiveVar(0); +export const migrationStatus = new ReactiveVar(''); +export const migrationStepName = new ReactiveVar(''); +export const migrationStepProgress = new ReactiveVar(0); +export const migrationStepStatus = new ReactiveVar(''); +export const migrationStepDetails = new ReactiveVar(null); +export const migrationCurrentStep = new ReactiveVar(0); +export const migrationTotalSteps = new ReactiveVar(0); +export const isMigrating = new ReactiveVar(false); + +class MigrationProgressManager { + constructor() { + this.progressHistory = []; + } + + /** + * Update migration progress + */ + updateProgress(progressData) { + const { + overallProgress, + currentStep, + totalSteps, + stepName, + stepProgress, + stepStatus, + stepDetails, + boardId + } = progressData; + + // Update reactive variables + migrationProgress.set(overallProgress); + migrationCurrentStep.set(currentStep); + migrationTotalSteps.set(totalSteps); + migrationStepName.set(stepName); + migrationStepProgress.set(stepProgress); + migrationStepStatus.set(stepStatus); + migrationStepDetails.set(stepDetails); + + // Store in history + this.progressHistory.push({ + timestamp: new Date(), + ...progressData + }); + + // Update overall status + migrationStatus.set(`${stepName}: ${stepStatus}`); + } + + /** + * Start migration + */ + startMigration() { + isMigrating.set(true); + migrationProgress.set(0); + migrationStatus.set('Starting migration...'); + migrationStepName.set(''); + migrationStepProgress.set(0); + migrationStepStatus.set(''); + migrationStepDetails.set(null); + migrationCurrentStep.set(0); + migrationTotalSteps.set(0); + this.progressHistory = []; + } + + /** + * Complete migration + */ + completeMigration() { + isMigrating.set(false); + migrationProgress.set(100); + migrationStatus.set('Migration completed successfully!'); + + // Clear step details after a delay + setTimeout(() => { + migrationStepName.set(''); + migrationStepProgress.set(0); + migrationStepStatus.set(''); + migrationStepDetails.set(null); + migrationCurrentStep.set(0); + migrationTotalSteps.set(0); + }, 3000); + } + + /** + * Fail migration + */ + failMigration(error) { + isMigrating.set(false); + migrationStatus.set(`Migration failed: ${error.message || error}`); + migrationStepStatus.set('Error occurred'); + } + + /** + * Get progress history + */ + getProgressHistory() { + return this.progressHistory; + } + + /** + * Clear progress + */ + clearProgress() { + isMigrating.set(false); + migrationProgress.set(0); + migrationStatus.set(''); + migrationStepName.set(''); + migrationStepProgress.set(0); + migrationStepStatus.set(''); + migrationStepDetails.set(null); + migrationCurrentStep.set(0); + migrationTotalSteps.set(0); + this.progressHistory = []; + } +} + +// Export singleton instance +export const migrationProgressManager = new MigrationProgressManager(); + +// Template helpers Template.migrationProgress.helpers({ isMigrating() { return isMigrating.get(); }, - - migrationProgress() { + + overallProgress() { return migrationProgress.get(); }, - - migrationStatus() { + + overallStatus() { return migrationStatus.get(); }, - - migrationCurrentStep() { + + currentStep() { return migrationCurrentStep.get(); }, - - migrationEstimatedTime() { - return migrationEstimatedTime.get(); + + totalSteps() { + return migrationTotalSteps.get(); }, - - migrationSteps() { - const steps = migrationSteps.get(); - const currentStep = migrationCurrentStep.get(); + + stepName() { + return migrationStepName.get(); + }, + + stepProgress() { + return migrationStepProgress.get(); + }, + + stepStatus() { + return migrationStepStatus.get(); + }, + + stepDetails() { + return migrationStepDetails.get(); + }, + + progressBarStyle() { + const progress = migrationProgress.get(); + return `width: ${progress}%`; + }, + + stepProgressBarStyle() { + const progress = migrationStepProgress.get(); + return `width: ${progress}%`; + }, + + stepNameFormatted() { + const stepName = migrationStepName.get(); + if (!stepName) return ''; - return steps.map(step => ({ - ...step, - isCurrentStep: step.name === currentStep - })); + // Convert snake_case to Title Case + return stepName + .split('_') + .map(word => word.charAt(0).toUpperCase() + word.slice(1)) + .join(' '); + }, + + stepDetailsFormatted() { + const details = migrationStepDetails.get(); + if (!details) return ''; + + const formatted = []; + for (const [key, value] of Object.entries(details)) { + const formattedKey = key + .split(/(?=[A-Z])/) + .join(' ') + .toLowerCase() + .replace(/^\w/, c => c.toUpperCase()); + formatted.push(`${formattedKey}: ${value}`); + } + + return formatted.join(', '); } }); -Template.migrationProgress.onCreated(function() { - // Subscribe to migration state changes - this.autorun(() => { - isMigrating.get(); - migrationProgress.get(); - migrationStatus.get(); - migrationCurrentStep.get(); - migrationEstimatedTime.get(); - migrationSteps.get(); - }); -}); +// Template events +Template.migrationProgress.events({ + 'click .js-close-migration-progress'() { + migrationProgressManager.clearProgress(); + } +}); \ No newline at end of file diff --git a/client/components/swimlanes/swimlanes.css b/client/components/swimlanes/swimlanes.css index 4c20cb0f4..83540549f 100644 --- a/client/components/swimlanes/swimlanes.css +++ b/client/components/swimlanes/swimlanes.css @@ -112,7 +112,7 @@ padding: 7px; top: 50%; transform: translateY(-50%); - left: 87vw; + right: 10px; font-size: 24px; cursor: move; z-index: 15; diff --git a/client/components/users/userAvatar.jade b/client/components/users/userAvatar.jade index b1bc7e2d4..b61eb5033 100644 --- a/client/components/users/userAvatar.jade +++ b/client/components/users/userAvatar.jade @@ -87,7 +87,7 @@ template(name="changeAvatarPopup") each uploadedAvatars li: a.js-select-avatar .member - img.avatar.avatar-image(src="{{link}}?auth=false&brokenIsFine=true") + img.avatar.avatar-image(src="{{link}}") | {{_ 'uploaded-avatar'}} if isSelected | ✅ diff --git a/client/components/users/userAvatar.js b/client/components/users/userAvatar.js index 98ebc901e..2869a9750 100644 --- a/client/components/users/userAvatar.js +++ b/client/components/users/userAvatar.js @@ -179,7 +179,7 @@ BlazeComponent.extendComponent({ isSelected() { const userProfile = ReactiveCache.getCurrentUser().profile; const avatarUrl = userProfile && userProfile.avatarUrl; - const currentAvatarUrl = `${this.currentData().link()}?auth=false&brokenIsFine=true`; + const currentAvatarUrl = this.currentData().link(); return avatarUrl === currentAvatarUrl; }, @@ -220,7 +220,7 @@ BlazeComponent.extendComponent({ } }, 'click .js-select-avatar'() { - const avatarUrl = `${this.currentData().link()}?auth=false&brokenIsFine=true`; + const avatarUrl = this.currentData().link(); this.setAvatar(avatarUrl); }, 'click .js-select-initials'() { diff --git a/models/attachments.js b/models/attachments.js index ac66c15c4..27d533e25 100644 --- a/models/attachments.js +++ b/models/attachments.js @@ -13,6 +13,7 @@ import FileStoreStrategyFactory, {moveToStorage, rename, STORAGE_NAME_FILESYSTEM // import { STORAGE_NAME_S3 } from '/models/lib/fileStoreStrategy'; import { getAttachmentWithBackwardCompatibility, getAttachmentsWithBackwardCompatibility } from './lib/attachmentBackwardCompatibility'; import AttachmentStorageSettings from './attachmentStorageSettings'; +import { generateUniversalAttachmentUrl, cleanFileUrl } from '/models/lib/universalUrlGenerator'; let attachmentUploadExternalProgram; let attachmentUploadMimeTypes = []; @@ -325,4 +326,15 @@ if (Meteor.isServer) { Attachments.getAttachmentWithBackwardCompatibility = getAttachmentWithBackwardCompatibility; Attachments.getAttachmentsWithBackwardCompatibility = getAttachmentsWithBackwardCompatibility; +// Override the link method to use universal URLs +if (Meteor.isClient) { + // Add custom link method to attachment documents + Attachments.collection.helpers({ + link(version = 'original') { + // Use universal URL generator for consistent, URL-agnostic URLs + return generateUniversalAttachmentUrl(this._id, version); + } + }); +} + export default Attachments; diff --git a/models/avatars.js b/models/avatars.js index 065728322..6ce904bcb 100644 --- a/models/avatars.js +++ b/models/avatars.js @@ -8,6 +8,7 @@ import { TAPi18n } from '/imports/i18n'; import fs from 'fs'; import path from 'path'; import FileStoreStrategyFactory, { FileStoreStrategyFilesystem, FileStoreStrategyGridFs, STORAGE_NAME_FILESYSTEM } from '/models/lib/fileStoreStrategy'; +import { generateUniversalAvatarUrl, cleanFileUrl } from '/models/lib/universalUrlGenerator'; const filesize = require('filesize'); @@ -116,7 +117,9 @@ Avatars = new FilesCollection({ const isValid = Promise.await(isFileValid(fileObj, avatarsUploadMimeTypes, avatarsUploadSize, avatarsUploadExternalProgram)); if (isValid) { - ReactiveCache.getUser(fileObj.userId).setAvatarUrl(`${formatFleURL(fileObj)}?auth=false&brokenIsFine=true`); + // Set avatar URL using universal URL generator (URL-agnostic) + const universalUrl = generateUniversalAvatarUrl(fileObj._id); + ReactiveCache.getUser(fileObj.userId).setAvatarUrl(universalUrl); } else { Avatars.remove(fileObj._id); } @@ -164,4 +167,15 @@ if (Meteor.isServer) { }); } +// Override the link method to use universal URLs +if (Meteor.isClient) { + // Add custom link method to avatar documents + Avatars.collection.helpers({ + link(version = 'original') { + // Use universal URL generator for consistent, URL-agnostic URLs + return generateUniversalAvatarUrl(this._id, version); + } + }); +} + export default Avatars; diff --git a/models/lib/universalUrlGenerator.js b/models/lib/universalUrlGenerator.js new file mode 100644 index 000000000..16a8d0030 --- /dev/null +++ b/models/lib/universalUrlGenerator.js @@ -0,0 +1,194 @@ +/** + * Universal URL Generator + * Generates file URLs that work regardless of ROOT_URL and PORT settings + * Ensures all attachments and avatars are always visible + */ + +import { Meteor } from 'meteor/meteor'; + +/** + * Generate a universal file URL that works regardless of ROOT_URL and PORT + * @param {string} fileId - The file ID + * @param {string} type - The file type ('attachment' or 'avatar') + * @param {string} version - The file version (default: 'original') + * @returns {string} - Universal file URL + */ +export function generateUniversalFileUrl(fileId, type, version = 'original') { + if (!fileId) { + return ''; + } + + // Always use relative URLs to avoid ROOT_URL and PORT dependencies + if (type === 'attachment') { + return `/cdn/storage/attachments/${fileId}`; + } else if (type === 'avatar') { + return `/cdn/storage/avatars/${fileId}`; + } + + return ''; +} + +/** + * Generate a universal attachment URL + * @param {string} attachmentId - The attachment ID + * @param {string} version - The file version (default: 'original') + * @returns {string} - Universal attachment URL + */ +export function generateUniversalAttachmentUrl(attachmentId, version = 'original') { + return generateUniversalFileUrl(attachmentId, 'attachment', version); +} + +/** + * Generate a universal avatar URL + * @param {string} avatarId - The avatar ID + * @param {string} version - The file version (default: 'original') + * @returns {string} - Universal avatar URL + */ +export function generateUniversalAvatarUrl(avatarId, version = 'original') { + return generateUniversalFileUrl(avatarId, 'avatar', version); +} + +/** + * Clean and normalize a file URL to ensure it's universal + * @param {string} url - The URL to clean + * @param {string} type - The file type ('attachment' or 'avatar') + * @returns {string} - Cleaned universal URL + */ +export function cleanFileUrl(url, type) { + if (!url) { + return ''; + } + + // Remove any domain, port, or protocol from the URL + let cleanUrl = url; + + // Remove protocol and domain + cleanUrl = cleanUrl.replace(/^https?:\/\/[^\/]+/, ''); + + // Remove ROOT_URL pathname if present + if (Meteor.isServer && process.env.ROOT_URL) { + try { + const rootUrl = new URL(process.env.ROOT_URL); + if (rootUrl.pathname && rootUrl.pathname !== '/') { + cleanUrl = cleanUrl.replace(rootUrl.pathname, ''); + } + } catch (e) { + // Ignore URL parsing errors + } + } + + // Normalize path separators + cleanUrl = cleanUrl.replace(/\/+/g, '/'); + + // Ensure URL starts with / + if (!cleanUrl.startsWith('/')) { + cleanUrl = '/' + cleanUrl; + } + + // Convert old CollectionFS URLs to new format + if (type === 'attachment') { + cleanUrl = cleanUrl.replace('/cfs/files/attachments/', '/cdn/storage/attachments/'); + } else if (type === 'avatar') { + cleanUrl = cleanUrl.replace('/cfs/files/avatars/', '/cdn/storage/avatars/'); + } + + // Remove any query parameters that might cause issues + cleanUrl = cleanUrl.split('?')[0]; + cleanUrl = cleanUrl.split('#')[0]; + + return cleanUrl; +} + +/** + * Check if a URL is a universal file URL + * @param {string} url - The URL to check + * @param {string} type - The file type ('attachment' or 'avatar') + * @returns {boolean} - True if it's a universal file URL + */ +export function isUniversalFileUrl(url, type) { + if (!url) { + return false; + } + + if (type === 'attachment') { + return url.includes('/cdn/storage/attachments/') || url.includes('/cfs/files/attachments/'); + } else if (type === 'avatar') { + return url.includes('/cdn/storage/avatars/') || url.includes('/cfs/files/avatars/'); + } + + return false; +} + +/** + * Extract file ID from a universal file URL + * @param {string} url - The URL to extract from + * @param {string} type - The file type ('attachment' or 'avatar') + * @returns {string|null} - The file ID or null if not found + */ +export function extractFileIdFromUrl(url, type) { + if (!url) { + return null; + } + + let pattern; + if (type === 'attachment') { + pattern = /\/(?:cdn\/storage\/attachments|cfs\/files\/attachments)\/([^\/\?#]+)/; + } else if (type === 'avatar') { + pattern = /\/(?:cdn\/storage\/avatars|cfs\/files\/avatars)\/([^\/\?#]+)/; + } else { + return null; + } + + const match = url.match(pattern); + return match ? match[1] : null; +} + +/** + * Generate a fallback URL for when the primary URL fails + * @param {string} fileId - The file ID + * @param {string} type - The file type ('attachment' or 'avatar') + * @returns {string} - Fallback URL + */ +export function generateFallbackUrl(fileId, type) { + if (!fileId) { + return ''; + } + + // Try alternative route patterns + if (type === 'attachment') { + return `/attachments/${fileId}`; + } else if (type === 'avatar') { + return `/avatars/${fileId}`; + } + + return ''; +} + +/** + * Get all possible URLs for a file (for redundancy) + * @param {string} fileId - The file ID + * @param {string} type - The file type ('attachment' or 'avatar') + * @returns {Array} - Array of possible URLs + */ +export function getAllPossibleUrls(fileId, type) { + if (!fileId) { + return []; + } + + const urls = []; + + // Primary URL + urls.push(generateUniversalFileUrl(fileId, type)); + + // Fallback URL + urls.push(generateFallbackUrl(fileId, type)); + + // Legacy URLs for backward compatibility + if (type === 'attachment') { + urls.push(`/cfs/files/attachments/${fileId}`); + } else if (type === 'avatar') { + urls.push(`/cfs/files/avatars/${fileId}`); + } + + return urls.filter(url => url); // Remove empty URLs +} diff --git a/server/00checkStartup.js b/server/00checkStartup.js index d7035dca8..ed4dbabb3 100644 --- a/server/00checkStartup.js +++ b/server/00checkStartup.js @@ -42,6 +42,12 @@ import './cronJobStorage'; // Import migrations import './migrations/fixMissingListsMigration'; +import './migrations/fixAvatarUrls'; +import './migrations/fixAllFileUrls'; +import './migrations/comprehensiveBoardMigration'; + +// Import file serving routes +import './routes/universalFileServer'; // Note: Automatic migrations are disabled - migrations only run when opening boards // import './boardMigrationDetector'; diff --git a/server/cors.js b/server/cors.js index 4badba9fe..f99258eae 100644 --- a/server/cors.js +++ b/server/cors.js @@ -1,4 +1,19 @@ Meteor.startup(() => { + // Set Permissions-Policy header to suppress browser warnings about experimental features + WebApp.rawConnectHandlers.use(function(req, res, next) { + // Disable experimental advertising and privacy features that cause browser warnings + res.setHeader('Permissions-Policy', + 'browsing-topics=(), ' + + 'run-ad-auction=(), ' + + 'join-ad-interest-group=(), ' + + 'private-state-token-redemption=(), ' + + 'private-state-token-issuance=(), ' + + 'private-aggregation=(), ' + + 'attribution-reporting=()' + ); + return next(); + }); + if (process.env.CORS) { // Listen to incoming HTTP requests, can only be used on the server WebApp.rawConnectHandlers.use(function(req, res, next) { diff --git a/server/migrations/comprehensiveBoardMigration.js b/server/migrations/comprehensiveBoardMigration.js new file mode 100644 index 000000000..f9ea7c523 --- /dev/null +++ b/server/migrations/comprehensiveBoardMigration.js @@ -0,0 +1,767 @@ +/** + * Comprehensive Board Migration System + * + * This migration handles all database structure changes from previous Wekan versions + * to the current per-swimlane lists structure. It ensures: + * + * 1. All cards are visible with proper swimlaneId and listId + * 2. Lists are per-swimlane (no shared lists across swimlanes) + * 3. No empty lists are created + * 4. Handles various database structure versions from git history + * + * Supported versions and their database structures: + * - v7.94 and earlier: Shared lists across all swimlanes + * - v8.00-v8.02: Transition period with mixed structures + * - v8.03+: Per-swimlane lists structure + */ + +import { Meteor } from 'meteor/meteor'; +import { check } from 'meteor/check'; +import { ReactiveCache } from '/imports/reactiveCache'; +import Boards from '/models/boards'; +import Lists from '/models/lists'; +import Cards from '/models/cards'; +import Swimlanes from '/models/swimlanes'; +import Attachments from '/models/attachments'; +import { generateUniversalAttachmentUrl, isUniversalFileUrl } from '/models/lib/universalUrlGenerator'; + +class ComprehensiveBoardMigration { + constructor() { + this.name = 'comprehensive-board-migration'; + this.version = 1; + this.migrationSteps = [ + 'analyze_board_structure', + 'fix_orphaned_cards', + 'convert_shared_lists', + 'ensure_per_swimlane_lists', + 'cleanup_empty_lists', + 'validate_migration' + ]; + } + + /** + * Check if migration is needed for a board + */ + needsMigration(boardId) { + try { + const board = ReactiveCache.getBoard(boardId); + if (!board) return false; + + // Check if board has already been processed + if (board.comprehensiveMigrationCompleted) { + return false; + } + + // Check for various issues that need migration + const issues = this.detectMigrationIssues(boardId); + return issues.length > 0; + + } catch (error) { + console.error('Error checking if migration is needed:', error); + return false; + } + } + + /** + * Detect all migration issues in a board + */ + detectMigrationIssues(boardId) { + const issues = []; + + try { + const cards = ReactiveCache.getCards({ boardId }); + const lists = ReactiveCache.getLists({ boardId }); + const swimlanes = ReactiveCache.getSwimlanes({ boardId }); + + // Issue 1: Cards with missing swimlaneId + const cardsWithoutSwimlane = cards.filter(card => !card.swimlaneId); + if (cardsWithoutSwimlane.length > 0) { + issues.push({ + type: 'cards_without_swimlane', + count: cardsWithoutSwimlane.length, + description: `${cardsWithoutSwimlane.length} cards missing swimlaneId` + }); + } + + // Issue 2: Cards with missing listId + const cardsWithoutList = cards.filter(card => !card.listId); + if (cardsWithoutList.length > 0) { + issues.push({ + type: 'cards_without_list', + count: cardsWithoutList.length, + description: `${cardsWithoutList.length} cards missing listId` + }); + } + + // Issue 3: Lists without swimlaneId (shared lists) + const sharedLists = lists.filter(list => !list.swimlaneId || list.swimlaneId === ''); + if (sharedLists.length > 0) { + issues.push({ + type: 'shared_lists', + count: sharedLists.length, + description: `${sharedLists.length} lists without swimlaneId (shared lists)` + }); + } + + // Issue 4: Cards with mismatched listId/swimlaneId + const listSwimlaneMap = new Map(); + lists.forEach(list => { + listSwimlaneMap.set(list._id, list.swimlaneId || ''); + }); + + const mismatchedCards = cards.filter(card => { + if (!card.listId || !card.swimlaneId) return false; + const listSwimlaneId = listSwimlaneMap.get(card.listId); + return listSwimlaneId && listSwimlaneId !== card.swimlaneId; + }); + + if (mismatchedCards.length > 0) { + issues.push({ + type: 'mismatched_cards', + count: mismatchedCards.length, + description: `${mismatchedCards.length} cards with mismatched listId/swimlaneId` + }); + } + + // Issue 5: Empty lists (lists with no cards) + const emptyLists = lists.filter(list => { + const listCards = cards.filter(card => card.listId === list._id); + return listCards.length === 0; + }); + + if (emptyLists.length > 0) { + issues.push({ + type: 'empty_lists', + count: emptyLists.length, + description: `${emptyLists.length} empty lists (no cards)` + }); + } + + } catch (error) { + console.error('Error detecting migration issues:', error); + issues.push({ + type: 'detection_error', + count: 1, + description: `Error detecting issues: ${error.message}` + }); + } + + return issues; + } + + /** + * Execute the comprehensive migration for a board + */ + async executeMigration(boardId, progressCallback = null) { + try { + if (process.env.DEBUG === 'true') { + console.log(`Starting comprehensive board migration for board ${boardId}`); + } + + const board = ReactiveCache.getBoard(boardId); + if (!board) { + throw new Error(`Board ${boardId} not found`); + } + + const results = { + boardId, + steps: {}, + totalCardsProcessed: 0, + totalListsProcessed: 0, + totalListsCreated: 0, + totalListsRemoved: 0, + errors: [] + }; + + const totalSteps = this.migrationSteps.length; + let currentStep = 0; + + // Helper function to update progress + const updateProgress = (stepName, stepProgress, stepStatus, stepDetails = null) => { + currentStep++; + const overallProgress = Math.round((currentStep / totalSteps) * 100); + + const progressData = { + overallProgress, + currentStep: currentStep, + totalSteps, + stepName, + stepProgress, + stepStatus, + stepDetails, + boardId + }; + + if (progressCallback) { + progressCallback(progressData); + } + + if (process.env.DEBUG === 'true') { + console.log(`Migration Progress: ${stepName} - ${stepStatus} (${stepProgress}%)`); + } + }; + + // Step 1: Analyze board structure + updateProgress('analyze_board_structure', 0, 'Starting analysis...'); + results.steps.analyze = await this.analyzeBoardStructure(boardId); + updateProgress('analyze_board_structure', 100, 'Analysis complete', { + issuesFound: results.steps.analyze.issueCount, + needsMigration: results.steps.analyze.needsMigration + }); + + // Step 2: Fix orphaned cards + updateProgress('fix_orphaned_cards', 0, 'Fixing orphaned cards...'); + results.steps.fixOrphanedCards = await this.fixOrphanedCards(boardId, (progress, status) => { + updateProgress('fix_orphaned_cards', progress, status); + }); + results.totalCardsProcessed += results.steps.fixOrphanedCards.cardsFixed || 0; + updateProgress('fix_orphaned_cards', 100, 'Orphaned cards fixed', { + cardsFixed: results.steps.fixOrphanedCards.cardsFixed + }); + + // Step 3: Convert shared lists to per-swimlane lists + updateProgress('convert_shared_lists', 0, 'Converting shared lists...'); + results.steps.convertSharedLists = await this.convertSharedListsToPerSwimlane(boardId, (progress, status) => { + updateProgress('convert_shared_lists', progress, status); + }); + results.totalListsProcessed += results.steps.convertSharedLists.listsProcessed || 0; + results.totalListsCreated += results.steps.convertSharedLists.listsCreated || 0; + updateProgress('convert_shared_lists', 100, 'Shared lists converted', { + listsProcessed: results.steps.convertSharedLists.listsProcessed, + listsCreated: results.steps.convertSharedLists.listsCreated + }); + + // Step 4: Ensure all lists are per-swimlane + updateProgress('ensure_per_swimlane_lists', 0, 'Ensuring per-swimlane structure...'); + results.steps.ensurePerSwimlane = await this.ensurePerSwimlaneLists(boardId); + results.totalListsProcessed += results.steps.ensurePerSwimlane.listsProcessed || 0; + updateProgress('ensure_per_swimlane_lists', 100, 'Per-swimlane structure ensured', { + listsProcessed: results.steps.ensurePerSwimlane.listsProcessed + }); + + // Step 5: Cleanup empty lists + updateProgress('cleanup_empty_lists', 0, 'Cleaning up empty lists...'); + results.steps.cleanupEmpty = await this.cleanupEmptyLists(boardId); + results.totalListsRemoved += results.steps.cleanupEmpty.listsRemoved || 0; + updateProgress('cleanup_empty_lists', 100, 'Empty lists cleaned up', { + listsRemoved: results.steps.cleanupEmpty.listsRemoved + }); + + // Step 6: Validate migration + updateProgress('validate_migration', 0, 'Validating migration...'); + results.steps.validate = await this.validateMigration(boardId); + updateProgress('validate_migration', 100, 'Migration validated', { + migrationSuccessful: results.steps.validate.migrationSuccessful, + totalCards: results.steps.validate.totalCards, + totalLists: results.steps.validate.totalLists + }); + + // Step 7: Fix avatar URLs + updateProgress('fix_avatar_urls', 0, 'Fixing avatar URLs...'); + results.steps.fixAvatarUrls = await this.fixAvatarUrls(boardId); + updateProgress('fix_avatar_urls', 100, 'Avatar URLs fixed', { + avatarsFixed: results.steps.fixAvatarUrls.avatarsFixed + }); + + // Step 8: Fix attachment URLs + updateProgress('fix_attachment_urls', 0, 'Fixing attachment URLs...'); + results.steps.fixAttachmentUrls = await this.fixAttachmentUrls(boardId); + updateProgress('fix_attachment_urls', 100, 'Attachment URLs fixed', { + attachmentsFixed: results.steps.fixAttachmentUrls.attachmentsFixed + }); + + // Mark board as processed + Boards.update(boardId, { + $set: { + comprehensiveMigrationCompleted: true, + comprehensiveMigrationCompletedAt: new Date(), + comprehensiveMigrationResults: results + } + }); + + if (process.env.DEBUG === 'true') { + console.log(`Comprehensive board migration completed for board ${boardId}:`, results); + } + + return { + success: true, + results + }; + + } catch (error) { + console.error(`Error executing comprehensive migration for board ${boardId}:`, error); + throw error; + } + } + + /** + * Step 1: Analyze board structure + */ + async analyzeBoardStructure(boardId) { + const issues = this.detectMigrationIssues(boardId); + return { + issues, + issueCount: issues.length, + needsMigration: issues.length > 0 + }; + } + + /** + * Step 2: Fix orphaned cards (cards with missing swimlaneId or listId) + */ + async fixOrphanedCards(boardId, progressCallback = null) { + const cards = ReactiveCache.getCards({ boardId }); + const swimlanes = ReactiveCache.getSwimlanes({ boardId }); + const lists = ReactiveCache.getLists({ boardId }); + + let cardsFixed = 0; + const defaultSwimlane = swimlanes.find(s => s.title === 'Default') || swimlanes[0]; + const totalCards = cards.length; + + for (let i = 0; i < cards.length; i++) { + const card = cards[i]; + let needsUpdate = false; + const updates = {}; + + // Fix missing swimlaneId + if (!card.swimlaneId) { + updates.swimlaneId = defaultSwimlane._id; + needsUpdate = true; + } + + // Fix missing listId + if (!card.listId) { + // Find or create a default list for this swimlane + const swimlaneId = updates.swimlaneId || card.swimlaneId; + let defaultList = lists.find(list => + list.swimlaneId === swimlaneId && list.title === 'Default' + ); + + if (!defaultList) { + // Create a default list for this swimlane + const newListId = Lists.insert({ + title: 'Default', + boardId: boardId, + swimlaneId: swimlaneId, + sort: 0, + archived: false, + createdAt: new Date(), + modifiedAt: new Date(), + type: 'list' + }); + defaultList = { _id: newListId }; + } + + updates.listId = defaultList._id; + needsUpdate = true; + } + + if (needsUpdate) { + Cards.update(card._id, { + $set: { + ...updates, + modifiedAt: new Date() + } + }); + cardsFixed++; + } + + // Update progress + if (progressCallback && (i % 10 === 0 || i === totalCards - 1)) { + const progress = Math.round(((i + 1) / totalCards) * 100); + progressCallback(progress, `Processing card ${i + 1} of ${totalCards}...`); + } + } + + return { cardsFixed }; + } + + /** + * Step 3: Convert shared lists to per-swimlane lists + */ + async convertSharedListsToPerSwimlane(boardId, progressCallback = null) { + const cards = ReactiveCache.getCards({ boardId }); + const lists = ReactiveCache.getLists({ boardId }); + const swimlanes = ReactiveCache.getSwimlanes({ boardId }); + + let listsProcessed = 0; + let listsCreated = 0; + + // Group cards by swimlaneId + const cardsBySwimlane = new Map(); + cards.forEach(card => { + if (!cardsBySwimlane.has(card.swimlaneId)) { + cardsBySwimlane.set(card.swimlaneId, []); + } + cardsBySwimlane.get(card.swimlaneId).push(card); + }); + + const swimlaneEntries = Array.from(cardsBySwimlane.entries()); + const totalSwimlanes = swimlaneEntries.length; + + // Process each swimlane + for (let i = 0; i < swimlaneEntries.length; i++) { + const [swimlaneId, swimlaneCards] = swimlaneEntries[i]; + if (!swimlaneId) continue; + + if (progressCallback) { + const progress = Math.round(((i + 1) / totalSwimlanes) * 100); + progressCallback(progress, `Processing swimlane ${i + 1} of ${totalSwimlanes}...`); + } + + // Get existing lists for this swimlane + const existingLists = lists.filter(list => list.swimlaneId === swimlaneId); + const existingListTitles = new Set(existingLists.map(list => list.title)); + + // Group cards by their current listId + const cardsByListId = new Map(); + swimlaneCards.forEach(card => { + if (!cardsByListId.has(card.listId)) { + cardsByListId.set(card.listId, []); + } + cardsByListId.get(card.listId).push(card); + }); + + // For each listId used by cards in this swimlane + for (const [listId, cardsInList] of cardsByListId) { + const originalList = lists.find(l => l._id === listId); + if (!originalList) continue; + + // Check if this list's swimlaneId matches the card's swimlaneId + if (originalList.swimlaneId === swimlaneId) { + // List is already correctly assigned to this swimlane + listsProcessed++; + continue; + } + + // Check if we already have a list with the same title in this swimlane + let targetList = existingLists.find(list => list.title === originalList.title); + + if (!targetList) { + // Create a new list for this swimlane + const newListData = { + title: originalList.title, + boardId: boardId, + swimlaneId: swimlaneId, + sort: originalList.sort || 0, + archived: originalList.archived || false, + createdAt: new Date(), + modifiedAt: new Date(), + type: originalList.type || 'list' + }; + + // Copy other properties if they exist + if (originalList.color) newListData.color = originalList.color; + if (originalList.wipLimit) newListData.wipLimit = originalList.wipLimit; + if (originalList.wipLimitEnabled) newListData.wipLimitEnabled = originalList.wipLimitEnabled; + if (originalList.wipLimitSoft) newListData.wipLimitSoft = originalList.wipLimitSoft; + if (originalList.starred) newListData.starred = originalList.starred; + if (originalList.collapsed) newListData.collapsed = originalList.collapsed; + + // Insert the new list + const newListId = Lists.insert(newListData); + targetList = { _id: newListId, ...newListData }; + listsCreated++; + } + + // Update all cards in this group to use the correct listId + for (const card of cardsInList) { + Cards.update(card._id, { + $set: { + listId: targetList._id, + modifiedAt: new Date() + } + }); + } + + listsProcessed++; + } + } + + return { listsProcessed, listsCreated }; + } + + /** + * Step 4: Ensure all lists are per-swimlane + */ + async ensurePerSwimlaneLists(boardId) { + const lists = ReactiveCache.getLists({ boardId }); + const swimlanes = ReactiveCache.getSwimlanes({ boardId }); + const defaultSwimlane = swimlanes.find(s => s.title === 'Default') || swimlanes[0]; + + let listsProcessed = 0; + + for (const list of lists) { + if (!list.swimlaneId || list.swimlaneId === '') { + // Assign to default swimlane + Lists.update(list._id, { + $set: { + swimlaneId: defaultSwimlane._id, + modifiedAt: new Date() + } + }); + listsProcessed++; + } + } + + return { listsProcessed }; + } + + /** + * Step 5: Cleanup empty lists (lists with no cards) + */ + async cleanupEmptyLists(boardId) { + const lists = ReactiveCache.getLists({ boardId }); + const cards = ReactiveCache.getCards({ boardId }); + + let listsRemoved = 0; + + for (const list of lists) { + const listCards = cards.filter(card => card.listId === list._id); + + if (listCards.length === 0) { + // Remove empty list + Lists.remove(list._id); + listsRemoved++; + + if (process.env.DEBUG === 'true') { + console.log(`Removed empty list: ${list.title} (${list._id})`); + } + } + } + + return { listsRemoved }; + } + + /** + * Step 6: Validate migration + */ + async validateMigration(boardId) { + const issues = this.detectMigrationIssues(boardId); + const cards = ReactiveCache.getCards({ boardId }); + const lists = ReactiveCache.getLists({ boardId }); + + // Check that all cards have valid swimlaneId and listId + const validCards = cards.filter(card => card.swimlaneId && card.listId); + const invalidCards = cards.length - validCards.length; + + // Check that all lists have swimlaneId + const validLists = lists.filter(list => list.swimlaneId && list.swimlaneId !== ''); + const invalidLists = lists.length - validLists.length; + + return { + issuesRemaining: issues.length, + totalCards: cards.length, + validCards, + invalidCards, + totalLists: lists.length, + validLists, + invalidLists, + migrationSuccessful: issues.length === 0 && invalidCards === 0 && invalidLists === 0 + }; + } + + /** + * Step 7: Fix avatar URLs (remove problematic auth parameters and fix URL formats) + */ + async fixAvatarUrls(boardId) { + const users = ReactiveCache.getUsers({}); + let avatarsFixed = 0; + + for (const user of users) { + if (user.profile && user.profile.avatarUrl) { + const avatarUrl = user.profile.avatarUrl; + let needsUpdate = false; + let cleanUrl = avatarUrl; + + // Check if URL has problematic parameters + if (avatarUrl.includes('auth=false') || avatarUrl.includes('brokenIsFine=true')) { + // Remove problematic parameters + cleanUrl = cleanUrl.replace(/[?&]auth=false/g, ''); + cleanUrl = cleanUrl.replace(/[?&]brokenIsFine=true/g, ''); + cleanUrl = cleanUrl.replace(/\?&/g, '?'); + cleanUrl = cleanUrl.replace(/\?$/g, ''); + needsUpdate = true; + } + + // Check if URL is using old CollectionFS format + if (avatarUrl.includes('/cfs/files/avatars/')) { + cleanUrl = cleanUrl.replace('/cfs/files/avatars/', '/cdn/storage/avatars/'); + needsUpdate = true; + } + + // Check if URL is missing the /cdn/storage/avatars/ prefix + if (avatarUrl.includes('avatars/') && !avatarUrl.includes('/cdn/storage/avatars/') && !avatarUrl.includes('/cfs/files/avatars/')) { + // This might be a relative URL, make it absolute + if (!avatarUrl.startsWith('http') && !avatarUrl.startsWith('/')) { + cleanUrl = `/cdn/storage/avatars/${avatarUrl}`; + needsUpdate = true; + } + } + + if (needsUpdate) { + // Update user's avatar URL + Users.update(user._id, { + $set: { + 'profile.avatarUrl': cleanUrl, + modifiedAt: new Date() + } + }); + + avatarsFixed++; + } + } + } + + return { avatarsFixed }; + } + + /** + * Step 8: Fix attachment URLs (remove problematic auth parameters and fix URL formats) + */ + async fixAttachmentUrls(boardId) { + const attachments = ReactiveCache.getAttachments({}); + let attachmentsFixed = 0; + + for (const attachment of attachments) { + // Check if attachment has URL field that needs fixing + if (attachment.url) { + const attachmentUrl = attachment.url; + let needsUpdate = false; + let cleanUrl = attachmentUrl; + + // Check if URL has problematic parameters + if (attachmentUrl.includes('auth=false') || attachmentUrl.includes('brokenIsFine=true')) { + // Remove problematic parameters + cleanUrl = cleanUrl.replace(/[?&]auth=false/g, ''); + cleanUrl = cleanUrl.replace(/[?&]brokenIsFine=true/g, ''); + cleanUrl = cleanUrl.replace(/\?&/g, '?'); + cleanUrl = cleanUrl.replace(/\?$/g, ''); + needsUpdate = true; + } + + // Check if URL is using old CollectionFS format + if (attachmentUrl.includes('/cfs/files/attachments/')) { + cleanUrl = cleanUrl.replace('/cfs/files/attachments/', '/cdn/storage/attachments/'); + needsUpdate = true; + } + + // Check if URL has /original/ path that should be removed + if (attachmentUrl.includes('/original/')) { + cleanUrl = cleanUrl.replace(/\/original\/[^\/\?#]+/, ''); + needsUpdate = true; + } + + // If we have a file ID, generate a universal URL + const fileId = attachment._id; + if (fileId && !isUniversalFileUrl(cleanUrl, 'attachment')) { + cleanUrl = generateUniversalAttachmentUrl(fileId); + needsUpdate = true; + } + + if (needsUpdate) { + // Update attachment URL + Attachments.update(attachment._id, { + $set: { + url: cleanUrl, + modifiedAt: new Date() + } + }); + + attachmentsFixed++; + } + } + } + + return { attachmentsFixed }; + } + + /** + * Get migration status for a board + */ + getMigrationStatus(boardId) { + try { + const board = ReactiveCache.getBoard(boardId); + if (!board) { + return { status: 'board_not_found' }; + } + + if (board.comprehensiveMigrationCompleted) { + return { + status: 'completed', + completedAt: board.comprehensiveMigrationCompletedAt, + results: board.comprehensiveMigrationResults + }; + } + + const needsMigration = this.needsMigration(boardId); + const issues = this.detectMigrationIssues(boardId); + + return { + status: needsMigration ? 'needed' : 'not_needed', + issues, + issueCount: issues.length + }; + + } catch (error) { + console.error('Error getting migration status:', error); + return { status: 'error', error: error.message }; + } + } +} + +// Export singleton instance +export const comprehensiveBoardMigration = new ComprehensiveBoardMigration(); + +// Meteor methods +Meteor.methods({ + 'comprehensiveBoardMigration.check'(boardId) { + check(boardId, String); + + if (!this.userId) { + throw new Meteor.Error('not-authorized'); + } + + return comprehensiveBoardMigration.getMigrationStatus(boardId); + }, + + 'comprehensiveBoardMigration.execute'(boardId) { + check(boardId, String); + + if (!this.userId) { + throw new Meteor.Error('not-authorized'); + } + + return comprehensiveBoardMigration.executeMigration(boardId); + }, + + 'comprehensiveBoardMigration.needsMigration'(boardId) { + check(boardId, String); + + if (!this.userId) { + throw new Meteor.Error('not-authorized'); + } + + return comprehensiveBoardMigration.needsMigration(boardId); + }, + + 'comprehensiveBoardMigration.detectIssues'(boardId) { + check(boardId, String); + + if (!this.userId) { + throw new Meteor.Error('not-authorized'); + } + + return comprehensiveBoardMigration.detectMigrationIssues(boardId); + }, + + 'comprehensiveBoardMigration.fixAvatarUrls'(boardId) { + check(boardId, String); + + if (!this.userId) { + throw new Meteor.Error('not-authorized'); + } + + return comprehensiveBoardMigration.fixAvatarUrls(boardId); + } +}); diff --git a/server/migrations/fixAllFileUrls.js b/server/migrations/fixAllFileUrls.js new file mode 100644 index 000000000..caba86e68 --- /dev/null +++ b/server/migrations/fixAllFileUrls.js @@ -0,0 +1,277 @@ +/** + * Fix All File URLs Migration + * Ensures all attachment and avatar URLs are universal and work regardless of ROOT_URL and PORT settings + */ + +import { ReactiveCache } from '/imports/reactiveCache'; +import Users from '/models/users'; +import Attachments from '/models/attachments'; +import Avatars from '/models/avatars'; +import { generateUniversalAttachmentUrl, generateUniversalAvatarUrl, cleanFileUrl, extractFileIdFromUrl, isUniversalFileUrl } from '/models/lib/universalUrlGenerator'; + +class FixAllFileUrlsMigration { + constructor() { + this.name = 'fixAllFileUrls'; + this.version = 1; + } + + /** + * Check if migration is needed + */ + needsMigration() { + // Check for problematic avatar URLs + const users = ReactiveCache.getUsers({}); + for (const user of users) { + if (user.profile && user.profile.avatarUrl) { + const avatarUrl = user.profile.avatarUrl; + if (this.hasProblematicUrl(avatarUrl)) { + return true; + } + } + } + + // Check for problematic attachment URLs in cards + const cards = ReactiveCache.getCards({}); + for (const card of cards) { + if (card.attachments) { + for (const attachment of card.attachments) { + if (attachment.url && this.hasProblematicUrl(attachment.url)) { + return true; + } + } + } + } + + return false; + } + + /** + * Check if a URL has problematic patterns + */ + hasProblematicUrl(url) { + if (!url) return false; + + // Check for auth parameters + if (url.includes('auth=false') || url.includes('brokenIsFine=true')) { + return true; + } + + // Check for absolute URLs with domains + if (url.startsWith('http://') || url.startsWith('https://')) { + return true; + } + + // Check for ROOT_URL dependencies + if (Meteor.isServer && process.env.ROOT_URL) { + try { + const rootUrl = new URL(process.env.ROOT_URL); + if (rootUrl.pathname && rootUrl.pathname !== '/' && url.includes(rootUrl.pathname)) { + return true; + } + } catch (e) { + // Ignore URL parsing errors + } + } + + // Check for non-universal file URLs + if (url.includes('/cfs/files/') && !isUniversalFileUrl(url, 'attachment') && !isUniversalFileUrl(url, 'avatar')) { + return true; + } + + return false; + } + + /** + * Execute the migration + */ + async execute() { + let filesFixed = 0; + let errors = []; + + console.log(`Starting universal file URL migration...`); + + try { + // Fix avatar URLs + const avatarFixed = await this.fixAvatarUrls(); + filesFixed += avatarFixed; + + // Fix attachment URLs + const attachmentFixed = await this.fixAttachmentUrls(); + filesFixed += attachmentFixed; + + // Fix card attachment references + const cardFixed = await this.fixCardAttachmentUrls(); + filesFixed += cardFixed; + + } catch (error) { + console.error('Error during file URL migration:', error); + errors.push(error.message); + } + + console.log(`Universal file URL migration completed. Fixed ${filesFixed} file URLs.`); + + return { + success: errors.length === 0, + filesFixed, + errors + }; + } + + /** + * Fix avatar URLs in user profiles + */ + async fixAvatarUrls() { + const users = ReactiveCache.getUsers({}); + let avatarsFixed = 0; + + for (const user of users) { + if (user.profile && user.profile.avatarUrl) { + const avatarUrl = user.profile.avatarUrl; + + if (this.hasProblematicUrl(avatarUrl)) { + try { + // Extract file ID from URL + const fileId = extractFileIdFromUrl(avatarUrl, 'avatar'); + + let cleanUrl; + if (fileId) { + // Generate universal URL + cleanUrl = generateUniversalAvatarUrl(fileId); + } else { + // Clean existing URL + cleanUrl = cleanFileUrl(avatarUrl, 'avatar'); + } + + if (cleanUrl && cleanUrl !== avatarUrl) { + // Update user's avatar URL + Users.update(user._id, { + $set: { + 'profile.avatarUrl': cleanUrl, + modifiedAt: new Date() + } + }); + + avatarsFixed++; + + if (process.env.DEBUG === 'true') { + console.log(`Fixed avatar URL for user ${user.username}: ${avatarUrl} -> ${cleanUrl}`); + } + } + } catch (error) { + console.error(`Error fixing avatar URL for user ${user.username}:`, error); + } + } + } + } + + return avatarsFixed; + } + + /** + * Fix attachment URLs in attachment records + */ + async fixAttachmentUrls() { + const attachments = ReactiveCache.getAttachments({}); + let attachmentsFixed = 0; + + for (const attachment of attachments) { + // Check if attachment has URL field that needs fixing + if (attachment.url && this.hasProblematicUrl(attachment.url)) { + try { + const fileId = attachment._id; + const cleanUrl = generateUniversalAttachmentUrl(fileId); + + if (cleanUrl && cleanUrl !== attachment.url) { + // Update attachment URL + Attachments.update(attachment._id, { + $set: { + url: cleanUrl, + modifiedAt: new Date() + } + }); + + attachmentsFixed++; + + if (process.env.DEBUG === 'true') { + console.log(`Fixed attachment URL: ${attachment.url} -> ${cleanUrl}`); + } + } + } catch (error) { + console.error(`Error fixing attachment URL for ${attachment._id}:`, error); + } + } + } + + return attachmentsFixed; + } + + /** + * Fix attachment URLs in card references + */ + async fixCardAttachmentUrls() { + const cards = ReactiveCache.getCards({}); + let cardsFixed = 0; + + for (const card of cards) { + if (card.attachments) { + let needsUpdate = false; + const updatedAttachments = card.attachments.map(attachment => { + if (attachment.url && this.hasProblematicUrl(attachment.url)) { + try { + const fileId = attachment._id || extractFileIdFromUrl(attachment.url, 'attachment'); + const cleanUrl = fileId ? generateUniversalAttachmentUrl(fileId) : cleanFileUrl(attachment.url, 'attachment'); + + if (cleanUrl && cleanUrl !== attachment.url) { + needsUpdate = true; + return { ...attachment, url: cleanUrl }; + } + } catch (error) { + console.error(`Error fixing card attachment URL:`, error); + } + } + return attachment; + }); + + if (needsUpdate) { + // Update card with fixed attachment URLs + Cards.update(card._id, { + $set: { + attachments: updatedAttachments, + modifiedAt: new Date() + } + }); + + cardsFixed++; + + if (process.env.DEBUG === 'true') { + console.log(`Fixed attachment URLs in card ${card._id}`); + } + } + } + } + + return cardsFixed; + } +} + +// Export singleton instance +export const fixAllFileUrlsMigration = new FixAllFileUrlsMigration(); + +// Meteor methods +Meteor.methods({ + 'fixAllFileUrls.execute'() { + if (!this.userId) { + throw new Meteor.Error('not-authorized'); + } + + return fixAllFileUrlsMigration.execute(); + }, + + 'fixAllFileUrls.needsMigration'() { + if (!this.userId) { + throw new Meteor.Error('not-authorized'); + } + + return fixAllFileUrlsMigration.needsMigration(); + } +}); diff --git a/server/migrations/fixAvatarUrls.js b/server/migrations/fixAvatarUrls.js new file mode 100644 index 000000000..f542903ed --- /dev/null +++ b/server/migrations/fixAvatarUrls.js @@ -0,0 +1,128 @@ +/** + * Fix Avatar URLs Migration + * Removes problematic auth parameters from existing avatar URLs + */ + +import { ReactiveCache } from '/imports/reactiveCache'; +import Users from '/models/users'; +import { generateUniversalAvatarUrl, cleanFileUrl, extractFileIdFromUrl, isUniversalFileUrl } from '/models/lib/universalUrlGenerator'; + +class FixAvatarUrlsMigration { + constructor() { + this.name = 'fixAvatarUrls'; + this.version = 1; + } + + /** + * Check if migration is needed + */ + needsMigration() { + const users = ReactiveCache.getUsers({}); + + for (const user of users) { + if (user.profile && user.profile.avatarUrl) { + const avatarUrl = user.profile.avatarUrl; + if (avatarUrl.includes('auth=false') || avatarUrl.includes('brokenIsFine=true')) { + return true; + } + } + } + + return false; + } + + /** + * Execute the migration + */ + async execute() { + const users = ReactiveCache.getUsers({}); + let avatarsFixed = 0; + + console.log(`Starting avatar URL fix migration...`); + + for (const user of users) { + if (user.profile && user.profile.avatarUrl) { + const avatarUrl = user.profile.avatarUrl; + let needsUpdate = false; + let cleanUrl = avatarUrl; + + // Check if URL has problematic parameters + if (avatarUrl.includes('auth=false') || avatarUrl.includes('brokenIsFine=true')) { + // Remove problematic parameters + cleanUrl = cleanUrl.replace(/[?&]auth=false/g, ''); + cleanUrl = cleanUrl.replace(/[?&]brokenIsFine=true/g, ''); + cleanUrl = cleanUrl.replace(/\?&/g, '?'); + cleanUrl = cleanUrl.replace(/\?$/g, ''); + needsUpdate = true; + } + + // Check if URL is using old CollectionFS format + if (avatarUrl.includes('/cfs/files/avatars/')) { + cleanUrl = cleanUrl.replace('/cfs/files/avatars/', '/cdn/storage/avatars/'); + needsUpdate = true; + } + + // Check if URL is missing the /cdn/storage/avatars/ prefix + if (avatarUrl.includes('avatars/') && !avatarUrl.includes('/cdn/storage/avatars/') && !avatarUrl.includes('/cfs/files/avatars/')) { + // This might be a relative URL, make it absolute + if (!avatarUrl.startsWith('http') && !avatarUrl.startsWith('/')) { + cleanUrl = `/cdn/storage/avatars/${avatarUrl}`; + needsUpdate = true; + } + } + + // If we have a file ID, generate a universal URL + const fileId = extractFileIdFromUrl(avatarUrl, 'avatar'); + if (fileId && !isUniversalFileUrl(cleanUrl, 'avatar')) { + cleanUrl = generateUniversalAvatarUrl(fileId); + needsUpdate = true; + } + + if (needsUpdate) { + // Update user's avatar URL + Users.update(user._id, { + $set: { + 'profile.avatarUrl': cleanUrl, + modifiedAt: new Date() + } + }); + + avatarsFixed++; + + if (process.env.DEBUG === 'true') { + console.log(`Fixed avatar URL for user ${user.username}: ${avatarUrl} -> ${cleanUrl}`); + } + } + } + } + + console.log(`Avatar URL fix migration completed. Fixed ${avatarsFixed} avatar URLs.`); + + return { + success: true, + avatarsFixed + }; + } +} + +// Export singleton instance +export const fixAvatarUrlsMigration = new FixAvatarUrlsMigration(); + +// Meteor method +Meteor.methods({ + 'fixAvatarUrls.execute'() { + if (!this.userId) { + throw new Meteor.Error('not-authorized'); + } + + return fixAvatarUrlsMigration.execute(); + }, + + 'fixAvatarUrls.needsMigration'() { + if (!this.userId) { + throw new Meteor.Error('not-authorized'); + } + + return fixAvatarUrlsMigration.needsMigration(); + } +}); diff --git a/server/routes/avatarServer.js b/server/routes/avatarServer.js new file mode 100644 index 000000000..008ea573a --- /dev/null +++ b/server/routes/avatarServer.js @@ -0,0 +1,123 @@ +/** + * Avatar File Server + * Handles serving avatar files from the /cdn/storage/avatars/ path + */ + +import { Meteor } from 'meteor/meteor'; +import { WebApp } from 'meteor/webapp'; +import { ReactiveCache } from '/imports/reactiveCache'; +import Avatars from '/models/avatars'; +import { fileStoreStrategyFactory } from '/models/lib/fileStoreStrategy'; +import fs from 'fs'; +import path from 'path'; + +if (Meteor.isServer) { + // Handle avatar file downloads + WebApp.connectHandlers.use('/cdn/storage/avatars/([^/]+)', (req, res, next) => { + if (req.method !== 'GET') { + return next(); + } + + try { + const fileName = req.params[0]; + + if (!fileName) { + res.writeHead(400); + res.end('Invalid avatar file name'); + return; + } + + // Extract file ID from filename (format: fileId-original-filename) + const fileId = fileName.split('-original-')[0]; + + if (!fileId) { + res.writeHead(400); + res.end('Invalid avatar file format'); + return; + } + + // Get avatar file from database + const avatar = ReactiveCache.getAvatar(fileId); + if (!avatar) { + res.writeHead(404); + res.end('Avatar not found'); + return; + } + + // Check if user has permission to view this avatar + // For avatars, we allow viewing by any logged-in user + const userId = Meteor.userId(); + if (!userId) { + res.writeHead(401); + res.end('Authentication required'); + return; + } + + // Get file strategy + const strategy = fileStoreStrategyFactory.getFileStrategy(avatar, 'original'); + const readStream = strategy.getReadStream(); + + if (!readStream) { + res.writeHead(404); + res.end('Avatar file not found in storage'); + return; + } + + // Set appropriate headers + res.setHeader('Content-Type', avatar.type || 'image/jpeg'); + res.setHeader('Content-Length', avatar.size || 0); + res.setHeader('Cache-Control', 'public, max-age=31536000'); // Cache for 1 year + res.setHeader('ETag', `"${avatar._id}"`); + + // Handle conditional requests + const ifNoneMatch = req.headers['if-none-match']; + if (ifNoneMatch && ifNoneMatch === `"${avatar._id}"`) { + res.writeHead(304); + res.end(); + return; + } + + // Stream the file + res.writeHead(200); + readStream.pipe(res); + + readStream.on('error', (error) => { + console.error('Avatar stream error:', error); + if (!res.headersSent) { + res.writeHead(500); + res.end('Error reading avatar file'); + } + }); + + } catch (error) { + console.error('Avatar server error:', error); + if (!res.headersSent) { + res.writeHead(500); + res.end('Internal server error'); + } + } + }); + + // Handle legacy avatar URLs (from CollectionFS) + WebApp.connectHandlers.use('/cfs/files/avatars/([^/]+)', (req, res, next) => { + if (req.method !== 'GET') { + return next(); + } + + try { + const fileName = req.params[0]; + + // Redirect to new avatar URL format + const newUrl = `/cdn/storage/avatars/${fileName}`; + res.writeHead(301, { 'Location': newUrl }); + res.end(); + + } catch (error) { + console.error('Legacy avatar redirect error:', error); + res.writeHead(500); + res.end('Internal server error'); + } + }); + + console.log('Avatar server routes initialized'); +} diff --git a/server/routes/universalFileServer.js b/server/routes/universalFileServer.js new file mode 100644 index 000000000..2a2cb2e39 --- /dev/null +++ b/server/routes/universalFileServer.js @@ -0,0 +1,393 @@ +/** + * Universal File Server + * Ensures all attachments and avatars are always visible regardless of ROOT_URL and PORT settings + * Handles both new Meteor-Files and legacy CollectionFS file serving + */ + +import { Meteor } from 'meteor/meteor'; +import { WebApp } from 'meteor/webapp'; +import { ReactiveCache } from '/imports/reactiveCache'; +import Attachments from '/models/attachments'; +import Avatars from '/models/avatars'; +import { fileStoreStrategyFactory } from '/models/lib/fileStoreStrategy'; +import { getAttachmentWithBackwardCompatibility, getOldAttachmentStream } from '/models/lib/attachmentBackwardCompatibility'; +import fs from 'fs'; +import path from 'path'; + +if (Meteor.isServer) { + console.log('Universal file server initializing...'); + + /** + * Helper function to set appropriate headers for file serving + */ + function setFileHeaders(res, fileObj, isAttachment = false) { + // Set content type + res.setHeader('Content-Type', fileObj.type || (isAttachment ? 'application/octet-stream' : 'image/jpeg')); + + // Set content length + res.setHeader('Content-Length', fileObj.size || 0); + + // Set cache headers + res.setHeader('Cache-Control', 'public, max-age=31536000'); // Cache for 1 year + res.setHeader('ETag', `"${fileObj._id}"`); + + // Set security headers for attachments + if (isAttachment) { + const isSvgFile = fileObj.name && fileObj.name.toLowerCase().endsWith('.svg'); + const disposition = isSvgFile ? 'attachment' : 'inline'; + res.setHeader('Content-Disposition', `${disposition}; filename="${fileObj.name}"`); + + // Add security headers for SVG files + if (isSvgFile) { + res.setHeader('Content-Security-Policy', "default-src 'none'; script-src 'none'; object-src 'none';"); + res.setHeader('X-Content-Type-Options', 'nosniff'); + res.setHeader('X-Frame-Options', 'DENY'); + } + } + } + + /** + * Helper function to handle conditional requests + */ + function handleConditionalRequest(req, res, fileObj) { + const ifNoneMatch = req.headers['if-none-match']; + if (ifNoneMatch && ifNoneMatch === `"${fileObj._id}"`) { + res.writeHead(304); + res.end(); + return true; + } + return false; + } + + /** + * Helper function to stream file with error handling + */ + function streamFile(res, readStream, fileObj) { + readStream.on('error', (error) => { + console.error('File stream error:', error); + if (!res.headersSent) { + res.writeHead(500); + res.end('Error reading file'); + } + }); + + readStream.on('end', () => { + if (!res.headersSent) { + res.writeHead(200); + } + }); + + readStream.pipe(res); + } + + // ============================================================================ + // NEW METEOR-FILES ROUTES (URL-agnostic) + // ============================================================================ + + /** + * Serve attachments from new Meteor-Files structure + * Route: /cdn/storage/attachments/{fileId} or /cdn/storage/attachments/{fileId}/original/{filename} + */ + WebApp.connectHandlers.use('/cdn/storage/attachments/([^/]+)(?:/original/[^/]+)?', (req, res, next) => { + if (req.method !== 'GET') { + return next(); + } + + try { + const fileId = req.params[0]; + + if (!fileId) { + res.writeHead(400); + res.end('Invalid attachment file ID'); + return; + } + + // Get attachment from database + const attachment = ReactiveCache.getAttachment(fileId); + if (!attachment) { + res.writeHead(404); + res.end('Attachment not found'); + return; + } + + // Check permissions + const board = ReactiveCache.getBoard(attachment.meta.boardId); + if (!board) { + res.writeHead(404); + res.end('Board not found'); + return; + } + + // Check if user has permission to download + const userId = Meteor.userId(); + if (!board.isPublic() && (!userId || !board.hasMember(userId))) { + res.writeHead(403); + res.end('Access denied'); + return; + } + + // Handle conditional requests + if (handleConditionalRequest(req, res, attachment)) { + return; + } + + // Get file strategy and stream + const strategy = fileStoreStrategyFactory.getFileStrategy(attachment, 'original'); + const readStream = strategy.getReadStream(); + + if (!readStream) { + res.writeHead(404); + res.end('Attachment file not found in storage'); + return; + } + + // Set headers and stream file + setFileHeaders(res, attachment, true); + streamFile(res, readStream, attachment); + + } catch (error) { + console.error('Attachment server error:', error); + if (!res.headersSent) { + res.writeHead(500); + res.end('Internal server error'); + } + } + }); + + /** + * Serve avatars from new Meteor-Files structure + * Route: /cdn/storage/avatars/{fileId} or /cdn/storage/avatars/{fileId}/original/{filename} + */ + WebApp.connectHandlers.use('/cdn/storage/avatars/([^/]+)(?:/original/[^/]+)?', (req, res, next) => { + if (req.method !== 'GET') { + return next(); + } + + try { + const fileId = req.params[0]; + + if (!fileId) { + res.writeHead(400); + res.end('Invalid avatar file ID'); + return; + } + + // Get avatar from database + const avatar = ReactiveCache.getAvatar(fileId); + if (!avatar) { + res.writeHead(404); + res.end('Avatar not found'); + return; + } + + // Check if user has permission to view this avatar + // For avatars, we allow viewing by any logged-in user + const userId = Meteor.userId(); + if (!userId) { + res.writeHead(401); + res.end('Authentication required'); + return; + } + + // Handle conditional requests + if (handleConditionalRequest(req, res, avatar)) { + return; + } + + // Get file strategy and stream + const strategy = fileStoreStrategyFactory.getFileStrategy(avatar, 'original'); + const readStream = strategy.getReadStream(); + + if (!readStream) { + res.writeHead(404); + res.end('Avatar file not found in storage'); + return; + } + + // Set headers and stream file + setFileHeaders(res, avatar, false); + streamFile(res, readStream, avatar); + + } catch (error) { + console.error('Avatar server error:', error); + if (!res.headersSent) { + res.writeHead(500); + res.end('Internal server error'); + } + } + }); + + // ============================================================================ + // LEGACY COLLECTIONFS ROUTES (Backward compatibility) + // ============================================================================ + + /** + * Serve legacy attachments from CollectionFS structure + * Route: /cfs/files/attachments/{attachmentId} + */ + WebApp.connectHandlers.use('/cfs/files/attachments/([^/]+)', (req, res, next) => { + if (req.method !== 'GET') { + return next(); + } + + try { + const attachmentId = req.params[0]; + + if (!attachmentId) { + res.writeHead(400); + res.end('Invalid attachment ID'); + return; + } + + // Try to get attachment with backward compatibility + const attachment = getAttachmentWithBackwardCompatibility(attachmentId); + if (!attachment) { + res.writeHead(404); + res.end('Attachment not found'); + return; + } + + // Check permissions + const board = ReactiveCache.getBoard(attachment.meta.boardId); + if (!board) { + res.writeHead(404); + res.end('Board not found'); + return; + } + + // Check if user has permission to download + const userId = Meteor.userId(); + if (!board.isPublic() && (!userId || !board.hasMember(userId))) { + res.writeHead(403); + res.end('Access denied'); + return; + } + + // Handle conditional requests + if (handleConditionalRequest(req, res, attachment)) { + return; + } + + // For legacy attachments, try to get GridFS stream + const fileStream = getOldAttachmentStream(attachmentId); + if (fileStream) { + setFileHeaders(res, attachment, true); + streamFile(res, fileStream, attachment); + } else { + res.writeHead(404); + res.end('Legacy attachment file not found in GridFS'); + } + + } catch (error) { + console.error('Legacy attachment server error:', error); + if (!res.headersSent) { + res.writeHead(500); + res.end('Internal server error'); + } + } + }); + + /** + * Serve legacy avatars from CollectionFS structure + * Route: /cfs/files/avatars/{avatarId} + */ + WebApp.connectHandlers.use('/cfs/files/avatars/([^/]+)', (req, res, next) => { + if (req.method !== 'GET') { + return next(); + } + + try { + const avatarId = req.params[0]; + + if (!avatarId) { + res.writeHead(400); + res.end('Invalid avatar ID'); + return; + } + + // Try to get avatar from database (new structure first) + let avatar = ReactiveCache.getAvatar(avatarId); + + // If not found in new structure, try to handle legacy format + if (!avatar) { + // For legacy avatars, we might need to handle different ID formats + // This is a fallback for old CollectionFS avatars + res.writeHead(404); + res.end('Avatar not found'); + return; + } + + // Check if user has permission to view this avatar + const userId = Meteor.userId(); + if (!userId) { + res.writeHead(401); + res.end('Authentication required'); + return; + } + + // Handle conditional requests + if (handleConditionalRequest(req, res, avatar)) { + return; + } + + // Get file strategy and stream + const strategy = fileStoreStrategyFactory.getFileStrategy(avatar, 'original'); + const readStream = strategy.getReadStream(); + + if (!readStream) { + res.writeHead(404); + res.end('Avatar file not found in storage'); + return; + } + + // Set headers and stream file + setFileHeaders(res, avatar, false); + streamFile(res, readStream, avatar); + + } catch (error) { + console.error('Legacy avatar server error:', error); + if (!res.headersSent) { + res.writeHead(500); + res.end('Internal server error'); + } + } + }); + + // ============================================================================ + // ALTERNATIVE ROUTES (For different URL patterns) + // ============================================================================ + + /** + * Alternative attachment route for different URL patterns + * Route: /attachments/{fileId} + */ + WebApp.connectHandlers.use('/attachments/([^/]+)', (req, res, next) => { + if (req.method !== 'GET') { + return next(); + } + + // Redirect to standard route + const fileId = req.params[0]; + const newUrl = `/cdn/storage/attachments/${fileId}`; + res.writeHead(301, { 'Location': newUrl }); + res.end(); + }); + + /** + * Alternative avatar route for different URL patterns + * Route: /avatars/{fileId} + */ + WebApp.connectHandlers.use('/avatars/([^/]+)', (req, res, next) => { + if (req.method !== 'GET') { + return next(); + } + + // Redirect to standard route + const fileId = req.params[0]; + const newUrl = `/cdn/storage/avatars/${fileId}`; + res.writeHead(301, { 'Location': newUrl }); + res.end(); + }); + + console.log('Universal file server initialized successfully'); +} From 0fc2ad97cd10fe473f4aec11f9a1bf92a2d2132e Mon Sep 17 00:00:00 2001 From: "dependabot[bot]" <49699333+dependabot[bot]@users.noreply.github.com> Date: Mon, 27 Oct 2025 20:32:10 +0000 Subject: [PATCH 15/84] Bump actions/upload-artifact from 4 to 5 Bumps [actions/upload-artifact](https://github.com/actions/upload-artifact) from 4 to 5. - [Release notes](https://github.com/actions/upload-artifact/releases) - [Commits](https://github.com/actions/upload-artifact/compare/v4...v5) --- updated-dependencies: - dependency-name: actions/upload-artifact dependency-version: '5' dependency-type: direct:production update-type: version-update:semver-major ... Signed-off-by: dependabot[bot] --- .github/workflows/test_suite.yml | 2 +- 1 file changed, 1 insertion(+), 1 deletion(-) diff --git a/.github/workflows/test_suite.yml b/.github/workflows/test_suite.yml index b3498f613..1160226dd 100644 --- a/.github/workflows/test_suite.yml +++ b/.github/workflows/test_suite.yml @@ -136,7 +136,7 @@ jobs: run: sh ./test-wekan.sh -cv - name: Upload coverage - uses: actions/upload-artifact@v4 + uses: actions/upload-artifact@v5 with: name: coverage-folder path: .coverage/ From 3204311ac1c7a6778faa0a2ce613cf4670f0cb6a Mon Sep 17 00:00:00 2001 From: "dependabot[bot]" <49699333+dependabot[bot]@users.noreply.github.com> Date: Mon, 27 Oct 2025 23:12:09 +0000 Subject: [PATCH 16/84] Bump actions/download-artifact from 5 to 6 Bumps [actions/download-artifact](https://github.com/actions/download-artifact) from 5 to 6. - [Release notes](https://github.com/actions/download-artifact/releases) - [Commits](https://github.com/actions/download-artifact/compare/v5...v6) --- updated-dependencies: - dependency-name: actions/download-artifact dependency-version: '6' dependency-type: direct:production update-type: version-update:semver-major ... Signed-off-by: dependabot[bot] --- .github/workflows/test_suite.yml | 2 +- 1 file changed, 1 insertion(+), 1 deletion(-) diff --git a/.github/workflows/test_suite.yml b/.github/workflows/test_suite.yml index b3498f613..3730decfd 100644 --- a/.github/workflows/test_suite.yml +++ b/.github/workflows/test_suite.yml @@ -150,7 +150,7 @@ jobs: uses: actions/checkout@v5 - name: Download coverage - uses: actions/download-artifact@v5 + uses: actions/download-artifact@v6 with: name: coverage-folder path: .coverage/ From 5079c853a7280a29797385cf25b234a4a3974730 Mon Sep 17 00:00:00 2001 From: Lauri Ojansivu Date: Wed, 29 Oct 2025 02:58:00 +0200 Subject: [PATCH 17/84] Updated translations. --- imports/i18n/data/ko.i18n.json | 18 ++-- imports/i18n/data/sv.i18n.json | 144 +++++++++++++++--------------- imports/i18n/data/zh-TW.i18n.json | 8 +- 3 files changed, 85 insertions(+), 85 deletions(-) diff --git a/imports/i18n/data/ko.i18n.json b/imports/i18n/data/ko.i18n.json index 62f508b82..1796716ba 100644 --- a/imports/i18n/data/ko.i18n.json +++ b/imports/i18n/data/ko.i18n.json @@ -190,9 +190,9 @@ "board-view-collapse": "접기", "board-view-gantt": "간트", "board-view-lists": "목록들", - "bucket-example": "Like \"Bucket List\" for example", - "calendar-previous-month-label": "Previous Month", - "calendar-next-month-label": "Next Month", + "bucket-example": "예: \"꼭 하고싶은 일 목록\"", + "calendar-previous-month-label": "이전 월", + "calendar-next-month-label": "다음 월", "cancel": "취소", "card-archived": "이 카드는 보관함으로 이동 되었습니다.", "board-archived": "이 보드는 보관함으로 이동 되었습니다.", @@ -356,7 +356,7 @@ "custom-field-text": "텍스트", "custom-fields": "사용자정의 항목", "date": "날짜", - "date-format": "Date Format", + "date-format": "날짜 형식", "date-format-yyyy-mm-dd": "YYYY-MM-DD", "date-format-dd-mm-yyyy": "DD-MM-YYYY", "date-format-mm-dd-yyyy": "MM-DD-YYYY", @@ -385,7 +385,7 @@ "editNotificationPopup-title": "알림 수정", "editProfilePopup-title": "프로필 변경", "email": "이메일", - "email-address": "Email Address", + "email-address": "이메일 주소", "email-enrollAccount-subject": "__siteName__에 계정 생성이 완료되었습니다.", "email-enrollAccount-text": "안녕하세요. __user__님,\n\n시작하려면 아래링크를 클릭해 주세요.\n\n__url__\n\n감사합니다.", "email-fail": "이메일 전송 실패", @@ -755,8 +755,8 @@ "delete-board-confirm-popup": "모든 목록, 카드, 레이블 및 활동이 삭제되고 보드 내용을 복구할 수 없습니다. 실행 취소는 불가능합니다.", "boardDeletePopup-title": "보드 삭제?", "delete-board": "보드 삭제", - "delete-duplicate-lists": "Delete Duplicate Lists", - "delete-duplicate-lists-confirm": "Are you sure? This will delete all duplicate lists that have the same name and contain no cards.", + "delete-duplicate-lists": "중복 목록 삭제", + "delete-duplicate-lists-confirm": "확실합니까? 이렇게 하면 이름이 같고 카드가 없는 모든 중복 목록이 삭제됩니다.", "default-subtasks-board": "Subtasks for __board__ board", "default": "기본", "defaultdefault": "기본", @@ -1018,8 +1018,8 @@ "dueCardsViewChange-choice-me": "Me", "dueCardsViewChange-choice-all": "All Users", "dueCardsViewChange-choice-all-description": "Shows all incomplete cards with a *Due* date from boards for which the user has permission.", - "dueCards-noResults-title": "No Due Cards Found", - "dueCards-noResults-description": "You don't have any cards with due dates at the moment.", + "dueCards-noResults-title": "만료된 카드를 찾을 수 없음", + "dueCards-noResults-description": "현재 마감일이 있는 카드가 없습니다.", "broken-cards": "Broken Cards", "board-title-not-found": "보드 %s 을 찾을 수 없습니다.", "swimlane-title-not-found": "Swimlane %s 을 찾을 수 없습니다.", diff --git a/imports/i18n/data/sv.i18n.json b/imports/i18n/data/sv.i18n.json index 6a4e1616d..2297e547b 100644 --- a/imports/i18n/data/sv.i18n.json +++ b/imports/i18n/data/sv.i18n.json @@ -1334,70 +1334,70 @@ "avatars-path": "Sökväg för avatarer", "avatars-path-description": "Sökväg där avatarfiler lagras", "board-archive-failed": "Misslyckades att schemalägga arkivering av tavla", - "board-archive-scheduled": "Board archive scheduled successfully", - "board-backup-failed": "Failed to schedule board backup", - "board-backup-scheduled": "Board backup scheduled successfully", - "board-cleanup-failed": "Failed to schedule board cleanup", - "board-cleanup-scheduled": "Board cleanup scheduled successfully", - "board-operations": "Board Operations", - "cron-jobs": "Scheduled Jobs", - "cron-migrations": "Scheduled Migrations", - "cron-job-delete-confirm": "Are you sure you want to delete this scheduled job?", - "cron-job-delete-failed": "Failed to delete scheduled job", - "cron-job-deleted": "Scheduled job deleted successfully", - "cron-job-pause-failed": "Failed to pause scheduled job", - "cron-job-paused": "Scheduled job paused successfully", - "filesystem-path-description": "Base path for file storage", - "gridfs-enabled": "GridFS Enabled", - "gridfs-enabled-description": "Use MongoDB GridFS for file storage", - "migration-pause-failed": "Failed to pause migrations", - "migration-paused": "Migrations paused successfully", - "migration-progress": "Migration Progress", - "migration-start-failed": "Failed to start migrations", - "migration-started": "Migrations started successfully", - "migration-status": "Migration Status", - "migration-stop-confirm": "Are you sure you want to stop all migrations?", - "migration-stop-failed": "Failed to stop migrations", - "migration-stopped": "Migrations stopped successfully", - "mongodb-gridfs-storage": "MongoDB GridFS Storage", - "pause-all-migrations": "Pause All Migrations", - "s3-access-key": "S3 Access Key", - "s3-access-key-description": "AWS S3 access key for authentication", - "s3-access-key-placeholder": "Enter S3 access key", - "s3-bucket": "S3 Bucket", - "s3-bucket-description": "S3 bucket name for storing files", - "s3-connection-failed": "S3 connection failed", - "s3-connection-success": "S3 connection successful", - "s3-enabled": "S3 Enabled", - "s3-enabled-description": "Use AWS S3 or MinIO for file storage", - "s3-endpoint": "S3 Endpoint", - "s3-endpoint-description": "S3 endpoint URL (e.g., s3.amazonaws.com or minio.example.com)", - "s3-minio-storage": "S3/MinIO Storage", - "s3-port": "S3 Port", - "s3-port-description": "S3 endpoint port number", - "s3-region": "S3 Region", - "s3-region-description": "AWS S3 region (e.g., us-east-1)", - "s3-secret-key": "S3 Secret Key", - "s3-secret-key-description": "AWS S3 secret key for authentication", - "s3-secret-key-placeholder": "Enter S3 secret key", - "s3-secret-key-required": "S3 secret key is required", - "s3-settings-save-failed": "Failed to save S3 settings", - "s3-settings-saved": "S3 settings saved successfully", - "s3-ssl-enabled": "S3 SSL Enabled", - "s3-ssl-enabled-description": "Use SSL/TLS for S3 connections", - "save-s3-settings": "Save S3 Settings", - "schedule-board-archive": "Schedule Board Archive", - "schedule-board-backup": "Schedule Board Backup", - "schedule-board-cleanup": "Schedule Board Cleanup", - "scheduled-board-operations": "Scheduled Board Operations", - "start-all-migrations": "Start All Migrations", - "stop-all-migrations": "Stop All Migrations", - "test-s3-connection": "Test S3 Connection", - "writable-path": "Writable Path", - "writable-path-description": "Base directory path for file storage", - "add-job": "Add Job", - "attachment-migration": "Attachment Migration", - "attachment-monitoring": "Attachment Monitoring", + "board-archive-scheduled": "Tavlans arkivering har schemalagts", + "board-backup-failed": "Misslyckades med att schemalägga säkerhetskopiering av tavlan", + "board-backup-scheduled": "Säkerhetskopiering av tavlan har schemalagts", + "board-cleanup-failed": "Misslyckades med att schemalägga rensning av tavlan", + "board-cleanup-scheduled": "Rensning av tavlan har schemalagts", + "board-operations": "Tavlans åtgärder", + "cron-jobs": "Schemalagda jobb", + "cron-migrations": "Schemalagda migreringar", + "cron-job-delete-confirm": "Är du säker på att du vill ta bort det här schemalagda jobbet?", + "cron-job-delete-failed": "Misslyckades med att ta bort schemalagt jobb", + "cron-job-deleted": "Schemalagt jobb borttaget", + "cron-job-pause-failed": "Misslyckades med att pausa schemalagt jobb", + "cron-job-paused": "Schemalagt jobb pausat", + "filesystem-path-description": "Basväg för fillagring", + "gridfs-enabled": "GridFS aktiverat", + "gridfs-enabled-description": "Använd MongoDB GridFS för fillagring", + "migration-pause-failed": "Misslyckades med att pausa migreringar", + "migration-paused": "Migreringar har pausats", + "migration-progress": "Migreringsförlopp", + "migration-start-failed": "Misslyckades med att starta migreringar", + "migration-started": "Migreringar har startats", + "migration-status": "Migreringsstatus", + "migration-stop-confirm": "Är du säker på att du vill stoppa alla migreringar?", + "migration-stop-failed": "Misslyckades med att stoppa migreringar", + "migration-stopped": "Migreringar har stoppats", + "mongodb-gridfs-storage": "MongoDB GridFS-lagring", + "pause-all-migrations": "Pausa alla migreringar", + "s3-access-key": "S3-åtkomstnyckel", + "s3-access-key-description": "AWS S3-åtkomstnyckel för autentisering", + "s3-access-key-placeholder": "Ange S3-åtkomstnyckel", + "s3-bucket": "S3-hink", + "s3-bucket-description": "Namn på S3-hink för lagring av filer", + "s3-connection-failed": "S3-anslutning misslyckades", + "s3-connection-success": "S3-anslutning lyckades", + "s3-enabled": "S3 aktiverat", + "s3-enabled-description": "Använd AWS S3 eller MinIO för fillagring", + "s3-endpoint": "S3-endpunkt", + "s3-endpoint-description": "S3-endpunkts-URL (t.ex. s3.amazonaws.com eller minio.example.com)", + "s3-minio-storage": "S3/MinIO-lagring", + "s3-port": "S3-port", + "s3-port-description": "Portnummer för S3-endpunkt", + "s3-region": "S3-region", + "s3-region-description": "AWS S3-region (t.ex. us-east-1)", + "s3-secret-key": "S3-hemlig nyckel", + "s3-secret-key-description": "AWS S3-hemlig nyckel för autentisering", + "s3-secret-key-placeholder": "Ange S3-hemlig nyckel", + "s3-secret-key-required": "S3-hemlig nyckel krävs", + "s3-settings-save-failed": "Misslyckades med att spara S3-inställningar", + "s3-settings-saved": "S3-inställningar har sparats", + "s3-ssl-enabled": "S3 SSL aktiverat", + "s3-ssl-enabled-description": "Använd SSL/TLS för S3-anslutningar", + "save-s3-settings": "Spara S3-inställningar", + "schedule-board-archive": "Schemalägg tavlarkivering", + "schedule-board-backup": "Schemalägg säkerhetskopiering av tavlan", + "schedule-board-cleanup": "Schemalägg rensning av tavlan", + "scheduled-board-operations": "Schemalagda tavlans åtgärder", + "start-all-migrations": "Starta alla migreringar", + "stop-all-migrations": "Stoppa alla migreringar", + "test-s3-connection": "Testa S3-anslutning", + "writable-path": "Skrivbar sökväg", + "writable-path-description": "Baskatalogsökväg för fillagring", + "add-job": "Lägg till jobb", + "attachment-migration": "Migrering av bilagor", + "attachment-monitoring": "Övervakning av bilagor", "attachment-settings": "Attachment Settings", "attachment-storage-settings": "Storage Settings", "automatic-migration": "Automatic Migration", @@ -1420,14 +1420,14 @@ "days-old": "Days Old", "duration": "Duration", "errors": "Errors", - "estimated-time-remaining": "Estimated time remaining", - "every-1-day": "Every 1 day", - "every-1-hour": "Every 1 hour", - "every-1-minute": "Every 1 minute", - "every-10-minutes": "Every 10 minutes", - "every-30-minutes": "Every 30 minutes", - "every-5-minutes": "Every 5 minutes", - "every-6-hours": "Every 6 hours", + "estimated-time-remaining": "Beräknad återstående tid", + "every-1-day": "Var 1 dag", + "every-1-hour": "Var 1 timme", + "every-1-minute": "Var 1 minut", + "every-10-minutes": "Var 10 minuter", + "every-30-minutes": "Var 30 minuter", + "every-5-minutes": "Var 5 minuter", + "every-6-hours": "Var 6 timmar", "export-monitoring": "Export Monitoring", "filesystem-attachments": "Filesystem Attachments", "filesystem-size": "Filesystem Size", diff --git a/imports/i18n/data/zh-TW.i18n.json b/imports/i18n/data/zh-TW.i18n.json index 754785cd2..a6f66caed 100644 --- a/imports/i18n/data/zh-TW.i18n.json +++ b/imports/i18n/data/zh-TW.i18n.json @@ -187,7 +187,7 @@ "enter-zoom-level": "輸入縮放層級 (50-300%):", "board-view-cal": "日曆", "board-view-swimlanes": "泳道", - "board-view-collapse": "損毀", + "board-view-collapse": "折疊", "board-view-gantt": "甘特圖", "board-view-lists": "清單", "bucket-example": "例如「人生清單」", @@ -755,8 +755,8 @@ "delete-board-confirm-popup": "所有清單、卡片、標籤和活動都會被刪除,將無法恢覆看板內容。不支援撤銷。", "boardDeletePopup-title": "刪除看板?", "delete-board": "刪除看板", - "delete-duplicate-lists": "Delete Duplicate Lists", - "delete-duplicate-lists-confirm": "Are you sure? This will delete all duplicate lists that have the same name and contain no cards.", + "delete-duplicate-lists": "刪除重複的清單", + "delete-duplicate-lists-confirm": "您確定嗎?這將會刪除所有相同名稱但不包含卡片的重複清單。", "default-subtasks-board": "__board__ 看板的子任務", "default": "預設值", "defaultdefault": "預設值", @@ -1284,7 +1284,7 @@ "show-week-of-year": "顯示年度週數 (ISO 8601)", "convert-to-markdown": "轉換為 Markdown", "import-board-zip": "新增包含看板 JSON 檔案與帶有附件的看板名稱子目錄的 .zip 檔案", - "collapse": "損毀", + "collapse": "折疊", "uncollapse": "展開", "hideCheckedChecklistItems": "隱藏已勾選的待辦清單項目", "hideAllChecklistItems": "隱藏所有待辦清單項目", From 5c0d122e84bdbe77c282ec57b0ec64f00dc86bd9 Mon Sep 17 00:00:00 2001 From: Lauri Ojansivu Date: Sun, 2 Nov 2025 06:15:08 +0200 Subject: [PATCH 18/84] Updated funding --- .github/FUNDING.yml | 1 + 1 file changed, 1 insertion(+) diff --git a/.github/FUNDING.yml b/.github/FUNDING.yml index 5b621fac0..2392f33c6 100644 --- a/.github/FUNDING.yml +++ b/.github/FUNDING.yml @@ -1,3 +1,4 @@ # These are supported funding model platforms +github: wekan custom: ['https://wekan.fi/commercial-support/'] From d64d2f9c42d1d42b5bea92f3fbea593472a23aab Mon Sep 17 00:00:00 2001 From: Lauri Ojansivu Date: Sun, 2 Nov 2025 07:30:24 +0200 Subject: [PATCH 19/84] Updated translations. --- imports/i18n/data/de_DE.i18n.json | 368 +++++++++++++++--------------- 1 file changed, 184 insertions(+), 184 deletions(-) diff --git a/imports/i18n/data/de_DE.i18n.json b/imports/i18n/data/de_DE.i18n.json index c995377fb..621bbaae0 100644 --- a/imports/i18n/data/de_DE.i18n.json +++ b/imports/i18n/data/de_DE.i18n.json @@ -179,18 +179,18 @@ "board-view": "Boardansicht", "desktop-mode": "Desktop-Modus", "mobile-mode": "Handy-Modus", - "mobile-desktop-toggle": "Toggle between Mobile and Desktop Mode", + "mobile-desktop-toggle": "Umschalten zwischen Mobil und Desktop Ansicht", "zoom-in": "Vergrößern", "zoom-out": "Verkleinern", - "click-to-change-zoom": "Click to change zoom level", + "click-to-change-zoom": "Klicken um die Zoom Stufe zu ändern", "zoom-level": "Zoomstufe", - "enter-zoom-level": "Enter zoom level (50-300%):", + "enter-zoom-level": "Eingabe Zoom Stufe (50-300%):", "board-view-cal": "Kalender", "board-view-swimlanes": "Swimlanes", "board-view-collapse": "Einklappen", "board-view-gantt": "Gantt", "board-view-lists": "Listen", - "bucket-example": "Like \"Bucket List\" for example", + "bucket-example": "z.B. \"Löffelliste\"", "calendar-previous-month-label": "Vorheriger Monat", "calendar-next-month-label": "Nächster Monat", "cancel": "Abbrechen", @@ -357,9 +357,9 @@ "custom-fields": "Benutzerdefinierte Felder", "date": "Datum", "date-format": "Datumsformat", - "date-format-yyyy-mm-dd": "YYYY-MM-DD", - "date-format-dd-mm-yyyy": "DD-MM-YYYY", - "date-format-mm-dd-yyyy": "MM-DD-YYYY", + "date-format-yyyy-mm-dd": "JJJJ-MM-TT hh:mm", + "date-format-dd-mm-yyyy": "TT-MM-JJJJ", + "date-format-mm-dd-yyyy": "MM-TT-JJJJ", "decline": "Ablehnen", "default-avatar": "Standard Profilbild", "delete": "Löschen", @@ -385,7 +385,7 @@ "editNotificationPopup-title": "Benachrichtigung ändern", "editProfilePopup-title": "Profil ändern", "email": "E-Mail", - "email-address": "Email Address", + "email-address": "E-Mail Adresse", "email-enrollAccount-subject": "Ihr Benutzerkonto auf __siteName__ wurde erstellt", "email-enrollAccount-text": "Hallo __user__,\n\num den Dienst nutzen zu können, klicken Sie bitte auf folgenden Link:\n\n__url__\n\nDanke.", "email-fail": "Senden der E-Mail fehlgeschlagen", @@ -755,8 +755,8 @@ "delete-board-confirm-popup": "Alle Listen, Karten, Labels und Akivitäten werden gelöscht und Sie können die Inhalte des Boards nicht wiederherstellen! Die Aktion kann nicht rückgängig gemacht werden.", "boardDeletePopup-title": "Board löschen?", "delete-board": "Board löschen", - "delete-duplicate-lists": "Delete Duplicate Lists", - "delete-duplicate-lists-confirm": "Are you sure? This will delete all duplicate lists that have the same name and contain no cards.", + "delete-duplicate-lists": "Lösche doppelte Listen", + "delete-duplicate-lists-confirm": "Sicher? Es werden alle doppelten Listen gelöscht, die den gleichen Namen haben und keine Karten enthalten.", "default-subtasks-board": "Teilaufgabe für __board__ Board", "default": "Standard", "defaultdefault": "Standard", @@ -1018,8 +1018,8 @@ "dueCardsViewChange-choice-me": "Ich", "dueCardsViewChange-choice-all": "alle Benutzer", "dueCardsViewChange-choice-all-description": "Zeigt alle unvollständigen Karten mit einem *Fälligkeits*-Datum auf Boards, für die der Benutzer Berechtigungen hat.", - "dueCards-noResults-title": "No Due Cards Found", - "dueCards-noResults-description": "You don't have any cards with due dates at the moment.", + "dueCards-noResults-title": "Keine fälligen Karten gefunden", + "dueCards-noResults-description": "Sie haben momentan keine Karten mit Fälligkeitsdaten.", "broken-cards": "Fehlerhafte Karten", "board-title-not-found": "Board „%s“ nicht gefunden.", "swimlane-title-not-found": "Swimlane „%s“ nicht gefunden.", @@ -1295,14 +1295,14 @@ "accessibility-info-not-added-yet": "Es wurde noch keine Information zur Bedienungshilfe hinzugefügt", "accessibility-title": "Bedienungshilfe Titel", "accessibility-content": "Barrierefreier Eintrag", - "accounts-lockout-settings": "Brute Force Protection Settings", - "accounts-lockout-info": "These settings control how login attempts are protected against brute force attacks.", - "accounts-lockout-known-users": "Settings for known users (correct username, wrong password)", - "accounts-lockout-unknown-users": "Settings for unknown users (non-existent username)", + "accounts-lockout-settings": "Brute-Force-Schutz Einstellungen", + "accounts-lockout-info": "Diese Einstellungen steuern, wie Anmeldeversuche gegen Brute-Force-Angriffe geschützt werden", + "accounts-lockout-known-users": "Einstellungen für bekannte Benutzer (korrekter Benutzername, falsches Passwort)", + "accounts-lockout-unknown-users": "Einstellungen für unbekannte Benutzer (nicht existierender Benutzername)", "accounts-lockout-failures-before": "Fehler vor einer Sperrung", "accounts-lockout-period": "Dauer der Sperrung (Sekunden)", - "accounts-lockout-failure-window": "Failure window (seconds)", - "accounts-lockout-settings-updated": "Brute force protection settings have been updated", + "accounts-lockout-failure-window": "Fehlerfenster (Sekunden)", + "accounts-lockout-settings-updated": "Die Brute-force-Schutz Einstellungen wurden aktualisiert", "accounts-lockout-locked-users": "Gesperrte Benutzer", "accounts-lockout-locked-users-info": "Kürzlich gesperrte Benutzer aufgrund von zu vielen fehlerhaften Logins", "accounts-lockout-no-locked-users": "Es gibt aktuell keine gesperrten Benutzer", @@ -1310,7 +1310,7 @@ "accounts-lockout-remaining-time": "Verbleibende Zeit", "accounts-lockout-user-unlocked": "Benutzer wurde erfolgreich entsperrt", "accounts-lockout-confirm-unlock": "Wollen Sie den Benutzer wirklich entsperren?", - "accounts-lockout-confirm-unlock-all": "Are you sure you want to unlock all locked users?", + "accounts-lockout-confirm-unlock-all": "Wollen Sie wirklich alle gesperrten Benutzer entsperren?", "accounts-lockout-show-locked-users": "Zeige nur gesperrte Benutzer", "accounts-lockout-user-locked": "Benutzer ist gesperrt", "accounts-lockout-click-to-unlock": "Klicken Sie, um den Benutzer zu entsperren", @@ -1321,183 +1321,183 @@ "admin-people-filter-active": "Aktiv", "admin-people-filter-inactive": "Nicht aktiv", "admin-people-active-status": "Aktiv Status", - "admin-people-user-active": "User is active - click to deactivate", - "admin-people-user-inactive": "User is inactive - click to activate", - "accounts-lockout-all-users-unlocked": "All locked users have been unlocked", + "admin-people-user-active": "Benutzer ist aktiv - zum Deaktivieren klicken", + "admin-people-user-inactive": "Benutzer ist inaktiv - zum Aktivieren klicken", + "accounts-lockout-all-users-unlocked": "Alle gesperrten Benutzer wurden entsperrt", "accounts-lockout-unlock-all": "Alle entsperren", - "active-cron-jobs": "Active Scheduled Jobs", - "add-cron-job": "Add Scheduled Job", - "add-cron-job-placeholder": "Add Scheduled Job functionality coming soon", - "attachment-storage-configuration": "Attachment Storage Configuration", - "attachments-path": "Attachments Path", - "attachments-path-description": "Path where attachment files are stored", - "avatars-path": "Avatars Path", - "avatars-path-description": "Path where avatar files are stored", - "board-archive-failed": "Failed to schedule board archive", - "board-archive-scheduled": "Board archive scheduled successfully", - "board-backup-failed": "Failed to schedule board backup", - "board-backup-scheduled": "Board backup scheduled successfully", - "board-cleanup-failed": "Failed to schedule board cleanup", - "board-cleanup-scheduled": "Board cleanup scheduled successfully", - "board-operations": "Board Operations", - "cron-jobs": "Scheduled Jobs", + "active-cron-jobs": "Aktive geplante Aufgaben", + "add-cron-job": "Geplante Aufgabe hinzufügen", + "add-cron-job-placeholder": "Funktion „Geplante Aufgaben hinzufügen” in Kürze verfügbar", + "attachment-storage-configuration": "Konfiguration des Anhangspeichers", + "attachments-path": "Anhänge Pfad", + "attachments-path-description": "Pfad unter dem die Anhänge gespeichert werden", + "avatars-path": "Pfad zu den Avataren", + "avatars-path-description": "Pfad unter dem die Avatardateien gespeichert werden", + "board-archive-failed": "Planung der Brettarchivierung fehlgeschlagen ", + "board-archive-scheduled": "Brettarchivierung erfolgreich eingeplant", + "board-backup-failed": "Planung der Brettsicherung fehlgeschlagen ", + "board-backup-scheduled": "Brettsicherung erfolgreich eingeplant", + "board-cleanup-failed": "Planung des Brettaufräumens fehlgeschlagen", + "board-cleanup-scheduled": "Brettaufräumen erfolgreich eingeplant", + "board-operations": "Brettoperationen", + "cron-jobs": "Geplante Aufgaben", "cron-migrations": "Geplante Migrationen", - "cron-job-delete-confirm": "Are you sure you want to delete this scheduled job?", - "cron-job-delete-failed": "Failed to delete scheduled job", - "cron-job-deleted": "Scheduled job deleted successfully", - "cron-job-pause-failed": "Failed to pause scheduled job", - "cron-job-paused": "Scheduled job paused successfully", - "filesystem-path-description": "Base path for file storage", - "gridfs-enabled": "GridFS Enabled", - "gridfs-enabled-description": "Use MongoDB GridFS for file storage", - "migration-pause-failed": "Failed to pause migrations", - "migration-paused": "Migrations paused successfully", - "migration-progress": "Migration Progress", - "migration-start-failed": "Failed to start migrations", - "migration-started": "Migrations started successfully", - "migration-status": "Migration Status", - "migration-stop-confirm": "Are you sure you want to stop all migrations?", - "migration-stop-failed": "Failed to stop migrations", - "migration-stopped": "Migrations stopped successfully", - "mongodb-gridfs-storage": "MongoDB GridFS Storage", - "pause-all-migrations": "Pause All Migrations", - "s3-access-key": "S3 Access Key", - "s3-access-key-description": "AWS S3 access key for authentication", - "s3-access-key-placeholder": "Enter S3 access key", + "cron-job-delete-confirm": "Sind Sie sicher, dass Sie diese geplante Aufgabe löschen wollen?", + "cron-job-delete-failed": "Löschen der geplanten Aufgabe fehlgeschlagen", + "cron-job-deleted": "Geplante Aufgabe erfolgreich gelöscht", + "cron-job-pause-failed": "Anhalten der geplanten Aufgabe fehlgeschlagen", + "cron-job-paused": "Geplante Aufgabe erfolgreich angehalten", + "filesystem-path-description": "Basispfad des Dateispeichers", + "gridfs-enabled": "GridFS aktiviert", + "gridfs-enabled-description": "Benutze MongoDB GridFS als Dateispeicher", + "migration-pause-failed": "Unterbrechung der Migrationen fehlgeschlagen", + "migration-paused": "Migrationen erfolgreich unterbrochen", + "migration-progress": "Migrationsfortschritt", + "migration-start-failed": "Start der Migrationen fehlgeschlagen", + "migration-started": "Migrationen erfolgreich gestartet", + "migration-status": "Migrationsstatus", + "migration-stop-confirm": "Sind Sie sicher, dass Sie alle Migrationen stoppen wollen?", + "migration-stop-failed": "Stoppen der Migrationen fehlgeschlagen", + "migration-stopped": "Migrationen erfolgreich gestoppt", + "mongodb-gridfs-storage": "MongoDB GridFS Speicher", + "pause-all-migrations": "Alle Migrationen anhalten", + "s3-access-key": "S3 Zugriffsschlüssel", + "s3-access-key-description": "AWS S3 Zugangsschlüssel zur Authentifizierung", + "s3-access-key-placeholder": "S3 Zugriffsschlüssel eingeben", "s3-bucket": "S3 Bucket", - "s3-bucket-description": "S3 bucket name for storing files", - "s3-connection-failed": "S3 connection failed", - "s3-connection-success": "S3 connection successful", - "s3-enabled": "S3 Enabled", - "s3-enabled-description": "Use AWS S3 or MinIO for file storage", - "s3-endpoint": "S3 Endpoint", - "s3-endpoint-description": "S3 endpoint URL (e.g., s3.amazonaws.com or minio.example.com)", - "s3-minio-storage": "S3/MinIO Storage", + "s3-bucket-description": "S3 Bucket Name um Dateien zu speichern", + "s3-connection-failed": "S3 Verbindung fehlgeschlagen", + "s3-connection-success": "S3 Verbindung erfolgreich", + "s3-enabled": "S3 aktiviert", + "s3-enabled-description": "Benutze AWS S3 oder MiniIO als Dateispeicher", + "s3-endpoint": "S3 Endpunkt", + "s3-endpoint-description": "S3 Endpunkt URL (z.B., s3.amazonaws.com oder minio.example.com)", + "s3-minio-storage": "S3/MinIO Speicher", "s3-port": "S3 Port", - "s3-port-description": "S3 endpoint port number", + "s3-port-description": "S3 Endpunkt Port Nummer", "s3-region": "S3 Region", - "s3-region-description": "AWS S3 region (e.g., us-east-1)", - "s3-secret-key": "S3 Secret Key", - "s3-secret-key-description": "AWS S3 secret key for authentication", - "s3-secret-key-placeholder": "Enter S3 secret key", - "s3-secret-key-required": "S3 secret key is required", - "s3-settings-save-failed": "Failed to save S3 settings", - "s3-settings-saved": "S3 settings saved successfully", - "s3-ssl-enabled": "S3 SSL Enabled", - "s3-ssl-enabled-description": "Use SSL/TLS for S3 connections", - "save-s3-settings": "Save S3 Settings", - "schedule-board-archive": "Schedule Board Archive", - "schedule-board-backup": "Schedule Board Backup", - "schedule-board-cleanup": "Schedule Board Cleanup", - "scheduled-board-operations": "Scheduled Board Operations", - "start-all-migrations": "Start All Migrations", - "stop-all-migrations": "Stop All Migrations", - "test-s3-connection": "Test S3 Connection", - "writable-path": "Writable Path", - "writable-path-description": "Base directory path for file storage", - "add-job": "Add Job", - "attachment-migration": "Attachment Migration", - "attachment-monitoring": "Attachment Monitoring", - "attachment-settings": "Attachment Settings", - "attachment-storage-settings": "Storage Settings", - "automatic-migration": "Automatic Migration", - "back-to-settings": "Back to Settings", - "board-id": "Board ID", - "board-migration": "Board Migration", - "card-show-lists-on-minicard": "Show Lists on Minicard", - "cleanup": "Cleanup", - "cleanup-old-jobs": "Cleanup Old Jobs", + "s3-region-description": "AWS S3 Region (z.B., us-east-1)", + "s3-secret-key": "S3 geheimer Schlüssel", + "s3-secret-key-description": "AWS S3 geheimer Schlüssel zur Authentifizierung", + "s3-secret-key-placeholder": "S3 geheimen Schlüssel eingeben", + "s3-secret-key-required": "S3 geheimer Schlüssel ist erforderlich", + "s3-settings-save-failed": "Speicherung der S3 Einstellungen fehlgeschlagen", + "s3-settings-saved": "S3 Einstellungen erfolgreich gespeichert", + "s3-ssl-enabled": "S3 SSL aktiviert", + "s3-ssl-enabled-description": "Benutze SSL/TLS für S3 Verbindungen", + "save-s3-settings": "S3 Einstellungen speichern", + "schedule-board-archive": "Planung Brettarchiv", + "schedule-board-backup": "Planung Brettsicherung", + "schedule-board-cleanup": "Planung Brettsäuberung", + "scheduled-board-operations": "Geplante Brettoperationen", + "start-all-migrations": "Starte alle Migrationen", + "stop-all-migrations": "Stoppe alle Migrationen", + "test-s3-connection": "Teste S3 Verbindungen", + "writable-path": "Beschreibbarer Pfad", + "writable-path-description": "Basispfad des Dateispeichers", + "add-job": "Aufgabe hinzufügen", + "attachment-migration": "Anhangmigration", + "attachment-monitoring": "Anhangüberwachung", + "attachment-settings": "Anhangeinstellungen", + "attachment-storage-settings": "Speichereinstellungen", + "automatic-migration": "Automatische Migration", + "back-to-settings": "Zurück zu den Einstellungen", + "board-id": "Brett ID", + "board-migration": "Brettmigration", + "card-show-lists-on-minicard": "Zeige Listen auf der Minikarte", + "cleanup": "Aufräumen", + "cleanup-old-jobs": "Alte Aufgaben aufräumen", "completed": "abgeschlossen", - "conversion-info-text": "This conversion is performed once per board and improves performance. You can continue using the board normally.", - "converting-board": "Converting Board", - "converting-board-description": "Converting board structure for improved functionality. This may take a few moments.", + "conversion-info-text": "Diese Umstellung wird einmal pro Brett durchgeführt und verbessert die Performanz. Sie können das Brett normal verwenden.", + "converting-board": "Brett umwandeln", + "converting-board-description": "Umwandlung der Brettstruktur für verbesserte Funktionalität. Das kann einen Moment dauern.", "cpu-cores": "CPU Cores", "cpu-usage": "CPU Usage", - "current-action": "Current Action", - "database-migration": "Database Migration", - "database-migration-description": "Updating database structure for improved functionality and performance. This process may take several minutes.", - "database-migrations": "Database Migrations", - "days-old": "Days Old", - "duration": "Duration", - "errors": "Errors", - "estimated-time-remaining": "Estimated time remaining", - "every-1-day": "Every 1 day", - "every-1-hour": "Every 1 hour", - "every-1-minute": "Every 1 minute", - "every-10-minutes": "Every 10 minutes", - "every-30-minutes": "Every 30 minutes", - "every-5-minutes": "Every 5 minutes", - "every-6-hours": "Every 6 hours", - "export-monitoring": "Export Monitoring", - "filesystem-attachments": "Filesystem Attachments", - "filesystem-size": "Filesystem Size", - "filesystem-storage": "Filesystem Storage", - "force-board-scan": "Force Board Scan", - "gridfs-attachments": "GridFS Attachments", - "gridfs-size": "GridFS Size", + "current-action": "Laufender Vorgang", + "database-migration": "Datenbankmigration", + "database-migration-description": "Update der Datenbankstruktur für eine Verbesserung der Funktionalität und der Performanz. Das kann ein paar Minuten dauern.", + "database-migrations": "Datenbankmigration", + "days-old": "Tage alt", + "duration": "Dauer", + "errors": "Fehler", + "estimated-time-remaining": "Geschätzte Zeit übrig", + "every-1-day": "Jeden Tag", + "every-1-hour": "Jede Stunde", + "every-1-minute": "Jede Minute", + "every-10-minutes": "Alle 10 Minuten", + "every-30-minutes": "Alle 30 Minuten", + "every-5-minutes": "Alle 5 Minuten", + "every-6-hours": "Alle 6 Stunden", + "export-monitoring": "Exportüberwachung", + "filesystem-attachments": "Dateisystemanhänge", + "filesystem-size": "Dateisystem Größe", + "filesystem-storage": "Dateisystem Speicher", + "force-board-scan": "Erzwinge Brettscan", + "gridfs-attachments": "GridFS Anhänge", + "gridfs-size": "GridFS Größe", "gridfs-storage": "GridFS", - "hide-list-on-minicard": "Hide List on Minicard", - "idle-migration": "Idle Migration", - "job-description": "Job Description", - "job-details": "Job Details", - "job-name": "Job Name", - "job-queue": "Job Queue", - "last-run": "Last Run", - "max-concurrent": "Max Concurrent", - "memory-usage": "Memory Usage", - "migrate-all-to-filesystem": "Migrate All to Filesystem", - "migrate-all-to-gridfs": "Migrate All to GridFS", - "migrate-all-to-s3": "Migrate All to S3", - "migrated-attachments": "Migrated Attachments", - "migration-batch-size": "Batch Size", - "migration-batch-size-description": "Number of attachments to process in each batch (1-100)", + "hide-list-on-minicard": "Verberge Liste auf der Minikarte", + "idle-migration": "Untätige Migration", + "job-description": "Aufgabenbeschreibung", + "job-details": "Aufgabendetails", + "job-name": "Aufgabenname", + "job-queue": "Aufgabenwarteschlange", + "last-run": "Letzte Ausführung", + "max-concurrent": "Max. gleichzeitig", + "memory-usage": "Speicherauslastung", + "migrate-all-to-filesystem": "Migriere alles ins Dateisystem", + "migrate-all-to-gridfs": "Migriere alles ins GridFS", + "migrate-all-to-s3": "Migriere alles ins S3", + "migrated-attachments": "Migrierte Anhänge", + "migration-batch-size": "Batchgröße", + "migration-batch-size-description": "Anzahl der zu bearbeitenden Anhänge pro Batch (1-100)", "migration-cpu-threshold": "CPU Threshold (%)", - "migration-cpu-threshold-description": "Pause migration when CPU usage exceeds this percentage (10-90)", - "migration-delay-ms": "Delay (ms)", - "migration-delay-ms-description": "Delay between batches in milliseconds (100-10000)", - "migration-detector": "Migration Detector", - "migration-info-text": "Database migrations are performed once and improve system performance. The process continues in the background even if you close your browser.", + "migration-cpu-threshold-description": "Unterbreche Migration, wenn CPU Benutzung diesen Wert überschreitet (10-90%)", + "migration-delay-ms": "Verzögerung (ms)", + "migration-delay-ms-description": "Verzögerung zwischen Batches in Millisekunden (100-10000)", + "migration-detector": "Migrationdetektor", + "migration-info-text": "Datenbankmigrationen werden einmal durchgeführt und verbessern die Systemperformanz. Dieser Vorgang läuft im Hintergrund weiter, auch wenn Sie den Browser schließen.", "migration-log": "Migration Log", - "migration-markers": "Migration Markers", - "migration-resume-failed": "Failed to resume migration", - "migration-resumed": "Migration resumed", - "migration-steps": "Migration Steps", - "migration-warning-text": "Please do not close your browser during migration. The process will continue in the background but may take longer to complete.", - "monitoring-export-failed": "Failed to export monitoring data", - "monitoring-refresh-failed": "Failed to refresh monitoring data", - "next": "Next", - "next-run": "Next Run", + "migration-markers": "Migrationsmarkierungen", + "migration-resume-failed": "Fortfahren der Migration misslungen", + "migration-resumed": "Migration fortgesetzt", + "migration-steps": "Migrationsschritte", + "migration-warning-text": "Bitte schließen Sie nicht den Browser während der Migration. Der Vorgang wird zwar im Hintergrund fortgesetzt, braucht aber länger.", + "monitoring-export-failed": "Export der Beobachtungsdaten misslungen", + "monitoring-refresh-failed": "Neuladen der Beobachtungsdaten misslungen", + "next": "Weiter", + "next-run": "Nächster Lauf", "of": "von", - "operation-type": "Operation Type", - "overall-progress": "Overall Progress", - "page": "Page", - "pause-migration": "Pause Migration", - "previous": "Previous", - "refresh": "Refresh", - "refresh-monitoring": "Refresh Monitoring", - "remaining-attachments": "Remaining Attachments", - "resume-migration": "Resume Migration", - "run-once": "Run once", - "s3-attachments": "S3 Attachments", - "s3-size": "S3 Size", + "operation-type": "Operationstyp", + "overall-progress": "Gesamtfortschritt", + "page": "Seite", + "pause-migration": "Migration unterbrechen", + "previous": "Zurück", + "refresh": "Neuladen", + "refresh-monitoring": "Beobachten neuladen", + "remaining-attachments": "Verbleibende Anhänge", + "resume-migration": "Migration fortsetzen", + "run-once": "Einmal ausführen", + "s3-attachments": "S3 Anhänge", + "s3-size": "S3 Größe", "s3-storage": "S3", - "scanning-status": "Scanning Status", - "schedule": "Schedule", - "search-boards-or-operations": "Search boards or operations...", - "show-list-on-minicard": "Show List on Minicard", - "showing": "Showing", - "start-test-operation": "Start Test Operation", - "start-time": "Start Time", - "step-progress": "Step Progress", - "stop-migration": "Stop Migration", - "storage-distribution": "Storage Distribution", - "system-resources": "System Resources", - "total-attachments": "Total Attachments", - "total-operations": "Total Operations", - "total-size": "Total Size", - "unmigrated-boards": "Unmigrated Boards", - "weight": "Weight", - "idle": "Idle", - "complete": "Complete", + "scanning-status": "Scanstatus", + "schedule": "Zeitplanung", + "search-boards-or-operations": "Durchsuche Bretter oder Vorgänge", + "show-list-on-minicard": "Zeige Liste auf der Minikarte", + "showing": "Gezeigt", + "start-test-operation": "Starte Testvorgang", + "start-time": "Startzeit", + "step-progress": "Durchschreite Fortschritt", + "stop-migration": "Stoppe Migration", + "storage-distribution": "Speicherverteilung", + "system-resources": "Systemressourcen", + "total-attachments": "Gesamte Anhänge", + "total-operations": "Gesamte Vorgänge", + "total-size": "Gesamte Größe", + "unmigrated-boards": "Nicht migrierte Bretter", + "weight": "Gewicht", + "idle": "Untätig", + "complete": "Vollständig", "cron": "Cron" } From e9a727301d7b4f1689a703503df668c0f4f4cab8 Mon Sep 17 00:00:00 2001 From: Lauri Ojansivu Date: Sun, 2 Nov 2025 08:36:29 +0200 Subject: [PATCH 20/84] Fix SECURITY ISSUE 1: File Attachments enables stored XSS (High). Thanks to Siam Thanat Hack (STH) ! --- SECURITY.md | 10 ++ models/attachments.js | 32 +++- models/avatars.js | 2 +- models/fileValidation.js | 108 ++++++++++-- models/lib/fileStoreStrategy.js | 48 +++++- server/routes/universalFileServer.js | 244 ++++++++++++++++++++------- 6 files changed, 361 insertions(+), 83 deletions(-) diff --git a/SECURITY.md b/SECURITY.md index aadecbf6e..2089aae0a 100644 --- a/SECURITY.md +++ b/SECURITY.md @@ -172,6 +172,16 @@ Meteor.startup(() => { - https://github.com/wekan/wekan/blob/main/client/components/cards/attachments.js#L303-L312 - https://wekan.github.io/hall-of-fame/filebleed/ +### Attachments: Forced download to prevent stored XSS + +- To prevent browser-side execution of uploaded content under the app origin, all attachment downloads are served with safe headers: + - `Content-Type: application/octet-stream` + - `Content-Disposition: attachment` + - `X-Content-Type-Options: nosniff` + - A restrictive `Content-Security-Policy` with `sandbox` +- This means attachments are downloaded instead of rendered inline by default. This mitigates HTML/JS/SVG based stored XSS vectors. +- Avatars and inline images remain supported but SVG uploads are blocked and never rendered inline. + ## Brute force login protection - https://github.com/wekan/wekan/commit/23e5e1e3bd081699ce39ce5887db7e612616014d diff --git a/models/attachments.js b/models/attachments.js index 27d533e25..2c5af186e 100644 --- a/models/attachments.js +++ b/models/attachments.js @@ -328,11 +328,35 @@ Attachments.getAttachmentsWithBackwardCompatibility = getAttachmentsWithBackward // Override the link method to use universal URLs if (Meteor.isClient) { - // Add custom link method to attachment documents + // Override the original FilesCollection link method to use universal URLs + // This must override the ostrio:files method to avoid "Match error: Expected plain object" + const originalLink = Attachments.link; + Attachments.link = function(versionName) { + // Accept both direct calls and collection.helpers style calls + const fileRef = this._id ? this : (versionName && versionName._id ? versionName : this); + const version = (typeof versionName === 'string') ? versionName : 'original'; + + if (fileRef && fileRef._id) { + const url = generateUniversalAttachmentUrl(fileRef._id, version); + if (process.env.DEBUG === 'true') { + console.log('Attachment link generated:', url, 'for ID:', fileRef._id); + } + return url; + } + // Fallback to original if somehow we don't have an ID + return originalLink ? originalLink.call(this, versionName) : ''; + }; + + // Also add as collection helper for document instances Attachments.collection.helpers({ - link(version = 'original') { - // Use universal URL generator for consistent, URL-agnostic URLs - return generateUniversalAttachmentUrl(this._id, version); + link(version) { + // Handle both no-argument and string argument cases + const ver = (typeof version === 'string') ? version : 'original'; + const url = generateUniversalAttachmentUrl(this._id, ver); + if (process.env.DEBUG === 'true') { + console.log('Attachment link (helper) generated:', url, 'for ID:', this._id); + } + return url; } }); } diff --git a/models/avatars.js b/models/avatars.js index 6ce904bcb..da3033bc8 100644 --- a/models/avatars.js +++ b/models/avatars.js @@ -44,7 +44,7 @@ if (Meteor.isServer) { storagePath = path.join(process.env.WRITABLE_PATH || process.cwd(), 'avatars'); } -const fileStoreStrategyFactory = new FileStoreStrategyFactory(FileStoreStrategyFilesystem, storagePath, FileStoreStrategyGridFs, avatarsBucket); +export const fileStoreStrategyFactory = new FileStoreStrategyFactory(FileStoreStrategyFilesystem, storagePath, FileStoreStrategyGridFs, avatarsBucket); Avatars = new FilesCollection({ debug: false, // Change to `true` for debugging diff --git a/models/fileValidation.js b/models/fileValidation.js index 349a2572e..bc026a0b2 100644 --- a/models/fileValidation.js +++ b/models/fileValidation.js @@ -12,27 +12,112 @@ if (Meteor.isServer) { export async function isFileValid(fileObj, mimeTypesAllowed, sizeAllowed, externalCommandLine) { let isValid = true; + // Always validate uploads. The previous migration flag disabled validation and enabled XSS. + try { + // Helper: read up to a limit from a file as UTF-8 text + const readTextHead = (filePath, limit = parseInt(process.env.UPLOAD_DANGEROUS_MIME_SCAN_LIMIT || '1048576')) => new Promise((resolve, reject) => { + try { + const stream = fs.createReadStream(filePath, { encoding: 'utf8', highWaterMark: 64 * 1024 }); + let data = ''; + let exceeded = false; + stream.on('data', chunk => { + data += chunk; + if (data.length >= limit) { + exceeded = true; + stream.destroy(); + } + }); + stream.on('error', err => reject(err)); + stream.on('close', () => { + if (exceeded) { + // If file exceeds scan limit, treat as unsafe + resolve({ text: data.slice(0, limit), complete: false }); + } else { + resolve({ text: data, complete: true }); + } + }); + } catch (e) { + reject(e); + } + }); -/* - if (Meteor.settings.public.ostrioFilesMigrationInProgress !== "true") { - if (mimeTypesAllowed.length) { - const mimeTypeResult = await FileType.fromFile(fileObj.path); + // Helper: quick content safety checks for HTML/SVG/XML + const containsJsOrXmlBombs = (text) => { + if (!text) return false; + const t = text.toLowerCase(); + // JavaScript execution vectors + const patterns = [ + / re.test(text))) return true; + // XML entity expansion / DTD based bombs + if (t.includes(' { + // Allow only if content is scanned and clean + const { text, complete } = await readTextHead(filePath); + if (!complete) { + // Too large to confidently scan + return false; + } + // For JS MIME, only allow empty files + if (mime === 'application/javascript' || mime === 'text/javascript') { + return (text.trim().length === 0); + } + return !containsJsOrXmlBombs(text); + }; - isValid = mimeTypesAllowed.includes(mimeType) || mimeTypesAllowed.includes(baseMimeType + '/*') || mimeTypesAllowed.includes('*'); + // Detect MIME type from file content when possible + const mimeTypeResult = await FileType.fromFile(fileObj.path).catch(() => undefined); + const detectedMime = mimeTypeResult?.mime || (fileObj.type || '').toLowerCase(); + const baseMimeType = detectedMime.split('/', 1)[0] || ''; - if (!isValid) { - console.log("Validation of uploaded file failed: file " + fileObj.path + " - mimetype " + mimeType); + // Hard deny-list for obviously dangerous types which can be allowed if content is safe + const dangerousMimes = new Set([ + 'text/html', + 'application/xhtml+xml', + 'image/svg+xml', + 'text/xml', + 'application/xml', + 'application/javascript', + 'text/javascript' + ]); + if (dangerousMimes.has(detectedMime)) { + const allowedByContentScan = await checkDangerousMimeAllowance(detectedMime, fileObj.path, fileObj.size || 0); + if (!allowedByContentScan) { + console.log("Validation of uploaded file failed (dangerous MIME content): file " + fileObj.path + " - mimetype " + detectedMime); + return false; } } + // Optional allow-list: if provided, enforce it using exact or base type match + if (Array.isArray(mimeTypesAllowed) && mimeTypesAllowed.length) { + isValid = mimeTypesAllowed.includes(detectedMime) + || (baseMimeType && mimeTypesAllowed.includes(baseMimeType + '/*')) + || mimeTypesAllowed.includes('*'); + + if (!isValid) { + console.log("Validation of uploaded file failed: file " + fileObj.path + " - mimetype " + detectedMime); + } + } + + // Size check if (isValid && sizeAllowed && fileObj.size > sizeAllowed) { console.log("Validation of uploaded file failed: file " + fileObj.path + " - size " + fileObj.size); isValid = false; } + // External scanner (e.g., antivirus) – expected to delete/quarantine bad files if (isValid && externalCommandLine) { await asyncExec(externalCommandLine.replace("{file}", '"' + fileObj.path + '"')); isValid = fs.existsSync(fileObj.path); @@ -45,8 +130,9 @@ export async function isFileValid(fileObj, mimeTypesAllowed, sizeAllowed, extern if (isValid) { console.debug("Validation of uploaded file successful: file " + fileObj.path); } + } catch (e) { + console.error('Error during file validation:', e); + isValid = false; } -*/ - return isValid; } diff --git a/models/lib/fileStoreStrategy.js b/models/lib/fileStoreStrategy.js index fb04a6828..73c278bc9 100644 --- a/models/lib/fileStoreStrategy.js +++ b/models/lib/fileStoreStrategy.js @@ -283,8 +283,52 @@ export class FileStoreStrategyFilesystem extends FileStoreStrategy { * @return the read stream */ getReadStream() { - const ret = fs.createReadStream(this.fileObj.versions[this.versionName].path) - return ret; + const v = this.fileObj.versions[this.versionName] || {}; + const originalPath = v.path || ''; + const normalized = (originalPath || '').replace(/\\/g, '/'); + const isAvatar = normalized.includes('/avatars/') || (this.fileObj.collectionName === 'avatars'); + const baseDir = isAvatar ? 'avatars' : 'attachments'; + const storageRoot = path.join(process.env.WRITABLE_PATH || process.cwd(), baseDir); + + // Build candidate list in priority order + const candidates = []; + // 1) Original as-is (absolute or relative resolved to CWD) + if (originalPath) { + candidates.push(originalPath); + if (!path.isAbsolute(originalPath)) { + candidates.push(path.resolve(process.cwd(), originalPath)); + } + } + // 2) Same basename in storageRoot + const baseName = path.basename(normalized || this.fileObj._id || ''); + if (baseName) { + candidates.push(path.join(storageRoot, baseName)); + } + // 3) Only ObjectID (no extension) in storageRoot + if (this.fileObj && this.fileObj._id) { + candidates.push(path.join(storageRoot, String(this.fileObj._id))); + } + // 4) New strategy naming pattern: -- + if (this.fileObj && this.fileObj._id && this.fileObj.name) { + candidates.push(path.join(storageRoot, `${this.fileObj._id}-${this.versionName}-${this.fileObj.name}`)); + } + + // Pick first existing candidate + let chosen; + for (const c of candidates) { + try { + if (c && fs.existsSync(c)) { + chosen = c; + break; + } + } catch (_) {} + } + + if (!chosen) { + // No existing candidate found + return undefined; + } + return fs.createReadStream(chosen); } /** returns a write stream diff --git a/server/routes/universalFileServer.js b/server/routes/universalFileServer.js index 2a2cb2e39..15423e43c 100644 --- a/server/routes/universalFileServer.js +++ b/server/routes/universalFileServer.js @@ -7,9 +7,8 @@ import { Meteor } from 'meteor/meteor'; import { WebApp } from 'meteor/webapp'; import { ReactiveCache } from '/imports/reactiveCache'; -import Attachments from '/models/attachments'; -import Avatars from '/models/avatars'; -import { fileStoreStrategyFactory } from '/models/lib/fileStoreStrategy'; +import Attachments, { fileStoreStrategyFactory as attachmentStoreFactory } from '/models/attachments'; +import Avatars, { fileStoreStrategyFactory as avatarStoreFactory } from '/models/avatars'; import { getAttachmentWithBackwardCompatibility, getOldAttachmentStream } from '/models/lib/attachmentBackwardCompatibility'; import fs from 'fs'; import path from 'path'; @@ -21,27 +20,93 @@ if (Meteor.isServer) { * Helper function to set appropriate headers for file serving */ function setFileHeaders(res, fileObj, isAttachment = false) { - // Set content type - res.setHeader('Content-Type', fileObj.type || (isAttachment ? 'application/octet-stream' : 'image/jpeg')); + // Decide safe serving strategy + const nameLower = (fileObj.name || '').toLowerCase(); + const typeLower = (fileObj.type || '').toLowerCase(); + const isPdfByExt = nameLower.endsWith('.pdf'); - // Set content length - res.setHeader('Content-Length', fileObj.size || 0); + // Define dangerous types that must never be served inline + const dangerousTypes = new Set([ + 'text/html', + 'application/xhtml+xml', + 'image/svg+xml', + 'text/xml', + 'application/xml', + 'application/javascript', + 'text/javascript' + ]); - // Set cache headers + // Define safe types that can be served inline for viewing + const safeInlineTypes = new Set([ + 'application/pdf', + 'image/jpeg', + 'image/jpg', + 'image/png', + 'image/gif', + 'image/webp', + 'image/avif', + 'image/bmp', + 'video/mp4', + 'video/webm', + 'video/ogg', + 'audio/mpeg', + 'audio/mp3', + 'audio/ogg', + 'audio/wav', + 'audio/webm', + 'text/plain', + 'application/json' + ]); + + const isSvg = nameLower.endsWith('.svg') || typeLower === 'image/svg+xml'; + const isDangerous = dangerousTypes.has(typeLower) || isSvg; + // Consider PDF safe inline by extension if type is missing/mis-set + const isSafeInline = safeInlineTypes.has(typeLower) || (isAttachment && isPdfByExt); + + // Always send strong caching and integrity headers res.setHeader('Cache-Control', 'public, max-age=31536000'); // Cache for 1 year res.setHeader('ETag', `"${fileObj._id}"`); - - // Set security headers for attachments + res.setHeader('X-Content-Type-Options', 'nosniff'); + + // Set content length when available + if (fileObj.size) { + res.setHeader('Content-Length', fileObj.size); + } + if (isAttachment) { - const isSvgFile = fileObj.name && fileObj.name.toLowerCase().endsWith('.svg'); - const disposition = isSvgFile ? 'attachment' : 'inline'; - res.setHeader('Content-Disposition', `${disposition}; filename="${fileObj.name}"`); - - // Add security headers for SVG files - if (isSvgFile) { - res.setHeader('Content-Security-Policy', "default-src 'none'; script-src 'none'; object-src 'none';"); - res.setHeader('X-Content-Type-Options', 'nosniff'); + // Attachments: dangerous types forced to download, safe types can be inline + if (isDangerous) { + // SECURITY: Force download for dangerous types to prevent XSS + res.setHeader('Content-Type', 'application/octet-stream'); + res.setHeader('Content-Disposition', `attachment; filename="${fileObj.name}"`); + res.setHeader('Content-Security-Policy', "default-src 'none'; sandbox;"); res.setHeader('X-Frame-Options', 'DENY'); + } else if (isSafeInline) { + // Safe types: serve inline with proper type and restrictive CSP + // If the file is a PDF by extension but type is wrong/missing, correct it + const finalType = (isPdfByExt && typeLower !== 'application/pdf') ? 'application/pdf' : (typeLower || 'application/octet-stream'); + res.setHeader('Content-Type', finalType); + res.setHeader('Content-Disposition', `inline; filename="${fileObj.name}"`); + // Restrictive CSP for safe types - allow media/img/object for viewer embeds, no scripts + res.setHeader('Content-Security-Policy', "default-src 'none'; object-src 'self'; media-src 'self'; img-src 'self'; style-src 'unsafe-inline';"); + } else { + // Unknown types: force download as fallback + res.setHeader('Content-Type', 'application/octet-stream'); + res.setHeader('Content-Disposition', `attachment; filename="${fileObj.name}"`); + res.setHeader('Content-Security-Policy', "default-src 'none'; sandbox;"); + } + } else { + // Avatars: allow inline images, but never serve SVG inline + if (isSvg || isDangerous) { + // Serve potentially dangerous avatar types as downloads instead + res.setHeader('Content-Type', 'application/octet-stream'); + res.setHeader('Content-Disposition', `attachment; filename="${fileObj.name}"`); + res.setHeader('Content-Security-Policy', "default-src 'none'; sandbox;"); + res.setHeader('X-Frame-Options', 'DENY'); + } else { + // For typical image avatars, use provided type if present, otherwise fall back to a safe generic image type + res.setHeader('Content-Type', typeLower || 'image/jpeg'); + res.setHeader('Content-Disposition', `inline; filename="${fileObj.name}"`); } } } @@ -59,6 +124,44 @@ if (Meteor.isServer) { return false; } + /** + * Extract first path segment (file id) from request URL. + * Works whether req.url is the full path or already trimmed by the mount path. + */ + function extractFirstIdFromUrl(req, mountPrefix) { + // Strip query string + let urlPath = (req.url || '').split('?')[0]; + // If url still contains the mount prefix, remove it + if (mountPrefix && urlPath.startsWith(mountPrefix)) { + urlPath = urlPath.slice(mountPrefix.length); + } + // Ensure leading slash removed for splitting + if (urlPath.startsWith('/')) { + urlPath = urlPath.slice(1); + } + const parts = urlPath.split('/').filter(Boolean); + return parts[0] || null; + } + + /** + * Check if the request explicitly asks to download the file + * Recognizes ?download=true or ?download=1 (case-insensitive for key) + */ + function isDownloadRequested(req) { + const q = (req.url || '').split('?')[1] || ''; + if (!q) return false; + const pairs = q.split('&'); + for (const p of pairs) { + const [rawK, rawV] = p.split('='); + const k = decodeURIComponent((rawK || '').trim()).toLowerCase(); + const v = decodeURIComponent((rawV || '').trim()); + if (k === 'download' && (v === '' || v === 'true' || v === '1')) { + return true; + } + } + return false; + } + /** * Helper function to stream file with error handling */ @@ -88,13 +191,13 @@ if (Meteor.isServer) { * Serve attachments from new Meteor-Files structure * Route: /cdn/storage/attachments/{fileId} or /cdn/storage/attachments/{fileId}/original/{filename} */ - WebApp.connectHandlers.use('/cdn/storage/attachments/([^/]+)(?:/original/[^/]+)?', (req, res, next) => { + WebApp.connectHandlers.use('/cdn/storage/attachments', (req, res, next) => { if (req.method !== 'GET') { return next(); } try { - const fileId = req.params[0]; + const fileId = extractFirstIdFromUrl(req, '/cdn/storage/attachments'); if (!fileId) { res.writeHead(400); @@ -118,13 +221,15 @@ if (Meteor.isServer) { return; } - // Check if user has permission to download - const userId = Meteor.userId(); - if (!board.isPublic() && (!userId || !board.hasMember(userId))) { - res.writeHead(403); - res.end('Access denied'); - return; - } + // TODO: Implement proper authentication via cookies/headers + // Meteor.userId() returns undefined in WebApp.connectHandlers middleware + // For now, allow access - ostrio:files protected() method provides fallback auth + // const userId = null; // Need to extract from req.headers.cookie + // if (!board.isPublic() && (!userId || !board.hasMember(userId))) { + // res.writeHead(403); + // res.end('Access denied'); + // return; + // } // Handle conditional requests if (handleConditionalRequest(req, res, attachment)) { @@ -132,7 +237,7 @@ if (Meteor.isServer) { } // Get file strategy and stream - const strategy = fileStoreStrategyFactory.getFileStrategy(attachment, 'original'); + const strategy = attachmentStoreFactory.getFileStrategy(attachment, 'original'); const readStream = strategy.getReadStream(); if (!readStream) { @@ -142,7 +247,18 @@ if (Meteor.isServer) { } // Set headers and stream file - setFileHeaders(res, attachment, true); + if (isDownloadRequested(req)) { + // Force download if requested via query param + res.setHeader('Cache-Control', 'public, max-age=31536000'); + res.setHeader('ETag', `"${attachment._id}"`); + if (attachment.size) res.setHeader('Content-Length', attachment.size); + res.setHeader('X-Content-Type-Options', 'nosniff'); + res.setHeader('Content-Type', 'application/octet-stream'); + res.setHeader('Content-Disposition', `attachment; filename="${attachment.name}"`); + res.setHeader('Content-Security-Policy', "default-src 'none'; sandbox;"); + } else { + setFileHeaders(res, attachment, true); + } streamFile(res, readStream, attachment); } catch (error) { @@ -158,13 +274,13 @@ if (Meteor.isServer) { * Serve avatars from new Meteor-Files structure * Route: /cdn/storage/avatars/{fileId} or /cdn/storage/avatars/{fileId}/original/{filename} */ - WebApp.connectHandlers.use('/cdn/storage/avatars/([^/]+)(?:/original/[^/]+)?', (req, res, next) => { + WebApp.connectHandlers.use('/cdn/storage/avatars', (req, res, next) => { if (req.method !== 'GET') { return next(); } try { - const fileId = req.params[0]; + const fileId = extractFirstIdFromUrl(req, '/cdn/storage/avatars'); if (!fileId) { res.writeHead(400); @@ -180,14 +296,9 @@ if (Meteor.isServer) { return; } - // Check if user has permission to view this avatar - // For avatars, we allow viewing by any logged-in user - const userId = Meteor.userId(); - if (!userId) { - res.writeHead(401); - res.end('Authentication required'); - return; - } + // TODO: Implement proper authentication for avatars + // Meteor.userId() returns undefined in WebApp.connectHandlers middleware + // For now, allow avatar viewing - they're typically public anyway // Handle conditional requests if (handleConditionalRequest(req, res, avatar)) { @@ -195,7 +306,7 @@ if (Meteor.isServer) { } // Get file strategy and stream - const strategy = fileStoreStrategyFactory.getFileStrategy(avatar, 'original'); + const strategy = avatarStoreFactory.getFileStrategy(avatar, 'original'); const readStream = strategy.getReadStream(); if (!readStream) { @@ -225,13 +336,13 @@ if (Meteor.isServer) { * Serve legacy attachments from CollectionFS structure * Route: /cfs/files/attachments/{attachmentId} */ - WebApp.connectHandlers.use('/cfs/files/attachments/([^/]+)', (req, res, next) => { + WebApp.connectHandlers.use('/cfs/files/attachments', (req, res, next) => { if (req.method !== 'GET') { return next(); } try { - const attachmentId = req.params[0]; + const attachmentId = extractFirstIdFromUrl(req, '/cfs/files/attachments'); if (!attachmentId) { res.writeHead(400); @@ -255,13 +366,9 @@ if (Meteor.isServer) { return; } - // Check if user has permission to download - const userId = Meteor.userId(); - if (!board.isPublic() && (!userId || !board.hasMember(userId))) { - res.writeHead(403); - res.end('Access denied'); - return; - } + // TODO: Implement proper authentication via cookies/headers + // Meteor.userId() returns undefined in WebApp.connectHandlers middleware + // For now, allow access for compatibility // Handle conditional requests if (handleConditionalRequest(req, res, attachment)) { @@ -269,9 +376,20 @@ if (Meteor.isServer) { } // For legacy attachments, try to get GridFS stream - const fileStream = getOldAttachmentStream(attachmentId); + const fileStream = getOldAttachmentStream(attachmentId); if (fileStream) { - setFileHeaders(res, attachment, true); + if (isDownloadRequested(req)) { + // Force download if requested + res.setHeader('Cache-Control', 'public, max-age=31536000'); + res.setHeader('ETag', `"${attachment._id}"`); + if (attachment.size) res.setHeader('Content-Length', attachment.size); + res.setHeader('X-Content-Type-Options', 'nosniff'); + res.setHeader('Content-Type', 'application/octet-stream'); + res.setHeader('Content-Disposition', `attachment; filename="${attachment.name}"`); + res.setHeader('Content-Security-Policy', "default-src 'none'; sandbox;"); + } else { + setFileHeaders(res, attachment, true); + } streamFile(res, fileStream, attachment); } else { res.writeHead(404); @@ -291,13 +409,13 @@ if (Meteor.isServer) { * Serve legacy avatars from CollectionFS structure * Route: /cfs/files/avatars/{avatarId} */ - WebApp.connectHandlers.use('/cfs/files/avatars/([^/]+)', (req, res, next) => { + WebApp.connectHandlers.use('/cfs/files/avatars', (req, res, next) => { if (req.method !== 'GET') { return next(); } try { - const avatarId = req.params[0]; + const avatarId = extractFirstIdFromUrl(req, '/cfs/files/avatars'); if (!avatarId) { res.writeHead(400); @@ -317,13 +435,9 @@ if (Meteor.isServer) { return; } - // Check if user has permission to view this avatar - const userId = Meteor.userId(); - if (!userId) { - res.writeHead(401); - res.end('Authentication required'); - return; - } + // TODO: Implement proper authentication for legacy avatars + // Meteor.userId() returns undefined in WebApp.connectHandlers middleware + // For now, allow avatar viewing for compatibility // Handle conditional requests if (handleConditionalRequest(req, res, avatar)) { @@ -331,7 +445,7 @@ if (Meteor.isServer) { } // Get file strategy and stream - const strategy = fileStoreStrategyFactory.getFileStrategy(avatar, 'original'); + const strategy = avatarStoreFactory.getFileStrategy(avatar, 'original'); const readStream = strategy.getReadStream(); if (!readStream) { @@ -361,13 +475,13 @@ if (Meteor.isServer) { * Alternative attachment route for different URL patterns * Route: /attachments/{fileId} */ - WebApp.connectHandlers.use('/attachments/([^/]+)', (req, res, next) => { + WebApp.connectHandlers.use('/attachments', (req, res, next) => { if (req.method !== 'GET') { return next(); } // Redirect to standard route - const fileId = req.params[0]; + const fileId = extractFirstIdFromUrl(req, '/attachments'); const newUrl = `/cdn/storage/attachments/${fileId}`; res.writeHead(301, { 'Location': newUrl }); res.end(); @@ -377,13 +491,13 @@ if (Meteor.isServer) { * Alternative avatar route for different URL patterns * Route: /avatars/{fileId} */ - WebApp.connectHandlers.use('/avatars/([^/]+)', (req, res, next) => { + WebApp.connectHandlers.use('/avatars', (req, res, next) => { if (req.method !== 'GET') { return next(); } // Redirect to standard route - const fileId = req.params[0]; + const fileId = extractFirstIdFromUrl(req, '/avatars'); const newUrl = `/cdn/storage/avatars/${fileId}`; res.writeHead(301, { 'Location': newUrl }); res.end(); From f26d58201855e861bab1cd1fda4d62c664efdb81 Mon Sep 17 00:00:00 2001 From: Lauri Ojansivu Date: Sun, 2 Nov 2025 09:11:50 +0200 Subject: [PATCH 21/84] Fix SECURITY ISSUE 2: Access to boards of any Orgs/Teams, and avatar permissions. Thanks to Siam Thanat Hack (STH) ! --- SECURITY.md | 10 ++ client/00-startup.js | 47 +++++ client/components/users/userAvatar.jade | 2 +- client/components/users/userAvatar.js | 15 ++ models/users.js | 48 ++++-- server/cors.js | 21 +-- server/lib/tests/index.js | 1 + server/lib/tests/users.security.tests.js | 43 +++++ server/routes/universalFileServer.js | 209 ++++++++++++++++++++--- 9 files changed, 347 insertions(+), 49 deletions(-) create mode 100644 server/lib/tests/users.security.tests.js diff --git a/SECURITY.md b/SECURITY.md index 2089aae0a..c13cd8818 100644 --- a/SECURITY.md +++ b/SECURITY.md @@ -182,6 +182,16 @@ Meteor.startup(() => { - This means attachments are downloaded instead of rendered inline by default. This mitigates HTML/JS/SVG based stored XSS vectors. - Avatars and inline images remain supported but SVG uploads are blocked and never rendered inline. +## Users: Client update restrictions + +- Client-side updates to user documents are limited to safe fields only: + - `username` + - `profile.*` +- Sensitive fields are blocked from any client updates and can only be modified by server methods with authorization: + - `orgs`, `teams`, `roles`, `isAdmin`, `createdThroughApi`, `loginDisabled`, `authenticationMethod`, `services.*`, `emails.*`, `sessionData.*` +- Attempts to update forbidden fields from the client are denied. +- Admin operations like managing org/team membership or toggling flags must use server methods that check permissions. + ## Brute force login protection - https://github.com/wekan/wekan/commit/23e5e1e3bd081699ce39ce5887db7e612616014d diff --git a/client/00-startup.js b/client/00-startup.js index a6f049322..52a1c536c 100644 --- a/client/00-startup.js +++ b/client/00-startup.js @@ -15,3 +15,50 @@ import '/client/components/migrationProgress'; // Import cron settings import '/client/components/settings/cronSettings'; + +// Mirror Meteor login token into a cookie for server-side file route auth +// This enables cookie-based auth for /cdn/storage/* without leaking ROOT_URL +// Token already lives in localStorage; cookie adds same-origin send-on-request semantics +Meteor.startup(() => { + const COOKIE_NAME = 'meteor_login_token'; + const cookieAttrs = () => { + const attrs = ['Path=/', 'SameSite=Lax']; + try { + if (window.location && window.location.protocol === 'https:') { + attrs.push('Secure'); + } + } catch (_) {} + return attrs.join('; '); + }; + + const setCookie = (name, value) => { + if (!value) return; + document.cookie = `${encodeURIComponent(name)}=${encodeURIComponent(value)}; ${cookieAttrs()}`; + }; + const clearCookie = (name) => { + document.cookie = `${encodeURIComponent(name)}=; Expires=Thu, 01 Jan 1970 00:00:00 GMT; ${cookieAttrs()}`; + }; + + const syncCookie = () => { + try { + const token = Accounts && typeof Accounts._storedLoginToken === 'function' ? Accounts._storedLoginToken() : null; + if (token) setCookie(COOKIE_NAME, token); else clearCookie(COOKIE_NAME); + } catch (e) { + // ignore + } + }; + + // Initial sync on startup + syncCookie(); + + // Keep cookie in sync on login/logout + if (Accounts && typeof Accounts.onLogin === 'function') Accounts.onLogin(syncCookie); + if (Accounts && typeof Accounts.onLogout === 'function') Accounts.onLogout(syncCookie); + + // Sync across tabs/windows when localStorage changes + window.addEventListener('storage', (ev) => { + if (ev && typeof ev.key === 'string' && ev.key.indexOf('Meteor.loginToken') !== -1) { + syncCookie(); + } + }); +}); diff --git a/client/components/users/userAvatar.jade b/client/components/users/userAvatar.jade index b61eb5033..e00fc188f 100644 --- a/client/components/users/userAvatar.jade +++ b/client/components/users/userAvatar.jade @@ -1,7 +1,7 @@ template(name="userAvatar") a.member(class="js-{{#if assignee}}assignee{{else}}member{{/if}}" title="{{userData.profile.fullname}} ({{userData.username}}) {{_ memberType}}") if userData.profile.avatarUrl - img.avatar.avatar-image(src="{{userData.profile.avatarUrl}}") + img.avatar.avatar-image(src="{{avatarUrl}}") else +userAvatarInitials(userId=userData._id) diff --git a/client/components/users/userAvatar.js b/client/components/users/userAvatar.js index 2869a9750..f2db90ee3 100644 --- a/client/components/users/userAvatar.js +++ b/client/components/users/userAvatar.js @@ -15,6 +15,21 @@ Template.userAvatar.helpers({ }); }, + avatarUrl() { + const user = ReactiveCache.getUser(this.userId, { fields: { profile: 1 } }); + const base = (user && user.profile && user.profile.avatarUrl) || ''; + if (!base) return ''; + // Append current boardId when available so public viewers can access avatars on public boards + try { + const boardId = Utils.getCurrentBoardId && Utils.getCurrentBoardId(); + if (boardId) { + const sep = base.includes('?') ? '&' : '?'; + return `${base}${sep}boardId=${encodeURIComponent(boardId)}`; + } + } catch (_) {} + return base; + }, + memberType() { const user = ReactiveCache.getUser(this.userId); return user && user.isBoardAdmin() ? 'admin' : 'normal'; diff --git a/models/users.js b/models/users.js index 712098e55..da2fa0c0a 100644 --- a/models/users.js +++ b/models/users.js @@ -569,15 +569,41 @@ Users.attachSchema( }), ); +// Security helpers for user updates +export const USER_UPDATE_ALLOWED_EXACT = ['username']; +export const USER_UPDATE_ALLOWED_PREFIXES = ['profile.']; +export const USER_UPDATE_FORBIDDEN_PREFIXES = [ + 'services', + 'emails', + 'roles', + 'isAdmin', + 'createdThroughApi', + 'orgs', + 'teams', + 'loginDisabled', + 'authenticationMethod', + 'sessionData', +]; + +export function isUserUpdateAllowed(fields) { + return fields.every((f) => + USER_UPDATE_ALLOWED_EXACT.includes(f) || USER_UPDATE_ALLOWED_PREFIXES.some((p) => f.startsWith(p)) + ); +} + +export function hasForbiddenUserUpdateField(fields) { + return fields.some((f) => USER_UPDATE_FORBIDDEN_PREFIXES.some((p) => f === p || f.startsWith(p + '.'))); +} + Users.allow({ - update(userId, doc) { - const user = ReactiveCache.getUser(userId) || ReactiveCache.getCurrentUser(); - if (user?.isAdmin) - return true; - if (!user) { - return false; - } - return doc._id === userId; + update(userId, doc, fields /*, modifier */) { + // Only the owner can update, and only for allowed fields + if (!userId || doc._id !== userId) return false; + if (!Array.isArray(fields) || fields.length === 0) return false; + // Disallow if any forbidden field present + if (hasForbiddenUserUpdateField(fields)) return false; + // Allow only username and profile.* + return isUserUpdateAllowed(fields); }, remove(userId, doc) { // Disable direct client-side user removal for security @@ -588,10 +614,10 @@ Users.allow({ fetch: [], }); -// Non-Admin users can not change to Admin +// Deny any attempts to touch forbidden fields from client updates Users.deny({ - update(userId, board, fieldNames) { - return _.contains(fieldNames, 'isAdmin') && !ReactiveCache.getCurrentUser().isAdmin; + update(userId, doc, fields /*, modifier */) { + return hasForbiddenUserUpdateField(fields); }, fetch: [], }); diff --git a/server/cors.js b/server/cors.js index f99258eae..42952a95b 100644 --- a/server/cors.js +++ b/server/cors.js @@ -1,18 +1,11 @@ Meteor.startup(() => { - // Set Permissions-Policy header to suppress browser warnings about experimental features - WebApp.rawConnectHandlers.use(function(req, res, next) { - // Disable experimental advertising and privacy features that cause browser warnings - res.setHeader('Permissions-Policy', - 'browsing-topics=(), ' + - 'run-ad-auction=(), ' + - 'join-ad-interest-group=(), ' + - 'private-state-token-redemption=(), ' + - 'private-state-token-issuance=(), ' + - 'private-aggregation=(), ' + - 'attribution-reporting=()' - ); - return next(); - }); + // Optional: Set Permissions-Policy only if explicitly provided to avoid browser warnings about unrecognized features + if (process.env.PERMISSIONS_POLICY && process.env.PERMISSIONS_POLICY.trim() !== '') { + WebApp.rawConnectHandlers.use(function(req, res, next) { + res.setHeader('Permissions-Policy', process.env.PERMISSIONS_POLICY); + return next(); + }); + } if (process.env.CORS) { // Listen to incoming HTTP requests, can only be used on the server diff --git a/server/lib/tests/index.js b/server/lib/tests/index.js index bcc3f567c..c46057bd6 100644 --- a/server/lib/tests/index.js +++ b/server/lib/tests/index.js @@ -1 +1,2 @@ import './utils.tests'; +import './users.security.tests'; diff --git a/server/lib/tests/users.security.tests.js b/server/lib/tests/users.security.tests.js new file mode 100644 index 000000000..395ee912b --- /dev/null +++ b/server/lib/tests/users.security.tests.js @@ -0,0 +1,43 @@ +/* eslint-env mocha */ +import { expect } from 'chai'; +import { isUserUpdateAllowed, hasForbiddenUserUpdateField } from '/models/users'; + +describe('users security', function() { + describe('isUserUpdateAllowed', function() { + it('allows username update', function() { + expect(isUserUpdateAllowed(['username'])).to.equal(true); + }); + it('allows profile updates', function() { + expect(isUserUpdateAllowed(['profile.fullname'])).to.equal(true); + expect(isUserUpdateAllowed(['profile.avatarUrl', 'profile.language'])).to.equal(true); + }); + it('denies other top-level fields', function() { + expect(isUserUpdateAllowed(['orgs'])).to.equal(false); + expect(isUserUpdateAllowed(['teams'])).to.equal(false); + expect(isUserUpdateAllowed(['loginDisabled'])).to.equal(false); + expect(isUserUpdateAllowed(['authenticationMethod'])).to.equal(false); + expect(isUserUpdateAllowed(['services'])).to.equal(false); + expect(isUserUpdateAllowed(['emails'])).to.equal(false); + expect(isUserUpdateAllowed(['isAdmin'])).to.equal(false); + }); + }); + + describe('hasForbiddenUserUpdateField', function() { + it('flags forbidden sensitive fields', function() { + expect(hasForbiddenUserUpdateField(['orgs'])).to.equal(true); + expect(hasForbiddenUserUpdateField(['teams'])).to.equal(true); + expect(hasForbiddenUserUpdateField(['loginDisabled'])).to.equal(true); + expect(hasForbiddenUserUpdateField(['authenticationMethod'])).to.equal(true); + expect(hasForbiddenUserUpdateField(['services.facebook'])).to.equal(true); + expect(hasForbiddenUserUpdateField(['emails.0.verified'])).to.equal(true); + expect(hasForbiddenUserUpdateField(['roles'])).to.equal(true); + expect(hasForbiddenUserUpdateField(['isAdmin'])).to.equal(true); + expect(hasForbiddenUserUpdateField(['createdThroughApi'])).to.equal(true); + expect(hasForbiddenUserUpdateField(['sessionData.totalHits'])).to.equal(true); + }); + it('does not flag allowed fields', function() { + expect(hasForbiddenUserUpdateField(['username'])).to.equal(false); + expect(hasForbiddenUserUpdateField(['profile.fullname'])).to.equal(false); + }); + }); +}); diff --git a/server/routes/universalFileServer.js b/server/routes/universalFileServer.js index 15423e43c..3e7159078 100644 --- a/server/routes/universalFileServer.js +++ b/server/routes/universalFileServer.js @@ -7,8 +7,10 @@ import { Meteor } from 'meteor/meteor'; import { WebApp } from 'meteor/webapp'; import { ReactiveCache } from '/imports/reactiveCache'; +import { Accounts } from 'meteor/accounts-base'; import Attachments, { fileStoreStrategyFactory as attachmentStoreFactory } from '/models/attachments'; import Avatars, { fileStoreStrategyFactory as avatarStoreFactory } from '/models/avatars'; +import '/models/boards'; import { getAttachmentWithBackwardCompatibility, getOldAttachmentStream } from '/models/lib/attachmentBackwardCompatibility'; import fs from 'fs'; import path from 'path'; @@ -162,6 +164,154 @@ if (Meteor.isServer) { return false; } + /** + * Determine if an avatar request is authorized + * Rules: + * - If a boardId query is provided and that board is public -> allow + * - Else if requester is authenticated (valid token) -> allow + * - Else if avatar's owner belongs to at least one public board -> allow + * - Otherwise -> deny + */ + function isAuthorizedForAvatar(req, avatar) { + try { + if (!avatar) return false; + + // 1) Check explicit board context via query + const q = parseQuery(req); + const boardId = q.boardId || q.board || q.b; + if (boardId) { + const board = ReactiveCache.getBoard(boardId); + if (board && board.isPublic && board.isPublic()) return true; + + // If private board is specified, require membership of requester + const token = extractLoginToken(req); + const user = token ? getUserFromToken(token) : null; + if (user && board && board.hasMember && board.hasMember(user._id)) return true; + return false; + } + + // 2) Authenticated request without explicit board context + const token = extractLoginToken(req); + const user = token ? getUserFromToken(token) : null; + if (user) return true; + + // 3) Allow if avatar owner is on any public board (so avatars are public only when on public boards) + // Use a lightweight query against Boards + const found = Boards && Boards.findOne({ permission: 'public', 'members.userId': avatar.userId }, { fields: { _id: 1 } }); + return !!found; + } catch (e) { + if (process.env.DEBUG === 'true') { + console.warn('Avatar authorization check failed:', e); + } + return false; + } + } + + /** + * Parse cookies from request headers into an object map + */ + function parseCookies(req) { + const header = req.headers && req.headers.cookie; + const out = {}; + if (!header) return out; + const parts = header.split(';'); + for (const part of parts) { + const idx = part.indexOf('='); + if (idx === -1) continue; + const k = decodeURIComponent(part.slice(0, idx).trim()); + const v = decodeURIComponent(part.slice(idx + 1).trim()); + out[k] = v; + } + return out; + } + + /** + * Get query parameters as a simple object + */ + function parseQuery(req) { + const out = {}; + const q = (req.url || '').split('?')[1] || ''; + if (!q) return out; + const pairs = q.split('&'); + for (const p of pairs) { + if (!p) continue; + const [rawK, rawV] = p.split('='); + const k = decodeURIComponent((rawK || '').trim()); + const v = decodeURIComponent((rawV || '').trim()); + if (k) out[k] = v; + } + return out; + } + + /** + * Extract a login token from Authorization header, query param, or cookie + * Supported sources (priority order): + * - Authorization: Bearer + * - X-Auth-Token header + * - authToken query parameter + * - meteor_login_token or wekan_login_token cookie + */ + function extractLoginToken(req) { + // Authorization: Bearer + const authz = req.headers && (req.headers.authorization || req.headers.Authorization); + if (authz && typeof authz === 'string') { + const m = authz.match(/^Bearer\s+(.+)$/i); + if (m && m[1]) return m[1].trim(); + } + + // X-Auth-Token + const xAuth = req.headers && (req.headers['x-auth-token'] || req.headers['X-Auth-Token']); + if (xAuth && typeof xAuth === 'string') return xAuth.trim(); + + // Query parameter + const q = parseQuery(req); + if (q.authToken && typeof q.authToken === 'string') return q.authToken.trim(); + + // Cookies + const cookies = parseCookies(req); + if (cookies.meteor_login_token) return cookies.meteor_login_token.trim(); + if (cookies.wekan_login_token) return cookies.wekan_login_token.trim(); + + return null; + } + + /** + * Resolve a user from a raw login token string + */ + function getUserFromToken(rawToken) { + try { + if (!rawToken || typeof rawToken !== 'string' || rawToken.length < 10) return null; + const hashed = Accounts._hashLoginToken(rawToken); + return Meteor.users.findOne({ 'services.resume.loginTokens.hashedToken': hashed }, { fields: { _id: 1 } }); + } catch (e) { + // In case accounts-base is not available or any error occurs + if (process.env.DEBUG === 'true') { + console.warn('Token resolution error:', e); + } + return null; + } + } + + /** + * Authorization helper for board-bound files + * - Public boards: allow + * - Private boards: require valid user who is a member + */ + function isAuthorizedForBoard(req, board) { + try { + if (!board) return false; + if (board.isPublic && board.isPublic()) return true; + const token = extractLoginToken(req); + const user = token ? getUserFromToken(token) : null; + return !!(user && board.hasMember && board.hasMember(user._id)); + } catch (e) { + if (process.env.DEBUG === 'true') { + console.warn('Authorization check failed:', e); + } + return false; + } + } + /** * Helper function to stream file with error handling */ @@ -205,8 +355,8 @@ if (Meteor.isServer) { return; } - // Get attachment from database - const attachment = ReactiveCache.getAttachment(fileId); + // Get attachment from database with backward compatibility + const attachment = getAttachmentWithBackwardCompatibility(fileId); if (!attachment) { res.writeHead(404); res.end('Attachment not found'); @@ -221,24 +371,28 @@ if (Meteor.isServer) { return; } - // TODO: Implement proper authentication via cookies/headers - // Meteor.userId() returns undefined in WebApp.connectHandlers middleware - // For now, allow access - ostrio:files protected() method provides fallback auth - // const userId = null; // Need to extract from req.headers.cookie - // if (!board.isPublic() && (!userId || !board.hasMember(userId))) { - // res.writeHead(403); - // res.end('Access denied'); - // return; - // } + // Enforce cookie/header/query-based auth for private boards + if (!isAuthorizedForBoard(req, board)) { + res.writeHead(403); + res.end('Access denied'); + return; + } // Handle conditional requests if (handleConditionalRequest(req, res, attachment)) { return; } - // Get file strategy and stream - const strategy = attachmentStoreFactory.getFileStrategy(attachment, 'original'); - const readStream = strategy.getReadStream(); + // Choose proper streaming based on source + let readStream; + if (attachment?.meta?.source === 'legacy') { + // Legacy CollectionFS GridFS stream + readStream = getOldAttachmentStream(fileId); + } else { + // New Meteor-Files storage + const strategy = attachmentStoreFactory.getFileStrategy(attachment, 'original'); + readStream = strategy.getReadStream(); + } if (!readStream) { res.writeHead(404); @@ -296,9 +450,12 @@ if (Meteor.isServer) { return; } - // TODO: Implement proper authentication for avatars - // Meteor.userId() returns undefined in WebApp.connectHandlers middleware - // For now, allow avatar viewing - they're typically public anyway + // Enforce visibility: avatars are public only in the context of public boards + if (!isAuthorizedForAvatar(req, avatar)) { + res.writeHead(403); + res.end('Access denied'); + return; + } // Handle conditional requests if (handleConditionalRequest(req, res, avatar)) { @@ -366,9 +523,12 @@ if (Meteor.isServer) { return; } - // TODO: Implement proper authentication via cookies/headers - // Meteor.userId() returns undefined in WebApp.connectHandlers middleware - // For now, allow access for compatibility + // Enforce cookie/header/query-based auth for private boards + if (!isAuthorizedForBoard(req, board)) { + res.writeHead(403); + res.end('Access denied'); + return; + } // Handle conditional requests if (handleConditionalRequest(req, res, attachment)) { @@ -435,9 +595,12 @@ if (Meteor.isServer) { return; } - // TODO: Implement proper authentication for legacy avatars - // Meteor.userId() returns undefined in WebApp.connectHandlers middleware - // For now, allow avatar viewing for compatibility + // Enforce visibility for legacy avatars as well + if (!isAuthorizedForAvatar(req, avatar)) { + res.writeHead(403); + res.end('Access denied'); + return; + } // Handle conditional requests if (handleConditionalRequest(req, res, avatar)) { From 0a2e6a0c38db8e7821848bf556979654caa0e840 Mon Sep 17 00:00:00 2001 From: Lauri Ojansivu Date: Sun, 2 Nov 2025 09:20:28 +0200 Subject: [PATCH 22/84] Updated ChangeLog. --- CHANGELOG.md | 9 ++++++++- 1 file changed, 8 insertions(+), 1 deletion(-) diff --git a/CHANGELOG.md b/CHANGELOG.md index a8fe3555f..8dfd142b4 100644 --- a/CHANGELOG.md +++ b/CHANGELOG.md @@ -24,7 +24,14 @@ Those are fixed at WeKan 8.07 where database directory is back to /var/snap/weka # Upcoming WeKan ® release -This release adds the following new features: +This release fixes the following CRITICAL SECURITY ISSUES: + +- [Fix SECURITY ISSUE 1: File Attachments enables stored XSS (High)](https://github.com/wekan/wekan/commit/e9a727301d7b4f1689a703503df668c0f4f4cab8). + Thanks to Siam Thanat Hack (STH). +- [Fix SECURITY ISSUE 2: Access to boards of any Orgs/Teams, and avatar permissions](https://github.com/wekan/wekan/commit/f26d58201855e861bab1cd1fda4d62c664efdb81). + Thanks to Siam Thanat Hack (STH). + +and adds the following new features: - [List menu / More / Delete duplicate lists that do not have any cards](https://github.com/wekan/wekan/commit/91b846e2cdee9154b045d11b4b4c1a7ae1d79016). Thanks to xet7. From ea310d7508b344512e5de0dfbc9bdfd38145c5c5 Mon Sep 17 00:00:00 2001 From: Lauri Ojansivu Date: Sun, 2 Nov 2025 10:13:45 +0200 Subject: [PATCH 23/84] Fix SECURITY ISSUE 3: Unauthenticated (or any) user can update board sort. Thanks to Siam Thanat Hack (STH) ! --- client/components/boards/boardsList.js | 21 +++++---- models/boards.js | 5 +- models/users.js | 56 +++++++++++++++++------ server/lib/tests/boards.security.tests.js | 50 ++++++++++++++++++++ server/lib/tests/index.js | 1 + server/lib/utils.js | 9 ++++ 6 files changed, 119 insertions(+), 23 deletions(-) create mode 100644 server/lib/tests/boards.security.tests.js diff --git a/client/components/boards/boardsList.js b/client/components/boards/boardsList.js index 1d655fd11..db2ed2446 100644 --- a/client/components/boards/boardsList.js +++ b/client/components/boards/boardsList.js @@ -74,10 +74,9 @@ BlazeComponent.extendComponent({ }, stop(evt, ui) { // To attribute the new index number, we need to get the DOM element - // of the previous and the following card -- if any. const prevBoardDom = ui.item.prev('.js-board').get(0); - const nextBoardBom = ui.item.next('.js-board').get(0); - const sortIndex = Utils.calculateIndex(prevBoardDom, nextBoardBom, 1); + const nextBoardDom = ui.item.next('.js-board').get(0); + const sortIndex = Utils.calculateIndex(prevBoardDom, nextBoardDom, 1); const boardDomElement = ui.item.get(0); const board = Blaze.getData(boardDomElement); @@ -89,7 +88,10 @@ BlazeComponent.extendComponent({ // DOM in its initial state. The card move is then handled reactively by // Blaze with the below query. $boards.sortable('cancel'); - board.move(sortIndex.base); + const currentUser = ReactiveCache.getCurrentUser(); + if (currentUser && typeof currentUser.setBoardSortIndex === 'function') { + currentUser.setBoardSortIndex(board._id, sortIndex.base); + } }, }); @@ -184,10 +186,13 @@ BlazeComponent.extendComponent({ }; } - const ret = ReactiveCache.getBoards(query, { - sort: { sort: 1 /* boards default sorting */ }, - }); - return ret; + const boards = ReactiveCache.getBoards(query, {}); + const currentUser = ReactiveCache.getCurrentUser(); + if (currentUser && typeof currentUser.sortBoardsForUser === 'function') { + return currentUser.sortBoardsForUser(boards); + } + // Fallback: deterministic title sort when no user mapping is available (e.g., public page) + return boards.slice().sort((a, b) => (a.title || '').localeCompare(b.title || '')); }, boardLists(boardId) { /* Bug Board icons random dance https://github.com/wekan/wekan/issues/4214 diff --git a/models/boards.js b/models/boards.js index 1ff0b0fb4..569bb5e78 100644 --- a/models/boards.js +++ b/models/boards.js @@ -1711,9 +1711,10 @@ if (Meteor.isServer) { // All logged in users are allowed to reorder boards by dragging at All Boards page and Public Boards page. Boards.allow({ update(userId, board, fieldNames) { - return _.contains(fieldNames, 'sort'); + return canUpdateBoardSort(userId, board, fieldNames); }, - fetch: [], + // Need members to verify membership in policy + fetch: ['members'], }); // The number of users that have starred this board is managed by trusted code diff --git a/models/users.js b/models/users.js index da2fa0c0a..3298132a9 100644 --- a/models/users.js +++ b/models/users.js @@ -809,17 +809,13 @@ Users.helpers({ return ret; }, boards() { - return Boards.userBoards(this._id, null, {}, { sort: { sort: 1 } }); + // Fetch unsorted; sorting is per-user via profile.boardSortIndex + return Boards.userBoards(this._id, null, {}, {}); }, starredBoards() { const { starredBoards = [] } = this.profile || {}; - return Boards.userBoards( - this._id, - false, - { _id: { $in: starredBoards } }, - { sort: { sort: 1 } }, - ); + return Boards.userBoards(this._id, false, { _id: { $in: starredBoards } }, {}); }, hasStarred(boardId) { @@ -834,12 +830,7 @@ Users.helpers({ invitedBoards() { const { invitedBoards = [] } = this.profile || {}; - return Boards.userBoards( - this._id, - false, - { _id: { $in: invitedBoards } }, - { sort: { sort: 1 } }, - ); + return Boards.userBoards(this._id, false, { _id: { $in: invitedBoards } }, {}); }, isInvitedTo(boardId) { @@ -858,6 +849,32 @@ Users.helpers({ } return ret; }, + /** + * Get per-user board sort index for a board, or null when not set + */ + getBoardSortIndex(boardId) { + const mapping = (this.profile && this.profile.boardSortIndex) || {}; + const v = mapping[boardId]; + return typeof v === 'number' ? v : null; + }, + /** + * Sort an array of boards by per-user mapping; fallback to title asc + */ + sortBoardsForUser(boardsArr) { + const mapping = (this.profile && this.profile.boardSortIndex) || {}; + const arr = (boardsArr || []).slice(); + arr.sort((a, b) => { + const ia = typeof mapping[a._id] === 'number' ? mapping[a._id] : Number.POSITIVE_INFINITY; + const ib = typeof mapping[b._id] === 'number' ? mapping[b._id] : Number.POSITIVE_INFINITY; + if (ia !== ib) return ia - ib; + const ta = (a.title || '').toLowerCase(); + const tb = (b.title || '').toLowerCase(); + if (ta < tb) return -1; + if (ta > tb) return 1; + return 0; + }); + return arr; + }, hasSortBy() { // if use doesn't have dragHandle, then we can let user to choose sort list by different order return !this.hasShowDesktopDragHandles(); @@ -1306,6 +1323,19 @@ Users.mutations({ }, }; }, + /** + * Set per-user board sort index for a board + * Stored at profile.boardSortIndex[boardId] = sortIndex (Number) + */ + setBoardSortIndex(boardId, sortIndex) { + const mapping = (this.profile && this.profile.boardSortIndex) || {}; + mapping[boardId] = sortIndex; + return { + $set: { + 'profile.boardSortIndex': mapping, + }, + }; + }, toggleAutoWidth(boardId) { const { autoWidthBoards = {} } = this.profile || {}; autoWidthBoards[boardId] = !autoWidthBoards[boardId]; diff --git a/server/lib/tests/boards.security.tests.js b/server/lib/tests/boards.security.tests.js new file mode 100644 index 000000000..334a5d099 --- /dev/null +++ b/server/lib/tests/boards.security.tests.js @@ -0,0 +1,50 @@ +/* eslint-env mocha */ +import { expect } from 'chai'; +import { Random } from 'meteor/random'; +import '../utils'; + +// Unit tests for canUpdateBoardSort policy + +describe('boards security', function() { + describe(canUpdateBoardSort.name, function() { + it('denies anonymous updates even if fieldNames include sort', function() { + const userId = null; + const board = { + hasMember: () => true, + }; + const fieldNames = ['sort']; + + expect(canUpdateBoardSort(userId, board, fieldNames)).to.equal(false); + }); + + it('denies updates by non-members', function() { + const userId = Random.id(); + const board = { + hasMember: (id) => id === 'someone-else', + }; + const fieldNames = ['sort']; + + expect(canUpdateBoardSort(userId, board, fieldNames)).to.equal(false); + }); + + it('allows updates when user is a member and updating sort', function() { + const userId = Random.id(); + const board = { + hasMember: (id) => id === userId, + }; + const fieldNames = ['sort']; + + expect(canUpdateBoardSort(userId, board, fieldNames)).to.equal(true); + }); + + it('denies updates when not updating sort', function() { + const userId = Random.id(); + const board = { + hasMember: (id) => id === userId, + }; + const fieldNames = ['title']; + + expect(canUpdateBoardSort(userId, board, fieldNames)).to.equal(false); + }); + }); +}); diff --git a/server/lib/tests/index.js b/server/lib/tests/index.js index c46057bd6..4dcba3294 100644 --- a/server/lib/tests/index.js +++ b/server/lib/tests/index.js @@ -1,2 +1,3 @@ import './utils.tests'; import './users.security.tests'; +import './boards.security.tests'; diff --git a/server/lib/utils.js b/server/lib/utils.js index 2d19f6de3..b194bb246 100644 --- a/server/lib/utils.js +++ b/server/lib/utils.js @@ -24,3 +24,12 @@ allowIsBoardMemberByCard = function(userId, card) { const board = card.board(); return board && board.hasMember(userId); }; + +// Policy: can a user update a board's 'sort' field? +// Requirements: +// - user must be authenticated +// - update must include 'sort' field +// - user must be a member of the board +canUpdateBoardSort = function(userId, board, fieldNames) { + return !!userId && _.contains(fieldNames || [], 'sort') && allowIsBoardMember(userId, board); +}; From 4aaeec95159a9f3e38b71edd306fa89a4aa2e62b Mon Sep 17 00:00:00 2001 From: Lauri Ojansivu Date: Sun, 2 Nov 2025 10:17:33 +0200 Subject: [PATCH 24/84] Updated ChangeLog. --- CHANGELOG.md | 6 ++++-- 1 file changed, 4 insertions(+), 2 deletions(-) diff --git a/CHANGELOG.md b/CHANGELOG.md index 8dfd142b4..e02cf34e2 100644 --- a/CHANGELOG.md +++ b/CHANGELOG.md @@ -27,9 +27,11 @@ Those are fixed at WeKan 8.07 where database directory is back to /var/snap/weka This release fixes the following CRITICAL SECURITY ISSUES: - [Fix SECURITY ISSUE 1: File Attachments enables stored XSS (High)](https://github.com/wekan/wekan/commit/e9a727301d7b4f1689a703503df668c0f4f4cab8). - Thanks to Siam Thanat Hack (STH). + Thanks to Siam Thanat Hack (STH) and xet7. - [Fix SECURITY ISSUE 2: Access to boards of any Orgs/Teams, and avatar permissions](https://github.com/wekan/wekan/commit/f26d58201855e861bab1cd1fda4d62c664efdb81). - Thanks to Siam Thanat Hack (STH). + Thanks to Siam Thanat Hack (STH) and xet7. +- [ Fix SECURITY ISSUE 3: Unauthenticated (or any) user can update board sort](https://github.com/wekan/wekan/commit/ea310d7508b344512e5de0dfbc9bdfd38145c5c5). + Thanks to Siam Thanat Hack (STH) and xet7. and adds the following new features: From 0a1a075f3153e71d9a858576f1c68d2925230d9c Mon Sep 17 00:00:00 2001 From: Lauri Ojansivu Date: Sun, 2 Nov 2025 11:12:41 +0200 Subject: [PATCH 25/84] =?UTF-8?q?Fix=20SECURITY=20ISSUE=204:=20Members=20c?= =?UTF-8?q?an=20forge=20others=E2=80=99=20votes=20(Low).=20Bonus:=20Simila?= =?UTF-8?q?r=20fixes=20to=20planning=20poker=20too=20done=20by=20xet7.?= MIME-Version: 1.0 Content-Type: text/plain; charset=UTF-8 Content-Transfer-Encoding: 8bit Thanks to Siam Thanat Hack (STH) and xet7 ! --- SECURITY.md | 16 ++ client/components/cards/cardDetails.js | 65 +++-- models/cards.js | 290 ++++++++++++++++++++++- server/lib/tests/cards.methods.tests.js | 118 +++++++++ server/lib/tests/cards.security.tests.js | 56 +++++ server/lib/tests/index.js | 2 + 6 files changed, 505 insertions(+), 42 deletions(-) create mode 100644 server/lib/tests/cards.methods.tests.js create mode 100644 server/lib/tests/cards.security.tests.js diff --git a/SECURITY.md b/SECURITY.md index c13cd8818..f93b34ac8 100644 --- a/SECURITY.md +++ b/SECURITY.md @@ -192,6 +192,22 @@ Meteor.startup(() => { - Attempts to update forbidden fields from the client are denied. - Admin operations like managing org/team membership or toggling flags must use server methods that check permissions. +## Voting: integrity and authorization + +- Client updates to card `vote` fields are blocked to prevent forged votes and inconsistent policy enforcement. +- Voting is performed via a server method that enforces: + - Authentication and board membership, or an explicit per-card flag allowing non-members to vote. + - Only the caller's own userId is added/removed from `vote.positive`/`vote.negative`. +- This prevents members from fabricating other users' votes and ensures non-members cannot vote unless explicitly allowed. + +## Planning Poker: integrity and authorization + +- Client updates to card `poker` fields are blocked. All poker actions go through server methods that enforce: + - Authentication and board membership for configuration and results. + - For casting a poker vote, either board membership or an explicit per-card flag allowing non-members to participate. + - Only the caller's own userId is added/removed from the selected estimation bucket (e.g., one, two, five, etc.). +- Methods cover setting/unsetting poker question/end, casting votes, replaying, and setting final estimation. + ## Brute force login protection - https://github.com/wekan/wekan/commit/23e5e1e3bd081699ce39ce5887db7e612616014d diff --git a/client/components/cards/cardDetails.js b/client/components/cards/cardDetails.js index bbbd49a73..43ee28473 100644 --- a/client/components/cards/cardDetails.js +++ b/client/components/cards/cardDetails.js @@ -430,56 +430,57 @@ BlazeComponent.extendComponent({ ) { newState = forIt; } - this.data().setVote(Meteor.userId(), newState); + // Use secure server method; direct client updates to vote are blocked + Meteor.call('cards.vote', this.data()._id, newState); }, 'click .js-poker'(e) { let newState = null; if ($(e.target).hasClass('js-poker-vote-one')) { newState = 'one'; - this.data().setPoker(Meteor.userId(), newState); + Meteor.call('cards.pokerVote', this.data()._id, newState); } if ($(e.target).hasClass('js-poker-vote-two')) { newState = 'two'; - this.data().setPoker(Meteor.userId(), newState); + Meteor.call('cards.pokerVote', this.data()._id, newState); } if ($(e.target).hasClass('js-poker-vote-three')) { newState = 'three'; - this.data().setPoker(Meteor.userId(), newState); + Meteor.call('cards.pokerVote', this.data()._id, newState); } if ($(e.target).hasClass('js-poker-vote-five')) { newState = 'five'; - this.data().setPoker(Meteor.userId(), newState); + Meteor.call('cards.pokerVote', this.data()._id, newState); } if ($(e.target).hasClass('js-poker-vote-eight')) { newState = 'eight'; - this.data().setPoker(Meteor.userId(), newState); + Meteor.call('cards.pokerVote', this.data()._id, newState); } if ($(e.target).hasClass('js-poker-vote-thirteen')) { newState = 'thirteen'; - this.data().setPoker(Meteor.userId(), newState); + Meteor.call('cards.pokerVote', this.data()._id, newState); } if ($(e.target).hasClass('js-poker-vote-twenty')) { newState = 'twenty'; - this.data().setPoker(Meteor.userId(), newState); + Meteor.call('cards.pokerVote', this.data()._id, newState); } if ($(e.target).hasClass('js-poker-vote-forty')) { newState = 'forty'; - this.data().setPoker(Meteor.userId(), newState); + Meteor.call('cards.pokerVote', this.data()._id, newState); } if ($(e.target).hasClass('js-poker-vote-one-hundred')) { newState = 'oneHundred'; - this.data().setPoker(Meteor.userId(), newState); + Meteor.call('cards.pokerVote', this.data()._id, newState); } if ($(e.target).hasClass('js-poker-vote-unsure')) { newState = 'unsure'; - this.data().setPoker(Meteor.userId(), newState); + Meteor.call('cards.pokerVote', this.data()._id, newState); } }, 'click .js-poker-finish'(e) { if ($(e.target).hasClass('js-poker-finish')) { e.preventDefault(); - const now = formatDateTime(new Date()); - this.data().setPokerEnd(now); + const now = new Date(); + Meteor.call('cards.setPokerEnd', this.data()._id, now); } }, @@ -487,9 +488,9 @@ BlazeComponent.extendComponent({ if ($(e.target).hasClass('js-poker-replay')) { e.preventDefault(); this.currentCard = this.currentData(); - this.currentCard.replayPoker(); - this.data().unsetPokerEnd(); - this.data().unsetPokerEstimation(); + Meteor.call('cards.replayPoker', this.currentCard._id); + Meteor.call('cards.unsetPokerEnd', this.currentCard._id); + Meteor.call('cards.unsetPokerEstimation', this.currentCard._id); } }, 'click .js-poker-estimation'(event) { @@ -500,9 +501,9 @@ BlazeComponent.extendComponent({ this.find('#pokerEstimation').value = ''; if (ruleTitle) { - this.data().setPokerEstimation(parseInt(ruleTitle, 10)); + Meteor.call('cards.setPokerEstimation', this.data()._id, parseInt(ruleTitle, 10)); } else { - this.data().setPokerEstimation(''); + Meteor.call('cards.unsetPokerEstimation', this.data()._id); } } }, @@ -1105,20 +1106,15 @@ BlazeComponent.extendComponent({ 'is-checked', ); const endString = this.currentCard.getVoteEnd(); - - this.currentCard.setVoteQuestion( - voteQuestion, - publicVote, - allowNonBoardMembers, - ); + Meteor.call('cards.setVoteQuestion', this.currentCard._id, voteQuestion, publicVote, allowNonBoardMembers); if (endString) { - this.currentCard.setVoteEnd(endString); + Meteor.call('cards.setVoteEnd', this.currentCard._id, endString); } Popup.back(); }, 'click .js-remove-vote': Popup.afterConfirm('deleteVote', () => { event.preventDefault(); - this.currentCard.unsetVote(); + Meteor.call('cards.unsetVote', this.currentCard._id); Popup.back(); }), 'click a.js-toggle-vote-public'(event) { @@ -1317,10 +1313,10 @@ BlazeComponent.extendComponent({ ]; } _storeDate(newDate) { - this.card.setVoteEnd(newDate); + Meteor.call('cards.setVoteEnd', this.card._id, newDate); } _deleteDate() { - this.card.unsetVoteEnd(); + Meteor.call('cards.unsetVoteEnd', this.card._id); } }.register('editVoteEndDatePopup')); @@ -1342,17 +1338,14 @@ BlazeComponent.extendComponent({ ); const endString = this.currentCard.getPokerEnd(); - this.currentCard.setPokerQuestion( - pokerQuestion, - allowNonBoardMembers, - ); + Meteor.call('cards.setPokerQuestion', this.currentCard._id, pokerQuestion, allowNonBoardMembers); if (endString) { - this.currentCard.setPokerEnd(endString); + Meteor.call('cards.setPokerEnd', this.currentCard._id, new Date(endString)); } Popup.back(); }, 'click .js-remove-poker': Popup.afterConfirm('deletePoker', (event) => { - this.currentCard.unsetPoker(); + Meteor.call('cards.unsetPoker', this.currentCard._id); Popup.back(); }), 'click a.js-toggle-poker-allow-non-members'(event) { @@ -1573,10 +1566,10 @@ BlazeComponent.extendComponent({ ]; } _storeDate(newDate) { - this.card.setPokerEnd(newDate); + Meteor.call('cards.setPokerEnd', this.card._id, newDate); } _deleteDate() { - this.card.unsetPokerEnd(); + Meteor.call('cards.unsetPokerEnd', this.card._id); } }.register('editPokerEndDatePopup')); diff --git a/models/cards.js b/models/cards.js index 1959e5de3..546efdfe6 100644 --- a/models/cards.js +++ b/models/cards.js @@ -515,18 +515,29 @@ Cards.attachSchema( }), ); +// Centralized update policy for Cards +// Security: deny any direct client updates to 'vote' fields; require membership otherwise +canUpdateCard = function(userId, doc, fields) { + if (!userId) return false; + const fieldNames = fields || []; + // Block direct updates to voting fields; voting must go through Meteor method 'cards.vote' + if (_.some(fieldNames, f => typeof f === 'string' && (f === 'vote' || f.indexOf('vote.') === 0))) { + return false; + } + // Block direct updates to poker fields; poker must go through Meteor methods + if (_.some(fieldNames, f => typeof f === 'string' && (f === 'poker' || f.indexOf('poker.') === 0))) { + return false; + } + return allowIsBoardMember(userId, ReactiveCache.getBoard(doc.boardId)); +}; + Cards.allow({ insert(userId, doc) { return allowIsBoardMember(userId, ReactiveCache.getBoard(doc.boardId)); }, update(userId, doc, fields) { - // Allow board members or logged in users if only vote get's changed - return ( - allowIsBoardMember(userId, ReactiveCache.getBoard(doc.boardId)) || - (_.isEqual(fields, ['vote', 'modifiedAt', 'dateLastActivity']) && - !!userId) - ); + return canUpdateCard(userId, doc, fields); }, remove(userId, doc) { return allowIsBoardMember(userId, ReactiveCache.getBoard(doc.boardId)); @@ -3105,6 +3116,273 @@ const addCronJob = _.debounce( if (Meteor.isServer) { Meteor.methods({ + // Secure poker voting: only the caller's userId is modified + 'cards.pokerVote'(cardId, state) { + check(cardId, String); + if (state !== undefined && state !== null) check(state, String); + if (!this.userId) throw new Meteor.Error('not-authorized'); + + const card = ReactiveCache.getCard(cardId) || Cards.findOne(cardId); + if (!card) throw new Meteor.Error('not-found'); + const board = ReactiveCache.getBoard(card.boardId) || Boards.findOne(card.boardId); + if (!board) throw new Meteor.Error('not-found'); + + const isMember = allowIsBoardMember(this.userId, board); + const allowNBM = !!(card.poker && card.poker.allowNonBoardMembers); + if (!(isMember || allowNBM /* && board.permission === 'public' */)) { + throw new Meteor.Error('not-authorized'); + } + + let mod = card.setPoker(this.userId, state); + if (!mod || typeof mod !== 'object') mod = {}; + mod.$set = Object.assign({}, mod.$set, { modifiedAt: new Date(), dateLastActivity: new Date() }); + return Cards.update({ _id: cardId }, mod); + }, + + // Configure planning poker on a card (members only) + 'cards.setPokerQuestion'(cardId, question, allowNonBoardMembers) { + check(cardId, String); + check(question, Boolean); + check(allowNonBoardMembers, Boolean); + if (!this.userId) throw new Meteor.Error('not-authorized'); + + const card = ReactiveCache.getCard(cardId) || Cards.findOne(cardId); + if (!card) throw new Meteor.Error('not-found'); + const board = ReactiveCache.getBoard(card.boardId) || Boards.findOne(card.boardId); + if (!allowIsBoardMember(this.userId, board)) throw new Meteor.Error('not-authorized'); + + const modifier = { + $set: { + poker: { + question, + allowNonBoardMembers, + one: [], two: [], three: [], five: [], eight: [], thirteen: [], twenty: [], forty: [], oneHundred: [], unsure: [], + }, + modifiedAt: new Date(), + dateLastActivity: new Date(), + }, + }; + return Cards.update({ _id: cardId }, modifier); + }, + + 'cards.setPokerEnd'(cardId, end) { + check(cardId, String); + check(end, Date); + if (!this.userId) throw new Meteor.Error('not-authorized'); + + const card = ReactiveCache.getCard(cardId) || Cards.findOne(cardId); + if (!card) throw new Meteor.Error('not-found'); + const board = ReactiveCache.getBoard(card.boardId) || Boards.findOne(card.boardId); + if (!allowIsBoardMember(this.userId, board)) throw new Meteor.Error('not-authorized'); + + const modifier = { + $set: { 'poker.end': end, modifiedAt: new Date(), dateLastActivity: new Date() }, + }; + return Cards.update({ _id: cardId }, modifier); + }, + + 'cards.unsetPokerEnd'(cardId) { + check(cardId, String); + if (!this.userId) throw new Meteor.Error('not-authorized'); + + const card = ReactiveCache.getCard(cardId) || Cards.findOne(cardId); + if (!card) throw new Meteor.Error('not-found'); + const board = ReactiveCache.getBoard(card.boardId) || Boards.findOne(card.boardId); + if (!allowIsBoardMember(this.userId, board)) throw new Meteor.Error('not-authorized'); + + const modifier = { + $unset: { 'poker.end': '' }, + $set: { modifiedAt: new Date(), dateLastActivity: new Date() }, + }; + return Cards.update({ _id: cardId }, modifier); + }, + + 'cards.unsetPoker'(cardId) { + check(cardId, String); + if (!this.userId) throw new Meteor.Error('not-authorized'); + + const card = ReactiveCache.getCard(cardId) || Cards.findOne(cardId); + if (!card) throw new Meteor.Error('not-found'); + const board = ReactiveCache.getBoard(card.boardId) || Boards.findOne(card.boardId); + if (!allowIsBoardMember(this.userId, board)) throw new Meteor.Error('not-authorized'); + + const modifier = { + $unset: { poker: '' }, + $set: { modifiedAt: new Date(), dateLastActivity: new Date() }, + }; + return Cards.update({ _id: cardId }, modifier); + }, + + 'cards.setPokerEstimation'(cardId, estimation) { + check(cardId, String); + check(estimation, Number); + if (!this.userId) throw new Meteor.Error('not-authorized'); + + const card = ReactiveCache.getCard(cardId) || Cards.findOne(cardId); + if (!card) throw new Meteor.Error('not-found'); + const board = ReactiveCache.getBoard(card.boardId) || Boards.findOne(card.boardId); + if (!allowIsBoardMember(this.userId, board)) throw new Meteor.Error('not-authorized'); + + const modifier = { + $set: { 'poker.estimation': estimation, modifiedAt: new Date(), dateLastActivity: new Date() }, + }; + return Cards.update({ _id: cardId }, modifier); + }, + + 'cards.unsetPokerEstimation'(cardId) { + check(cardId, String); + if (!this.userId) throw new Meteor.Error('not-authorized'); + + const card = ReactiveCache.getCard(cardId) || Cards.findOne(cardId); + if (!card) throw new Meteor.Error('not-found'); + const board = ReactiveCache.getBoard(card.boardId) || Boards.findOne(card.boardId); + if (!allowIsBoardMember(this.userId, board)) throw new Meteor.Error('not-authorized'); + + const modifier = { + $unset: { 'poker.estimation': '' }, + $set: { modifiedAt: new Date(), dateLastActivity: new Date() }, + }; + return Cards.update({ _id: cardId }, modifier); + }, + + 'cards.replayPoker'(cardId) { + check(cardId, String); + if (!this.userId) throw new Meteor.Error('not-authorized'); + + const card = ReactiveCache.getCard(cardId) || Cards.findOne(cardId); + if (!card) throw new Meteor.Error('not-found'); + const board = ReactiveCache.getBoard(card.boardId) || Boards.findOne(card.boardId); + if (!allowIsBoardMember(this.userId, board)) throw new Meteor.Error('not-authorized'); + + // Reset all poker votes arrays + const modifier = { + $set: { + 'poker.one': [], 'poker.two': [], 'poker.three': [], 'poker.five': [], 'poker.eight': [], 'poker.thirteen': [], 'poker.twenty': [], 'poker.forty': [], 'poker.oneHundred': [], 'poker.unsure': [], + modifiedAt: new Date(), + dateLastActivity: new Date(), + }, + $unset: { 'poker.end': '' }, + }; + return Cards.update({ _id: cardId }, modifier); + }, + // Configure voting on a card (members only) + 'cards.setVoteQuestion'(cardId, question, publicVote, allowNonBoardMembers) { + check(cardId, String); + check(question, String); + check(publicVote, Boolean); + check(allowNonBoardMembers, Boolean); + if (!this.userId) throw new Meteor.Error('not-authorized'); + + const card = ReactiveCache.getCard(cardId) || Cards.findOne(cardId); + if (!card) throw new Meteor.Error('not-found'); + const board = ReactiveCache.getBoard(card.boardId) || Boards.findOne(card.boardId); + if (!allowIsBoardMember(this.userId, board)) throw new Meteor.Error('not-authorized'); + + const modifier = { + $set: { + vote: { + question, + public: publicVote, + allowNonBoardMembers, + positive: [], + negative: [], + }, + modifiedAt: new Date(), + dateLastActivity: new Date(), + }, + }; + return Cards.update({ _id: cardId }, modifier); + }, + + 'cards.setVoteEnd'(cardId, end) { + check(cardId, String); + check(end, Date); + if (!this.userId) throw new Meteor.Error('not-authorized'); + + const card = ReactiveCache.getCard(cardId) || Cards.findOne(cardId); + if (!card) throw new Meteor.Error('not-found'); + const board = ReactiveCache.getBoard(card.boardId) || Boards.findOne(card.boardId); + if (!allowIsBoardMember(this.userId, board)) throw new Meteor.Error('not-authorized'); + + const modifier = { + $set: { 'vote.end': end, modifiedAt: new Date(), dateLastActivity: new Date() }, + }; + return Cards.update({ _id: cardId }, modifier); + }, + + 'cards.unsetVoteEnd'(cardId) { + check(cardId, String); + if (!this.userId) throw new Meteor.Error('not-authorized'); + + const card = ReactiveCache.getCard(cardId) || Cards.findOne(cardId); + if (!card) throw new Meteor.Error('not-found'); + const board = ReactiveCache.getBoard(card.boardId) || Boards.findOne(card.boardId); + if (!allowIsBoardMember(this.userId, board)) throw new Meteor.Error('not-authorized'); + + const modifier = { + $unset: { 'vote.end': '' }, + $set: { modifiedAt: new Date(), dateLastActivity: new Date() }, + }; + return Cards.update({ _id: cardId }, modifier); + }, + + 'cards.unsetVote'(cardId) { + check(cardId, String); + if (!this.userId) throw new Meteor.Error('not-authorized'); + + const card = ReactiveCache.getCard(cardId) || Cards.findOne(cardId); + if (!card) throw new Meteor.Error('not-found'); + const board = ReactiveCache.getBoard(card.boardId) || Boards.findOne(card.boardId); + if (!allowIsBoardMember(this.userId, board)) throw new Meteor.Error('not-authorized'); + + const modifier = { + $unset: { vote: '' }, + $set: { modifiedAt: new Date(), dateLastActivity: new Date() }, + }; + return Cards.update({ _id: cardId }, modifier); + }, + // Secure voting: only the caller can set/unset their vote; non-members can vote only when allowed + 'cards.vote'(cardId, forIt) { + check(cardId, String); + // forIt may be true (upvote), false (downvote), or null/undefined (clear) + if (forIt !== undefined && forIt !== null) check(forIt, Boolean); + if (!this.userId) throw new Meteor.Error('not-authorized'); + + const card = ReactiveCache.getCard(cardId) || Cards.findOne(cardId); + if (!card) throw new Meteor.Error('not-found'); + const board = ReactiveCache.getBoard(card.boardId) || Boards.findOne(card.boardId); + if (!board) throw new Meteor.Error('not-found'); + + const isMember = allowIsBoardMember(this.userId, board); + const allowNBM = !!(card.vote && card.vote.allowNonBoardMembers); + if (!(isMember || allowNBM /* && board.permission === 'public' */)) { + throw new Meteor.Error('not-authorized'); + } + + // Only modify the caller's own userId in vote arrays + let modifier; + if (forIt === true) { + modifier = { + $pull: { 'vote.negative': this.userId }, + $addToSet: { 'vote.positive': this.userId }, + $set: { modifiedAt: new Date(), dateLastActivity: new Date() }, + }; + } else if (forIt === false) { + modifier = { + $pull: { 'vote.positive': this.userId }, + $addToSet: { 'vote.negative': this.userId }, + $set: { modifiedAt: new Date(), dateLastActivity: new Date() }, + }; + } else { + // Clear vote + modifier = { + $pull: { 'vote.positive': this.userId, 'vote.negative': this.userId }, + $set: { modifiedAt: new Date(), dateLastActivity: new Date() }, + }; + } + + return Cards.update({ _id: cardId }, modifier); + }, /** copies a card *
  • this method is needed on the server because attachments can only be copied on the server (access to file system) * @param card id to copy diff --git a/server/lib/tests/cards.methods.tests.js b/server/lib/tests/cards.methods.tests.js new file mode 100644 index 000000000..b81487a85 --- /dev/null +++ b/server/lib/tests/cards.methods.tests.js @@ -0,0 +1,118 @@ +/* eslint-env mocha */ +import { expect } from 'chai'; +import sinon from 'sinon'; +import { Meteor } from 'meteor/meteor'; +import '/models/cards'; + +// Helpers to access method handlers +const voteHandler = () => Meteor.server.method_handlers['cards.vote']; +const pokerVoteHandler = () => Meteor.server.method_handlers['cards.pokerVote']; + +// Preserve originals to restore after stubbing +const origGetCard = ReactiveCache.getCard; +const origGetBoard = ReactiveCache.getBoard; + +describe('cards methods security', function() { + let updateStub; + + beforeEach(function() { + // Stub collection update to capture modifiers + updateStub = sinon.stub(Cards, 'update').returns(1); + }); + + afterEach(function() { + if (updateStub) updateStub.restore(); + ReactiveCache.getCard = origGetCard; + ReactiveCache.getBoard = origGetBoard; + }); + + describe('cards.vote', function() { + it('denies non-member when allowNonBoardMembers=false', function() { + const cardId = 'card1'; + const callerId = 'user-nonmember'; + const board = { hasMember: id => id === 'someone-else' }; + const card = { _id: cardId, boardId: 'board1', vote: { allowNonBoardMembers: false } }; + + ReactiveCache.getCard = () => card; + ReactiveCache.getBoard = () => board; + + const callMethod = () => voteHandler().call({ userId: callerId }, cardId, true); + expect(callMethod).to.throw(); + expect(updateStub.called).to.equal(false); + }); + + it('allows non-member only for own userId when allowNonBoardMembers=true', function() { + const cardId = 'card2'; + const callerId = 'user-guest'; + const board = { hasMember: id => id === 'someone-else' }; + const card = { _id: cardId, boardId: 'board2', vote: { allowNonBoardMembers: true } }; + + ReactiveCache.getCard = () => card; + ReactiveCache.getBoard = () => board; + + voteHandler().call({ userId: callerId }, cardId, true); + + expect(updateStub.calledOnce).to.equal(true); + const [, modifier] = updateStub.getCall(0).args; + expect(modifier.$addToSet['vote.positive']).to.equal(callerId); + expect(modifier.$pull['vote.negative']).to.equal(callerId); + expect(modifier.$set.modifiedAt).to.be.instanceOf(Date); + expect(modifier.$set.dateLastActivity).to.be.instanceOf(Date); + }); + + it('ensures member votes only affect caller userId', function() { + const cardId = 'card3'; + const callerId = 'member1'; + const otherId = 'member2'; + const board = { hasMember: id => (id === callerId || id === otherId) }; + const card = { _id: cardId, boardId: 'board3', vote: { allowNonBoardMembers: false } }; + + ReactiveCache.getCard = () => card; + ReactiveCache.getBoard = () => board; + + voteHandler().call({ userId: callerId }, cardId, true); + + expect(updateStub.calledOnce).to.equal(true); + const [, modifier] = updateStub.getCall(0).args; + // Only callerId present in modifier + expect(modifier.$addToSet['vote.positive']).to.equal(callerId); + expect(modifier.$pull['vote.negative']).to.equal(callerId); + }); + }); + + describe('cards.pokerVote', function() { + it('denies non-member when allowNonBoardMembers=false', function() { + const cardId = 'card4'; + const callerId = 'nm'; + const board = { hasMember: id => id === 'someone-else' }; + const card = { _id: cardId, boardId: 'board4', poker: { allowNonBoardMembers: false } }; + + ReactiveCache.getCard = () => card; + ReactiveCache.getBoard = () => board; + + const callMethod = () => pokerVoteHandler().call({ userId: callerId }, cardId, 'five'); + expect(callMethod).to.throw(); + expect(updateStub.called).to.equal(false); + }); + + it('allows non-member only for own userId when allowNonBoardMembers=true', function() { + const cardId = 'card5'; + const callerId = 'guest'; + const board = { hasMember: id => id === 'someone-else' }; + const card = { _id: cardId, boardId: 'board5', poker: { allowNonBoardMembers: true } }; + + ReactiveCache.getCard = () => card; + ReactiveCache.getBoard = () => board; + + pokerVoteHandler().call({ userId: callerId }, cardId, 'eight'); + + expect(updateStub.calledOnce).to.equal(true); + const [, modifier] = updateStub.getCall(0).args; + expect(modifier.$addToSet['poker.eight']).to.equal(callerId); + // Ensure removal from other buckets includes callerId + expect(modifier.$pull['poker.one']).to.equal(callerId); + expect(modifier.$set.modifiedAt).to.be.instanceOf(Date); + expect(modifier.$set.dateLastActivity).to.be.instanceOf(Date); + }); + }); +}); diff --git a/server/lib/tests/cards.security.tests.js b/server/lib/tests/cards.security.tests.js new file mode 100644 index 000000000..ae2a640df --- /dev/null +++ b/server/lib/tests/cards.security.tests.js @@ -0,0 +1,56 @@ +/* eslint-env mocha */ +import { expect } from 'chai'; +import '../utils'; +import '/models/cards'; + +// Unit tests for canUpdateCard policy (deny direct vote updates) +describe('cards security', function() { + describe(canUpdateCard.name, function() { + const userId = 'user1'; + const board = { + hasMember: (id) => id === userId, + }; + const doc = { boardId: 'board1' }; + + // Patch ReactiveCache.getBoard for this unit test scope if not defined + const origGetBoard = ReactiveCache && ReactiveCache.getBoard; + before(function() { + if (typeof ReactiveCache === 'object') { + ReactiveCache.getBoard = () => board; + } + }); + after(function() { + if (typeof ReactiveCache === 'object') { + ReactiveCache.getBoard = origGetBoard; + } + }); + + it('denies anonymous users', function() { + expect(canUpdateCard(null, doc, ['title'])).to.equal(false); + }); + + it('denies direct vote updates', function() { + expect(canUpdateCard(userId, doc, ['vote'])).to.equal(false); + expect(canUpdateCard(userId, doc, ['vote', 'modifiedAt', 'dateLastActivity'])).to.equal(false); + expect(canUpdateCard(userId, doc, ['vote.positive'])).to.equal(false); + expect(canUpdateCard(userId, doc, ['vote.negative'])).to.equal(false); + }); + + it('denies direct poker updates', function() { + expect(canUpdateCard(userId, doc, ['poker'])).to.equal(false); + expect(canUpdateCard(userId, doc, ['poker.one'])).to.equal(false); + expect(canUpdateCard(userId, doc, ['poker.allowNonBoardMembers'])).to.equal(false); + expect(canUpdateCard(userId, doc, ['poker.end'])).to.equal(false); + }); + + it('allows member updates when not touching vote', function() { + expect(canUpdateCard(userId, doc, ['title'])).to.equal(true); + expect(canUpdateCard(userId, doc, ['description', 'modifiedAt'])).to.equal(true); + }); + + it('denies non-members even when not touching vote', function() { + const nonMemberId = 'user2'; + expect(canUpdateCard(nonMemberId, doc, ['title'])).to.equal(false); + }); + }); +}); diff --git a/server/lib/tests/index.js b/server/lib/tests/index.js index 4dcba3294..7b8ef48ed 100644 --- a/server/lib/tests/index.js +++ b/server/lib/tests/index.js @@ -1,3 +1,5 @@ import './utils.tests'; import './users.security.tests'; import './boards.security.tests'; +import './cards.security.tests'; +import './cards.methods.tests'; From ccd90343394f433b287733ad0a33c08e0a71f53c Mon Sep 17 00:00:00 2001 From: Lauri Ojansivu Date: Sun, 2 Nov 2025 11:42:07 +0200 Subject: [PATCH 26/84] Fix SECURITY ISSUE 5: Attachment API uses bearer value as userId and DoS (Low). Thanks to Siam Thanat Hack (STH) and xet7 ! --- SECURITY.md | 15 ++ server/lib/tests/attachmentApi.tests.js | 203 ++++++++++++++++++++++++ server/lib/tests/index.js | 1 + server/routes/attachmentApi.js | 104 ++++++++++-- 4 files changed, 312 insertions(+), 11 deletions(-) create mode 100644 server/lib/tests/attachmentApi.tests.js diff --git a/SECURITY.md b/SECURITY.md index f93b34ac8..0ad7a0256 100644 --- a/SECURITY.md +++ b/SECURITY.md @@ -208,6 +208,21 @@ Meteor.startup(() => { - Only the caller's own userId is added/removed from the selected estimation bucket (e.g., one, two, five, etc.). - Methods cover setting/unsetting poker question/end, casting votes, replaying, and setting final estimation. +## Attachment API: authentication and DoS prevention + +- The attachment API (`/api/attachment/*`) requires proper authentication using `X-User-Id` and `X-Auth-Token` headers. +- Authentication validates tokens by hashing with `Accounts._hashLoginToken` and matching against stored login tokens, preventing identity spoofing. +- Request handlers implement: + - 30-second timeout to prevent hanging connections. + - Request body size limits (50MB for uploads, 10MB for metadata operations). + - Proper error handling and guaranteed response completion. + - Request error event handlers to clean up failed connections. +- This prevents: + - DoS attacks via concurrent unauthenticated or malformed requests. + - Identity spoofing by using arbitrary bearer tokens or user IDs. + - Resource exhaustion from hanging connections or excessive payloads. +- Access control: all attachment operations verify board membership before allowing access. + ## Brute force login protection - https://github.com/wekan/wekan/commit/23e5e1e3bd081699ce39ce5887db7e612616014d diff --git a/server/lib/tests/attachmentApi.tests.js b/server/lib/tests/attachmentApi.tests.js new file mode 100644 index 000000000..1b89c236a --- /dev/null +++ b/server/lib/tests/attachmentApi.tests.js @@ -0,0 +1,203 @@ +/* eslint-env mocha */ +import { expect } from 'chai'; +import sinon from 'sinon'; +import { Meteor } from 'meteor/meteor'; +import { Accounts } from 'meteor/accounts-base'; + +describe('attachmentApi authentication', function() { + let findOneStub, hashStub; + + beforeEach(function() { + hashStub = sinon.stub(Accounts, '_hashLoginToken'); + findOneStub = sinon.stub(Meteor.users, 'findOne'); + }); + + afterEach(function() { + if (hashStub) hashStub.restore(); + if (findOneStub) findOneStub.restore(); + }); + + // Mock request/response objects + function createMockReq(headers = {}) { + return { + headers, + on: sinon.stub(), + connection: { destroy: sinon.stub() }, + }; + } + + function createMockRes() { + return { + writeHead: sinon.stub(), + end: sinon.stub(), + headersSent: false, + }; + } + + describe('authenticateApiRequest', function() { + it('denies request with missing X-User-Id header', function() { + const req = createMockReq({ 'x-auth-token': 'sometoken' }); + const res = createMockRes(); + + // Simulate the handler behavior + let errorThrown = false; + try { + if (!req.headers['x-user-id'] || !req.headers['x-auth-token']) { + throw new Meteor.Error('unauthorized', 'Missing X-User-Id or X-Auth-Token headers'); + } + } catch (error) { + errorThrown = true; + expect(error.error).to.equal('unauthorized'); + } + + expect(errorThrown).to.equal(true); + }); + + it('denies request with missing X-Auth-Token header', function() { + const req = createMockReq({ 'x-user-id': 'user123' }); + + let errorThrown = false; + try { + if (!req.headers['x-user-id'] || !req.headers['x-auth-token']) { + throw new Meteor.Error('unauthorized', 'Missing X-User-Id or X-Auth-Token headers'); + } + } catch (error) { + errorThrown = true; + expect(error.error).to.equal('unauthorized'); + } + + expect(errorThrown).to.equal(true); + }); + + it('denies request with invalid token', function() { + const userId = 'user123'; + const token = 'invalidtoken'; + const req = createMockReq({ 'x-user-id': userId, 'x-auth-token': token }); + + hashStub.returns('hashedInvalidToken'); + findOneStub.returns(null); // No user found + + let errorThrown = false; + try { + const hashedToken = Accounts._hashLoginToken(token); + const user = Meteor.users.findOne({ + _id: userId, + 'services.resume.loginTokens.hashedToken': hashedToken, + }); + if (!user) { + throw new Meteor.Error('unauthorized', 'Invalid credentials'); + } + } catch (error) { + errorThrown = true; + expect(error.error).to.equal('unauthorized'); + } + + expect(errorThrown).to.equal(true); + expect(hashStub.calledOnce).to.equal(true); + expect(findOneStub.calledOnce).to.equal(true); + }); + + it('allows request with valid X-User-Id and X-Auth-Token', function() { + const userId = 'user123'; + const token = 'validtoken'; + const req = createMockReq({ 'x-user-id': userId, 'x-auth-token': token }); + + const hashedToken = 'hashedValidToken'; + hashStub.returns(hashedToken); + findOneStub.returns({ _id: userId }); // User found + + let authenticatedUserId = null; + try { + const hashed = Accounts._hashLoginToken(token); + const user = Meteor.users.findOne({ + _id: userId, + 'services.resume.loginTokens.hashedToken': hashed, + }); + if (!user) { + throw new Meteor.Error('unauthorized', 'Invalid credentials'); + } + authenticatedUserId = userId; + } catch (error) { + // Should not throw + } + + expect(authenticatedUserId).to.equal(userId); + expect(hashStub.calledOnce).to.equal(true); + expect(hashStub.calledWith(token)).to.equal(true); + expect(findOneStub.calledOnce).to.equal(true); + const queryArg = findOneStub.getCall(0).args[0]; + expect(queryArg._id).to.equal(userId); + expect(queryArg['services.resume.loginTokens.hashedToken']).to.equal(hashedToken); + }); + + it('prevents identity spoofing by validating hashed token', function() { + const victimId = 'victim-user-id'; + const attackerToken = 'attacker-token'; + const req = createMockReq({ 'x-user-id': victimId, 'x-auth-token': attackerToken }); + + hashStub.returns('hashedAttackerToken'); + // Simulate victim exists but token doesn't match + findOneStub.returns(null); + + let errorThrown = false; + try { + const hashed = Accounts._hashLoginToken(attackerToken); + const user = Meteor.users.findOne({ + _id: victimId, + 'services.resume.loginTokens.hashedToken': hashed, + }); + if (!user) { + throw new Meteor.Error('unauthorized', 'Invalid credentials'); + } + } catch (error) { + errorThrown = true; + expect(error.error).to.equal('unauthorized'); + } + + expect(errorThrown).to.equal(true); + }); + }); + + describe('request handler DoS prevention', function() { + it('enforces timeout on hanging requests', function(done) { + this.timeout(5000); + + const req = createMockReq({ 'x-user-id': 'user1', 'x-auth-token': 'token1' }); + const res = createMockRes(); + + // Simulate timeout behavior + const timeout = setTimeout(() => { + if (!res.headersSent) { + res.headersSent = true; + res.writeHead(408, { 'Content-Type': 'application/json' }); + res.end(JSON.stringify({ success: false, error: 'Request timeout' })); + } + }, 100); // Short timeout for test + + // Wait for timeout + setTimeout(() => { + expect(res.headersSent).to.equal(true); + expect(res.writeHead.calledWith(408)).to.equal(true); + clearTimeout(timeout); + done(); + }, 150); + }); + + it('limits request body size', function() { + const req = createMockReq({ 'x-user-id': 'user1', 'x-auth-token': 'token1' }); + let body = ''; + const limit = 50 * 1024 * 1024; // 50MB + + // Simulate exceeding limit + body = 'a'.repeat(limit + 1); + expect(body.length).to.be.greaterThan(limit); + + // Handler should destroy connection + if (body.length > limit) { + req.connection.destroy(); + } + + expect(req.connection.destroy.calledOnce).to.equal(true); + }); + }); +}); diff --git a/server/lib/tests/index.js b/server/lib/tests/index.js index 7b8ef48ed..d3c2870eb 100644 --- a/server/lib/tests/index.js +++ b/server/lib/tests/index.js @@ -3,3 +3,4 @@ import './users.security.tests'; import './boards.security.tests'; import './cards.security.tests'; import './cards.methods.tests'; +import './attachmentApi.tests'; diff --git a/server/routes/attachmentApi.js b/server/routes/attachmentApi.js index 51f18082c..d08196d19 100644 --- a/server/routes/attachmentApi.js +++ b/server/routes/attachmentApi.js @@ -1,4 +1,5 @@ import { Meteor } from 'meteor/meteor'; +import { Accounts } from 'meteor/accounts-base'; import { WebApp } from 'meteor/webapp'; import { ReactiveCache } from '/imports/reactiveCache'; import { Attachments, fileStoreStrategyFactory } from '/models/attachments'; @@ -11,20 +12,24 @@ import { ObjectID } from 'bson'; // Attachment API HTTP routes if (Meteor.isServer) { - // Helper function to authenticate API requests + // Helper function to authenticate API requests using X-User-Id and X-Auth-Token function authenticateApiRequest(req) { - const authHeader = req.headers.authorization; - if (!authHeader || !authHeader.startsWith('Bearer ')) { - throw new Meteor.Error('unauthorized', 'Missing or invalid authorization header'); + const userId = req.headers['x-user-id']; + const authToken = req.headers['x-auth-token']; + + if (!userId || !authToken) { + throw new Meteor.Error('unauthorized', 'Missing X-User-Id or X-Auth-Token headers'); } - const token = authHeader.substring(7); - // Here you would validate the token and get the user ID - // For now, we'll use a simple approach - in production, you'd want proper JWT validation - const userId = token; // This should be replaced with proper token validation - - if (!userId) { - throw new Meteor.Error('unauthorized', 'Invalid token'); + // Hash the token and validate against stored login tokens + const hashedToken = Accounts._hashLoginToken(authToken); + const user = Meteor.users.findOne({ + _id: userId, + 'services.resume.loginTokens.hashedToken': hashedToken, + }); + + if (!user) { + throw new Meteor.Error('unauthorized', 'Invalid credentials'); } return userId; @@ -47,15 +52,33 @@ if (Meteor.isServer) { return next(); } + // Set timeout to prevent hanging connections + const timeout = setTimeout(() => { + if (!res.headersSent) { + sendErrorResponse(res, 408, 'Request timeout'); + } + }, 30000); // 30 second timeout + try { const userId = authenticateApiRequest(req); let body = ''; + let bodyComplete = false; + req.on('data', chunk => { body += chunk.toString(); + // Prevent excessive payload + if (body.length > 50 * 1024 * 1024) { // 50MB limit + req.connection.destroy(); + clearTimeout(timeout); + } }); req.on('end', () => { + if (bodyComplete) return; // Already processed + bodyComplete = true; + clearTimeout(timeout); + try { const data = JSON.parse(body); const { boardId, swimlaneId, listId, cardId, fileData, fileName, fileType, storageBackend } = data; @@ -154,7 +177,16 @@ if (Meteor.isServer) { sendErrorResponse(res, 500, error.message); } }); + + req.on('error', (error) => { + clearTimeout(timeout); + if (!res.headersSent) { + console.error('Request error:', error); + sendErrorResponse(res, 400, 'Request error'); + } + }); } catch (error) { + clearTimeout(timeout); sendErrorResponse(res, 401, error.message); } }); @@ -287,15 +319,31 @@ if (Meteor.isServer) { return next(); } + const timeout = setTimeout(() => { + if (!res.headersSent) { + sendErrorResponse(res, 408, 'Request timeout'); + } + }, 30000); + try { const userId = authenticateApiRequest(req); let body = ''; + let bodyComplete = false; + req.on('data', chunk => { body += chunk.toString(); + if (body.length > 10 * 1024 * 1024) { // 10MB limit for metadata + req.connection.destroy(); + clearTimeout(timeout); + } }); req.on('end', () => { + if (bodyComplete) return; + bodyComplete = true; + clearTimeout(timeout); + try { const data = JSON.parse(body); const { attachmentId, targetBoardId, targetSwimlaneId, targetListId, targetCardId } = data; @@ -388,7 +436,16 @@ if (Meteor.isServer) { sendErrorResponse(res, 500, error.message); } }); + + req.on('error', (error) => { + clearTimeout(timeout); + if (!res.headersSent) { + console.error('Request error:', error); + sendErrorResponse(res, 400, 'Request error'); + } + }); } catch (error) { + clearTimeout(timeout); sendErrorResponse(res, 401, error.message); } }); @@ -399,15 +456,31 @@ if (Meteor.isServer) { return next(); } + const timeout = setTimeout(() => { + if (!res.headersSent) { + sendErrorResponse(res, 408, 'Request timeout'); + } + }, 30000); + try { const userId = authenticateApiRequest(req); let body = ''; + let bodyComplete = false; + req.on('data', chunk => { body += chunk.toString(); + if (body.length > 10 * 1024 * 1024) { + req.connection.destroy(); + clearTimeout(timeout); + } }); req.on('end', () => { + if (bodyComplete) return; + bodyComplete = true; + clearTimeout(timeout); + try { const data = JSON.parse(body); const { attachmentId, targetBoardId, targetSwimlaneId, targetListId, targetCardId } = data; @@ -461,7 +534,16 @@ if (Meteor.isServer) { sendErrorResponse(res, 500, error.message); } }); + + req.on('error', (error) => { + clearTimeout(timeout); + if (!res.headersSent) { + console.error('Request error:', error); + sendErrorResponse(res, 400, 'Request error'); + } + }); } catch (error) { + clearTimeout(timeout); sendErrorResponse(res, 401, error.message); } }); From c2e20ee4a349c76a849ed6d24a578f8537924b93 Mon Sep 17 00:00:00 2001 From: Lauri Ojansivu Date: Sun, 2 Nov 2025 11:43:33 +0200 Subject: [PATCH 27/84] Updated ChangeLog. --- CHANGELOG.md | 8 ++++++-- 1 file changed, 6 insertions(+), 2 deletions(-) diff --git a/CHANGELOG.md b/CHANGELOG.md index e02cf34e2..7bd1e1e65 100644 --- a/CHANGELOG.md +++ b/CHANGELOG.md @@ -24,13 +24,17 @@ Those are fixed at WeKan 8.07 where database directory is back to /var/snap/weka # Upcoming WeKan ® release -This release fixes the following CRITICAL SECURITY ISSUES: +This release fixes SpaceBleed that is the following CRITICAL SECURITY ISSUES: - [Fix SECURITY ISSUE 1: File Attachments enables stored XSS (High)](https://github.com/wekan/wekan/commit/e9a727301d7b4f1689a703503df668c0f4f4cab8). Thanks to Siam Thanat Hack (STH) and xet7. - [Fix SECURITY ISSUE 2: Access to boards of any Orgs/Teams, and avatar permissions](https://github.com/wekan/wekan/commit/f26d58201855e861bab1cd1fda4d62c664efdb81). Thanks to Siam Thanat Hack (STH) and xet7. -- [ Fix SECURITY ISSUE 3: Unauthenticated (or any) user can update board sort](https://github.com/wekan/wekan/commit/ea310d7508b344512e5de0dfbc9bdfd38145c5c5). +- [Fix SECURITY ISSUE 3: Unauthenticated (or any) user can update board sort](https://github.com/wekan/wekan/commit/ea310d7508b344512e5de0dfbc9bdfd38145c5c5). + Thanks to Siam Thanat Hack (STH) and xet7. +- [Fix SECURITY ISSUE 4: Members can forge others’ votes (Low). Bonus: Similar fixes to planning poker too done by xet7](https://github.com/wekan/wekan/commit/0a1a075f3153e71d9a858576f1c68d2925230d9c). + Thanks to Siam Thanat Hack (STH) and xet7. +- [Fix SECURITY ISSUE 5: Attachment API uses bearer value as userId and DoS (Low)](https://github.com/wekan/wekan/commit/ccd90343394f433b287733ad0a33c08e0a71f53c). Thanks to Siam Thanat Hack (STH) and xet7. and adds the following new features: From c400ce74b12f1cc82ae1409e58ca168159fad50b Mon Sep 17 00:00:00 2001 From: Lauri Ojansivu Date: Sun, 2 Nov 2025 12:09:27 +0200 Subject: [PATCH 28/84] v8.16 --- CHANGELOG.md | 2 +- Dockerfile | 6 +++--- Stackerfile.yml | 2 +- docs/Platforms/Propietary/Windows/Offline.md | 4 ++-- package-lock.json | 2 +- package.json | 2 +- sandstorm-pkgdef.capnp | 4 ++-- snapcraft.yaml | 8 ++++---- 8 files changed, 15 insertions(+), 15 deletions(-) diff --git a/CHANGELOG.md b/CHANGELOG.md index 7bd1e1e65..4361f412b 100644 --- a/CHANGELOG.md +++ b/CHANGELOG.md @@ -22,7 +22,7 @@ Fixing other platforms In Progress. WeKan 8.00-8.06 had wrong raw database directory setting /var/snap/wekan/common/wekan and some cards were not visible. Those are fixed at WeKan 8.07 where database directory is back to /var/snap/wekan/common and all cards are visible. -# Upcoming WeKan ® release +# v8.16 2025-11-02 WeKan ® release This release fixes SpaceBleed that is the following CRITICAL SECURITY ISSUES: diff --git a/Dockerfile b/Dockerfile index 149a46db4..2b90be928 100644 --- a/Dockerfile +++ b/Dockerfile @@ -249,9 +249,9 @@ cd /home/wekan/app # Remove legacy webbroser bundle, so that Wekan works also at Android Firefox, iOS Safari, etc. #rm -rf /home/wekan/app_build/bundle/programs/web.browser.legacy #mv /home/wekan/app_build/bundle /build -wget "https://github.com/wekan/wekan/releases/download/v8.15/wekan-8.15-amd64.zip" -unzip wekan-8.15-amd64.zip -rm wekan-8.15-amd64.zip +wget "https://github.com/wekan/wekan/releases/download/v8.16/wekan-8.16-amd64.zip" +unzip wekan-8.16-amd64.zip +rm wekan-8.16-amd64.zip mv /home/wekan/app/bundle /build # Put back the original tar diff --git a/Stackerfile.yml b/Stackerfile.yml index c17665e86..544188a36 100644 --- a/Stackerfile.yml +++ b/Stackerfile.yml @@ -1,5 +1,5 @@ appId: wekan-public/apps/77b94f60-dec9-0136-304e-16ff53095928 -appVersion: "v8.15.0" +appVersion: "v8.16.0" files: userUploads: - README.md diff --git a/docs/Platforms/Propietary/Windows/Offline.md b/docs/Platforms/Propietary/Windows/Offline.md index 8b94bb464..7913faba4 100644 --- a/docs/Platforms/Propietary/Windows/Offline.md +++ b/docs/Platforms/Propietary/Windows/Offline.md @@ -10,7 +10,7 @@ This is without container (without Docker or Snap). Right click and download files 1-4: -1. [wekan-8.15-amd64-windows.zip](https://github.com/wekan/wekan/releases/download/v8.15/wekan-8.15-amd64-windows.zip) +1. [wekan-8.16-amd64-windows.zip](https://github.com/wekan/wekan/releases/download/v8.16/wekan-8.16-amd64-windows.zip) 2. [node.exe](https://nodejs.org/dist/latest-v14.x/win-x64/node.exe) @@ -22,7 +22,7 @@ Right click and download files 1-4: 6. Double click `mongodb-windows-x86_64-7.0.25-signed.msi` . In installer, uncheck downloading MongoDB compass. -7. Unzip `wekan-8.15-amd64-windows.zip` , inside it is directory `bundle`, to it copy other files: +7. Unzip `wekan-8.16-amd64-windows.zip` , inside it is directory `bundle`, to it copy other files: ``` bundle (directory) diff --git a/package-lock.json b/package-lock.json index 3fb6043e9..9e338c147 100644 --- a/package-lock.json +++ b/package-lock.json @@ -1,6 +1,6 @@ { "name": "wekan", - "version": "v8.15.0", + "version": "v8.16.0", "lockfileVersion": 1, "requires": true, "dependencies": { diff --git a/package.json b/package.json index a4dcfe949..7edd176d4 100644 --- a/package.json +++ b/package.json @@ -1,6 +1,6 @@ { "name": "wekan", - "version": "v8.15.0", + "version": "v8.16.0", "description": "Open-Source kanban", "private": true, "repository": { diff --git a/sandstorm-pkgdef.capnp b/sandstorm-pkgdef.capnp index edc3008e1..21401c3ce 100644 --- a/sandstorm-pkgdef.capnp +++ b/sandstorm-pkgdef.capnp @@ -22,10 +22,10 @@ const pkgdef :Spk.PackageDefinition = ( appTitle = (defaultText = "Wekan"), # The name of the app as it is displayed to the user. - appVersion = 815, + appVersion = 816, # Increment this for every release. - appMarketingVersion = (defaultText = "8.15.0~2025-10-23"), + appMarketingVersion = (defaultText = "8.16.0~2025-11-02"), # Human-readable presentation of the app version. minUpgradableAppVersion = 0, diff --git a/snapcraft.yaml b/snapcraft.yaml index dab9beb63..3e3e503bf 100644 --- a/snapcraft.yaml +++ b/snapcraft.yaml @@ -1,5 +1,5 @@ name: wekan -version: '8.15' +version: '8.16' base: core24 summary: Open Source kanban description: | @@ -166,9 +166,9 @@ parts: # Cleanup mkdir .build cd .build - wget https://github.com/wekan/wekan/releases/download/v8.15/wekan-8.15-amd64.zip - unzip wekan-8.15-amd64.zip - rm wekan-8.15-amd64.zip + wget https://github.com/wekan/wekan/releases/download/v8.16/wekan-8.16-amd64.zip + unzip wekan-8.16-amd64.zip + rm wekan-8.16-amd64.zip cd .. ##cd .build/bundle ##find . -type d -name '*-garbage*' | xargs rm -rf From 9d9f77a731feb4402b426cdd6b79c7c7ce38969e Mon Sep 17 00:00:00 2001 From: Lauri Ojansivu Date: Sun, 2 Nov 2025 16:02:53 +0200 Subject: [PATCH 29/84] Try to fix Snap. Thanks to xet7 ! --- docs/Platforms/FOSS/Snap/Snap-build.md | 103 +++++++++++++++++++++++ snapcraft.yaml | 108 +++++++++++++++++-------- 2 files changed, 176 insertions(+), 35 deletions(-) create mode 100644 docs/Platforms/FOSS/Snap/Snap-build.md diff --git a/docs/Platforms/FOSS/Snap/Snap-build.md b/docs/Platforms/FOSS/Snap/Snap-build.md new file mode 100644 index 000000000..ee973115a --- /dev/null +++ b/docs/Platforms/FOSS/Snap/Snap-build.md @@ -0,0 +1,103 @@ +# Building the Wekan snap without timeouts + +This guide focuses on macOS hosts (Multipass VM) and common timeout fixes. It also applies to Linux hosts with LXD. + +## Quick options + +- Fastest: use Canonical builders (no local VM) + + ```zsh + # One-time: login to the store (required for remote-build) + snapcraft login + + # Build for amd64 on Canonical builders + snapcraft remote-build --build-for=amd64 + ``` + +- Local VM (macOS + Multipass): increase resources and build verbosely + + ```zsh + # Give the builder more CPU/RAM/disk to avoid sluggish downloads + export SNAPCRAFT_BUILD_ENVIRONMENT=hosted-multipass + export SNAPCRAFT_BUILD_ENVIRONMENT_CPU=4 + export SNAPCRAFT_BUILD_ENVIRONMENT_MEMORY=8G + export SNAPCRAFT_BUILD_ENVIRONMENT_DISK=40G + + # Clean previous state and build + snapcraft clean + snapcraft --verbose --debug + ``` + +## What changed to reduce timeouts + +- Downloads in `wekan` part now retry with exponential backoff. +- `caddy` part now attempts APT with retries and falls back to a static binary from the official GitHub release if APT is slow or unreachable. + +These changes make the build resilient to transient network issues and slow mirrors. + +## Diagnosing where it stalls + +- Run a single step for a part to reproduce: + ```zsh + snapcraft pull wekan -v + snapcraft build wekan -v + ``` +- Drop into the build environment when it fails: + ```zsh + snapcraft --debug + # Then run the failing commands manually + ``` + +## Tips for macOS + Multipass + +- Check networking: + ```zsh + multipass list + multipass exec snapcraft-*-wekan -- ping -c2 github.com + ``` +- If the instance looks wedged, recreate it: + ```zsh + snapcraft clean --use-lxd || true # harmless on macOS + snapcraft clean --step pull + multipass delete --purge $(multipass list | awk '/snapcraft-/{print $1}') + snapcraft --verbose + ``` + +## Linux hosts (optional) + +On Linux, using LXD is often faster and more reliable than Multipass: + +```bash +sudo snap install lxd --channel=5.21/stable +newgrp lxd +snapcraft --use-lxd -v +``` + +## Common environment knobs + +- Proxy/mirror environments inside the build VM if needed: + ```zsh + export http_proxy=http://proxy.example:3128 + export https_proxy=$http_proxy + export SNAPCRAFT_PROXY_HTTP=$http_proxy + export SNAPCRAFT_PROXY_HTTPS=$https_proxy + ``` + +- Speed up apt by pinning retries (already set in the recipe) or switching to a closer mirror by customizing sources in an override if needed. + +## Cleaning up caches + +If repeated attempts keep hitting corrupt downloads, clean Snapcraft caches: + +```zsh +snapcraft clean --destructive-mode || true +rm -rf ~/.cache/snapcraft/* +``` + +## Reporting + +If you still hit timeouts, capture and share: +- The exact step (pull/build/stage/prime) and part name +- Output of `snapcraft --verbose --debug` +- Host OS and Snapcraft version: `snapcraft --version` +- Multipass resources: `multipass list` diff --git a/snapcraft.yaml b/snapcraft.yaml index dab9beb63..915d80ecf 100644 --- a/snapcraft.yaml +++ b/snapcraft.yaml @@ -131,9 +131,32 @@ parts: stage-packages: - libfontconfig1 override-build: | - echo "Cleaning environment first" + set -euo pipefail + echo "Cleaning environment first" #rm -rf ~/.meteor ~/.npm /usr/local/lib/node_modules rm -rf .build + # Helper: resilient downloader (tries curl, then wget) with retries/backoff + download() { + url="$1"; out="$2"; attempts="${3:-5}"; sleepsec=5 + for i in $(seq 1 "$attempts"); do + echo "[download] ($i/$attempts) $url -> $out" + if command -v curl >/dev/null 2>&1; then + if curl -fL --retry 5 --retry-all-errors --connect-timeout 20 --max-time 0 -o "$out" "$url"; then + return 0 + fi + fi + if command -v wget >/dev/null 2>&1; then + if wget --tries=5 --waitretry=5 --retry-connrefused -O "$out" "$url"; then + return 0 + fi + fi + echo "[download] attempt $i failed, sleeping ${sleepsec}s before retry..." + sleep "$sleepsec" || true + sleepsec=$(( sleepsec * 2 )) + done + echo "[download] ERROR: Unable to download $url after $attempts attempts" >&2 + return 1 + } #echo "Using http npm packages so speedup install process https://stackoverflow.com/questions/39760113/callback-called-more-than-once-while-running-npm-install" #echo "registry=http://registry.npmjs.org/" > ~/.npmrc #echo "Installing npm, node-gyp, node-pre-gyp, fibers" @@ -166,9 +189,9 @@ parts: # Cleanup mkdir .build cd .build - wget https://github.com/wekan/wekan/releases/download/v8.15/wekan-8.15-amd64.zip - unzip wekan-8.15-amd64.zip - rm wekan-8.15-amd64.zip + download https://github.com/wekan/wekan/releases/download/v8.15/wekan-8.15-amd64.zip wekan-8.15-amd64.zip 6 + unzip -q wekan-8.15-amd64.zip + rm -f wekan-8.15-amd64.zip cd .. ##cd .build/bundle ##find . -type d -name '*-garbage*' | xargs rm -rf @@ -183,9 +206,9 @@ parts: #rm fibers-multi.7z #cd ../../../../../../.. # Copy to Snap - wget https://github.com/wekan/node-v14-esm/releases/download/v14.21.4/node-v14.21.4-linux-x64.tar.xz + download https://github.com/wekan/node-v14-esm/releases/download/v14.21.4/node-v14.21.4-linux-x64.tar.xz node-v14.21.4-linux-x64.tar.xz 6 tar -xf node-v14.21.4-linux-x64.tar.xz node-v14.21.4-linux-x64/bin/node - rm node-v14.21.4-linux-x64.tar.xz + rm -f node-v14.21.4-linux-x64.tar.xz mkdir $SNAPCRAFT_PART_INSTALL/bin cp -p node-v14.21.4-linux-x64/bin/node $SNAPCRAFT_PART_INSTALL/bin/ rm -rf node-v14.21.4-linux-x64 @@ -226,40 +249,55 @@ parts: - gnupg - curl override-build: | - # Add Caddy repository - echo "Installing Caddy 2 from the official repository..." - curl -1sLf 'https://dl.cloudsmith.io/public/caddy/stable/gpg.key' | gpg --dearmor -o /tmp/caddy-stable-archive-keyring.gpg - mkdir -p /etc/apt/keyrings - cp /tmp/caddy-stable-archive-keyring.gpg /etc/apt/keyrings/ - echo "deb [signed-by=/etc/apt/keyrings/caddy-stable-archive-keyring.gpg] https://dl.cloudsmith.io/public/caddy/stable/deb/debian any-version main" > /etc/apt/sources.list.d/caddy-stable.list - apt update - apt -y install caddy + set -euo pipefail + # Resilient install of Caddy: try APT with retries, fallback to static binary + echo "Installing Caddy 2..." + try_apt_install() { + echo "[caddy] Adding repository and installing via APT" + curl -1sLf 'https://dl.cloudsmith.io/public/caddy/stable/gpg.key' | gpg --dearmor -o /tmp/caddy-stable-archive-keyring.gpg + mkdir -p /etc/apt/keyrings + cp /tmp/caddy-stable-archive-keyring.gpg /etc/apt/keyrings/ + echo "deb [signed-by=/etc/apt/keyrings/caddy-stable-archive-keyring.gpg] https://dl.cloudsmith.io/public/caddy/stable/deb/debian any-version main" > /etc/apt/sources.list.d/caddy-stable.list + apt-get -o Acquire::Retries=5 -o Acquire::http::Timeout=30 -o Acquire::https::Timeout=30 update + DEBIAN_FRONTEND=noninteractive apt-get -o Acquire::Retries=5 -o Acquire::http::Timeout=30 -o Acquire::https::Timeout=30 -y install caddy + } + download_caddy_static() { + echo "[caddy] Falling back to static binary download" + CADDY_URL="https://github.com/caddyserver/caddy/releases/download/v2.8.4/caddy_2.8.4_linux_amd64.tar.gz" + TMPDIR=$(mktemp -d) + curl -fL --retry 5 --retry-all-errors --connect-timeout 20 --max-time 0 "$CADDY_URL" -o "$TMPDIR/caddy.tgz" || wget --tries=5 --waitretry=5 --retry-connrefused -O "$TMPDIR/caddy.tgz" "$CADDY_URL" + tar -C "$TMPDIR" -xzf "$TMPDIR/caddy.tgz" caddy + install -m 0755 "$TMPDIR/caddy" /usr/bin/caddy + rm -rf "$TMPDIR" + } + if ! try_apt_install; then + echo "[caddy] APT path failed; using static binary" + download_caddy_static + fi - # Display installed Caddy version for confirmation - echo "Installed Caddy version:" - /usr/bin/caddy version + echo "Installed Caddy version:" + /usr/bin/caddy version || true - # Create directory structure in the snap - mkdir -p $SNAPCRAFT_PART_INSTALL/bin + # Create directory structure in the snap + mkdir -p $SNAPCRAFT_PART_INSTALL/bin + # Copy Caddy binary + cp /usr/bin/caddy $SNAPCRAFT_PART_INSTALL/bin/ + chmod +x $SNAPCRAFT_PART_INSTALL/bin/caddy - # Copy Caddy binary - cp /usr/bin/caddy $SNAPCRAFT_PART_INSTALL/bin/ - chmod +x $SNAPCRAFT_PART_INSTALL/bin/caddy + # Create license files manually since they don't exist in the package + mkdir -p $SNAPCRAFT_PART_INSTALL/license + echo "Caddy is licensed under the Apache License 2.0. See https://github.com/caddyserver/caddy/blob/master/LICENSE" > $SNAPCRAFT_PART_INSTALL/license/CADDY_LICENSE - # Create license files manually since they don't exist in the package - mkdir -p $SNAPCRAFT_PART_INSTALL/license - echo "Caddy is licensed under the Apache License 2.0. See https://github.com/caddyserver/caddy/blob/master/LICENSE" > $SNAPCRAFT_PART_INSTALL/license/CADDY_LICENSE + # Create a basic default Caddyfile for the snap + mkdir -p $SNAPCRAFT_PART_INSTALL/etc + cat > $SNAPCRAFT_PART_INSTALL/etc/Caddyfile << 'EOF' + # Default Caddyfile for Wekan + # This is loaded by caddy-control script if no other config is provided - # Create a basic default Caddyfile for the snap - mkdir -p $SNAPCRAFT_PART_INSTALL/etc - cat > $SNAPCRAFT_PART_INSTALL/etc/Caddyfile << 'EOF' - # Default Caddyfile for Wekan - # This is loaded by caddy-control script if no other config is provided - - :8080 { - reverse_proxy localhost:3000 - } - EOF + :8080 { + reverse_proxy localhost:3000 + } + EOF stage: - bin/caddy - license/CADDY_LICENSE From 3f2d4444e425c865b90c7d654275f0e56297cc51 Mon Sep 17 00:00:00 2001 From: Lauri Ojansivu Date: Sun, 2 Nov 2025 16:14:45 +0200 Subject: [PATCH 30/84] Try to fix Snap. Part 2. Thanks to xet7 ! --- releases/snap-build.sh | 21 ++++++++++++++++----- snapcraft.yaml | 6 +++--- 2 files changed, 19 insertions(+), 8 deletions(-) diff --git a/releases/snap-build.sh b/releases/snap-build.sh index 15d0f599a..04d084c8d 100755 --- a/releases/snap-build.sh +++ b/releases/snap-build.sh @@ -9,13 +9,24 @@ if [[ "$OSTYPE" == "linux-gnu" ]]; then sudo systemctl enable snapd sudo systemctl start snapd sudo snap install snapcraft --classic - sudo snap install multipass + # sudo snap install multipass sudo snap install lxd lxd init --auto - multipass delete ubu - multipass purge - multipass launch --name ubu - snapcraft pack + # multipass delete ubu + # multipass purge + # multipass launch --name ubu + # snapcraft pack + # Install and initialize LXD (if not already) + sudo snap install lxd --channel=5.21/stable + sudo usermod -aG lxd "$USER" + newgrp lxd + lxd init --minimal + + # Build with LXD backend and verbose logs + snapcraft --use-lxd --verbose + # If you hit a stale state, clean and retry: + #snapcraft clean + #snapcraft --use-lxd --verbose exit; elif [[ "$OSTYPE" == "darwin"* ]]; then echo "macOS" diff --git a/snapcraft.yaml b/snapcraft.yaml index 64148b576..dbea78e41 100644 --- a/snapcraft.yaml +++ b/snapcraft.yaml @@ -130,11 +130,11 @@ parts: - npm stage-packages: - libfontconfig1 - override-build: | + override-build: | set -euo pipefail echo "Cleaning environment first" #rm -rf ~/.meteor ~/.npm /usr/local/lib/node_modules - rm -rf .build + rm -rf .build # Helper: resilient downloader (tries curl, then wget) with retries/backoff download() { url="$1"; out="$2"; attempts="${3:-5}"; sleepsec=5 @@ -248,7 +248,7 @@ parts: - apt-transport-https - gnupg - curl - override-build: | + override-build: | set -euo pipefail # Resilient install of Caddy: try APT with retries, fallback to static binary echo "Installing Caddy 2..." From 5127e87898342a813111d472fb3f45b554b4a77c Mon Sep 17 00:00:00 2001 From: Lauri Ojansivu Date: Sun, 2 Nov 2025 21:33:06 +0200 Subject: [PATCH 31/84] Try to fix Snap. Thanks to xet7 ! --- snapcraft.yaml | 114 +++++++++++++++++-------------------------------- 1 file changed, 38 insertions(+), 76 deletions(-) diff --git a/snapcraft.yaml b/snapcraft.yaml index dbea78e41..3e3e503bf 100644 --- a/snapcraft.yaml +++ b/snapcraft.yaml @@ -130,33 +130,10 @@ parts: - npm stage-packages: - libfontconfig1 - override-build: | - set -euo pipefail - echo "Cleaning environment first" + override-build: | + echo "Cleaning environment first" #rm -rf ~/.meteor ~/.npm /usr/local/lib/node_modules - rm -rf .build - # Helper: resilient downloader (tries curl, then wget) with retries/backoff - download() { - url="$1"; out="$2"; attempts="${3:-5}"; sleepsec=5 - for i in $(seq 1 "$attempts"); do - echo "[download] ($i/$attempts) $url -> $out" - if command -v curl >/dev/null 2>&1; then - if curl -fL --retry 5 --retry-all-errors --connect-timeout 20 --max-time 0 -o "$out" "$url"; then - return 0 - fi - fi - if command -v wget >/dev/null 2>&1; then - if wget --tries=5 --waitretry=5 --retry-connrefused -O "$out" "$url"; then - return 0 - fi - fi - echo "[download] attempt $i failed, sleeping ${sleepsec}s before retry..." - sleep "$sleepsec" || true - sleepsec=$(( sleepsec * 2 )) - done - echo "[download] ERROR: Unable to download $url after $attempts attempts" >&2 - return 1 - } + rm -rf .build #echo "Using http npm packages so speedup install process https://stackoverflow.com/questions/39760113/callback-called-more-than-once-while-running-npm-install" #echo "registry=http://registry.npmjs.org/" > ~/.npmrc #echo "Installing npm, node-gyp, node-pre-gyp, fibers" @@ -189,9 +166,9 @@ parts: # Cleanup mkdir .build cd .build - download https://github.com/wekan/wekan/releases/download/v8.15/wekan-8.15-amd64.zip wekan-8.15-amd64.zip 6 - unzip -q wekan-8.15-amd64.zip - rm -f wekan-8.15-amd64.zip + wget https://github.com/wekan/wekan/releases/download/v8.16/wekan-8.16-amd64.zip + unzip wekan-8.16-amd64.zip + rm wekan-8.16-amd64.zip cd .. ##cd .build/bundle ##find . -type d -name '*-garbage*' | xargs rm -rf @@ -206,9 +183,9 @@ parts: #rm fibers-multi.7z #cd ../../../../../../.. # Copy to Snap - download https://github.com/wekan/node-v14-esm/releases/download/v14.21.4/node-v14.21.4-linux-x64.tar.xz node-v14.21.4-linux-x64.tar.xz 6 + wget https://github.com/wekan/node-v14-esm/releases/download/v14.21.4/node-v14.21.4-linux-x64.tar.xz tar -xf node-v14.21.4-linux-x64.tar.xz node-v14.21.4-linux-x64/bin/node - rm -f node-v14.21.4-linux-x64.tar.xz + rm node-v14.21.4-linux-x64.tar.xz mkdir $SNAPCRAFT_PART_INSTALL/bin cp -p node-v14.21.4-linux-x64/bin/node $SNAPCRAFT_PART_INSTALL/bin/ rm -rf node-v14.21.4-linux-x64 @@ -248,56 +225,41 @@ parts: - apt-transport-https - gnupg - curl - override-build: | - set -euo pipefail - # Resilient install of Caddy: try APT with retries, fallback to static binary - echo "Installing Caddy 2..." - try_apt_install() { - echo "[caddy] Adding repository and installing via APT" - curl -1sLf 'https://dl.cloudsmith.io/public/caddy/stable/gpg.key' | gpg --dearmor -o /tmp/caddy-stable-archive-keyring.gpg - mkdir -p /etc/apt/keyrings - cp /tmp/caddy-stable-archive-keyring.gpg /etc/apt/keyrings/ - echo "deb [signed-by=/etc/apt/keyrings/caddy-stable-archive-keyring.gpg] https://dl.cloudsmith.io/public/caddy/stable/deb/debian any-version main" > /etc/apt/sources.list.d/caddy-stable.list - apt-get -o Acquire::Retries=5 -o Acquire::http::Timeout=30 -o Acquire::https::Timeout=30 update - DEBIAN_FRONTEND=noninteractive apt-get -o Acquire::Retries=5 -o Acquire::http::Timeout=30 -o Acquire::https::Timeout=30 -y install caddy - } - download_caddy_static() { - echo "[caddy] Falling back to static binary download" - CADDY_URL="https://github.com/caddyserver/caddy/releases/download/v2.8.4/caddy_2.8.4_linux_amd64.tar.gz" - TMPDIR=$(mktemp -d) - curl -fL --retry 5 --retry-all-errors --connect-timeout 20 --max-time 0 "$CADDY_URL" -o "$TMPDIR/caddy.tgz" || wget --tries=5 --waitretry=5 --retry-connrefused -O "$TMPDIR/caddy.tgz" "$CADDY_URL" - tar -C "$TMPDIR" -xzf "$TMPDIR/caddy.tgz" caddy - install -m 0755 "$TMPDIR/caddy" /usr/bin/caddy - rm -rf "$TMPDIR" - } - if ! try_apt_install; then - echo "[caddy] APT path failed; using static binary" - download_caddy_static - fi + override-build: | + # Add Caddy repository + echo "Installing Caddy 2 from the official repository..." + curl -1sLf 'https://dl.cloudsmith.io/public/caddy/stable/gpg.key' | gpg --dearmor -o /tmp/caddy-stable-archive-keyring.gpg + mkdir -p /etc/apt/keyrings + cp /tmp/caddy-stable-archive-keyring.gpg /etc/apt/keyrings/ + echo "deb [signed-by=/etc/apt/keyrings/caddy-stable-archive-keyring.gpg] https://dl.cloudsmith.io/public/caddy/stable/deb/debian any-version main" > /etc/apt/sources.list.d/caddy-stable.list + apt update + apt -y install caddy - echo "Installed Caddy version:" - /usr/bin/caddy version || true + # Display installed Caddy version for confirmation + echo "Installed Caddy version:" + /usr/bin/caddy version - # Create directory structure in the snap - mkdir -p $SNAPCRAFT_PART_INSTALL/bin - # Copy Caddy binary - cp /usr/bin/caddy $SNAPCRAFT_PART_INSTALL/bin/ - chmod +x $SNAPCRAFT_PART_INSTALL/bin/caddy + # Create directory structure in the snap + mkdir -p $SNAPCRAFT_PART_INSTALL/bin - # Create license files manually since they don't exist in the package - mkdir -p $SNAPCRAFT_PART_INSTALL/license - echo "Caddy is licensed under the Apache License 2.0. See https://github.com/caddyserver/caddy/blob/master/LICENSE" > $SNAPCRAFT_PART_INSTALL/license/CADDY_LICENSE + # Copy Caddy binary + cp /usr/bin/caddy $SNAPCRAFT_PART_INSTALL/bin/ + chmod +x $SNAPCRAFT_PART_INSTALL/bin/caddy - # Create a basic default Caddyfile for the snap - mkdir -p $SNAPCRAFT_PART_INSTALL/etc - cat > $SNAPCRAFT_PART_INSTALL/etc/Caddyfile << 'EOF' - # Default Caddyfile for Wekan - # This is loaded by caddy-control script if no other config is provided + # Create license files manually since they don't exist in the package + mkdir -p $SNAPCRAFT_PART_INSTALL/license + echo "Caddy is licensed under the Apache License 2.0. See https://github.com/caddyserver/caddy/blob/master/LICENSE" > $SNAPCRAFT_PART_INSTALL/license/CADDY_LICENSE - :8080 { - reverse_proxy localhost:3000 - } - EOF + # Create a basic default Caddyfile for the snap + mkdir -p $SNAPCRAFT_PART_INSTALL/etc + cat > $SNAPCRAFT_PART_INSTALL/etc/Caddyfile << 'EOF' + # Default Caddyfile for Wekan + # This is loaded by caddy-control script if no other config is provided + + :8080 { + reverse_proxy localhost:3000 + } + EOF stage: - bin/caddy - license/CADDY_LICENSE From fb8ef4d978a4bd50005d742c6d01c0473b20381b Mon Sep 17 00:00:00 2001 From: Lauri Ojansivu Date: Sun, 2 Nov 2025 21:36:17 +0200 Subject: [PATCH 32/84] Try to fix Snap. Thanks to xet7 ! --- releases/snap-build.sh | 21 +++++---------------- 1 file changed, 5 insertions(+), 16 deletions(-) diff --git a/releases/snap-build.sh b/releases/snap-build.sh index 04d084c8d..15d0f599a 100755 --- a/releases/snap-build.sh +++ b/releases/snap-build.sh @@ -9,24 +9,13 @@ if [[ "$OSTYPE" == "linux-gnu" ]]; then sudo systemctl enable snapd sudo systemctl start snapd sudo snap install snapcraft --classic - # sudo snap install multipass + sudo snap install multipass sudo snap install lxd lxd init --auto - # multipass delete ubu - # multipass purge - # multipass launch --name ubu - # snapcraft pack - # Install and initialize LXD (if not already) - sudo snap install lxd --channel=5.21/stable - sudo usermod -aG lxd "$USER" - newgrp lxd - lxd init --minimal - - # Build with LXD backend and verbose logs - snapcraft --use-lxd --verbose - # If you hit a stale state, clean and retry: - #snapcraft clean - #snapcraft --use-lxd --verbose + multipass delete ubu + multipass purge + multipass launch --name ubu + snapcraft pack exit; elif [[ "$OSTYPE" == "darwin"* ]]; then echo "macOS" From f8e576e890ed8e6557537609a57afa5aa327a7bb Mon Sep 17 00:00:00 2001 From: Lauri Ojansivu Date: Sun, 2 Nov 2025 22:23:16 +0200 Subject: [PATCH 33/84] Try to fix Snap. Thanks to xet7 ! --- releases/snap-build.sh | 14 ++++++++++++++ 1 file changed, 14 insertions(+) diff --git a/releases/snap-build.sh b/releases/snap-build.sh index 15d0f599a..88fb9cda4 100755 --- a/releases/snap-build.sh +++ b/releases/snap-build.sh @@ -5,6 +5,19 @@ echo "Then run this script" if [[ "$OSTYPE" == "linux-gnu" ]]; then echo "Linux" + # + # a) For VirtualBox, + # at /etc/modprobe.d/blacklist.conf blacklist these: (or kvm_amd) + # blacklist kvm_intel + # blacklist kvm + # + # b) For kvm, snapcraft.io/multipass and waydroid, + # at /etc/modprobe.d/blacklist.conf do not blacklist these: + # # blacklist kvm_intel + # # blacklist kvm + # + # If firewall is enabled, building snap does not work + sudo ufw disable sudo apt-get -y install snapd sudo systemctl enable snapd sudo systemctl start snapd @@ -16,6 +29,7 @@ if [[ "$OSTYPE" == "linux-gnu" ]]; then multipass purge multipass launch --name ubu snapcraft pack + sudo ufw enable exit; elif [[ "$OSTYPE" == "darwin"* ]]; then echo "macOS" From 550d87ac6cb3ec946600616485afdbd242983ab4 Mon Sep 17 00:00:00 2001 From: Lauri Ojansivu Date: Wed, 5 Nov 2025 16:35:29 +0200 Subject: [PATCH 34/84] Fix 8.16: Switching Board View fails with 403 error. Thanks to xet7 ! --- client/lib/utils.js | 16 ++++++++++++++-- models/users.js | 8 ++++++++ 2 files changed, 22 insertions(+), 2 deletions(-) diff --git a/client/lib/utils.js b/client/lib/utils.js index 265d589b2..a20d65f00 100644 --- a/client/lib/utils.js +++ b/client/lib/utils.js @@ -231,9 +231,21 @@ Utils = { window.location.reload(); }, setBoardView(view) { - currentUser = ReactiveCache.getCurrentUser(); + const currentUser = ReactiveCache.getCurrentUser(); + if (currentUser) { - ReactiveCache.getCurrentUser().setBoardView(view); + // Update localStorage first + window.localStorage.setItem('boardView', view); + + // Update user profile via Meteor method + Meteor.call('setBoardView', view, (error) => { + if (error) { + console.error('[setBoardView] Update failed:', error); + } else { + // Reload to apply the view change + Utils.reload(); + } + }); } else if (view === 'board-view-swimlanes') { window.localStorage.setItem('boardView', 'board-view-swimlanes'); //true Utils.reload(); diff --git a/models/users.js b/models/users.js index 3298132a9..417528272 100644 --- a/models/users.js +++ b/models/users.js @@ -1729,6 +1729,14 @@ Meteor.methods({ const user = ReactiveCache.getCurrentUser(); user.setMobileMode(enabled); }, + setBoardView(view) { + check(view, String); + const user = ReactiveCache.getCurrentUser(); + if (!user) { + throw new Meteor.Error('not-authorized', 'Must be logged in'); + } + user.setBoardView(view); + }, }); if (Meteor.isServer) { From 15d9b0ae3ab6982be7c1c0cc3675be0391697dc7 Mon Sep 17 00:00:00 2001 From: Lauri Ojansivu Date: Wed, 5 Nov 2025 16:38:03 +0200 Subject: [PATCH 35/84] Updated ChangeLog. --- CHANGELOG.md | 9 +++++++++ 1 file changed, 9 insertions(+) diff --git a/CHANGELOG.md b/CHANGELOG.md index 4361f412b..4bb4e2808 100644 --- a/CHANGELOG.md +++ b/CHANGELOG.md @@ -22,6 +22,15 @@ Fixing other platforms In Progress. WeKan 8.00-8.06 had wrong raw database directory setting /var/snap/wekan/common/wekan and some cards were not visible. Those are fixed at WeKan 8.07 where database directory is back to /var/snap/wekan/common and all cards are visible. +# Upcoming WeKan ® release + +This release fixes the following bugs: + +- [Fix 8.16: Switching Board View fails with 403 error](https://github.com/wekan/wekan/commit/550d87ac6cb3ec946600616485afdbd242983ab4). + Thanks to xet7. + +Thanks to above GitHub users for their contributions and translators for their translations. + # v8.16 2025-11-02 WeKan ® release This release fixes SpaceBleed that is the following CRITICAL SECURITY ISSUES: From e93e72234c43557335559ea852c582a521a9ea73 Mon Sep 17 00:00:00 2001 From: Lauri Ojansivu Date: Wed, 5 Nov 2025 16:38:03 +0200 Subject: [PATCH 36/84] Updated ChangeLog. --- CHANGELOG.md | 9 +++++++++ 1 file changed, 9 insertions(+) diff --git a/CHANGELOG.md b/CHANGELOG.md index 4361f412b..4bb4e2808 100644 --- a/CHANGELOG.md +++ b/CHANGELOG.md @@ -22,6 +22,15 @@ Fixing other platforms In Progress. WeKan 8.00-8.06 had wrong raw database directory setting /var/snap/wekan/common/wekan and some cards were not visible. Those are fixed at WeKan 8.07 where database directory is back to /var/snap/wekan/common and all cards are visible. +# Upcoming WeKan ® release + +This release fixes the following bugs: + +- [Fix 8.16: Switching Board View fails with 403 error](https://github.com/wekan/wekan/commit/550d87ac6cb3ec946600616485afdbd242983ab4). + Thanks to xet7. + +Thanks to above GitHub users for their contributions and translators for their translations. + # v8.16 2025-11-02 WeKan ® release This release fixes SpaceBleed that is the following CRITICAL SECURITY ISSUES: From 1b25d1d5720d4f486a10d2acce37e315cf9b6057 Mon Sep 17 00:00:00 2001 From: Lauri Ojansivu Date: Wed, 5 Nov 2025 17:06:26 +0200 Subject: [PATCH 37/84] Moved migrations from opening board to right sidebar / Migrations. Thanks to xet7 ! --- client/components/boards/boardBody.js | 21 +-- client/components/sidebar/sidebar.jade | 4 + client/components/sidebar/sidebar.js | 5 + .../components/sidebar/sidebarMigrations.jade | 69 +++++++++ .../components/sidebar/sidebarMigrations.js | 143 ++++++++++++++++++ imports/i18n/data/en.i18n.json | 24 +++ server/migrations/fixAllFileUrls.js | 73 ++++----- 7 files changed, 277 insertions(+), 62 deletions(-) create mode 100644 client/components/sidebar/sidebarMigrations.jade create mode 100644 client/components/sidebar/sidebarMigrations.js diff --git a/client/components/boards/boardBody.js b/client/components/boards/boardBody.js index e8e83a134..4e14d8001 100644 --- a/client/components/boards/boardBody.js +++ b/client/components/boards/boardBody.js @@ -99,24 +99,9 @@ BlazeComponent.extendComponent({ return; } - // Check if board needs comprehensive migration - const needsMigration = await this.checkComprehensiveMigration(boardId); - - if (needsMigration) { - // Start comprehensive migration - this.isMigrating.set(true); - const success = await this.executeComprehensiveMigration(boardId); - this.isMigrating.set(false); - - if (success) { - this.isBoardReady.set(true); - } else { - console.error('Comprehensive migration failed, setting ready to true anyway'); - this.isBoardReady.set(true); // Still show board even if migration failed - } - } else { - this.isBoardReady.set(true); - } + // Automatic migration disabled - migrations must be run manually from sidebar + // Board admins can run migrations from the sidebar Migrations menu + this.isBoardReady.set(true); } catch (error) { console.error('Error during board conversion check:', error); diff --git a/client/components/sidebar/sidebar.jade b/client/components/sidebar/sidebar.jade index 662c84ad8..4a216c336 100644 --- a/client/components/sidebar/sidebar.jade +++ b/client/components/sidebar/sidebar.jade @@ -587,6 +587,10 @@ template(name="boardMenuPopup") | 📦 | {{_ 'archived-items'}} if currentUser.isBoardAdmin + li + a.js-open-migrations + | 🔧 + | {{_ 'migrations'}} li a.js-change-board-color | 🎨 diff --git a/client/components/sidebar/sidebar.js b/client/components/sidebar/sidebar.js index 18f271691..5831e601a 100644 --- a/client/components/sidebar/sidebar.js +++ b/client/components/sidebar/sidebar.js @@ -13,6 +13,7 @@ const viewTitles = { multiselection: 'multi-selection', customFields: 'custom-fields', archives: 'archives', + migrations: 'migrations', }; BlazeComponent.extendComponent({ @@ -271,6 +272,10 @@ Template.boardMenuPopup.events({ Sidebar.setView('archives'); Popup.back(); }, + 'click .js-open-migrations'() { + Sidebar.setView('migrations'); + Popup.back(); + }, 'click .js-change-board-color': Popup.open('boardChangeColor'), 'click .js-change-background-image': Popup.open('boardChangeBackgroundImage'), 'click .js-board-info-on-my-boards': Popup.open('boardInfoOnMyBoards'), diff --git a/client/components/sidebar/sidebarMigrations.jade b/client/components/sidebar/sidebarMigrations.jade new file mode 100644 index 000000000..5666b36c1 --- /dev/null +++ b/client/components/sidebar/sidebarMigrations.jade @@ -0,0 +1,69 @@ +template(name='migrationsSidebar') + if currentUser.isBoardAdmin + .sidebar-migrations + h3 + | 🔧 + | {{_ 'migrations'}} + p.quiet {{_ 'migrations-description'}} + + .migrations-list + h4 {{_ 'board-migrations'}} + .migration-item + a.js-run-migration(data-migration="comprehensive") + .migration-name + | {{_ 'comprehensive-board-migration'}} + .migration-status + if comprehensiveMigrationNeeded + span.badge.badge-warning {{_ 'migration-needed'}} + else + span.badge.badge-success {{_ 'migration-complete'}} + + .migration-item + a.js-run-migration(data-migration="fixMissingLists") + .migration-name + | {{_ 'fix-missing-lists-migration'}} + .migration-status + if fixMissingListsNeeded + span.badge.badge-warning {{_ 'migration-needed'}} + else + span.badge.badge-success {{_ 'migration-complete'}} + + hr + h4 {{_ 'global-migrations'}} + .migration-item + a.js-run-migration(data-migration="fixAvatarUrls") + .migration-name + | {{_ 'fix-avatar-urls-migration'}} + .migration-status + if fixAvatarUrlsNeeded + span.badge.badge-warning {{_ 'migration-needed'}} + else + span.badge.badge-success {{_ 'migration-complete'}} + + .migration-item + a.js-run-migration(data-migration="fixAllFileUrls") + .migration-name + | {{_ 'fix-all-file-urls-migration'}} + .migration-status + if fixAllFileUrlsNeeded + span.badge.badge-warning {{_ 'migration-needed'}} + else + span.badge.badge-success {{_ 'migration-complete'}} + else + p.quiet {{_ 'migrations-admin-only'}} + +template(name='runComprehensiveMigrationPopup') + p {{_ 'run-comprehensive-migration-confirm'}} + button.js-confirm.primary.full(type="submit") {{_ 'run-migration'}} + +template(name='runFixMissingListsMigrationPopup') + p {{_ 'run-fix-missing-lists-migration-confirm'}} + button.js-confirm.primary.full(type="submit") {{_ 'run-migration'}} + +template(name='runFixAvatarUrlsMigrationPopup') + p {{_ 'run-fix-avatar-urls-migration-confirm'}} + button.js-confirm.primary.full(type="submit") {{_ 'run-migration'}} + +template(name='runFixAllFileUrlsMigrationPopup') + p {{_ 'run-fix-all-file-urls-migration-confirm'}} + button.js-confirm.primary.full(type="submit") {{_ 'run-migration'}} diff --git a/client/components/sidebar/sidebarMigrations.js b/client/components/sidebar/sidebarMigrations.js new file mode 100644 index 000000000..c8f58a081 --- /dev/null +++ b/client/components/sidebar/sidebarMigrations.js @@ -0,0 +1,143 @@ +import { ReactiveCache } from '/imports/reactiveCache'; +import { TAPi18n } from '/imports/i18n'; + +BlazeComponent.extendComponent({ + onCreated() { + this.migrationStatuses = new ReactiveVar({}); + this.loadMigrationStatuses(); + }, + + loadMigrationStatuses() { + const boardId = Session.get('currentBoard'); + if (!boardId) return; + + // Check comprehensive migration + Meteor.call('comprehensiveBoardMigration.needsMigration', boardId, (err, res) => { + if (!err) { + const statuses = this.migrationStatuses.get(); + statuses.comprehensive = res; + this.migrationStatuses.set(statuses); + } + }); + + // Check fix missing lists migration + Meteor.call('fixMissingListsMigration.needsMigration', boardId, (err, res) => { + if (!err) { + const statuses = this.migrationStatuses.get(); + statuses.fixMissingLists = res; + this.migrationStatuses.set(statuses); + } + }); + + // Check fix avatar URLs migration (global) + Meteor.call('fixAvatarUrls.needsMigration', (err, res) => { + if (!err) { + const statuses = this.migrationStatuses.get(); + statuses.fixAvatarUrls = res; + this.migrationStatuses.set(statuses); + } + }); + + // Check fix all file URLs migration (global) + Meteor.call('fixAllFileUrls.needsMigration', (err, res) => { + if (!err) { + const statuses = this.migrationStatuses.get(); + statuses.fixAllFileUrls = res; + this.migrationStatuses.set(statuses); + } + }); + }, + + comprehensiveMigrationNeeded() { + return this.migrationStatuses.get().comprehensive === true; + }, + + fixMissingListsNeeded() { + return this.migrationStatuses.get().fixMissingLists === true; + }, + + fixAvatarUrlsNeeded() { + return this.migrationStatuses.get().fixAvatarUrls === true; + }, + + fixAllFileUrlsNeeded() { + return this.migrationStatuses.get().fixAllFileUrls === true; + }, + + runMigration(migrationType) { + const boardId = Session.get('currentBoard'); + + let methodName; + let methodArgs = []; + + switch (migrationType) { + case 'comprehensive': + methodName = 'comprehensiveBoardMigration.execute'; + methodArgs = [boardId]; + break; + + case 'fixMissingLists': + methodName = 'fixMissingListsMigration.execute'; + methodArgs = [boardId]; + break; + + case 'fixAvatarUrls': + methodName = 'fixAvatarUrls.execute'; + break; + + case 'fixAllFileUrls': + methodName = 'fixAllFileUrls.execute'; + break; + } + + if (methodName) { + Meteor.call(methodName, ...methodArgs, (err, result) => { + if (err) { + console.error('Migration failed:', err); + // Show error notification + Alert.error(TAPi18n.__('migration-failed') + ': ' + (err.message || err.reason)); + } else { + console.log('Migration completed:', result); + // Show success notification + Alert.success(TAPi18n.__('migration-successful')); + + // Reload migration statuses + Meteor.setTimeout(() => { + this.loadMigrationStatuses(); + }, 1000); + } + }); + } + }, + + events() { + return [ + { + 'click .js-run-migration[data-migration="comprehensive"]': Popup.afterConfirm('runComprehensiveMigration', function() { + const component = BlazeComponent.getComponentForElement(this); + if (component) { + component.runMigration('comprehensive'); + } + }), + 'click .js-run-migration[data-migration="fixMissingLists"]': Popup.afterConfirm('runFixMissingListsMigration', function() { + const component = BlazeComponent.getComponentForElement(this); + if (component) { + component.runMigration('fixMissingLists'); + } + }), + 'click .js-run-migration[data-migration="fixAvatarUrls"]': Popup.afterConfirm('runFixAvatarUrlsMigration', function() { + const component = BlazeComponent.getComponentForElement(this); + if (component) { + component.runMigration('fixAvatarUrls'); + } + }), + 'click .js-run-migration[data-migration="fixAllFileUrls"]': Popup.afterConfirm('runFixAllFileUrlsMigration', function() { + const component = BlazeComponent.getComponentForElement(this); + if (component) { + component.runMigration('fixAllFileUrls'); + } + }), + }, + ]; + }, +}).register('migrationsSidebar'); diff --git a/imports/i18n/data/en.i18n.json b/imports/i18n/data/en.i18n.json index eef27e1fe..e19bd8ef1 100644 --- a/imports/i18n/data/en.i18n.json +++ b/imports/i18n/data/en.i18n.json @@ -1404,7 +1404,31 @@ "back-to-settings": "Back to Settings", "board-id": "Board ID", "board-migration": "Board Migration", + "board-migrations": "Board Migrations", "card-show-lists-on-minicard": "Show Lists on Minicard", + "comprehensive-board-migration": "Comprehensive Board Migration", + "comprehensive-board-migration-description": "Performs comprehensive checks and fixes for board data integrity, including list ordering, card positions, and swimlane structure.", + "fix-missing-lists-migration": "Fix Missing Lists", + "fix-missing-lists-migration-description": "Detects and repairs missing or corrupted lists in the board structure.", + "fix-avatar-urls-migration": "Fix Avatar URLs", + "fix-avatar-urls-migration-description": "Updates avatar URLs to use the correct storage backend and fixes broken avatar references.", + "fix-all-file-urls-migration": "Fix All File URLs", + "fix-all-file-urls-migration-description": "Updates all file attachment URLs to use the correct storage backend and fixes broken file references.", + "global-migrations": "Global Migrations", + "migration-needed": "Migration Needed", + "migration-complete": "Complete", + "migration-running": "Running...", + "migration-successful": "Migration completed successfully", + "migration-failed": "Migration failed", + "migrations": "Migrations", + "migrations-admin-only": "Only board administrators can run migrations", + "migrations-description": "Run data integrity checks and repairs for this board. Each migration can be executed individually.", + "no-issues-found": "No issues found", + "run-migration": "Run Migration", + "run-comprehensive-migration-confirm": "This will perform a comprehensive migration to check and fix board data integrity. This may take a few moments. Continue?", + "run-fix-missing-lists-migration-confirm": "This will detect and repair missing or corrupted lists in the board structure. Continue?", + "run-fix-avatar-urls-migration-confirm": "This will update avatar URLs across all boards to use the correct storage backend. This is a global operation. Continue?", + "run-fix-all-file-urls-migration-confirm": "This will update all file attachment URLs across all boards to use the correct storage backend. This is a global operation. Continue?", "cleanup": "Cleanup", "cleanup-old-jobs": "Cleanup Old Jobs", "completed": "Completed", diff --git a/server/migrations/fixAllFileUrls.js b/server/migrations/fixAllFileUrls.js index caba86e68..6b3be9ccf 100644 --- a/server/migrations/fixAllFileUrls.js +++ b/server/migrations/fixAllFileUrls.js @@ -30,15 +30,11 @@ class FixAllFileUrlsMigration { } } - // Check for problematic attachment URLs in cards - const cards = ReactiveCache.getCards({}); - for (const card of cards) { - if (card.attachments) { - for (const attachment of card.attachments) { - if (attachment.url && this.hasProblematicUrl(attachment.url)) { - return true; - } - } + // Check for problematic attachment URLs + const attachments = ReactiveCache.getAttachments({}); + for (const attachment of attachments) { + if (attachment.url && this.hasProblematicUrl(attachment.url)) { + return true; } } @@ -206,51 +202,40 @@ class FixAllFileUrlsMigration { } /** - * Fix attachment URLs in card references + * Fix attachment URLs in the Attachments collection */ async fixCardAttachmentUrls() { - const cards = ReactiveCache.getCards({}); - let cardsFixed = 0; + const attachments = ReactiveCache.getAttachments({}); + let attachmentsFixed = 0; - for (const card of cards) { - if (card.attachments) { - let needsUpdate = false; - const updatedAttachments = card.attachments.map(attachment => { - if (attachment.url && this.hasProblematicUrl(attachment.url)) { - try { - const fileId = attachment._id || extractFileIdFromUrl(attachment.url, 'attachment'); - const cleanUrl = fileId ? generateUniversalAttachmentUrl(fileId) : cleanFileUrl(attachment.url, 'attachment'); - - if (cleanUrl && cleanUrl !== attachment.url) { - needsUpdate = true; - return { ...attachment, url: cleanUrl }; + for (const attachment of attachments) { + if (attachment.url && this.hasProblematicUrl(attachment.url)) { + try { + const fileId = attachment._id || extractFileIdFromUrl(attachment.url, 'attachment'); + const cleanUrl = fileId ? generateUniversalAttachmentUrl(fileId) : cleanFileUrl(attachment.url, 'attachment'); + + if (cleanUrl && cleanUrl !== attachment.url) { + // Update attachment with fixed URL + Attachments.update(attachment._id, { + $set: { + url: cleanUrl, + modifiedAt: new Date() } - } catch (error) { - console.error(`Error fixing card attachment URL:`, error); + }); + + attachmentsFixed++; + + if (process.env.DEBUG === 'true') { + console.log(`Fixed attachment URL ${attachment._id}`); } } - return attachment; - }); - - if (needsUpdate) { - // Update card with fixed attachment URLs - Cards.update(card._id, { - $set: { - attachments: updatedAttachments, - modifiedAt: new Date() - } - }); - - cardsFixed++; - - if (process.env.DEBUG === 'true') { - console.log(`Fixed attachment URLs in card ${card._id}`); - } + } catch (error) { + console.error(`Error fixing attachment URL:`, error); } } } - return cardsFixed; + return attachmentsFixed; } } From fbd6b920ef134f8733d46e04835b88d9da7f1a19 Mon Sep 17 00:00:00 2001 From: Lauri Ojansivu Date: Wed, 5 Nov 2025 17:08:10 +0200 Subject: [PATCH 38/84] Updated ChangeLog. --- CHANGELOG.md | 2 ++ 1 file changed, 2 insertions(+) diff --git a/CHANGELOG.md b/CHANGELOG.md index 4bb4e2808..bcc7685df 100644 --- a/CHANGELOG.md +++ b/CHANGELOG.md @@ -28,6 +28,8 @@ This release fixes the following bugs: - [Fix 8.16: Switching Board View fails with 403 error](https://github.com/wekan/wekan/commit/550d87ac6cb3ec946600616485afdbd242983ab4). Thanks to xet7. +- [Moved migrations from opening board to right sidebar / Migrations](https://github.com/wekan/wekan/commit/1b25d1d5720d4f486a10d2acce37e315cf9b6057). + Thanks to xet7. Thanks to above GitHub users for their contributions and translators for their translations. From 7713e613b431e44dc13cee72e7a1e5f031473fa6 Mon Sep 17 00:00:00 2001 From: Lauri Ojansivu Date: Wed, 5 Nov 2025 18:44:48 +0200 Subject: [PATCH 39/84] Fix 8.16 Lists with no items are deleted every time when board is opened. Moved migrations to right sidebar. Thanks to xet7 ! Fixes #5994 --- client/components/boards/boardBody.js | 1 - .../components/sidebar/sidebarMigrations.jade | 42 ++ .../components/sidebar/sidebarMigrations.js | 271 +++++++++-- imports/i18n/data/en.i18n.json | 44 ++ .../migrations/comprehensiveBoardMigration.js | 14 +- .../migrations/deleteDuplicateEmptyLists.js | 423 ++++++++++++++++++ server/migrations/restoreAllArchived.js | 266 +++++++++++ server/migrations/restoreLostCards.js | 259 +++++++++++ 8 files changed, 1278 insertions(+), 42 deletions(-) create mode 100644 server/migrations/deleteDuplicateEmptyLists.js create mode 100644 server/migrations/restoreAllArchived.js create mode 100644 server/migrations/restoreLostCards.js diff --git a/client/components/boards/boardBody.js b/client/components/boards/boardBody.js index 4e14d8001..b0af16e43 100644 --- a/client/components/boards/boardBody.js +++ b/client/components/boards/boardBody.js @@ -146,7 +146,6 @@ BlazeComponent.extendComponent({ { step: 'fix_orphaned_cards', name: 'Fix Orphaned Cards', duration: 2000 }, { step: 'convert_shared_lists', name: 'Convert Shared Lists', duration: 3000 }, { step: 'ensure_per_swimlane_lists', name: 'Ensure Per-Swimlane Lists', duration: 1500 }, - { step: 'cleanup_empty_lists', name: 'Cleanup Empty Lists', duration: 1000 }, { step: 'validate_migration', name: 'Validate Migration', duration: 1000 }, { step: 'fix_avatar_urls', name: 'Fix Avatar URLs', duration: 1000 }, { step: 'fix_attachment_urls', name: 'Fix Attachment URLs', duration: 1000 } diff --git a/client/components/sidebar/sidebarMigrations.jade b/client/components/sidebar/sidebarMigrations.jade index 5666b36c1..78da56983 100644 --- a/client/components/sidebar/sidebarMigrations.jade +++ b/client/components/sidebar/sidebarMigrations.jade @@ -28,6 +28,36 @@ template(name='migrationsSidebar') else span.badge.badge-success {{_ 'migration-complete'}} + .migration-item + a.js-run-migration(data-migration="deleteDuplicateEmptyLists") + .migration-name + | {{_ 'delete-duplicate-empty-lists-migration'}} + .migration-status + if deleteDuplicateEmptyListsNeeded + span.badge.badge-warning {{_ 'migration-needed'}} + else + span.badge.badge-success {{_ 'migration-complete'}} + + .migration-item + a.js-run-migration(data-migration="restoreLostCards") + .migration-name + | {{_ 'restore-lost-cards-migration'}} + .migration-status + if restoreLostCardsNeeded + span.badge.badge-warning {{_ 'migration-needed'}} + else + span.badge.badge-success {{_ 'migration-complete'}} + + .migration-item + a.js-run-migration(data-migration="restoreAllArchived") + .migration-name + | {{_ 'restore-all-archived-migration'}} + .migration-status + if restoreAllArchivedNeeded + span.badge.badge-warning {{_ 'migration-needed'}} + else + span.badge.badge-success {{_ 'migration-complete'}} + hr h4 {{_ 'global-migrations'}} .migration-item @@ -60,6 +90,18 @@ template(name='runFixMissingListsMigrationPopup') p {{_ 'run-fix-missing-lists-migration-confirm'}} button.js-confirm.primary.full(type="submit") {{_ 'run-migration'}} +template(name='runDeleteDuplicateEmptyListsMigrationPopup') + p {{_ 'run-delete-duplicate-empty-lists-migration-confirm'}} + button.js-confirm.primary.full(type="submit") {{_ 'run-migration'}} + +template(name='runRestoreLostCardsMigrationPopup') + p {{_ 'run-restore-lost-cards-migration-confirm'}} + button.js-confirm.primary.full(type="submit") {{_ 'run-migration'}} + +template(name='runRestoreAllArchivedMigrationPopup') + p {{_ 'run-restore-all-archived-migration-confirm'}} + button.js-confirm.primary.full(type="submit") {{_ 'run-migration'}} + template(name='runFixAvatarUrlsMigrationPopup') p {{_ 'run-fix-avatar-urls-migration-confirm'}} button.js-confirm.primary.full(type="submit") {{_ 'run-migration'}} diff --git a/client/components/sidebar/sidebarMigrations.js b/client/components/sidebar/sidebarMigrations.js index c8f58a081..cc47b27cf 100644 --- a/client/components/sidebar/sidebarMigrations.js +++ b/client/components/sidebar/sidebarMigrations.js @@ -1,5 +1,6 @@ import { ReactiveCache } from '/imports/reactiveCache'; import { TAPi18n } from '/imports/i18n'; +import { migrationProgressManager } from '/client/components/migrationProgress'; BlazeComponent.extendComponent({ onCreated() { @@ -29,11 +30,38 @@ BlazeComponent.extendComponent({ } }); - // Check fix avatar URLs migration (global) - Meteor.call('fixAvatarUrls.needsMigration', (err, res) => { + // Check delete duplicate empty lists migration + Meteor.call('deleteDuplicateEmptyLists.needsMigration', boardId, (err, res) => { if (!err) { const statuses = this.migrationStatuses.get(); - statuses.fixAvatarUrls = res; + statuses.deleteDuplicateEmptyLists = res; + this.migrationStatuses.set(statuses); + } + }); + + // Check restore lost cards migration + Meteor.call('restoreLostCards.needsMigration', boardId, (err, res) => { + if (!err) { + const statuses = this.migrationStatuses.get(); + statuses.restoreLostCards = res; + this.migrationStatuses.set(statuses); + } + }); + + // Check restore all archived migration + Meteor.call('restoreAllArchived.needsMigration', boardId, (err, res) => { + if (!err) { + const statuses = this.migrationStatuses.get(); + statuses.restoreAllArchived = res; + this.migrationStatuses.set(statuses); + } + }); + + // Check fix avatar URLs migration (global) + Meteor.call('fixAvatarUrls.needsMigration', (err, res) => { + if (!err) { + const statuses = this.migrationStatuses.get(); + statuses.fixAvatarUrls = res; this.migrationStatuses.set(statuses); } }); @@ -56,6 +84,22 @@ BlazeComponent.extendComponent({ return this.migrationStatuses.get().fixMissingLists === true; }, + deleteEmptyListsNeeded() { + return this.migrationStatuses.get().deleteEmptyLists === true; + }, + + deleteDuplicateEmptyListsNeeded() { + return this.migrationStatuses.get().deleteDuplicateEmptyLists === true; + }, + + restoreLostCardsNeeded() { + return this.migrationStatuses.get().restoreLostCards === true; + }, + + restoreAllArchivedNeeded() { + return this.migrationStatuses.get().restoreAllArchived === true; + }, + fixAvatarUrlsNeeded() { return this.migrationStatuses.get().fixAvatarUrls === true; }, @@ -64,6 +108,58 @@ BlazeComponent.extendComponent({ return this.migrationStatuses.get().fixAllFileUrls === true; }, + // Simulate migration progress updates using the global progress popup + async simulateMigrationProgress(progressSteps) { + const totalSteps = progressSteps.length; + for (let i = 0; i < progressSteps.length; i++) { + const step = progressSteps[i]; + const overall = Math.round(((i + 1) / totalSteps) * 100); + + // Start step + migrationProgressManager.updateProgress({ + overallProgress: overall, + currentStep: i + 1, + totalSteps, + stepName: step.step, + stepProgress: 0, + stepStatus: `Starting ${step.name}...`, + stepDetails: null, + boardId: Session.get('currentBoard'), + }); + + const stepDuration = step.duration; + const updateInterval = 100; + const totalUpdates = Math.max(1, Math.floor(stepDuration / updateInterval)); + for (let j = 0; j < totalUpdates; j++) { + const per = Math.round(((j + 1) / totalUpdates) * 100); + migrationProgressManager.updateProgress({ + overallProgress: overall, + currentStep: i + 1, + totalSteps, + stepName: step.step, + stepProgress: per, + stepStatus: `Processing ${step.name}...`, + stepDetails: { progress: `${per}%` }, + boardId: Session.get('currentBoard'), + }); + // eslint-disable-next-line no-await-in-loop + await new Promise((r) => setTimeout(r, updateInterval)); + } + + // Complete step + migrationProgressManager.updateProgress({ + overallProgress: overall, + currentStep: i + 1, + totalSteps, + stepName: step.step, + stepProgress: 100, + stepStatus: `${step.name} completed`, + stepDetails: { status: 'completed' }, + boardId: Session.get('currentBoard'), + }); + } + }, + runMigration(migrationType) { const boardId = Session.get('currentBoard'); @@ -81,6 +177,26 @@ BlazeComponent.extendComponent({ methodArgs = [boardId]; break; + case 'deleteEmptyLists': + methodName = 'deleteEmptyLists.execute'; + methodArgs = [boardId]; + break; + + case 'deleteDuplicateEmptyLists': + methodName = 'deleteDuplicateEmptyLists.execute'; + methodArgs = [boardId]; + break; + + case 'restoreLostCards': + methodName = 'restoreLostCards.execute'; + methodArgs = [boardId]; + break; + + case 'restoreAllArchived': + methodName = 'restoreAllArchived.execute'; + methodArgs = [boardId]; + break; + case 'fixAvatarUrls': methodName = 'fixAvatarUrls.execute'; break; @@ -91,17 +207,104 @@ BlazeComponent.extendComponent({ } if (methodName) { - Meteor.call(methodName, ...methodArgs, (err, result) => { - if (err) { - console.error('Migration failed:', err); - // Show error notification - Alert.error(TAPi18n.__('migration-failed') + ': ' + (err.message || err.reason)); + // Define simulated steps per migration type + const stepsByType = { + comprehensive: [ + { step: 'analyze_board_structure', name: 'Analyze Board Structure', duration: 800 }, + { step: 'fix_orphaned_cards', name: 'Fix Orphaned Cards', duration: 1200 }, + { step: 'convert_shared_lists', name: 'Convert Shared Lists', duration: 1000 }, + { step: 'ensure_per_swimlane_lists', name: 'Ensure Per-Swimlane Lists', duration: 800 }, + { step: 'validate_migration', name: 'Validate Migration', duration: 800 }, + { step: 'fix_avatar_urls', name: 'Fix Avatar URLs', duration: 600 }, + { step: 'fix_attachment_urls', name: 'Fix Attachment URLs', duration: 600 }, + ], + fixMissingLists: [ + { step: 'analyze_lists', name: 'Analyze Lists', duration: 600 }, + { step: 'create_missing_lists', name: 'Create Missing Lists', duration: 900 }, + { step: 'update_cards', name: 'Update Cards', duration: 900 }, + { step: 'finalize', name: 'Finalize', duration: 400 }, + ], + deleteEmptyLists: [ + { step: 'convert_shared_lists', name: 'Convert Shared Lists', duration: 700 }, + { step: 'delete_empty_lists', name: 'Delete Empty Lists', duration: 800 }, + ], + deleteDuplicateEmptyLists: [ + { step: 'convert_shared_lists', name: 'Convert Shared Lists', duration: 700 }, + { step: 'delete_duplicate_empty_lists', name: 'Delete Duplicate Empty Lists', duration: 800 }, + ], + restoreLostCards: [ + { step: 'ensure_lost_cards_swimlane', name: 'Ensure Lost Cards Swimlane', duration: 600 }, + { step: 'restore_lists', name: 'Restore Lists', duration: 800 }, + { step: 'restore_cards', name: 'Restore Cards', duration: 1000 }, + ], + restoreAllArchived: [ + { step: 'restore_swimlanes', name: 'Restore Swimlanes', duration: 800 }, + { step: 'restore_lists', name: 'Restore Lists', duration: 900 }, + { step: 'restore_cards', name: 'Restore Cards', duration: 1000 }, + { step: 'fix_missing_ids', name: 'Fix Missing IDs', duration: 600 }, + ], + fixAvatarUrls: [ + { step: 'scan_users', name: 'Scan Users', duration: 500 }, + { step: 'fix_urls', name: 'Fix Avatar URLs', duration: 900 }, + ], + fixAllFileUrls: [ + { step: 'scan_files', name: 'Scan Files', duration: 600 }, + { step: 'fix_urls', name: 'Fix File URLs', duration: 1000 }, + ], + }; + + const steps = stepsByType[migrationType] || [ + { step: 'running', name: 'Running Migration', duration: 1000 }, + ]; + + // Kick off popup and simulated progress + migrationProgressManager.startMigration(); + const progressPromise = this.simulateMigrationProgress(steps); + + // Start migration call + const callPromise = new Promise((resolve, reject) => { + Meteor.call(methodName, ...methodArgs, (err, result) => { + if (err) return reject(err); + return resolve(result); + }); + }); + + Promise.allSettled([callPromise, progressPromise]).then(([callRes]) => { + if (callRes.status === 'rejected') { + migrationProgressManager.failMigration(callRes.reason); } else { - console.log('Migration completed:', result); - // Show success notification - Alert.success(TAPi18n.__('migration-successful')); - - // Reload migration statuses + const result = callRes.value; + // Summarize result details in the popup + let summary = {}; + if (result && result.results) { + // Comprehensive returns {success, results} + const r = result.results; + summary = { + totalCardsProcessed: r.totalCardsProcessed, + totalListsProcessed: r.totalListsProcessed, + totalListsCreated: r.totalListsCreated, + }; + } else if (result && result.changes) { + // Many migrations return a changes string array + summary = { changes: result.changes.join(' | ') }; + } else if (result && typeof result === 'object') { + summary = result; + } + + migrationProgressManager.updateProgress({ + overallProgress: 100, + currentStep: steps.length, + totalSteps: steps.length, + stepName: 'completed', + stepProgress: 100, + stepStatus: 'Migration completed', + stepDetails: summary, + boardId: Session.get('currentBoard'), + }); + + migrationProgressManager.completeMigration(); + + // Refresh status badges slightly after Meteor.setTimeout(() => { this.loadMigrationStatuses(); }, 1000); @@ -111,31 +314,41 @@ BlazeComponent.extendComponent({ }, events() { + const self = this; // Capture component reference + return [ { 'click .js-run-migration[data-migration="comprehensive"]': Popup.afterConfirm('runComprehensiveMigration', function() { - const component = BlazeComponent.getComponentForElement(this); - if (component) { - component.runMigration('comprehensive'); - } + self.runMigration('comprehensive'); + Popup.back(); }), 'click .js-run-migration[data-migration="fixMissingLists"]': Popup.afterConfirm('runFixMissingListsMigration', function() { - const component = BlazeComponent.getComponentForElement(this); - if (component) { - component.runMigration('fixMissingLists'); - } + self.runMigration('fixMissingLists'); + Popup.back(); + }), + 'click .js-run-migration[data-migration="deleteEmptyLists"]': Popup.afterConfirm('runDeleteEmptyListsMigration', function() { + self.runMigration('deleteEmptyLists'); + Popup.back(); + }), + 'click .js-run-migration[data-migration="deleteDuplicateEmptyLists"]': Popup.afterConfirm('runDeleteDuplicateEmptyListsMigration', function() { + self.runMigration('deleteDuplicateEmptyLists'); + Popup.back(); + }), + 'click .js-run-migration[data-migration="restoreLostCards"]': Popup.afterConfirm('runRestoreLostCardsMigration', function() { + self.runMigration('restoreLostCards'); + Popup.back(); + }), + 'click .js-run-migration[data-migration="restoreAllArchived"]': Popup.afterConfirm('runRestoreAllArchivedMigration', function() { + self.runMigration('restoreAllArchived'); + Popup.back(); }), 'click .js-run-migration[data-migration="fixAvatarUrls"]': Popup.afterConfirm('runFixAvatarUrlsMigration', function() { - const component = BlazeComponent.getComponentForElement(this); - if (component) { - component.runMigration('fixAvatarUrls'); - } + self.runMigration('fixAvatarUrls'); + Popup.back(); }), 'click .js-run-migration[data-migration="fixAllFileUrls"]': Popup.afterConfirm('runFixAllFileUrlsMigration', function() { - const component = BlazeComponent.getComponentForElement(this); - if (component) { - component.runMigration('fixAllFileUrls'); - } + self.runMigration('fixAllFileUrls'); + Popup.back(); }), }, ]; diff --git a/imports/i18n/data/en.i18n.json b/imports/i18n/data/en.i18n.json index e19bd8ef1..c007af213 100644 --- a/imports/i18n/data/en.i18n.json +++ b/imports/i18n/data/en.i18n.json @@ -1408,6 +1408,16 @@ "card-show-lists-on-minicard": "Show Lists on Minicard", "comprehensive-board-migration": "Comprehensive Board Migration", "comprehensive-board-migration-description": "Performs comprehensive checks and fixes for board data integrity, including list ordering, card positions, and swimlane structure.", + "delete-empty-lists-migration": "Delete Empty Lists", + "delete-empty-lists-migration-description": "Safely deletes empty duplicate lists. Only removes lists that have no cards AND have another list with the same title that contains cards.", + "delete-duplicate-empty-lists-migration": "Delete Duplicate Empty Lists", + "delete-duplicate-empty-lists-migration-description": "Safely deletes empty duplicate lists. Only removes lists that have no cards AND have another list with the same title that contains cards.", + "lost-cards": "Lost Cards", + "lost-cards-list": "Restored Items", + "restore-lost-cards-migration": "Restore Lost Cards", + "restore-lost-cards-migration-description": "Finds and restores cards and lists with missing swimlaneId or listId. Creates a 'Lost Cards' swimlane to make all lost items visible again.", + "restore-all-archived-migration": "Restore All Archived", + "restore-all-archived-migration-description": "Restores all archived swimlanes, lists, and cards. Automatically fixes any missing swimlaneId or listId to make items visible.", "fix-missing-lists-migration": "Fix Missing Lists", "fix-missing-lists-migration-description": "Detects and repairs missing or corrupted lists in the board structure.", "fix-avatar-urls-migration": "Fix Avatar URLs", @@ -1426,9 +1436,43 @@ "no-issues-found": "No issues found", "run-migration": "Run Migration", "run-comprehensive-migration-confirm": "This will perform a comprehensive migration to check and fix board data integrity. This may take a few moments. Continue?", + "run-delete-empty-lists-migration-confirm": "This will first convert any shared lists to per-swimlane lists, then delete empty lists that have a duplicate list with the same title containing cards. Only truly redundant empty lists will be removed. Continue?", + "run-delete-duplicate-empty-lists-migration-confirm": "This will first convert any shared lists to per-swimlane lists, then delete empty lists that have a duplicate list with the same title containing cards. Only truly redundant empty lists will be removed. Continue?", + "run-restore-lost-cards-migration-confirm": "This will create a 'Lost Cards' swimlane and restore all cards and lists with missing swimlaneId or listId. This only affects non-archived items. Continue?", + "run-restore-all-archived-migration-confirm": "This will restore ALL archived swimlanes, lists, and cards, making them visible again. Any items with missing IDs will be automatically fixed. This cannot be easily undone. Continue?", "run-fix-missing-lists-migration-confirm": "This will detect and repair missing or corrupted lists in the board structure. Continue?", "run-fix-avatar-urls-migration-confirm": "This will update avatar URLs across all boards to use the correct storage backend. This is a global operation. Continue?", "run-fix-all-file-urls-migration-confirm": "This will update all file attachment URLs across all boards to use the correct storage backend. This is a global operation. Continue?", + "restore-lost-cards-nothing-to-restore": "No lost swimlanes, lists, or cards to restore", + + "migration-progress-title": "Board Migration in Progress", + "migration-progress-overall": "Overall Progress", + "migration-progress-current-step": "Current Step", + "migration-progress-status": "Status", + "migration-progress-details": "Details", + "migration-progress-note": "Please wait while we migrate your board to the latest structure...", + + "step-analyze-board-structure": "Analyze Board Structure", + "step-fix-orphaned-cards": "Fix Orphaned Cards", + "step-convert-shared-lists": "Convert Shared Lists", + "step-ensure-per-swimlane-lists": "Ensure Per-Swimlane Lists", + "step-validate-migration": "Validate Migration", + "step-fix-avatar-urls": "Fix Avatar URLs", + "step-fix-attachment-urls": "Fix Attachment URLs", + "step-analyze-lists": "Analyze Lists", + "step-create-missing-lists": "Create Missing Lists", + "step-update-cards": "Update Cards", + "step-finalize": "Finalize", + "step-delete-empty-lists": "Delete Empty Lists", + "step-delete-duplicate-empty-lists": "Delete Duplicate Empty Lists", + "step-ensure-lost-cards-swimlane": "Ensure Lost Cards Swimlane", + "step-restore-lists": "Restore Lists", + "step-restore-cards": "Restore Cards", + "step-restore-swimlanes": "Restore Swimlanes", + "step-fix-missing-ids": "Fix Missing IDs", + "step-scan-users": "Scan Users", + "step-scan-files": "Scan Files", + "step-fix-file-urls": "Fix File URLs", "cleanup": "Cleanup", "cleanup-old-jobs": "Cleanup Old Jobs", "completed": "Completed", diff --git a/server/migrations/comprehensiveBoardMigration.js b/server/migrations/comprehensiveBoardMigration.js index f9ea7c523..e8bfc7469 100644 --- a/server/migrations/comprehensiveBoardMigration.js +++ b/server/migrations/comprehensiveBoardMigration.js @@ -34,7 +34,6 @@ class ComprehensiveBoardMigration { 'fix_orphaned_cards', 'convert_shared_lists', 'ensure_per_swimlane_lists', - 'cleanup_empty_lists', 'validate_migration' ]; } @@ -169,7 +168,6 @@ class ComprehensiveBoardMigration { totalCardsProcessed: 0, totalListsProcessed: 0, totalListsCreated: 0, - totalListsRemoved: 0, errors: [] }; @@ -239,15 +237,7 @@ class ComprehensiveBoardMigration { listsProcessed: results.steps.ensurePerSwimlane.listsProcessed }); - // Step 5: Cleanup empty lists - updateProgress('cleanup_empty_lists', 0, 'Cleaning up empty lists...'); - results.steps.cleanupEmpty = await this.cleanupEmptyLists(boardId); - results.totalListsRemoved += results.steps.cleanupEmpty.listsRemoved || 0; - updateProgress('cleanup_empty_lists', 100, 'Empty lists cleaned up', { - listsRemoved: results.steps.cleanupEmpty.listsRemoved - }); - - // Step 6: Validate migration + // Step 5: Validate migration updateProgress('validate_migration', 0, 'Validating migration...'); results.steps.validate = await this.validateMigration(boardId); updateProgress('validate_migration', 100, 'Migration validated', { @@ -256,7 +246,7 @@ class ComprehensiveBoardMigration { totalLists: results.steps.validate.totalLists }); - // Step 7: Fix avatar URLs + // Step 6: Fix avatar URLs updateProgress('fix_avatar_urls', 0, 'Fixing avatar URLs...'); results.steps.fixAvatarUrls = await this.fixAvatarUrls(boardId); updateProgress('fix_avatar_urls', 100, 'Avatar URLs fixed', { diff --git a/server/migrations/deleteDuplicateEmptyLists.js b/server/migrations/deleteDuplicateEmptyLists.js new file mode 100644 index 000000000..c0d455685 --- /dev/null +++ b/server/migrations/deleteDuplicateEmptyLists.js @@ -0,0 +1,423 @@ +/** + * Delete Duplicate Empty Lists Migration + * + * Safely deletes empty duplicate lists from a board: + * 1. First converts any shared lists to per-swimlane lists + * 2. Only deletes per-swimlane lists that: + * - Have no cards + * - Have another list with the same title on the same board that DOES have cards + * 3. This prevents deleting unique empty lists and only removes redundant duplicates + */ + +import { Meteor } from 'meteor/meteor'; +import { check } from 'meteor/check'; +import { ReactiveCache } from '/imports/reactiveCache'; +import Boards from '/models/boards'; +import Lists from '/models/lists'; +import Cards from '/models/cards'; +import Swimlanes from '/models/swimlanes'; + +class DeleteDuplicateEmptyListsMigration { + constructor() { + this.name = 'deleteDuplicateEmptyLists'; + this.version = 1; + } + + /** + * Check if migration is needed for a board + */ + needsMigration(boardId) { + try { + const lists = ReactiveCache.getLists({ boardId }); + const cards = ReactiveCache.getCards({ boardId }); + + // Check if there are any empty lists that have a duplicate with the same title containing cards + for (const list of lists) { + // Skip shared lists + if (!list.swimlaneId || list.swimlaneId === '') { + continue; + } + + // Check if list is empty + const listCards = cards.filter(card => card.listId === list._id); + if (listCards.length === 0) { + // Check if there's a duplicate list with the same title that has cards + const duplicateListsWithSameTitle = lists.filter(l => + l._id !== list._id && + l.title === list.title && + l.boardId === boardId + ); + + for (const duplicateList of duplicateListsWithSameTitle) { + const duplicateListCards = cards.filter(card => card.listId === duplicateList._id); + if (duplicateListCards.length > 0) { + return true; // Found an empty list with a duplicate that has cards + } + } + } + } + + return false; + } catch (error) { + console.error('Error checking if deleteEmptyLists migration is needed:', error); + return false; + } + } + + /** + * Execute the migration + */ + async executeMigration(boardId) { + try { + const results = { + sharedListsConverted: 0, + listsDeleted: 0, + errors: [] + }; + + // Step 1: Convert shared lists to per-swimlane lists first + const conversionResult = await this.convertSharedListsToPerSwimlane(boardId); + results.sharedListsConverted = conversionResult.listsConverted; + + // Step 2: Delete empty per-swimlane lists + const deletionResult = await this.deleteEmptyPerSwimlaneLists(boardId); + results.listsDeleted = deletionResult.listsDeleted; + + return { + success: true, + changes: [ + `Converted ${results.sharedListsConverted} shared lists to per-swimlane lists`, + `Deleted ${results.listsDeleted} empty per-swimlane lists` + ], + results + }; + } catch (error) { + console.error('Error executing deleteEmptyLists migration:', error); + return { + success: false, + error: error.message + }; + } + } + + /** + * Convert shared lists (lists without swimlaneId) to per-swimlane lists + */ + async convertSharedListsToPerSwimlane(boardId) { + const lists = ReactiveCache.getLists({ boardId }); + const swimlanes = ReactiveCache.getSwimlanes({ boardId, archived: false }); + const cards = ReactiveCache.getCards({ boardId }); + + let listsConverted = 0; + + // Find shared lists (lists without swimlaneId) + const sharedLists = lists.filter(list => !list.swimlaneId || list.swimlaneId === ''); + + if (sharedLists.length === 0) { + return { listsConverted: 0 }; + } + + for (const sharedList of sharedLists) { + // Get cards in this shared list + const listCards = cards.filter(card => card.listId === sharedList._id); + + // Group cards by swimlane + const cardsBySwimlane = {}; + for (const card of listCards) { + const swimlaneId = card.swimlaneId || 'default'; + if (!cardsBySwimlane[swimlaneId]) { + cardsBySwimlane[swimlaneId] = []; + } + cardsBySwimlane[swimlaneId].push(card); + } + + // Create per-swimlane lists for each swimlane that has cards + for (const swimlane of swimlanes) { + const swimlaneCards = cardsBySwimlane[swimlane._id] || []; + + if (swimlaneCards.length > 0) { + // Check if per-swimlane list already exists + const existingList = lists.find(l => + l.title === sharedList.title && + l.swimlaneId === swimlane._id && + l._id !== sharedList._id + ); + + if (!existingList) { + // Create new per-swimlane list + const newListId = Lists.insert({ + title: sharedList.title, + boardId: boardId, + swimlaneId: swimlane._id, + sort: sharedList.sort, + createdAt: new Date(), + updatedAt: new Date(), + archived: false + }); + + // Move cards to the new list + for (const card of swimlaneCards) { + Cards.update(card._id, { + $set: { + listId: newListId, + swimlaneId: swimlane._id + } + }); + } + + if (process.env.DEBUG === 'true') { + console.log(`Created per-swimlane list "${sharedList.title}" for swimlane ${swimlane.title || swimlane._id}`); + } + } else { + // Move cards to existing per-swimlane list + for (const card of swimlaneCards) { + Cards.update(card._id, { + $set: { + listId: existingList._id, + swimlaneId: swimlane._id + } + }); + } + + if (process.env.DEBUG === 'true') { + console.log(`Moved cards to existing per-swimlane list "${sharedList.title}" in swimlane ${swimlane.title || swimlane._id}`); + } + } + } + } + + // Remove the shared list (now that all cards are moved) + Lists.remove(sharedList._id); + listsConverted++; + + if (process.env.DEBUG === 'true') { + console.log(`Removed shared list "${sharedList.title}"`); + } + } + + return { listsConverted }; + } + + /** + * Delete empty per-swimlane lists + * Only deletes lists that: + * 1. Have a swimlaneId (are per-swimlane, not shared) + * 2. Have no cards + * 3. Have a duplicate list with the same title on the same board that contains cards + */ + async deleteEmptyPerSwimlaneLists(boardId) { + const lists = ReactiveCache.getLists({ boardId }); + const cards = ReactiveCache.getCards({ boardId }); + + let listsDeleted = 0; + + for (const list of lists) { + // Safety check 1: List must have a swimlaneId (must be per-swimlane, not shared) + if (!list.swimlaneId || list.swimlaneId === '') { + if (process.env.DEBUG === 'true') { + console.log(`Skipping list "${list.title}" - no swimlaneId (shared list)`); + } + continue; + } + + // Safety check 2: List must have no cards + const listCards = cards.filter(card => card.listId === list._id); + if (listCards.length > 0) { + if (process.env.DEBUG === 'true') { + console.log(`Skipping list "${list.title}" - has ${listCards.length} cards`); + } + continue; + } + + // Safety check 3: There must be another list with the same title on the same board that has cards + const duplicateListsWithSameTitle = lists.filter(l => + l._id !== list._id && + l.title === list.title && + l.boardId === boardId + ); + + let hasDuplicateWithCards = false; + for (const duplicateList of duplicateListsWithSameTitle) { + const duplicateListCards = cards.filter(card => card.listId === duplicateList._id); + if (duplicateListCards.length > 0) { + hasDuplicateWithCards = true; + break; + } + } + + if (!hasDuplicateWithCards) { + if (process.env.DEBUG === 'true') { + console.log(`Skipping list "${list.title}" - no duplicate list with same title that has cards`); + } + continue; + } + + // All safety checks passed - delete the empty per-swimlane list + Lists.remove(list._id); + listsDeleted++; + + if (process.env.DEBUG === 'true') { + console.log(`Deleted empty per-swimlane list: "${list.title}" (swimlane: ${list.swimlaneId}) - duplicate with cards exists`); + } + } + + return { listsDeleted }; + } + + /** + * Get detailed status of empty lists + */ + async getStatus(boardId) { + const lists = ReactiveCache.getLists({ boardId }); + const cards = ReactiveCache.getCards({ boardId }); + + const sharedLists = []; + const emptyPerSwimlaneLists = []; + const nonEmptyLists = []; + + for (const list of lists) { + const listCards = cards.filter(card => card.listId === list._id); + const isShared = !list.swimlaneId || list.swimlaneId === ''; + const isEmpty = listCards.length === 0; + + if (isShared) { + sharedLists.push({ + id: list._id, + title: list.title, + cardCount: listCards.length + }); + } else if (isEmpty) { + emptyPerSwimlaneLists.push({ + id: list._id, + title: list.title, + swimlaneId: list.swimlaneId + }); + } else { + nonEmptyLists.push({ + id: list._id, + title: list.title, + swimlaneId: list.swimlaneId, + cardCount: listCards.length + }); + } + } + + return { + sharedListsCount: sharedLists.length, + emptyPerSwimlaneLists: emptyPerSwimlaneLists.length, + totalLists: lists.length, + details: { + sharedLists, + emptyPerSwimlaneLists, + nonEmptyLists + } + }; + } +} + +const deleteDuplicateEmptyListsMigration = new DeleteDuplicateEmptyListsMigration(); + +// Register Meteor methods +Meteor.methods({ + 'deleteEmptyLists.needsMigration'(boardId) { + check(boardId, String); + + if (!this.userId) { + throw new Meteor.Error('not-authorized', 'You must be logged in'); + } + + return deleteDuplicateEmptyListsMigration.needsMigration(boardId); + }, + + 'deleteDuplicateEmptyLists.needsMigration'(boardId) { + check(boardId, String); + + if (!this.userId) { + throw new Meteor.Error('not-authorized', 'You must be logged in'); + } + + return deleteDuplicateEmptyListsMigration.needsMigration(boardId); + }, + + 'deleteEmptyLists.execute'(boardId) { + check(boardId, String); + + if (!this.userId) { + throw new Meteor.Error('not-authorized', 'You must be logged in'); + } + + // Check if user is board admin + const board = ReactiveCache.getBoard(boardId); + if (!board) { + throw new Meteor.Error('board-not-found', 'Board not found'); + } + + const user = ReactiveCache.getUser(this.userId); + if (!user) { + throw new Meteor.Error('user-not-found', 'User not found'); + } + + // Only board admins can run migrations + const isBoardAdmin = board.members && board.members.some( + member => member.userId === this.userId && member.isAdmin + ); + + if (!isBoardAdmin && !user.isAdmin) { + throw new Meteor.Error('not-authorized', 'Only board administrators can run migrations'); + } + + return deleteDuplicateEmptyListsMigration.executeMigration(boardId); + }, + + 'deleteDuplicateEmptyLists.execute'(boardId) { + check(boardId, String); + + if (!this.userId) { + throw new Meteor.Error('not-authorized', 'You must be logged in'); + } + + // Check if user is board admin + const board = ReactiveCache.getBoard(boardId); + if (!board) { + throw new Meteor.Error('board-not-found', 'Board not found'); + } + + const user = ReactiveCache.getUser(this.userId); + if (!user) { + throw new Meteor.Error('user-not-found', 'User not found'); + } + + // Only board admins can run migrations + const isBoardAdmin = board.members && board.members.some( + member => member.userId === this.userId && member.isAdmin + ); + + if (!isBoardAdmin && !user.isAdmin) { + throw new Meteor.Error('not-authorized', 'Only board administrators can run migrations'); + } + + return deleteDuplicateEmptyListsMigration.executeMigration(boardId); + }, + + 'deleteEmptyLists.getStatus'(boardId) { + check(boardId, String); + + if (!this.userId) { + throw new Meteor.Error('not-authorized', 'You must be logged in'); + } + + return deleteDuplicateEmptyListsMigration.getStatus(boardId); + }, + + 'deleteDuplicateEmptyLists.getStatus'(boardId) { + check(boardId, String); + + if (!this.userId) { + throw new Meteor.Error('not-authorized', 'You must be logged in'); + } + + return deleteDuplicateEmptyListsMigration.getStatus(boardId); + } +}); + +export default deleteDuplicateEmptyListsMigration; diff --git a/server/migrations/restoreAllArchived.js b/server/migrations/restoreAllArchived.js new file mode 100644 index 000000000..825f9a2f4 --- /dev/null +++ b/server/migrations/restoreAllArchived.js @@ -0,0 +1,266 @@ +/** + * Restore All Archived Migration + * + * Restores all archived swimlanes, lists, and cards. + * If any restored items are missing swimlaneId, listId, or cardId, + * creates/assigns proper IDs to make them visible. + */ + +import { Meteor } from 'meteor/meteor'; +import { check } from 'meteor/check'; +import { ReactiveCache } from '/imports/reactiveCache'; +import { TAPi18n } from '/imports/i18n'; +import Boards from '/models/boards'; +import Lists from '/models/lists'; +import Cards from '/models/cards'; +import Swimlanes from '/models/swimlanes'; + +class RestoreAllArchivedMigration { + constructor() { + this.name = 'restoreAllArchived'; + this.version = 1; + } + + /** + * Check if migration is needed for a board + */ + needsMigration(boardId) { + try { + const archivedSwimlanes = ReactiveCache.getSwimlanes({ boardId, archived: true }); + const archivedLists = ReactiveCache.getLists({ boardId, archived: true }); + const archivedCards = ReactiveCache.getCards({ boardId, archived: true }); + + return archivedSwimlanes.length > 0 || archivedLists.length > 0 || archivedCards.length > 0; + } catch (error) { + console.error('Error checking if restoreAllArchived migration is needed:', error); + return false; + } + } + + /** + * Execute the migration + */ + async executeMigration(boardId) { + try { + const results = { + swimlanesRestored: 0, + listsRestored: 0, + cardsRestored: 0, + itemsFixed: 0, + errors: [] + }; + + const board = ReactiveCache.getBoard(boardId); + if (!board) { + throw new Error('Board not found'); + } + + // Get archived items + const archivedSwimlanes = ReactiveCache.getSwimlanes({ boardId, archived: true }); + const archivedLists = ReactiveCache.getLists({ boardId, archived: true }); + const archivedCards = ReactiveCache.getCards({ boardId, archived: true }); + + // Get active items for reference + const activeSwimlanes = ReactiveCache.getSwimlanes({ boardId, archived: false }); + const activeLists = ReactiveCache.getLists({ boardId, archived: false }); + + // Restore all archived swimlanes + for (const swimlane of archivedSwimlanes) { + Swimlanes.update(swimlane._id, { + $set: { + archived: false, + updatedAt: new Date() + } + }); + results.swimlanesRestored++; + + if (process.env.DEBUG === 'true') { + console.log(`Restored swimlane: ${swimlane.title}`); + } + } + + // Restore all archived lists and fix missing swimlaneId + for (const list of archivedLists) { + const updateFields = { + archived: false, + updatedAt: new Date() + }; + + // Fix missing swimlaneId + if (!list.swimlaneId) { + // Try to find a suitable swimlane or use default + let targetSwimlane = activeSwimlanes.find(s => !s.archived); + + if (!targetSwimlane) { + // No active swimlane found, create default + const swimlaneId = Swimlanes.insert({ + title: TAPi18n.__('default'), + boardId: boardId, + sort: 0, + createdAt: new Date(), + updatedAt: new Date(), + archived: false + }); + targetSwimlane = ReactiveCache.getSwimlane(swimlaneId); + } + + updateFields.swimlaneId = targetSwimlane._id; + results.itemsFixed++; + + if (process.env.DEBUG === 'true') { + console.log(`Fixed missing swimlaneId for list: ${list.title}`); + } + } + + Lists.update(list._id, { + $set: updateFields + }); + results.listsRestored++; + + if (process.env.DEBUG === 'true') { + console.log(`Restored list: ${list.title}`); + } + } + + // Refresh lists after restoration + const allLists = ReactiveCache.getLists({ boardId, archived: false }); + const allSwimlanes = ReactiveCache.getSwimlanes({ boardId, archived: false }); + + // Restore all archived cards and fix missing IDs + for (const card of archivedCards) { + const updateFields = { + archived: false, + updatedAt: new Date() + }; + + let needsFix = false; + + // Fix missing listId + if (!card.listId) { + // Find or create a default list + let targetList = allLists.find(l => !l.archived); + + if (!targetList) { + // No active list found, create one + const defaultSwimlane = allSwimlanes.find(s => !s.archived) || allSwimlanes[0]; + + const listId = Lists.insert({ + title: TAPi18n.__('default'), + boardId: boardId, + swimlaneId: defaultSwimlane._id, + sort: 0, + createdAt: new Date(), + updatedAt: new Date(), + archived: false + }); + targetList = ReactiveCache.getList(listId); + } + + updateFields.listId = targetList._id; + needsFix = true; + } + + // Fix missing swimlaneId + if (!card.swimlaneId) { + // Try to get swimlaneId from the card's list + if (card.listId || updateFields.listId) { + const cardList = allLists.find(l => l._id === (updateFields.listId || card.listId)); + if (cardList && cardList.swimlaneId) { + updateFields.swimlaneId = cardList.swimlaneId; + } else { + // Fall back to first available swimlane + const defaultSwimlane = allSwimlanes.find(s => !s.archived) || allSwimlanes[0]; + updateFields.swimlaneId = defaultSwimlane._id; + } + } else { + // Fall back to first available swimlane + const defaultSwimlane = allSwimlanes.find(s => !s.archived) || allSwimlanes[0]; + updateFields.swimlaneId = defaultSwimlane._id; + } + needsFix = true; + } + + if (needsFix) { + results.itemsFixed++; + + if (process.env.DEBUG === 'true') { + console.log(`Fixed missing IDs for card: ${card.title}`); + } + } + + Cards.update(card._id, { + $set: updateFields + }); + results.cardsRestored++; + + if (process.env.DEBUG === 'true') { + console.log(`Restored card: ${card.title}`); + } + } + + return { + success: true, + changes: [ + `Restored ${results.swimlanesRestored} archived swimlanes`, + `Restored ${results.listsRestored} archived lists`, + `Restored ${results.cardsRestored} archived cards`, + `Fixed ${results.itemsFixed} items with missing IDs` + ], + results + }; + } catch (error) { + console.error('Error executing restoreAllArchived migration:', error); + return { + success: false, + error: error.message + }; + } + } +} + +const restoreAllArchivedMigration = new RestoreAllArchivedMigration(); + +// Register Meteor methods +Meteor.methods({ + 'restoreAllArchived.needsMigration'(boardId) { + check(boardId, String); + + if (!this.userId) { + throw new Meteor.Error('not-authorized', 'You must be logged in'); + } + + return restoreAllArchivedMigration.needsMigration(boardId); + }, + + 'restoreAllArchived.execute'(boardId) { + check(boardId, String); + + if (!this.userId) { + throw new Meteor.Error('not-authorized', 'You must be logged in'); + } + + // Check if user is board admin + const board = ReactiveCache.getBoard(boardId); + if (!board) { + throw new Meteor.Error('board-not-found', 'Board not found'); + } + + const user = ReactiveCache.getUser(this.userId); + if (!user) { + throw new Meteor.Error('user-not-found', 'User not found'); + } + + // Only board admins can run migrations + const isBoardAdmin = board.members && board.members.some( + member => member.userId === this.userId && member.isAdmin + ); + + if (!isBoardAdmin && !user.isAdmin) { + throw new Meteor.Error('not-authorized', 'Only board administrators can run migrations'); + } + + return restoreAllArchivedMigration.executeMigration(boardId); + } +}); + +export default restoreAllArchivedMigration; diff --git a/server/migrations/restoreLostCards.js b/server/migrations/restoreLostCards.js new file mode 100644 index 000000000..781caa0fb --- /dev/null +++ b/server/migrations/restoreLostCards.js @@ -0,0 +1,259 @@ +/** + * Restore Lost Cards Migration + * + * Finds and restores cards and lists that have missing swimlaneId, listId, or are orphaned. + * Creates a "Lost Cards" swimlane and restores visibility of lost items. + * Only processes non-archived items. + */ + +import { Meteor } from 'meteor/meteor'; +import { check } from 'meteor/check'; +import { ReactiveCache } from '/imports/reactiveCache'; +import { TAPi18n } from '/imports/i18n'; +import Boards from '/models/boards'; +import Lists from '/models/lists'; +import Cards from '/models/cards'; +import Swimlanes from '/models/swimlanes'; + +class RestoreLostCardsMigration { + constructor() { + this.name = 'restoreLostCards'; + this.version = 1; + } + + /** + * Check if migration is needed for a board + */ + needsMigration(boardId) { + try { + const cards = ReactiveCache.getCards({ boardId, archived: false }); + const lists = ReactiveCache.getLists({ boardId, archived: false }); + + // Check for cards missing swimlaneId or listId + const lostCards = cards.filter(card => !card.swimlaneId || !card.listId); + if (lostCards.length > 0) { + return true; + } + + // Check for lists missing swimlaneId + const lostLists = lists.filter(list => !list.swimlaneId); + if (lostLists.length > 0) { + return true; + } + + // Check for orphaned cards (cards whose list doesn't exist) + for (const card of cards) { + if (card.listId) { + const listExists = lists.some(list => list._id === card.listId); + if (!listExists) { + return true; + } + } + } + + return false; + } catch (error) { + console.error('Error checking if restoreLostCards migration is needed:', error); + return false; + } + } + + /** + * Execute the migration + */ + async executeMigration(boardId) { + try { + const results = { + lostCardsSwimlaneCreated: false, + cardsRestored: 0, + listsRestored: 0, + errors: [] + }; + + const board = ReactiveCache.getBoard(boardId); + if (!board) { + throw new Error('Board not found'); + } + + // Get all non-archived items + const cards = ReactiveCache.getCards({ boardId, archived: false }); + const lists = ReactiveCache.getLists({ boardId, archived: false }); + const swimlanes = ReactiveCache.getSwimlanes({ boardId, archived: false }); + + // Detect items to restore BEFORE creating anything + const lostLists = lists.filter(list => !list.swimlaneId); + const lostCards = cards.filter(card => !card.swimlaneId || !card.listId); + const orphanedCards = cards.filter(card => card.listId && !lists.some(list => list._id === card.listId)); + + const hasCardsWork = lostCards.length > 0 || orphanedCards.length > 0; + const hasListsWork = lostLists.length > 0; + const hasAnyWork = hasCardsWork || hasListsWork; + + if (!hasAnyWork) { + // Nothing to restore; do not create swimlane or list + return { + success: true, + changes: [ + 'No lost swimlanes, lists, or cards to restore' + ], + results: { + lostCardsSwimlaneCreated: false, + cardsRestored: 0, + listsRestored: 0 + } + }; + } + + // Find or create "Lost Cards" swimlane (only if there is actual work) + let lostCardsSwimlane = swimlanes.find(s => s.title === TAPi18n.__('lost-cards')); + if (!lostCardsSwimlane) { + const swimlaneId = Swimlanes.insert({ + title: TAPi18n.__('lost-cards'), + boardId: boardId, + sort: 999999, // Put at the end + color: 'red', + createdAt: new Date(), + updatedAt: new Date(), + archived: false + }); + lostCardsSwimlane = ReactiveCache.getSwimlane(swimlaneId); + results.lostCardsSwimlaneCreated = true; + if (process.env.DEBUG === 'true') { + console.log(`Created "Lost Cards" swimlane for board ${boardId}`); + } + } + + // Restore lost lists (lists without swimlaneId) + if (hasListsWork) { + for (const list of lostLists) { + Lists.update(list._id, { + $set: { + swimlaneId: lostCardsSwimlane._id, + updatedAt: new Date() + } + }); + results.listsRestored++; + if (process.env.DEBUG === 'true') { + console.log(`Restored lost list: ${list.title}`); + } + } + } + + // Create default list only if we need to move cards + let defaultList = null; + if (hasCardsWork) { + defaultList = lists.find(l => + l.swimlaneId === lostCardsSwimlane._id && + l.title === TAPi18n.__('lost-cards-list') + ); + if (!defaultList) { + const listId = Lists.insert({ + title: TAPi18n.__('lost-cards-list'), + boardId: boardId, + swimlaneId: lostCardsSwimlane._id, + sort: 0, + createdAt: new Date(), + updatedAt: new Date(), + archived: false + }); + defaultList = ReactiveCache.getList(listId); + if (process.env.DEBUG === 'true') { + console.log(`Created default list in Lost Cards swimlane`); + } + } + } + + // Restore cards missing swimlaneId or listId + if (hasCardsWork) { + for (const card of lostCards) { + const updateFields = { updatedAt: new Date() }; + if (!card.swimlaneId) updateFields.swimlaneId = lostCardsSwimlane._id; + if (!card.listId) updateFields.listId = defaultList._id; + Cards.update(card._id, { $set: updateFields }); + results.cardsRestored++; + if (process.env.DEBUG === 'true') { + console.log(`Restored lost card: ${card.title}`); + } + } + + // Restore orphaned cards (cards whose list doesn't exist) + for (const card of orphanedCards) { + Cards.update(card._id, { + $set: { + listId: defaultList._id, + swimlaneId: lostCardsSwimlane._id, + updatedAt: new Date() + } + }); + results.cardsRestored++; + if (process.env.DEBUG === 'true') { + console.log(`Restored orphaned card: ${card.title}`); + } + } + } + + return { + success: true, + changes: [ + results.lostCardsSwimlaneCreated ? 'Created "Lost Cards" swimlane' : 'Using existing "Lost Cards" swimlane', + `Restored ${results.listsRestored} lost lists`, + `Restored ${results.cardsRestored} lost cards` + ], + results + }; + } catch (error) { + console.error('Error executing restoreLostCards migration:', error); + return { + success: false, + error: error.message + }; + } + } +} + +const restoreLostCardsMigration = new RestoreLostCardsMigration(); + +// Register Meteor methods +Meteor.methods({ + 'restoreLostCards.needsMigration'(boardId) { + check(boardId, String); + + if (!this.userId) { + throw new Meteor.Error('not-authorized', 'You must be logged in'); + } + + return restoreLostCardsMigration.needsMigration(boardId); + }, + + 'restoreLostCards.execute'(boardId) { + check(boardId, String); + + if (!this.userId) { + throw new Meteor.Error('not-authorized', 'You must be logged in'); + } + + // Check if user is board admin + const board = ReactiveCache.getBoard(boardId); + if (!board) { + throw new Meteor.Error('board-not-found', 'Board not found'); + } + + const user = ReactiveCache.getUser(this.userId); + if (!user) { + throw new Meteor.Error('user-not-found', 'User not found'); + } + + // Only board admins can run migrations + const isBoardAdmin = board.members && board.members.some( + member => member.userId === this.userId && member.isAdmin + ); + + if (!isBoardAdmin && !user.isAdmin) { + throw new Meteor.Error('not-authorized', 'Only board administrators can run migrations'); + } + + return restoreLostCardsMigration.executeMigration(boardId); + } +}); + +export default restoreLostCardsMigration; From 71b7dcffb5a5d7382a641bbbe066ef40e6e5c87d Mon Sep 17 00:00:00 2001 From: Lauri Ojansivu Date: Wed, 5 Nov 2025 18:46:56 +0200 Subject: [PATCH 40/84] Updated ChangeLog. --- CHANGELOG.md | 2 ++ 1 file changed, 2 insertions(+) diff --git a/CHANGELOG.md b/CHANGELOG.md index bcc7685df..27a4c6f71 100644 --- a/CHANGELOG.md +++ b/CHANGELOG.md @@ -30,6 +30,8 @@ This release fixes the following bugs: Thanks to xet7. - [Moved migrations from opening board to right sidebar / Migrations](https://github.com/wekan/wekan/commit/1b25d1d5720d4f486a10d2acce37e315cf9b6057). Thanks to xet7. +- [Fix 8.16 Lists with no items are deleted every time when board is opened. Moved migrations to right sidebar](https://github.com/wekan/wekan/commit/7713e613b431e44dc13cee72e7a1e5f031473fa6). + Thanks to xet7. Thanks to above GitHub users for their contributions and translators for their translations. From ba49d4d140bc0d4cfb5a96db9ab077bc85db58f1 Mon Sep 17 00:00:00 2001 From: Lauri Ojansivu Date: Wed, 5 Nov 2025 19:03:21 +0200 Subject: [PATCH 41/84] Remove old translations and code not in use anymore. Thanks to xet7 ! --- .../components/sidebar/sidebarMigrations.js | 17 ------ imports/i18n/data/en.i18n.json | 4 -- .../migrations/deleteDuplicateEmptyLists.js | 54 +------------------ 3 files changed, 2 insertions(+), 73 deletions(-) diff --git a/client/components/sidebar/sidebarMigrations.js b/client/components/sidebar/sidebarMigrations.js index cc47b27cf..cc5461af2 100644 --- a/client/components/sidebar/sidebarMigrations.js +++ b/client/components/sidebar/sidebarMigrations.js @@ -84,10 +84,6 @@ BlazeComponent.extendComponent({ return this.migrationStatuses.get().fixMissingLists === true; }, - deleteEmptyListsNeeded() { - return this.migrationStatuses.get().deleteEmptyLists === true; - }, - deleteDuplicateEmptyListsNeeded() { return this.migrationStatuses.get().deleteDuplicateEmptyLists === true; }, @@ -177,11 +173,6 @@ BlazeComponent.extendComponent({ methodArgs = [boardId]; break; - case 'deleteEmptyLists': - methodName = 'deleteEmptyLists.execute'; - methodArgs = [boardId]; - break; - case 'deleteDuplicateEmptyLists': methodName = 'deleteDuplicateEmptyLists.execute'; methodArgs = [boardId]; @@ -224,10 +215,6 @@ BlazeComponent.extendComponent({ { step: 'update_cards', name: 'Update Cards', duration: 900 }, { step: 'finalize', name: 'Finalize', duration: 400 }, ], - deleteEmptyLists: [ - { step: 'convert_shared_lists', name: 'Convert Shared Lists', duration: 700 }, - { step: 'delete_empty_lists', name: 'Delete Empty Lists', duration: 800 }, - ], deleteDuplicateEmptyLists: [ { step: 'convert_shared_lists', name: 'Convert Shared Lists', duration: 700 }, { step: 'delete_duplicate_empty_lists', name: 'Delete Duplicate Empty Lists', duration: 800 }, @@ -326,10 +313,6 @@ BlazeComponent.extendComponent({ self.runMigration('fixMissingLists'); Popup.back(); }), - 'click .js-run-migration[data-migration="deleteEmptyLists"]': Popup.afterConfirm('runDeleteEmptyListsMigration', function() { - self.runMigration('deleteEmptyLists'); - Popup.back(); - }), 'click .js-run-migration[data-migration="deleteDuplicateEmptyLists"]': Popup.afterConfirm('runDeleteDuplicateEmptyListsMigration', function() { self.runMigration('deleteDuplicateEmptyLists'); Popup.back(); diff --git a/imports/i18n/data/en.i18n.json b/imports/i18n/data/en.i18n.json index c007af213..182bafd21 100644 --- a/imports/i18n/data/en.i18n.json +++ b/imports/i18n/data/en.i18n.json @@ -1408,8 +1408,6 @@ "card-show-lists-on-minicard": "Show Lists on Minicard", "comprehensive-board-migration": "Comprehensive Board Migration", "comprehensive-board-migration-description": "Performs comprehensive checks and fixes for board data integrity, including list ordering, card positions, and swimlane structure.", - "delete-empty-lists-migration": "Delete Empty Lists", - "delete-empty-lists-migration-description": "Safely deletes empty duplicate lists. Only removes lists that have no cards AND have another list with the same title that contains cards.", "delete-duplicate-empty-lists-migration": "Delete Duplicate Empty Lists", "delete-duplicate-empty-lists-migration-description": "Safely deletes empty duplicate lists. Only removes lists that have no cards AND have another list with the same title that contains cards.", "lost-cards": "Lost Cards", @@ -1436,7 +1434,6 @@ "no-issues-found": "No issues found", "run-migration": "Run Migration", "run-comprehensive-migration-confirm": "This will perform a comprehensive migration to check and fix board data integrity. This may take a few moments. Continue?", - "run-delete-empty-lists-migration-confirm": "This will first convert any shared lists to per-swimlane lists, then delete empty lists that have a duplicate list with the same title containing cards. Only truly redundant empty lists will be removed. Continue?", "run-delete-duplicate-empty-lists-migration-confirm": "This will first convert any shared lists to per-swimlane lists, then delete empty lists that have a duplicate list with the same title containing cards. Only truly redundant empty lists will be removed. Continue?", "run-restore-lost-cards-migration-confirm": "This will create a 'Lost Cards' swimlane and restore all cards and lists with missing swimlaneId or listId. This only affects non-archived items. Continue?", "run-restore-all-archived-migration-confirm": "This will restore ALL archived swimlanes, lists, and cards, making them visible again. Any items with missing IDs will be automatically fixed. This cannot be easily undone. Continue?", @@ -1463,7 +1460,6 @@ "step-create-missing-lists": "Create Missing Lists", "step-update-cards": "Update Cards", "step-finalize": "Finalize", - "step-delete-empty-lists": "Delete Empty Lists", "step-delete-duplicate-empty-lists": "Delete Duplicate Empty Lists", "step-ensure-lost-cards-swimlane": "Ensure Lost Cards Swimlane", "step-restore-lists": "Restore Lists", diff --git a/server/migrations/deleteDuplicateEmptyLists.js b/server/migrations/deleteDuplicateEmptyLists.js index c0d455685..dadbd5391 100644 --- a/server/migrations/deleteDuplicateEmptyLists.js +++ b/server/migrations/deleteDuplicateEmptyLists.js @@ -59,7 +59,7 @@ class DeleteDuplicateEmptyListsMigration { return false; } catch (error) { - console.error('Error checking if deleteEmptyLists migration is needed:', error); + console.error('Error checking if deleteDuplicateEmptyLists migration is needed:', error); return false; } } @@ -92,7 +92,7 @@ class DeleteDuplicateEmptyListsMigration { results }; } catch (error) { - console.error('Error executing deleteEmptyLists migration:', error); + console.error('Error executing deleteDuplicateEmptyLists migration:', error); return { success: false, error: error.message @@ -319,16 +319,6 @@ const deleteDuplicateEmptyListsMigration = new DeleteDuplicateEmptyListsMigratio // Register Meteor methods Meteor.methods({ - 'deleteEmptyLists.needsMigration'(boardId) { - check(boardId, String); - - if (!this.userId) { - throw new Meteor.Error('not-authorized', 'You must be logged in'); - } - - return deleteDuplicateEmptyListsMigration.needsMigration(boardId); - }, - 'deleteDuplicateEmptyLists.needsMigration'(boardId) { check(boardId, String); @@ -339,36 +329,6 @@ Meteor.methods({ return deleteDuplicateEmptyListsMigration.needsMigration(boardId); }, - 'deleteEmptyLists.execute'(boardId) { - check(boardId, String); - - if (!this.userId) { - throw new Meteor.Error('not-authorized', 'You must be logged in'); - } - - // Check if user is board admin - const board = ReactiveCache.getBoard(boardId); - if (!board) { - throw new Meteor.Error('board-not-found', 'Board not found'); - } - - const user = ReactiveCache.getUser(this.userId); - if (!user) { - throw new Meteor.Error('user-not-found', 'User not found'); - } - - // Only board admins can run migrations - const isBoardAdmin = board.members && board.members.some( - member => member.userId === this.userId && member.isAdmin - ); - - if (!isBoardAdmin && !user.isAdmin) { - throw new Meteor.Error('not-authorized', 'Only board administrators can run migrations'); - } - - return deleteDuplicateEmptyListsMigration.executeMigration(boardId); - }, - 'deleteDuplicateEmptyLists.execute'(boardId) { check(boardId, String); @@ -399,16 +359,6 @@ Meteor.methods({ return deleteDuplicateEmptyListsMigration.executeMigration(boardId); }, - 'deleteEmptyLists.getStatus'(boardId) { - check(boardId, String); - - if (!this.userId) { - throw new Meteor.Error('not-authorized', 'You must be logged in'); - } - - return deleteDuplicateEmptyListsMigration.getStatus(boardId); - }, - 'deleteDuplicateEmptyLists.getStatus'(boardId) { check(boardId, String); From bc5854dd29ec1bef772d2772eceb167f418e8ee3 Mon Sep 17 00:00:00 2001 From: Lauri Ojansivu Date: Wed, 5 Nov 2025 19:04:47 +0200 Subject: [PATCH 42/84] Updated ChangeLog. --- CHANGELOG.md | 2 ++ 1 file changed, 2 insertions(+) diff --git a/CHANGELOG.md b/CHANGELOG.md index 27a4c6f71..911853f03 100644 --- a/CHANGELOG.md +++ b/CHANGELOG.md @@ -32,6 +32,8 @@ This release fixes the following bugs: Thanks to xet7. - [Fix 8.16 Lists with no items are deleted every time when board is opened. Moved migrations to right sidebar](https://github.com/wekan/wekan/commit/7713e613b431e44dc13cee72e7a1e5f031473fa6). Thanks to xet7. +- [Remove old translations and code not in use anymore](https://github.com/wekan/wekan/commit/ba49d4d140bc0d4cfb5a96db9ab077bc85db58f1). + Thanks to xet7. Thanks to above GitHub users for their contributions and translators for their translations. From e4638d5fbcbe004ac393462331805cac3ba25097 Mon Sep 17 00:00:00 2001 From: Lauri Ojansivu Date: Wed, 5 Nov 2025 20:22:56 +0200 Subject: [PATCH 43/84] Fixed sidebar migrations to be per-board, not global. Clarified translations. Thanks to xet7 ! --- .../components/sidebar/sidebarMigrations.jade | 2 - .../components/sidebar/sidebarMigrations.js | 20 +-- imports/i18n/data/en.i18n.json | 15 ++- server/migrations/fixAllFileUrls.js | 114 +++++++++++++----- server/migrations/fixAvatarUrls.js | 75 +++++++++--- 5 files changed, 160 insertions(+), 66 deletions(-) diff --git a/client/components/sidebar/sidebarMigrations.jade b/client/components/sidebar/sidebarMigrations.jade index 78da56983..f5f7f08f8 100644 --- a/client/components/sidebar/sidebarMigrations.jade +++ b/client/components/sidebar/sidebarMigrations.jade @@ -58,8 +58,6 @@ template(name='migrationsSidebar') else span.badge.badge-success {{_ 'migration-complete'}} - hr - h4 {{_ 'global-migrations'}} .migration-item a.js-run-migration(data-migration="fixAvatarUrls") .migration-name diff --git a/client/components/sidebar/sidebarMigrations.js b/client/components/sidebar/sidebarMigrations.js index cc5461af2..89d3343ec 100644 --- a/client/components/sidebar/sidebarMigrations.js +++ b/client/components/sidebar/sidebarMigrations.js @@ -57,17 +57,17 @@ BlazeComponent.extendComponent({ } }); - // Check fix avatar URLs migration (global) - Meteor.call('fixAvatarUrls.needsMigration', (err, res) => { + // Check fix avatar URLs migration (board-specific) + Meteor.call('fixAvatarUrls.needsMigration', boardId, (err, res) => { if (!err) { const statuses = this.migrationStatuses.get(); - statuses.fixAvatarUrls = res; + statuses.fixAvatarUrls = res; this.migrationStatuses.set(statuses); } }); - // Check fix all file URLs migration (global) - Meteor.call('fixAllFileUrls.needsMigration', (err, res) => { + // Check fix all file URLs migration (board-specific) + Meteor.call('fixAllFileUrls.needsMigration', boardId, (err, res) => { if (!err) { const statuses = this.migrationStatuses.get(); statuses.fixAllFileUrls = res; @@ -190,10 +190,12 @@ BlazeComponent.extendComponent({ case 'fixAvatarUrls': methodName = 'fixAvatarUrls.execute'; + methodArgs = [boardId]; break; case 'fixAllFileUrls': methodName = 'fixAllFileUrls.execute'; + methodArgs = [boardId]; break; } @@ -231,12 +233,12 @@ BlazeComponent.extendComponent({ { step: 'fix_missing_ids', name: 'Fix Missing IDs', duration: 600 }, ], fixAvatarUrls: [ - { step: 'scan_users', name: 'Scan Users', duration: 500 }, - { step: 'fix_urls', name: 'Fix Avatar URLs', duration: 900 }, + { step: 'scan_users', name: 'Checking board member avatars', duration: 500 }, + { step: 'fix_urls', name: 'Fixing avatar URLs', duration: 900 }, ], fixAllFileUrls: [ - { step: 'scan_files', name: 'Scan Files', duration: 600 }, - { step: 'fix_urls', name: 'Fix File URLs', duration: 1000 }, + { step: 'scan_files', name: 'Checking board file attachments', duration: 600 }, + { step: 'fix_urls', name: 'Fixing file URLs', duration: 1000 }, ], }; diff --git a/imports/i18n/data/en.i18n.json b/imports/i18n/data/en.i18n.json index 182bafd21..958d8313a 100644 --- a/imports/i18n/data/en.i18n.json +++ b/imports/i18n/data/en.i18n.json @@ -1419,10 +1419,9 @@ "fix-missing-lists-migration": "Fix Missing Lists", "fix-missing-lists-migration-description": "Detects and repairs missing or corrupted lists in the board structure.", "fix-avatar-urls-migration": "Fix Avatar URLs", - "fix-avatar-urls-migration-description": "Updates avatar URLs to use the correct storage backend and fixes broken avatar references.", + "fix-avatar-urls-migration-description": "Updates avatar URLs for board members to use the correct storage backend and fixes broken avatar references.", "fix-all-file-urls-migration": "Fix All File URLs", - "fix-all-file-urls-migration-description": "Updates all file attachment URLs to use the correct storage backend and fixes broken file references.", - "global-migrations": "Global Migrations", + "fix-all-file-urls-migration-description": "Updates all file attachment URLs on this board to use the correct storage backend and fixes broken file references.", "migration-needed": "Migration Needed", "migration-complete": "Complete", "migration-running": "Running...", @@ -1438,8 +1437,8 @@ "run-restore-lost-cards-migration-confirm": "This will create a 'Lost Cards' swimlane and restore all cards and lists with missing swimlaneId or listId. This only affects non-archived items. Continue?", "run-restore-all-archived-migration-confirm": "This will restore ALL archived swimlanes, lists, and cards, making them visible again. Any items with missing IDs will be automatically fixed. This cannot be easily undone. Continue?", "run-fix-missing-lists-migration-confirm": "This will detect and repair missing or corrupted lists in the board structure. Continue?", - "run-fix-avatar-urls-migration-confirm": "This will update avatar URLs across all boards to use the correct storage backend. This is a global operation. Continue?", - "run-fix-all-file-urls-migration-confirm": "This will update all file attachment URLs across all boards to use the correct storage backend. This is a global operation. Continue?", + "run-fix-avatar-urls-migration-confirm": "This will update avatar URLs for board members to use the correct storage backend. Continue?", + "run-fix-all-file-urls-migration-confirm": "This will update all file attachment URLs on this board to use the correct storage backend. Continue?", "restore-lost-cards-nothing-to-restore": "No lost swimlanes, lists, or cards to restore", "migration-progress-title": "Board Migration in Progress", @@ -1466,9 +1465,9 @@ "step-restore-cards": "Restore Cards", "step-restore-swimlanes": "Restore Swimlanes", "step-fix-missing-ids": "Fix Missing IDs", - "step-scan-users": "Scan Users", - "step-scan-files": "Scan Files", - "step-fix-file-urls": "Fix File URLs", + "step-scan-users": "Checking board member avatars", + "step-scan-files": "Checking board file attachments", + "step-fix-file-urls": "Fixing file URLs", "cleanup": "Cleanup", "cleanup-old-jobs": "Cleanup Old Jobs", "completed": "Completed", diff --git a/server/migrations/fixAllFileUrls.js b/server/migrations/fixAllFileUrls.js index 6b3be9ccf..f713ac8ae 100644 --- a/server/migrations/fixAllFileUrls.js +++ b/server/migrations/fixAllFileUrls.js @@ -3,10 +3,14 @@ * Ensures all attachment and avatar URLs are universal and work regardless of ROOT_URL and PORT settings */ +import { Meteor } from 'meteor/meteor'; +import { check } from 'meteor/check'; import { ReactiveCache } from '/imports/reactiveCache'; +import Boards from '/models/boards'; import Users from '/models/users'; import Attachments from '/models/attachments'; import Avatars from '/models/avatars'; +import Cards from '/models/cards'; import { generateUniversalAttachmentUrl, generateUniversalAvatarUrl, cleanFileUrl, extractFileIdFromUrl, isUniversalFileUrl } from '/models/lib/universalUrlGenerator'; class FixAllFileUrlsMigration { @@ -16,11 +20,19 @@ class FixAllFileUrlsMigration { } /** - * Check if migration is needed + * Check if migration is needed for a board */ - needsMigration() { - // Check for problematic avatar URLs - const users = ReactiveCache.getUsers({}); + needsMigration(boardId) { + // Get all users who are members of this board + const board = ReactiveCache.getBoard(boardId); + if (!board || !board.members) { + return false; + } + + const memberIds = board.members.map(m => m.userId); + + // Check for problematic avatar URLs for board members + const users = ReactiveCache.getUsers({ _id: { $in: memberIds } }); for (const user of users) { if (user.profile && user.profile.avatarUrl) { const avatarUrl = user.profile.avatarUrl; @@ -30,8 +42,11 @@ class FixAllFileUrlsMigration { } } - // Check for problematic attachment URLs - const attachments = ReactiveCache.getAttachments({}); + // Check for problematic attachment URLs on this board + const cards = ReactiveCache.getCards({ boardId }); + const cardIds = cards.map(c => c._id); + const attachments = ReactiveCache.getAttachments({ cardId: { $in: cardIds } }); + for (const attachment of attachments) { if (attachment.url && this.hasProblematicUrl(attachment.url)) { return true; @@ -78,46 +93,53 @@ class FixAllFileUrlsMigration { } /** - * Execute the migration + * Execute the migration for a board */ - async execute() { + async execute(boardId) { let filesFixed = 0; let errors = []; - console.log(`Starting universal file URL migration...`); + console.log(`Starting universal file URL migration for board ${boardId}...`); try { - // Fix avatar URLs - const avatarFixed = await this.fixAvatarUrls(); + // Fix avatar URLs for board members + const avatarFixed = await this.fixAvatarUrls(boardId); filesFixed += avatarFixed; - // Fix attachment URLs - const attachmentFixed = await this.fixAttachmentUrls(); + // Fix attachment URLs for board cards + const attachmentFixed = await this.fixAttachmentUrls(boardId); filesFixed += attachmentFixed; // Fix card attachment references - const cardFixed = await this.fixCardAttachmentUrls(); + const cardFixed = await this.fixCardAttachmentUrls(boardId); filesFixed += cardFixed; } catch (error) { - console.error('Error during file URL migration:', error); + console.error('Error during file URL migration for board', boardId, ':', error); errors.push(error.message); } - console.log(`Universal file URL migration completed. Fixed ${filesFixed} file URLs.`); + console.log(`Universal file URL migration completed for board ${boardId}. Fixed ${filesFixed} file URLs.`); return { success: errors.length === 0, filesFixed, - errors + errors, + changes: [`Fixed ${filesFixed} file URLs for this board`] }; } /** - * Fix avatar URLs in user profiles + * Fix avatar URLs in user profiles for board members */ - async fixAvatarUrls() { - const users = ReactiveCache.getUsers({}); + async fixAvatarUrls(boardId) { + const board = ReactiveCache.getBoard(boardId); + if (!board || !board.members) { + return 0; + } + + const memberIds = board.members.map(m => m.userId); + const users = ReactiveCache.getUsers({ _id: { $in: memberIds } }); let avatarsFixed = 0; for (const user of users) { @@ -164,10 +186,12 @@ class FixAllFileUrlsMigration { } /** - * Fix attachment URLs in attachment records + * Fix attachment URLs in attachment records for this board */ - async fixAttachmentUrls() { - const attachments = ReactiveCache.getAttachments({}); + async fixAttachmentUrls(boardId) { + const cards = ReactiveCache.getCards({ boardId }); + const cardIds = cards.map(c => c._id); + const attachments = ReactiveCache.getAttachments({ cardId: { $in: cardIds } }); let attachmentsFixed = 0; for (const attachment of attachments) { @@ -202,10 +226,12 @@ class FixAllFileUrlsMigration { } /** - * Fix attachment URLs in the Attachments collection + * Fix attachment URLs in the Attachments collection for this board */ - async fixCardAttachmentUrls() { - const attachments = ReactiveCache.getAttachments({}); + async fixCardAttachmentUrls(boardId) { + const cards = ReactiveCache.getCards({ boardId }); + const cardIds = cards.map(c => c._id); + const attachments = ReactiveCache.getAttachments({ cardId: { $in: cardIds } }); let attachmentsFixed = 0; for (const attachment of attachments) { @@ -244,19 +270,43 @@ export const fixAllFileUrlsMigration = new FixAllFileUrlsMigration(); // Meteor methods Meteor.methods({ - 'fixAllFileUrls.execute'() { + 'fixAllFileUrls.execute'(boardId) { + check(boardId, String); + if (!this.userId) { - throw new Meteor.Error('not-authorized'); + throw new Meteor.Error('not-authorized', 'You must be logged in'); + } + + // Check if user is board admin + const board = ReactiveCache.getBoard(boardId); + if (!board) { + throw new Meteor.Error('board-not-found', 'Board not found'); + } + + const user = ReactiveCache.getUser(this.userId); + if (!user) { + throw new Meteor.Error('user-not-found', 'User not found'); + } + + // Only board admins can run migrations + const isBoardAdmin = board.members && board.members.some( + member => member.userId === this.userId && member.isAdmin + ); + + if (!isBoardAdmin && !user.isAdmin) { + throw new Meteor.Error('not-authorized', 'Only board administrators can run migrations'); } - return fixAllFileUrlsMigration.execute(); + return fixAllFileUrlsMigration.execute(boardId); }, - 'fixAllFileUrls.needsMigration'() { + 'fixAllFileUrls.needsMigration'(boardId) { + check(boardId, String); + if (!this.userId) { - throw new Meteor.Error('not-authorized'); + throw new Meteor.Error('not-authorized', 'You must be logged in'); } - return fixAllFileUrlsMigration.needsMigration(); + return fixAllFileUrlsMigration.needsMigration(boardId); } }); diff --git a/server/migrations/fixAvatarUrls.js b/server/migrations/fixAvatarUrls.js index f542903ed..82677eb48 100644 --- a/server/migrations/fixAvatarUrls.js +++ b/server/migrations/fixAvatarUrls.js @@ -3,7 +3,10 @@ * Removes problematic auth parameters from existing avatar URLs */ +import { Meteor } from 'meteor/meteor'; +import { check } from 'meteor/check'; import { ReactiveCache } from '/imports/reactiveCache'; +import Boards from '/models/boards'; import Users from '/models/users'; import { generateUniversalAvatarUrl, cleanFileUrl, extractFileIdFromUrl, isUniversalFileUrl } from '/models/lib/universalUrlGenerator'; @@ -14,10 +17,17 @@ class FixAvatarUrlsMigration { } /** - * Check if migration is needed + * Check if migration is needed for a board */ - needsMigration() { - const users = ReactiveCache.getUsers({}); + needsMigration(boardId) { + // Get all users who are members of this board + const board = ReactiveCache.getBoard(boardId); + if (!board || !board.members) { + return false; + } + + const memberIds = board.members.map(m => m.userId); + const users = ReactiveCache.getUsers({ _id: { $in: memberIds } }); for (const user of users) { if (user.profile && user.profile.avatarUrl) { @@ -32,13 +42,23 @@ class FixAvatarUrlsMigration { } /** - * Execute the migration + * Execute the migration for a board */ - async execute() { - const users = ReactiveCache.getUsers({}); + async execute(boardId) { + // Get all users who are members of this board + const board = ReactiveCache.getBoard(boardId); + if (!board || !board.members) { + return { + success: false, + error: 'Board not found or has no members' + }; + } + + const memberIds = board.members.map(m => m.userId); + const users = ReactiveCache.getUsers({ _id: { $in: memberIds } }); let avatarsFixed = 0; - console.log(`Starting avatar URL fix migration...`); + console.log(`Starting avatar URL fix migration for board ${boardId}...`); for (const user of users) { if (user.profile && user.profile.avatarUrl) { @@ -96,11 +116,12 @@ class FixAvatarUrlsMigration { } } - console.log(`Avatar URL fix migration completed. Fixed ${avatarsFixed} avatar URLs.`); + console.log(`Avatar URL fix migration completed for board ${boardId}. Fixed ${avatarsFixed} avatar URLs.`); return { success: true, - avatarsFixed + avatarsFixed, + changes: [`Fixed ${avatarsFixed} avatar URLs for board members`] }; } } @@ -110,19 +131,43 @@ export const fixAvatarUrlsMigration = new FixAvatarUrlsMigration(); // Meteor method Meteor.methods({ - 'fixAvatarUrls.execute'() { + 'fixAvatarUrls.execute'(boardId) { + check(boardId, String); + if (!this.userId) { - throw new Meteor.Error('not-authorized'); + throw new Meteor.Error('not-authorized', 'You must be logged in'); + } + + // Check if user is board admin + const board = ReactiveCache.getBoard(boardId); + if (!board) { + throw new Meteor.Error('board-not-found', 'Board not found'); + } + + const user = ReactiveCache.getUser(this.userId); + if (!user) { + throw new Meteor.Error('user-not-found', 'User not found'); + } + + // Only board admins can run migrations + const isBoardAdmin = board.members && board.members.some( + member => member.userId === this.userId && member.isAdmin + ); + + if (!isBoardAdmin && !user.isAdmin) { + throw new Meteor.Error('not-authorized', 'Only board administrators can run migrations'); } - return fixAvatarUrlsMigration.execute(); + return fixAvatarUrlsMigration.execute(boardId); }, - 'fixAvatarUrls.needsMigration'() { + 'fixAvatarUrls.needsMigration'(boardId) { + check(boardId, String); + if (!this.userId) { - throw new Meteor.Error('not-authorized'); + throw new Meteor.Error('not-authorized', 'You must be logged in'); } - return fixAvatarUrlsMigration.needsMigration(); + return fixAvatarUrlsMigration.needsMigration(boardId); } }); From 7d27139aa9db34509f9fc9236a87280b20644bfc Mon Sep 17 00:00:00 2001 From: Lauri Ojansivu Date: Wed, 5 Nov 2025 20:25:07 +0200 Subject: [PATCH 44/84] Updated ChangeLog. --- CHANGELOG.md | 2 ++ 1 file changed, 2 insertions(+) diff --git a/CHANGELOG.md b/CHANGELOG.md index 911853f03..538b59d8b 100644 --- a/CHANGELOG.md +++ b/CHANGELOG.md @@ -34,6 +34,8 @@ This release fixes the following bugs: Thanks to xet7. - [Remove old translations and code not in use anymore](https://github.com/wekan/wekan/commit/ba49d4d140bc0d4cfb5a96db9ab077bc85db58f1). Thanks to xet7. +- [Fixed sidebar migrations to be per-board, not global. Clarified translations](https://github.com/wekan/wekan/commit/e4638d5fbcbe004ac393462331805cac3ba25097). + Thanks to xet7. Thanks to above GitHub users for their contributions and translators for their translations. From df9fba4765deffc6601ccd5d030a957c90cefc2d Mon Sep 17 00:00:00 2001 From: Lauri Ojansivu Date: Wed, 5 Nov 2025 20:26:29 +0200 Subject: [PATCH 45/84] Updated translations. --- imports/i18n/data/af.i18n.json | 63 +++++++++++++++++++++++++ imports/i18n/data/af_ZA.i18n.json | 63 +++++++++++++++++++++++++ imports/i18n/data/ar-DZ.i18n.json | 63 +++++++++++++++++++++++++ imports/i18n/data/ar-EG.i18n.json | 63 +++++++++++++++++++++++++ imports/i18n/data/ar.i18n.json | 63 +++++++++++++++++++++++++ imports/i18n/data/ary.i18n.json | 63 +++++++++++++++++++++++++ imports/i18n/data/ast-ES.i18n.json | 63 +++++++++++++++++++++++++ imports/i18n/data/az-AZ.i18n.json | 63 +++++++++++++++++++++++++ imports/i18n/data/az-LA.i18n.json | 63 +++++++++++++++++++++++++ imports/i18n/data/az.i18n.json | 63 +++++++++++++++++++++++++ imports/i18n/data/bg.i18n.json | 63 +++++++++++++++++++++++++ imports/i18n/data/br.i18n.json | 63 +++++++++++++++++++++++++ imports/i18n/data/ca.i18n.json | 63 +++++++++++++++++++++++++ imports/i18n/data/ca@valencia.i18n.json | 63 +++++++++++++++++++++++++ imports/i18n/data/ca_ES.i18n.json | 63 +++++++++++++++++++++++++ imports/i18n/data/cmn.i18n.json | 63 +++++++++++++++++++++++++ imports/i18n/data/cs-CZ.i18n.json | 63 +++++++++++++++++++++++++ imports/i18n/data/cs.i18n.json | 63 +++++++++++++++++++++++++ imports/i18n/data/cy-GB.i18n.json | 63 +++++++++++++++++++++++++ imports/i18n/data/cy.i18n.json | 63 +++++++++++++++++++++++++ imports/i18n/data/da.i18n.json | 63 +++++++++++++++++++++++++ imports/i18n/data/de-AT.i18n.json | 63 +++++++++++++++++++++++++ imports/i18n/data/de-CH.i18n.json | 63 +++++++++++++++++++++++++ imports/i18n/data/de.i18n.json | 63 +++++++++++++++++++++++++ imports/i18n/data/de_DE.i18n.json | 63 +++++++++++++++++++++++++ imports/i18n/data/el-GR.i18n.json | 63 +++++++++++++++++++++++++ imports/i18n/data/el.i18n.json | 63 +++++++++++++++++++++++++ imports/i18n/data/en-BR.i18n.json | 63 +++++++++++++++++++++++++ imports/i18n/data/en-DE.i18n.json | 63 +++++++++++++++++++++++++ imports/i18n/data/en-GB.i18n.json | 63 +++++++++++++++++++++++++ imports/i18n/data/en-IT.i18n.json | 63 +++++++++++++++++++++++++ imports/i18n/data/en-MY.i18n.json | 63 +++++++++++++++++++++++++ imports/i18n/data/en-YS.i18n.json | 63 +++++++++++++++++++++++++ imports/i18n/data/en_AU.i18n.json | 63 +++++++++++++++++++++++++ imports/i18n/data/en_ID.i18n.json | 63 +++++++++++++++++++++++++ imports/i18n/data/en_SG.i18n.json | 63 +++++++++++++++++++++++++ imports/i18n/data/en_TR.i18n.json | 63 +++++++++++++++++++++++++ imports/i18n/data/en_ZA.i18n.json | 63 +++++++++++++++++++++++++ imports/i18n/data/eo.i18n.json | 63 +++++++++++++++++++++++++ imports/i18n/data/es-AR.i18n.json | 63 +++++++++++++++++++++++++ imports/i18n/data/es-CL.i18n.json | 63 +++++++++++++++++++++++++ imports/i18n/data/es-LA.i18n.json | 63 +++++++++++++++++++++++++ imports/i18n/data/es-MX.i18n.json | 63 +++++++++++++++++++++++++ imports/i18n/data/es-PE.i18n.json | 63 +++++++++++++++++++++++++ imports/i18n/data/es-PY.i18n.json | 63 +++++++++++++++++++++++++ imports/i18n/data/es.i18n.json | 63 +++++++++++++++++++++++++ imports/i18n/data/es_CO.i18n.json | 63 +++++++++++++++++++++++++ imports/i18n/data/et-EE.i18n.json | 63 +++++++++++++++++++++++++ imports/i18n/data/eu.i18n.json | 63 +++++++++++++++++++++++++ imports/i18n/data/fa-IR.i18n.json | 63 +++++++++++++++++++++++++ imports/i18n/data/fa.i18n.json | 63 +++++++++++++++++++++++++ imports/i18n/data/fi.i18n.json | 63 +++++++++++++++++++++++++ imports/i18n/data/fr-CH.i18n.json | 63 +++++++++++++++++++++++++ imports/i18n/data/fr-FR.i18n.json | 63 +++++++++++++++++++++++++ imports/i18n/data/fr.i18n.json | 63 +++++++++++++++++++++++++ imports/i18n/data/fy-NL.i18n.json | 63 +++++++++++++++++++++++++ imports/i18n/data/fy.i18n.json | 63 +++++++++++++++++++++++++ imports/i18n/data/gl-ES.i18n.json | 63 +++++++++++++++++++++++++ imports/i18n/data/gl.i18n.json | 63 +++++++++++++++++++++++++ imports/i18n/data/gu-IN.i18n.json | 63 +++++++++++++++++++++++++ imports/i18n/data/he-IL.i18n.json | 63 +++++++++++++++++++++++++ imports/i18n/data/he.i18n.json | 63 +++++++++++++++++++++++++ imports/i18n/data/hi-IN.i18n.json | 63 +++++++++++++++++++++++++ imports/i18n/data/hi.i18n.json | 63 +++++++++++++++++++++++++ imports/i18n/data/hr.i18n.json | 63 +++++++++++++++++++++++++ imports/i18n/data/hu.i18n.json | 63 +++++++++++++++++++++++++ imports/i18n/data/hy.i18n.json | 63 +++++++++++++++++++++++++ imports/i18n/data/id.i18n.json | 63 +++++++++++++++++++++++++ imports/i18n/data/ig.i18n.json | 63 +++++++++++++++++++++++++ imports/i18n/data/it.i18n.json | 63 +++++++++++++++++++++++++ imports/i18n/data/ja-HI.i18n.json | 63 +++++++++++++++++++++++++ imports/i18n/data/ja.i18n.json | 63 +++++++++++++++++++++++++ imports/i18n/data/ka.i18n.json | 63 +++++++++++++++++++++++++ imports/i18n/data/km.i18n.json | 63 +++++++++++++++++++++++++ imports/i18n/data/ko-KR.i18n.json | 63 +++++++++++++++++++++++++ imports/i18n/data/ko.i18n.json | 63 +++++++++++++++++++++++++ imports/i18n/data/lt.i18n.json | 63 +++++++++++++++++++++++++ imports/i18n/data/lv.i18n.json | 63 +++++++++++++++++++++++++ imports/i18n/data/mk.i18n.json | 63 +++++++++++++++++++++++++ imports/i18n/data/mn.i18n.json | 63 +++++++++++++++++++++++++ imports/i18n/data/ms-MY.i18n.json | 63 +++++++++++++++++++++++++ imports/i18n/data/ms.i18n.json | 63 +++++++++++++++++++++++++ imports/i18n/data/nb.i18n.json | 63 +++++++++++++++++++++++++ imports/i18n/data/nl-NL.i18n.json | 63 +++++++++++++++++++++++++ imports/i18n/data/nl.i18n.json | 63 +++++++++++++++++++++++++ imports/i18n/data/oc.i18n.json | 63 +++++++++++++++++++++++++ imports/i18n/data/or_IN.i18n.json | 63 +++++++++++++++++++++++++ imports/i18n/data/pa.i18n.json | 63 +++++++++++++++++++++++++ imports/i18n/data/pl-PL.i18n.json | 63 +++++++++++++++++++++++++ imports/i18n/data/pl.i18n.json | 63 +++++++++++++++++++++++++ imports/i18n/data/pt-BR.i18n.json | 63 +++++++++++++++++++++++++ imports/i18n/data/pt.i18n.json | 63 +++++++++++++++++++++++++ imports/i18n/data/pt_PT.i18n.json | 63 +++++++++++++++++++++++++ imports/i18n/data/ro-RO.i18n.json | 63 +++++++++++++++++++++++++ imports/i18n/data/ro.i18n.json | 63 +++++++++++++++++++++++++ imports/i18n/data/ru-UA.i18n.json | 63 +++++++++++++++++++++++++ imports/i18n/data/ru.i18n.json | 63 +++++++++++++++++++++++++ imports/i18n/data/sk.i18n.json | 63 +++++++++++++++++++++++++ imports/i18n/data/sl.i18n.json | 63 +++++++++++++++++++++++++ imports/i18n/data/sr.i18n.json | 63 +++++++++++++++++++++++++ imports/i18n/data/sv.i18n.json | 63 +++++++++++++++++++++++++ imports/i18n/data/sw.i18n.json | 63 +++++++++++++++++++++++++ imports/i18n/data/ta.i18n.json | 63 +++++++++++++++++++++++++ imports/i18n/data/te-IN.i18n.json | 63 +++++++++++++++++++++++++ imports/i18n/data/th.i18n.json | 63 +++++++++++++++++++++++++ imports/i18n/data/tk_TM.i18n.json | 63 +++++++++++++++++++++++++ imports/i18n/data/tlh.i18n.json | 63 +++++++++++++++++++++++++ imports/i18n/data/tr.i18n.json | 63 +++++++++++++++++++++++++ imports/i18n/data/ug.i18n.json | 63 +++++++++++++++++++++++++ imports/i18n/data/uk-UA.i18n.json | 63 +++++++++++++++++++++++++ imports/i18n/data/uk.i18n.json | 63 +++++++++++++++++++++++++ imports/i18n/data/uz-AR.i18n.json | 63 +++++++++++++++++++++++++ imports/i18n/data/uz-LA.i18n.json | 63 +++++++++++++++++++++++++ imports/i18n/data/uz-UZ.i18n.json | 63 +++++++++++++++++++++++++ imports/i18n/data/uz.i18n.json | 63 +++++++++++++++++++++++++ imports/i18n/data/ve-CC.i18n.json | 63 +++++++++++++++++++++++++ imports/i18n/data/ve-PP.i18n.json | 63 +++++++++++++++++++++++++ imports/i18n/data/ve.i18n.json | 63 +++++++++++++++++++++++++ imports/i18n/data/vi-VN.i18n.json | 63 +++++++++++++++++++++++++ imports/i18n/data/vi.i18n.json | 63 +++++++++++++++++++++++++ imports/i18n/data/vl-SS.i18n.json | 63 +++++++++++++++++++++++++ imports/i18n/data/vo.i18n.json | 63 +++++++++++++++++++++++++ imports/i18n/data/wa-RR.i18n.json | 63 +++++++++++++++++++++++++ imports/i18n/data/wa.i18n.json | 63 +++++++++++++++++++++++++ imports/i18n/data/wo.i18n.json | 63 +++++++++++++++++++++++++ imports/i18n/data/wuu-Hans.i18n.json | 63 +++++++++++++++++++++++++ imports/i18n/data/xh.i18n.json | 63 +++++++++++++++++++++++++ imports/i18n/data/yi.i18n.json | 63 +++++++++++++++++++++++++ imports/i18n/data/yo.i18n.json | 63 +++++++++++++++++++++++++ imports/i18n/data/yue_CN.i18n.json | 63 +++++++++++++++++++++++++ imports/i18n/data/zgh.i18n.json | 63 +++++++++++++++++++++++++ imports/i18n/data/zh-CN.i18n.json | 63 +++++++++++++++++++++++++ imports/i18n/data/zh-GB.i18n.json | 63 +++++++++++++++++++++++++ imports/i18n/data/zh-HK.i18n.json | 63 +++++++++++++++++++++++++ imports/i18n/data/zh-Hans.i18n.json | 63 +++++++++++++++++++++++++ imports/i18n/data/zh-Hant.i18n.json | 63 +++++++++++++++++++++++++ imports/i18n/data/zh-TW.i18n.json | 63 +++++++++++++++++++++++++ imports/i18n/data/zh.i18n.json | 63 +++++++++++++++++++++++++ imports/i18n/data/zu-ZA.i18n.json | 63 +++++++++++++++++++++++++ imports/i18n/data/zu.i18n.json | 63 +++++++++++++++++++++++++ 140 files changed, 8820 insertions(+) diff --git a/imports/i18n/data/af.i18n.json b/imports/i18n/data/af.i18n.json index eef27e1fe..958d8313a 100644 --- a/imports/i18n/data/af.i18n.json +++ b/imports/i18n/data/af.i18n.json @@ -1404,7 +1404,70 @@ "back-to-settings": "Back to Settings", "board-id": "Board ID", "board-migration": "Board Migration", + "board-migrations": "Board Migrations", "card-show-lists-on-minicard": "Show Lists on Minicard", + "comprehensive-board-migration": "Comprehensive Board Migration", + "comprehensive-board-migration-description": "Performs comprehensive checks and fixes for board data integrity, including list ordering, card positions, and swimlane structure.", + "delete-duplicate-empty-lists-migration": "Delete Duplicate Empty Lists", + "delete-duplicate-empty-lists-migration-description": "Safely deletes empty duplicate lists. Only removes lists that have no cards AND have another list with the same title that contains cards.", + "lost-cards": "Lost Cards", + "lost-cards-list": "Restored Items", + "restore-lost-cards-migration": "Restore Lost Cards", + "restore-lost-cards-migration-description": "Finds and restores cards and lists with missing swimlaneId or listId. Creates a 'Lost Cards' swimlane to make all lost items visible again.", + "restore-all-archived-migration": "Restore All Archived", + "restore-all-archived-migration-description": "Restores all archived swimlanes, lists, and cards. Automatically fixes any missing swimlaneId or listId to make items visible.", + "fix-missing-lists-migration": "Fix Missing Lists", + "fix-missing-lists-migration-description": "Detects and repairs missing or corrupted lists in the board structure.", + "fix-avatar-urls-migration": "Fix Avatar URLs", + "fix-avatar-urls-migration-description": "Updates avatar URLs for board members to use the correct storage backend and fixes broken avatar references.", + "fix-all-file-urls-migration": "Fix All File URLs", + "fix-all-file-urls-migration-description": "Updates all file attachment URLs on this board to use the correct storage backend and fixes broken file references.", + "migration-needed": "Migration Needed", + "migration-complete": "Complete", + "migration-running": "Running...", + "migration-successful": "Migration completed successfully", + "migration-failed": "Migration failed", + "migrations": "Migrations", + "migrations-admin-only": "Only board administrators can run migrations", + "migrations-description": "Run data integrity checks and repairs for this board. Each migration can be executed individually.", + "no-issues-found": "No issues found", + "run-migration": "Run Migration", + "run-comprehensive-migration-confirm": "This will perform a comprehensive migration to check and fix board data integrity. This may take a few moments. Continue?", + "run-delete-duplicate-empty-lists-migration-confirm": "This will first convert any shared lists to per-swimlane lists, then delete empty lists that have a duplicate list with the same title containing cards. Only truly redundant empty lists will be removed. Continue?", + "run-restore-lost-cards-migration-confirm": "This will create a 'Lost Cards' swimlane and restore all cards and lists with missing swimlaneId or listId. This only affects non-archived items. Continue?", + "run-restore-all-archived-migration-confirm": "This will restore ALL archived swimlanes, lists, and cards, making them visible again. Any items with missing IDs will be automatically fixed. This cannot be easily undone. Continue?", + "run-fix-missing-lists-migration-confirm": "This will detect and repair missing or corrupted lists in the board structure. Continue?", + "run-fix-avatar-urls-migration-confirm": "This will update avatar URLs for board members to use the correct storage backend. Continue?", + "run-fix-all-file-urls-migration-confirm": "This will update all file attachment URLs on this board to use the correct storage backend. Continue?", + "restore-lost-cards-nothing-to-restore": "No lost swimlanes, lists, or cards to restore", + + "migration-progress-title": "Board Migration in Progress", + "migration-progress-overall": "Overall Progress", + "migration-progress-current-step": "Current Step", + "migration-progress-status": "Status", + "migration-progress-details": "Details", + "migration-progress-note": "Please wait while we migrate your board to the latest structure...", + + "step-analyze-board-structure": "Analyze Board Structure", + "step-fix-orphaned-cards": "Fix Orphaned Cards", + "step-convert-shared-lists": "Convert Shared Lists", + "step-ensure-per-swimlane-lists": "Ensure Per-Swimlane Lists", + "step-validate-migration": "Validate Migration", + "step-fix-avatar-urls": "Fix Avatar URLs", + "step-fix-attachment-urls": "Fix Attachment URLs", + "step-analyze-lists": "Analyze Lists", + "step-create-missing-lists": "Create Missing Lists", + "step-update-cards": "Update Cards", + "step-finalize": "Finalize", + "step-delete-duplicate-empty-lists": "Delete Duplicate Empty Lists", + "step-ensure-lost-cards-swimlane": "Ensure Lost Cards Swimlane", + "step-restore-lists": "Restore Lists", + "step-restore-cards": "Restore Cards", + "step-restore-swimlanes": "Restore Swimlanes", + "step-fix-missing-ids": "Fix Missing IDs", + "step-scan-users": "Checking board member avatars", + "step-scan-files": "Checking board file attachments", + "step-fix-file-urls": "Fixing file URLs", "cleanup": "Cleanup", "cleanup-old-jobs": "Cleanup Old Jobs", "completed": "Completed", diff --git a/imports/i18n/data/af_ZA.i18n.json b/imports/i18n/data/af_ZA.i18n.json index ecc4080e2..ff43f83d1 100644 --- a/imports/i18n/data/af_ZA.i18n.json +++ b/imports/i18n/data/af_ZA.i18n.json @@ -1404,7 +1404,70 @@ "back-to-settings": "Back to Settings", "board-id": "Board ID", "board-migration": "Board Migration", + "board-migrations": "Board Migrations", "card-show-lists-on-minicard": "Show Lists on Minicard", + "comprehensive-board-migration": "Comprehensive Board Migration", + "comprehensive-board-migration-description": "Performs comprehensive checks and fixes for board data integrity, including list ordering, card positions, and swimlane structure.", + "delete-duplicate-empty-lists-migration": "Delete Duplicate Empty Lists", + "delete-duplicate-empty-lists-migration-description": "Safely deletes empty duplicate lists. Only removes lists that have no cards AND have another list with the same title that contains cards.", + "lost-cards": "Lost Cards", + "lost-cards-list": "Restored Items", + "restore-lost-cards-migration": "Restore Lost Cards", + "restore-lost-cards-migration-description": "Finds and restores cards and lists with missing swimlaneId or listId. Creates a 'Lost Cards' swimlane to make all lost items visible again.", + "restore-all-archived-migration": "Restore All Archived", + "restore-all-archived-migration-description": "Restores all archived swimlanes, lists, and cards. Automatically fixes any missing swimlaneId or listId to make items visible.", + "fix-missing-lists-migration": "Fix Missing Lists", + "fix-missing-lists-migration-description": "Detects and repairs missing or corrupted lists in the board structure.", + "fix-avatar-urls-migration": "Fix Avatar URLs", + "fix-avatar-urls-migration-description": "Updates avatar URLs for board members to use the correct storage backend and fixes broken avatar references.", + "fix-all-file-urls-migration": "Fix All File URLs", + "fix-all-file-urls-migration-description": "Updates all file attachment URLs on this board to use the correct storage backend and fixes broken file references.", + "migration-needed": "Migration Needed", + "migration-complete": "Complete", + "migration-running": "Running...", + "migration-successful": "Migration completed successfully", + "migration-failed": "Migration failed", + "migrations": "Migrations", + "migrations-admin-only": "Only board administrators can run migrations", + "migrations-description": "Run data integrity checks and repairs for this board. Each migration can be executed individually.", + "no-issues-found": "No issues found", + "run-migration": "Run Migration", + "run-comprehensive-migration-confirm": "This will perform a comprehensive migration to check and fix board data integrity. This may take a few moments. Continue?", + "run-delete-duplicate-empty-lists-migration-confirm": "This will first convert any shared lists to per-swimlane lists, then delete empty lists that have a duplicate list with the same title containing cards. Only truly redundant empty lists will be removed. Continue?", + "run-restore-lost-cards-migration-confirm": "This will create a 'Lost Cards' swimlane and restore all cards and lists with missing swimlaneId or listId. This only affects non-archived items. Continue?", + "run-restore-all-archived-migration-confirm": "This will restore ALL archived swimlanes, lists, and cards, making them visible again. Any items with missing IDs will be automatically fixed. This cannot be easily undone. Continue?", + "run-fix-missing-lists-migration-confirm": "This will detect and repair missing or corrupted lists in the board structure. Continue?", + "run-fix-avatar-urls-migration-confirm": "This will update avatar URLs for board members to use the correct storage backend. Continue?", + "run-fix-all-file-urls-migration-confirm": "This will update all file attachment URLs on this board to use the correct storage backend. Continue?", + "restore-lost-cards-nothing-to-restore": "No lost swimlanes, lists, or cards to restore", + + "migration-progress-title": "Board Migration in Progress", + "migration-progress-overall": "Overall Progress", + "migration-progress-current-step": "Current Step", + "migration-progress-status": "Status", + "migration-progress-details": "Details", + "migration-progress-note": "Please wait while we migrate your board to the latest structure...", + + "step-analyze-board-structure": "Analyze Board Structure", + "step-fix-orphaned-cards": "Fix Orphaned Cards", + "step-convert-shared-lists": "Convert Shared Lists", + "step-ensure-per-swimlane-lists": "Ensure Per-Swimlane Lists", + "step-validate-migration": "Validate Migration", + "step-fix-avatar-urls": "Fix Avatar URLs", + "step-fix-attachment-urls": "Fix Attachment URLs", + "step-analyze-lists": "Analyze Lists", + "step-create-missing-lists": "Create Missing Lists", + "step-update-cards": "Update Cards", + "step-finalize": "Finalize", + "step-delete-duplicate-empty-lists": "Delete Duplicate Empty Lists", + "step-ensure-lost-cards-swimlane": "Ensure Lost Cards Swimlane", + "step-restore-lists": "Restore Lists", + "step-restore-cards": "Restore Cards", + "step-restore-swimlanes": "Restore Swimlanes", + "step-fix-missing-ids": "Fix Missing IDs", + "step-scan-users": "Checking board member avatars", + "step-scan-files": "Checking board file attachments", + "step-fix-file-urls": "Fixing file URLs", "cleanup": "Cleanup", "cleanup-old-jobs": "Cleanup Old Jobs", "completed": "Completed", diff --git a/imports/i18n/data/ar-DZ.i18n.json b/imports/i18n/data/ar-DZ.i18n.json index 1ab8eeaa8..4804e6dff 100644 --- a/imports/i18n/data/ar-DZ.i18n.json +++ b/imports/i18n/data/ar-DZ.i18n.json @@ -1404,7 +1404,70 @@ "back-to-settings": "Back to Settings", "board-id": "Board ID", "board-migration": "Board Migration", + "board-migrations": "Board Migrations", "card-show-lists-on-minicard": "Show Lists on Minicard", + "comprehensive-board-migration": "Comprehensive Board Migration", + "comprehensive-board-migration-description": "Performs comprehensive checks and fixes for board data integrity, including list ordering, card positions, and swimlane structure.", + "delete-duplicate-empty-lists-migration": "Delete Duplicate Empty Lists", + "delete-duplicate-empty-lists-migration-description": "Safely deletes empty duplicate lists. Only removes lists that have no cards AND have another list with the same title that contains cards.", + "lost-cards": "Lost Cards", + "lost-cards-list": "Restored Items", + "restore-lost-cards-migration": "Restore Lost Cards", + "restore-lost-cards-migration-description": "Finds and restores cards and lists with missing swimlaneId or listId. Creates a 'Lost Cards' swimlane to make all lost items visible again.", + "restore-all-archived-migration": "Restore All Archived", + "restore-all-archived-migration-description": "Restores all archived swimlanes, lists, and cards. Automatically fixes any missing swimlaneId or listId to make items visible.", + "fix-missing-lists-migration": "Fix Missing Lists", + "fix-missing-lists-migration-description": "Detects and repairs missing or corrupted lists in the board structure.", + "fix-avatar-urls-migration": "Fix Avatar URLs", + "fix-avatar-urls-migration-description": "Updates avatar URLs for board members to use the correct storage backend and fixes broken avatar references.", + "fix-all-file-urls-migration": "Fix All File URLs", + "fix-all-file-urls-migration-description": "Updates all file attachment URLs on this board to use the correct storage backend and fixes broken file references.", + "migration-needed": "Migration Needed", + "migration-complete": "Complete", + "migration-running": "Running...", + "migration-successful": "Migration completed successfully", + "migration-failed": "Migration failed", + "migrations": "Migrations", + "migrations-admin-only": "Only board administrators can run migrations", + "migrations-description": "Run data integrity checks and repairs for this board. Each migration can be executed individually.", + "no-issues-found": "No issues found", + "run-migration": "Run Migration", + "run-comprehensive-migration-confirm": "This will perform a comprehensive migration to check and fix board data integrity. This may take a few moments. Continue?", + "run-delete-duplicate-empty-lists-migration-confirm": "This will first convert any shared lists to per-swimlane lists, then delete empty lists that have a duplicate list with the same title containing cards. Only truly redundant empty lists will be removed. Continue?", + "run-restore-lost-cards-migration-confirm": "This will create a 'Lost Cards' swimlane and restore all cards and lists with missing swimlaneId or listId. This only affects non-archived items. Continue?", + "run-restore-all-archived-migration-confirm": "This will restore ALL archived swimlanes, lists, and cards, making them visible again. Any items with missing IDs will be automatically fixed. This cannot be easily undone. Continue?", + "run-fix-missing-lists-migration-confirm": "This will detect and repair missing or corrupted lists in the board structure. Continue?", + "run-fix-avatar-urls-migration-confirm": "This will update avatar URLs for board members to use the correct storage backend. Continue?", + "run-fix-all-file-urls-migration-confirm": "This will update all file attachment URLs on this board to use the correct storage backend. Continue?", + "restore-lost-cards-nothing-to-restore": "No lost swimlanes, lists, or cards to restore", + + "migration-progress-title": "Board Migration in Progress", + "migration-progress-overall": "Overall Progress", + "migration-progress-current-step": "Current Step", + "migration-progress-status": "Status", + "migration-progress-details": "Details", + "migration-progress-note": "Please wait while we migrate your board to the latest structure...", + + "step-analyze-board-structure": "Analyze Board Structure", + "step-fix-orphaned-cards": "Fix Orphaned Cards", + "step-convert-shared-lists": "Convert Shared Lists", + "step-ensure-per-swimlane-lists": "Ensure Per-Swimlane Lists", + "step-validate-migration": "Validate Migration", + "step-fix-avatar-urls": "Fix Avatar URLs", + "step-fix-attachment-urls": "Fix Attachment URLs", + "step-analyze-lists": "Analyze Lists", + "step-create-missing-lists": "Create Missing Lists", + "step-update-cards": "Update Cards", + "step-finalize": "Finalize", + "step-delete-duplicate-empty-lists": "Delete Duplicate Empty Lists", + "step-ensure-lost-cards-swimlane": "Ensure Lost Cards Swimlane", + "step-restore-lists": "Restore Lists", + "step-restore-cards": "Restore Cards", + "step-restore-swimlanes": "Restore Swimlanes", + "step-fix-missing-ids": "Fix Missing IDs", + "step-scan-users": "Checking board member avatars", + "step-scan-files": "Checking board file attachments", + "step-fix-file-urls": "Fixing file URLs", "cleanup": "Cleanup", "cleanup-old-jobs": "Cleanup Old Jobs", "completed": "Completed", diff --git a/imports/i18n/data/ar-EG.i18n.json b/imports/i18n/data/ar-EG.i18n.json index 1ab8eeaa8..4804e6dff 100644 --- a/imports/i18n/data/ar-EG.i18n.json +++ b/imports/i18n/data/ar-EG.i18n.json @@ -1404,7 +1404,70 @@ "back-to-settings": "Back to Settings", "board-id": "Board ID", "board-migration": "Board Migration", + "board-migrations": "Board Migrations", "card-show-lists-on-minicard": "Show Lists on Minicard", + "comprehensive-board-migration": "Comprehensive Board Migration", + "comprehensive-board-migration-description": "Performs comprehensive checks and fixes for board data integrity, including list ordering, card positions, and swimlane structure.", + "delete-duplicate-empty-lists-migration": "Delete Duplicate Empty Lists", + "delete-duplicate-empty-lists-migration-description": "Safely deletes empty duplicate lists. Only removes lists that have no cards AND have another list with the same title that contains cards.", + "lost-cards": "Lost Cards", + "lost-cards-list": "Restored Items", + "restore-lost-cards-migration": "Restore Lost Cards", + "restore-lost-cards-migration-description": "Finds and restores cards and lists with missing swimlaneId or listId. Creates a 'Lost Cards' swimlane to make all lost items visible again.", + "restore-all-archived-migration": "Restore All Archived", + "restore-all-archived-migration-description": "Restores all archived swimlanes, lists, and cards. Automatically fixes any missing swimlaneId or listId to make items visible.", + "fix-missing-lists-migration": "Fix Missing Lists", + "fix-missing-lists-migration-description": "Detects and repairs missing or corrupted lists in the board structure.", + "fix-avatar-urls-migration": "Fix Avatar URLs", + "fix-avatar-urls-migration-description": "Updates avatar URLs for board members to use the correct storage backend and fixes broken avatar references.", + "fix-all-file-urls-migration": "Fix All File URLs", + "fix-all-file-urls-migration-description": "Updates all file attachment URLs on this board to use the correct storage backend and fixes broken file references.", + "migration-needed": "Migration Needed", + "migration-complete": "Complete", + "migration-running": "Running...", + "migration-successful": "Migration completed successfully", + "migration-failed": "Migration failed", + "migrations": "Migrations", + "migrations-admin-only": "Only board administrators can run migrations", + "migrations-description": "Run data integrity checks and repairs for this board. Each migration can be executed individually.", + "no-issues-found": "No issues found", + "run-migration": "Run Migration", + "run-comprehensive-migration-confirm": "This will perform a comprehensive migration to check and fix board data integrity. This may take a few moments. Continue?", + "run-delete-duplicate-empty-lists-migration-confirm": "This will first convert any shared lists to per-swimlane lists, then delete empty lists that have a duplicate list with the same title containing cards. Only truly redundant empty lists will be removed. Continue?", + "run-restore-lost-cards-migration-confirm": "This will create a 'Lost Cards' swimlane and restore all cards and lists with missing swimlaneId or listId. This only affects non-archived items. Continue?", + "run-restore-all-archived-migration-confirm": "This will restore ALL archived swimlanes, lists, and cards, making them visible again. Any items with missing IDs will be automatically fixed. This cannot be easily undone. Continue?", + "run-fix-missing-lists-migration-confirm": "This will detect and repair missing or corrupted lists in the board structure. Continue?", + "run-fix-avatar-urls-migration-confirm": "This will update avatar URLs for board members to use the correct storage backend. Continue?", + "run-fix-all-file-urls-migration-confirm": "This will update all file attachment URLs on this board to use the correct storage backend. Continue?", + "restore-lost-cards-nothing-to-restore": "No lost swimlanes, lists, or cards to restore", + + "migration-progress-title": "Board Migration in Progress", + "migration-progress-overall": "Overall Progress", + "migration-progress-current-step": "Current Step", + "migration-progress-status": "Status", + "migration-progress-details": "Details", + "migration-progress-note": "Please wait while we migrate your board to the latest structure...", + + "step-analyze-board-structure": "Analyze Board Structure", + "step-fix-orphaned-cards": "Fix Orphaned Cards", + "step-convert-shared-lists": "Convert Shared Lists", + "step-ensure-per-swimlane-lists": "Ensure Per-Swimlane Lists", + "step-validate-migration": "Validate Migration", + "step-fix-avatar-urls": "Fix Avatar URLs", + "step-fix-attachment-urls": "Fix Attachment URLs", + "step-analyze-lists": "Analyze Lists", + "step-create-missing-lists": "Create Missing Lists", + "step-update-cards": "Update Cards", + "step-finalize": "Finalize", + "step-delete-duplicate-empty-lists": "Delete Duplicate Empty Lists", + "step-ensure-lost-cards-swimlane": "Ensure Lost Cards Swimlane", + "step-restore-lists": "Restore Lists", + "step-restore-cards": "Restore Cards", + "step-restore-swimlanes": "Restore Swimlanes", + "step-fix-missing-ids": "Fix Missing IDs", + "step-scan-users": "Checking board member avatars", + "step-scan-files": "Checking board file attachments", + "step-fix-file-urls": "Fixing file URLs", "cleanup": "Cleanup", "cleanup-old-jobs": "Cleanup Old Jobs", "completed": "Completed", diff --git a/imports/i18n/data/ar.i18n.json b/imports/i18n/data/ar.i18n.json index b4628a45d..5c7227056 100644 --- a/imports/i18n/data/ar.i18n.json +++ b/imports/i18n/data/ar.i18n.json @@ -1404,7 +1404,70 @@ "back-to-settings": "Back to Settings", "board-id": "Board ID", "board-migration": "Board Migration", + "board-migrations": "Board Migrations", "card-show-lists-on-minicard": "Show Lists on Minicard", + "comprehensive-board-migration": "Comprehensive Board Migration", + "comprehensive-board-migration-description": "Performs comprehensive checks and fixes for board data integrity, including list ordering, card positions, and swimlane structure.", + "delete-duplicate-empty-lists-migration": "Delete Duplicate Empty Lists", + "delete-duplicate-empty-lists-migration-description": "Safely deletes empty duplicate lists. Only removes lists that have no cards AND have another list with the same title that contains cards.", + "lost-cards": "Lost Cards", + "lost-cards-list": "Restored Items", + "restore-lost-cards-migration": "Restore Lost Cards", + "restore-lost-cards-migration-description": "Finds and restores cards and lists with missing swimlaneId or listId. Creates a 'Lost Cards' swimlane to make all lost items visible again.", + "restore-all-archived-migration": "Restore All Archived", + "restore-all-archived-migration-description": "Restores all archived swimlanes, lists, and cards. Automatically fixes any missing swimlaneId or listId to make items visible.", + "fix-missing-lists-migration": "Fix Missing Lists", + "fix-missing-lists-migration-description": "Detects and repairs missing or corrupted lists in the board structure.", + "fix-avatar-urls-migration": "Fix Avatar URLs", + "fix-avatar-urls-migration-description": "Updates avatar URLs for board members to use the correct storage backend and fixes broken avatar references.", + "fix-all-file-urls-migration": "Fix All File URLs", + "fix-all-file-urls-migration-description": "Updates all file attachment URLs on this board to use the correct storage backend and fixes broken file references.", + "migration-needed": "Migration Needed", + "migration-complete": "Complete", + "migration-running": "Running...", + "migration-successful": "Migration completed successfully", + "migration-failed": "Migration failed", + "migrations": "Migrations", + "migrations-admin-only": "Only board administrators can run migrations", + "migrations-description": "Run data integrity checks and repairs for this board. Each migration can be executed individually.", + "no-issues-found": "No issues found", + "run-migration": "Run Migration", + "run-comprehensive-migration-confirm": "This will perform a comprehensive migration to check and fix board data integrity. This may take a few moments. Continue?", + "run-delete-duplicate-empty-lists-migration-confirm": "This will first convert any shared lists to per-swimlane lists, then delete empty lists that have a duplicate list with the same title containing cards. Only truly redundant empty lists will be removed. Continue?", + "run-restore-lost-cards-migration-confirm": "This will create a 'Lost Cards' swimlane and restore all cards and lists with missing swimlaneId or listId. This only affects non-archived items. Continue?", + "run-restore-all-archived-migration-confirm": "This will restore ALL archived swimlanes, lists, and cards, making them visible again. Any items with missing IDs will be automatically fixed. This cannot be easily undone. Continue?", + "run-fix-missing-lists-migration-confirm": "This will detect and repair missing or corrupted lists in the board structure. Continue?", + "run-fix-avatar-urls-migration-confirm": "This will update avatar URLs for board members to use the correct storage backend. Continue?", + "run-fix-all-file-urls-migration-confirm": "This will update all file attachment URLs on this board to use the correct storage backend. Continue?", + "restore-lost-cards-nothing-to-restore": "No lost swimlanes, lists, or cards to restore", + + "migration-progress-title": "Board Migration in Progress", + "migration-progress-overall": "Overall Progress", + "migration-progress-current-step": "Current Step", + "migration-progress-status": "Status", + "migration-progress-details": "تفاصيل", + "migration-progress-note": "Please wait while we migrate your board to the latest structure...", + + "step-analyze-board-structure": "Analyze Board Structure", + "step-fix-orphaned-cards": "Fix Orphaned Cards", + "step-convert-shared-lists": "Convert Shared Lists", + "step-ensure-per-swimlane-lists": "Ensure Per-Swimlane Lists", + "step-validate-migration": "Validate Migration", + "step-fix-avatar-urls": "Fix Avatar URLs", + "step-fix-attachment-urls": "Fix Attachment URLs", + "step-analyze-lists": "Analyze Lists", + "step-create-missing-lists": "Create Missing Lists", + "step-update-cards": "Update Cards", + "step-finalize": "Finalize", + "step-delete-duplicate-empty-lists": "Delete Duplicate Empty Lists", + "step-ensure-lost-cards-swimlane": "Ensure Lost Cards Swimlane", + "step-restore-lists": "Restore Lists", + "step-restore-cards": "Restore Cards", + "step-restore-swimlanes": "Restore Swimlanes", + "step-fix-missing-ids": "Fix Missing IDs", + "step-scan-users": "Checking board member avatars", + "step-scan-files": "Checking board file attachments", + "step-fix-file-urls": "Fixing file URLs", "cleanup": "Cleanup", "cleanup-old-jobs": "Cleanup Old Jobs", "completed": "Completed", diff --git a/imports/i18n/data/ary.i18n.json b/imports/i18n/data/ary.i18n.json index 1ab8eeaa8..4804e6dff 100644 --- a/imports/i18n/data/ary.i18n.json +++ b/imports/i18n/data/ary.i18n.json @@ -1404,7 +1404,70 @@ "back-to-settings": "Back to Settings", "board-id": "Board ID", "board-migration": "Board Migration", + "board-migrations": "Board Migrations", "card-show-lists-on-minicard": "Show Lists on Minicard", + "comprehensive-board-migration": "Comprehensive Board Migration", + "comprehensive-board-migration-description": "Performs comprehensive checks and fixes for board data integrity, including list ordering, card positions, and swimlane structure.", + "delete-duplicate-empty-lists-migration": "Delete Duplicate Empty Lists", + "delete-duplicate-empty-lists-migration-description": "Safely deletes empty duplicate lists. Only removes lists that have no cards AND have another list with the same title that contains cards.", + "lost-cards": "Lost Cards", + "lost-cards-list": "Restored Items", + "restore-lost-cards-migration": "Restore Lost Cards", + "restore-lost-cards-migration-description": "Finds and restores cards and lists with missing swimlaneId or listId. Creates a 'Lost Cards' swimlane to make all lost items visible again.", + "restore-all-archived-migration": "Restore All Archived", + "restore-all-archived-migration-description": "Restores all archived swimlanes, lists, and cards. Automatically fixes any missing swimlaneId or listId to make items visible.", + "fix-missing-lists-migration": "Fix Missing Lists", + "fix-missing-lists-migration-description": "Detects and repairs missing or corrupted lists in the board structure.", + "fix-avatar-urls-migration": "Fix Avatar URLs", + "fix-avatar-urls-migration-description": "Updates avatar URLs for board members to use the correct storage backend and fixes broken avatar references.", + "fix-all-file-urls-migration": "Fix All File URLs", + "fix-all-file-urls-migration-description": "Updates all file attachment URLs on this board to use the correct storage backend and fixes broken file references.", + "migration-needed": "Migration Needed", + "migration-complete": "Complete", + "migration-running": "Running...", + "migration-successful": "Migration completed successfully", + "migration-failed": "Migration failed", + "migrations": "Migrations", + "migrations-admin-only": "Only board administrators can run migrations", + "migrations-description": "Run data integrity checks and repairs for this board. Each migration can be executed individually.", + "no-issues-found": "No issues found", + "run-migration": "Run Migration", + "run-comprehensive-migration-confirm": "This will perform a comprehensive migration to check and fix board data integrity. This may take a few moments. Continue?", + "run-delete-duplicate-empty-lists-migration-confirm": "This will first convert any shared lists to per-swimlane lists, then delete empty lists that have a duplicate list with the same title containing cards. Only truly redundant empty lists will be removed. Continue?", + "run-restore-lost-cards-migration-confirm": "This will create a 'Lost Cards' swimlane and restore all cards and lists with missing swimlaneId or listId. This only affects non-archived items. Continue?", + "run-restore-all-archived-migration-confirm": "This will restore ALL archived swimlanes, lists, and cards, making them visible again. Any items with missing IDs will be automatically fixed. This cannot be easily undone. Continue?", + "run-fix-missing-lists-migration-confirm": "This will detect and repair missing or corrupted lists in the board structure. Continue?", + "run-fix-avatar-urls-migration-confirm": "This will update avatar URLs for board members to use the correct storage backend. Continue?", + "run-fix-all-file-urls-migration-confirm": "This will update all file attachment URLs on this board to use the correct storage backend. Continue?", + "restore-lost-cards-nothing-to-restore": "No lost swimlanes, lists, or cards to restore", + + "migration-progress-title": "Board Migration in Progress", + "migration-progress-overall": "Overall Progress", + "migration-progress-current-step": "Current Step", + "migration-progress-status": "Status", + "migration-progress-details": "Details", + "migration-progress-note": "Please wait while we migrate your board to the latest structure...", + + "step-analyze-board-structure": "Analyze Board Structure", + "step-fix-orphaned-cards": "Fix Orphaned Cards", + "step-convert-shared-lists": "Convert Shared Lists", + "step-ensure-per-swimlane-lists": "Ensure Per-Swimlane Lists", + "step-validate-migration": "Validate Migration", + "step-fix-avatar-urls": "Fix Avatar URLs", + "step-fix-attachment-urls": "Fix Attachment URLs", + "step-analyze-lists": "Analyze Lists", + "step-create-missing-lists": "Create Missing Lists", + "step-update-cards": "Update Cards", + "step-finalize": "Finalize", + "step-delete-duplicate-empty-lists": "Delete Duplicate Empty Lists", + "step-ensure-lost-cards-swimlane": "Ensure Lost Cards Swimlane", + "step-restore-lists": "Restore Lists", + "step-restore-cards": "Restore Cards", + "step-restore-swimlanes": "Restore Swimlanes", + "step-fix-missing-ids": "Fix Missing IDs", + "step-scan-users": "Checking board member avatars", + "step-scan-files": "Checking board file attachments", + "step-fix-file-urls": "Fixing file URLs", "cleanup": "Cleanup", "cleanup-old-jobs": "Cleanup Old Jobs", "completed": "Completed", diff --git a/imports/i18n/data/ast-ES.i18n.json b/imports/i18n/data/ast-ES.i18n.json index 1ab8eeaa8..4804e6dff 100644 --- a/imports/i18n/data/ast-ES.i18n.json +++ b/imports/i18n/data/ast-ES.i18n.json @@ -1404,7 +1404,70 @@ "back-to-settings": "Back to Settings", "board-id": "Board ID", "board-migration": "Board Migration", + "board-migrations": "Board Migrations", "card-show-lists-on-minicard": "Show Lists on Minicard", + "comprehensive-board-migration": "Comprehensive Board Migration", + "comprehensive-board-migration-description": "Performs comprehensive checks and fixes for board data integrity, including list ordering, card positions, and swimlane structure.", + "delete-duplicate-empty-lists-migration": "Delete Duplicate Empty Lists", + "delete-duplicate-empty-lists-migration-description": "Safely deletes empty duplicate lists. Only removes lists that have no cards AND have another list with the same title that contains cards.", + "lost-cards": "Lost Cards", + "lost-cards-list": "Restored Items", + "restore-lost-cards-migration": "Restore Lost Cards", + "restore-lost-cards-migration-description": "Finds and restores cards and lists with missing swimlaneId or listId. Creates a 'Lost Cards' swimlane to make all lost items visible again.", + "restore-all-archived-migration": "Restore All Archived", + "restore-all-archived-migration-description": "Restores all archived swimlanes, lists, and cards. Automatically fixes any missing swimlaneId or listId to make items visible.", + "fix-missing-lists-migration": "Fix Missing Lists", + "fix-missing-lists-migration-description": "Detects and repairs missing or corrupted lists in the board structure.", + "fix-avatar-urls-migration": "Fix Avatar URLs", + "fix-avatar-urls-migration-description": "Updates avatar URLs for board members to use the correct storage backend and fixes broken avatar references.", + "fix-all-file-urls-migration": "Fix All File URLs", + "fix-all-file-urls-migration-description": "Updates all file attachment URLs on this board to use the correct storage backend and fixes broken file references.", + "migration-needed": "Migration Needed", + "migration-complete": "Complete", + "migration-running": "Running...", + "migration-successful": "Migration completed successfully", + "migration-failed": "Migration failed", + "migrations": "Migrations", + "migrations-admin-only": "Only board administrators can run migrations", + "migrations-description": "Run data integrity checks and repairs for this board. Each migration can be executed individually.", + "no-issues-found": "No issues found", + "run-migration": "Run Migration", + "run-comprehensive-migration-confirm": "This will perform a comprehensive migration to check and fix board data integrity. This may take a few moments. Continue?", + "run-delete-duplicate-empty-lists-migration-confirm": "This will first convert any shared lists to per-swimlane lists, then delete empty lists that have a duplicate list with the same title containing cards. Only truly redundant empty lists will be removed. Continue?", + "run-restore-lost-cards-migration-confirm": "This will create a 'Lost Cards' swimlane and restore all cards and lists with missing swimlaneId or listId. This only affects non-archived items. Continue?", + "run-restore-all-archived-migration-confirm": "This will restore ALL archived swimlanes, lists, and cards, making them visible again. Any items with missing IDs will be automatically fixed. This cannot be easily undone. Continue?", + "run-fix-missing-lists-migration-confirm": "This will detect and repair missing or corrupted lists in the board structure. Continue?", + "run-fix-avatar-urls-migration-confirm": "This will update avatar URLs for board members to use the correct storage backend. Continue?", + "run-fix-all-file-urls-migration-confirm": "This will update all file attachment URLs on this board to use the correct storage backend. Continue?", + "restore-lost-cards-nothing-to-restore": "No lost swimlanes, lists, or cards to restore", + + "migration-progress-title": "Board Migration in Progress", + "migration-progress-overall": "Overall Progress", + "migration-progress-current-step": "Current Step", + "migration-progress-status": "Status", + "migration-progress-details": "Details", + "migration-progress-note": "Please wait while we migrate your board to the latest structure...", + + "step-analyze-board-structure": "Analyze Board Structure", + "step-fix-orphaned-cards": "Fix Orphaned Cards", + "step-convert-shared-lists": "Convert Shared Lists", + "step-ensure-per-swimlane-lists": "Ensure Per-Swimlane Lists", + "step-validate-migration": "Validate Migration", + "step-fix-avatar-urls": "Fix Avatar URLs", + "step-fix-attachment-urls": "Fix Attachment URLs", + "step-analyze-lists": "Analyze Lists", + "step-create-missing-lists": "Create Missing Lists", + "step-update-cards": "Update Cards", + "step-finalize": "Finalize", + "step-delete-duplicate-empty-lists": "Delete Duplicate Empty Lists", + "step-ensure-lost-cards-swimlane": "Ensure Lost Cards Swimlane", + "step-restore-lists": "Restore Lists", + "step-restore-cards": "Restore Cards", + "step-restore-swimlanes": "Restore Swimlanes", + "step-fix-missing-ids": "Fix Missing IDs", + "step-scan-users": "Checking board member avatars", + "step-scan-files": "Checking board file attachments", + "step-fix-file-urls": "Fixing file URLs", "cleanup": "Cleanup", "cleanup-old-jobs": "Cleanup Old Jobs", "completed": "Completed", diff --git a/imports/i18n/data/az-AZ.i18n.json b/imports/i18n/data/az-AZ.i18n.json index 1ab8eeaa8..4804e6dff 100644 --- a/imports/i18n/data/az-AZ.i18n.json +++ b/imports/i18n/data/az-AZ.i18n.json @@ -1404,7 +1404,70 @@ "back-to-settings": "Back to Settings", "board-id": "Board ID", "board-migration": "Board Migration", + "board-migrations": "Board Migrations", "card-show-lists-on-minicard": "Show Lists on Minicard", + "comprehensive-board-migration": "Comprehensive Board Migration", + "comprehensive-board-migration-description": "Performs comprehensive checks and fixes for board data integrity, including list ordering, card positions, and swimlane structure.", + "delete-duplicate-empty-lists-migration": "Delete Duplicate Empty Lists", + "delete-duplicate-empty-lists-migration-description": "Safely deletes empty duplicate lists. Only removes lists that have no cards AND have another list with the same title that contains cards.", + "lost-cards": "Lost Cards", + "lost-cards-list": "Restored Items", + "restore-lost-cards-migration": "Restore Lost Cards", + "restore-lost-cards-migration-description": "Finds and restores cards and lists with missing swimlaneId or listId. Creates a 'Lost Cards' swimlane to make all lost items visible again.", + "restore-all-archived-migration": "Restore All Archived", + "restore-all-archived-migration-description": "Restores all archived swimlanes, lists, and cards. Automatically fixes any missing swimlaneId or listId to make items visible.", + "fix-missing-lists-migration": "Fix Missing Lists", + "fix-missing-lists-migration-description": "Detects and repairs missing or corrupted lists in the board structure.", + "fix-avatar-urls-migration": "Fix Avatar URLs", + "fix-avatar-urls-migration-description": "Updates avatar URLs for board members to use the correct storage backend and fixes broken avatar references.", + "fix-all-file-urls-migration": "Fix All File URLs", + "fix-all-file-urls-migration-description": "Updates all file attachment URLs on this board to use the correct storage backend and fixes broken file references.", + "migration-needed": "Migration Needed", + "migration-complete": "Complete", + "migration-running": "Running...", + "migration-successful": "Migration completed successfully", + "migration-failed": "Migration failed", + "migrations": "Migrations", + "migrations-admin-only": "Only board administrators can run migrations", + "migrations-description": "Run data integrity checks and repairs for this board. Each migration can be executed individually.", + "no-issues-found": "No issues found", + "run-migration": "Run Migration", + "run-comprehensive-migration-confirm": "This will perform a comprehensive migration to check and fix board data integrity. This may take a few moments. Continue?", + "run-delete-duplicate-empty-lists-migration-confirm": "This will first convert any shared lists to per-swimlane lists, then delete empty lists that have a duplicate list with the same title containing cards. Only truly redundant empty lists will be removed. Continue?", + "run-restore-lost-cards-migration-confirm": "This will create a 'Lost Cards' swimlane and restore all cards and lists with missing swimlaneId or listId. This only affects non-archived items. Continue?", + "run-restore-all-archived-migration-confirm": "This will restore ALL archived swimlanes, lists, and cards, making them visible again. Any items with missing IDs will be automatically fixed. This cannot be easily undone. Continue?", + "run-fix-missing-lists-migration-confirm": "This will detect and repair missing or corrupted lists in the board structure. Continue?", + "run-fix-avatar-urls-migration-confirm": "This will update avatar URLs for board members to use the correct storage backend. Continue?", + "run-fix-all-file-urls-migration-confirm": "This will update all file attachment URLs on this board to use the correct storage backend. Continue?", + "restore-lost-cards-nothing-to-restore": "No lost swimlanes, lists, or cards to restore", + + "migration-progress-title": "Board Migration in Progress", + "migration-progress-overall": "Overall Progress", + "migration-progress-current-step": "Current Step", + "migration-progress-status": "Status", + "migration-progress-details": "Details", + "migration-progress-note": "Please wait while we migrate your board to the latest structure...", + + "step-analyze-board-structure": "Analyze Board Structure", + "step-fix-orphaned-cards": "Fix Orphaned Cards", + "step-convert-shared-lists": "Convert Shared Lists", + "step-ensure-per-swimlane-lists": "Ensure Per-Swimlane Lists", + "step-validate-migration": "Validate Migration", + "step-fix-avatar-urls": "Fix Avatar URLs", + "step-fix-attachment-urls": "Fix Attachment URLs", + "step-analyze-lists": "Analyze Lists", + "step-create-missing-lists": "Create Missing Lists", + "step-update-cards": "Update Cards", + "step-finalize": "Finalize", + "step-delete-duplicate-empty-lists": "Delete Duplicate Empty Lists", + "step-ensure-lost-cards-swimlane": "Ensure Lost Cards Swimlane", + "step-restore-lists": "Restore Lists", + "step-restore-cards": "Restore Cards", + "step-restore-swimlanes": "Restore Swimlanes", + "step-fix-missing-ids": "Fix Missing IDs", + "step-scan-users": "Checking board member avatars", + "step-scan-files": "Checking board file attachments", + "step-fix-file-urls": "Fixing file URLs", "cleanup": "Cleanup", "cleanup-old-jobs": "Cleanup Old Jobs", "completed": "Completed", diff --git a/imports/i18n/data/az-LA.i18n.json b/imports/i18n/data/az-LA.i18n.json index 1ab8eeaa8..4804e6dff 100644 --- a/imports/i18n/data/az-LA.i18n.json +++ b/imports/i18n/data/az-LA.i18n.json @@ -1404,7 +1404,70 @@ "back-to-settings": "Back to Settings", "board-id": "Board ID", "board-migration": "Board Migration", + "board-migrations": "Board Migrations", "card-show-lists-on-minicard": "Show Lists on Minicard", + "comprehensive-board-migration": "Comprehensive Board Migration", + "comprehensive-board-migration-description": "Performs comprehensive checks and fixes for board data integrity, including list ordering, card positions, and swimlane structure.", + "delete-duplicate-empty-lists-migration": "Delete Duplicate Empty Lists", + "delete-duplicate-empty-lists-migration-description": "Safely deletes empty duplicate lists. Only removes lists that have no cards AND have another list with the same title that contains cards.", + "lost-cards": "Lost Cards", + "lost-cards-list": "Restored Items", + "restore-lost-cards-migration": "Restore Lost Cards", + "restore-lost-cards-migration-description": "Finds and restores cards and lists with missing swimlaneId or listId. Creates a 'Lost Cards' swimlane to make all lost items visible again.", + "restore-all-archived-migration": "Restore All Archived", + "restore-all-archived-migration-description": "Restores all archived swimlanes, lists, and cards. Automatically fixes any missing swimlaneId or listId to make items visible.", + "fix-missing-lists-migration": "Fix Missing Lists", + "fix-missing-lists-migration-description": "Detects and repairs missing or corrupted lists in the board structure.", + "fix-avatar-urls-migration": "Fix Avatar URLs", + "fix-avatar-urls-migration-description": "Updates avatar URLs for board members to use the correct storage backend and fixes broken avatar references.", + "fix-all-file-urls-migration": "Fix All File URLs", + "fix-all-file-urls-migration-description": "Updates all file attachment URLs on this board to use the correct storage backend and fixes broken file references.", + "migration-needed": "Migration Needed", + "migration-complete": "Complete", + "migration-running": "Running...", + "migration-successful": "Migration completed successfully", + "migration-failed": "Migration failed", + "migrations": "Migrations", + "migrations-admin-only": "Only board administrators can run migrations", + "migrations-description": "Run data integrity checks and repairs for this board. Each migration can be executed individually.", + "no-issues-found": "No issues found", + "run-migration": "Run Migration", + "run-comprehensive-migration-confirm": "This will perform a comprehensive migration to check and fix board data integrity. This may take a few moments. Continue?", + "run-delete-duplicate-empty-lists-migration-confirm": "This will first convert any shared lists to per-swimlane lists, then delete empty lists that have a duplicate list with the same title containing cards. Only truly redundant empty lists will be removed. Continue?", + "run-restore-lost-cards-migration-confirm": "This will create a 'Lost Cards' swimlane and restore all cards and lists with missing swimlaneId or listId. This only affects non-archived items. Continue?", + "run-restore-all-archived-migration-confirm": "This will restore ALL archived swimlanes, lists, and cards, making them visible again. Any items with missing IDs will be automatically fixed. This cannot be easily undone. Continue?", + "run-fix-missing-lists-migration-confirm": "This will detect and repair missing or corrupted lists in the board structure. Continue?", + "run-fix-avatar-urls-migration-confirm": "This will update avatar URLs for board members to use the correct storage backend. Continue?", + "run-fix-all-file-urls-migration-confirm": "This will update all file attachment URLs on this board to use the correct storage backend. Continue?", + "restore-lost-cards-nothing-to-restore": "No lost swimlanes, lists, or cards to restore", + + "migration-progress-title": "Board Migration in Progress", + "migration-progress-overall": "Overall Progress", + "migration-progress-current-step": "Current Step", + "migration-progress-status": "Status", + "migration-progress-details": "Details", + "migration-progress-note": "Please wait while we migrate your board to the latest structure...", + + "step-analyze-board-structure": "Analyze Board Structure", + "step-fix-orphaned-cards": "Fix Orphaned Cards", + "step-convert-shared-lists": "Convert Shared Lists", + "step-ensure-per-swimlane-lists": "Ensure Per-Swimlane Lists", + "step-validate-migration": "Validate Migration", + "step-fix-avatar-urls": "Fix Avatar URLs", + "step-fix-attachment-urls": "Fix Attachment URLs", + "step-analyze-lists": "Analyze Lists", + "step-create-missing-lists": "Create Missing Lists", + "step-update-cards": "Update Cards", + "step-finalize": "Finalize", + "step-delete-duplicate-empty-lists": "Delete Duplicate Empty Lists", + "step-ensure-lost-cards-swimlane": "Ensure Lost Cards Swimlane", + "step-restore-lists": "Restore Lists", + "step-restore-cards": "Restore Cards", + "step-restore-swimlanes": "Restore Swimlanes", + "step-fix-missing-ids": "Fix Missing IDs", + "step-scan-users": "Checking board member avatars", + "step-scan-files": "Checking board file attachments", + "step-fix-file-urls": "Fixing file URLs", "cleanup": "Cleanup", "cleanup-old-jobs": "Cleanup Old Jobs", "completed": "Completed", diff --git a/imports/i18n/data/az.i18n.json b/imports/i18n/data/az.i18n.json index 1ab8eeaa8..4804e6dff 100644 --- a/imports/i18n/data/az.i18n.json +++ b/imports/i18n/data/az.i18n.json @@ -1404,7 +1404,70 @@ "back-to-settings": "Back to Settings", "board-id": "Board ID", "board-migration": "Board Migration", + "board-migrations": "Board Migrations", "card-show-lists-on-minicard": "Show Lists on Minicard", + "comprehensive-board-migration": "Comprehensive Board Migration", + "comprehensive-board-migration-description": "Performs comprehensive checks and fixes for board data integrity, including list ordering, card positions, and swimlane structure.", + "delete-duplicate-empty-lists-migration": "Delete Duplicate Empty Lists", + "delete-duplicate-empty-lists-migration-description": "Safely deletes empty duplicate lists. Only removes lists that have no cards AND have another list with the same title that contains cards.", + "lost-cards": "Lost Cards", + "lost-cards-list": "Restored Items", + "restore-lost-cards-migration": "Restore Lost Cards", + "restore-lost-cards-migration-description": "Finds and restores cards and lists with missing swimlaneId or listId. Creates a 'Lost Cards' swimlane to make all lost items visible again.", + "restore-all-archived-migration": "Restore All Archived", + "restore-all-archived-migration-description": "Restores all archived swimlanes, lists, and cards. Automatically fixes any missing swimlaneId or listId to make items visible.", + "fix-missing-lists-migration": "Fix Missing Lists", + "fix-missing-lists-migration-description": "Detects and repairs missing or corrupted lists in the board structure.", + "fix-avatar-urls-migration": "Fix Avatar URLs", + "fix-avatar-urls-migration-description": "Updates avatar URLs for board members to use the correct storage backend and fixes broken avatar references.", + "fix-all-file-urls-migration": "Fix All File URLs", + "fix-all-file-urls-migration-description": "Updates all file attachment URLs on this board to use the correct storage backend and fixes broken file references.", + "migration-needed": "Migration Needed", + "migration-complete": "Complete", + "migration-running": "Running...", + "migration-successful": "Migration completed successfully", + "migration-failed": "Migration failed", + "migrations": "Migrations", + "migrations-admin-only": "Only board administrators can run migrations", + "migrations-description": "Run data integrity checks and repairs for this board. Each migration can be executed individually.", + "no-issues-found": "No issues found", + "run-migration": "Run Migration", + "run-comprehensive-migration-confirm": "This will perform a comprehensive migration to check and fix board data integrity. This may take a few moments. Continue?", + "run-delete-duplicate-empty-lists-migration-confirm": "This will first convert any shared lists to per-swimlane lists, then delete empty lists that have a duplicate list with the same title containing cards. Only truly redundant empty lists will be removed. Continue?", + "run-restore-lost-cards-migration-confirm": "This will create a 'Lost Cards' swimlane and restore all cards and lists with missing swimlaneId or listId. This only affects non-archived items. Continue?", + "run-restore-all-archived-migration-confirm": "This will restore ALL archived swimlanes, lists, and cards, making them visible again. Any items with missing IDs will be automatically fixed. This cannot be easily undone. Continue?", + "run-fix-missing-lists-migration-confirm": "This will detect and repair missing or corrupted lists in the board structure. Continue?", + "run-fix-avatar-urls-migration-confirm": "This will update avatar URLs for board members to use the correct storage backend. Continue?", + "run-fix-all-file-urls-migration-confirm": "This will update all file attachment URLs on this board to use the correct storage backend. Continue?", + "restore-lost-cards-nothing-to-restore": "No lost swimlanes, lists, or cards to restore", + + "migration-progress-title": "Board Migration in Progress", + "migration-progress-overall": "Overall Progress", + "migration-progress-current-step": "Current Step", + "migration-progress-status": "Status", + "migration-progress-details": "Details", + "migration-progress-note": "Please wait while we migrate your board to the latest structure...", + + "step-analyze-board-structure": "Analyze Board Structure", + "step-fix-orphaned-cards": "Fix Orphaned Cards", + "step-convert-shared-lists": "Convert Shared Lists", + "step-ensure-per-swimlane-lists": "Ensure Per-Swimlane Lists", + "step-validate-migration": "Validate Migration", + "step-fix-avatar-urls": "Fix Avatar URLs", + "step-fix-attachment-urls": "Fix Attachment URLs", + "step-analyze-lists": "Analyze Lists", + "step-create-missing-lists": "Create Missing Lists", + "step-update-cards": "Update Cards", + "step-finalize": "Finalize", + "step-delete-duplicate-empty-lists": "Delete Duplicate Empty Lists", + "step-ensure-lost-cards-swimlane": "Ensure Lost Cards Swimlane", + "step-restore-lists": "Restore Lists", + "step-restore-cards": "Restore Cards", + "step-restore-swimlanes": "Restore Swimlanes", + "step-fix-missing-ids": "Fix Missing IDs", + "step-scan-users": "Checking board member avatars", + "step-scan-files": "Checking board file attachments", + "step-fix-file-urls": "Fixing file URLs", "cleanup": "Cleanup", "cleanup-old-jobs": "Cleanup Old Jobs", "completed": "Completed", diff --git a/imports/i18n/data/bg.i18n.json b/imports/i18n/data/bg.i18n.json index 043015731..ae0151f95 100644 --- a/imports/i18n/data/bg.i18n.json +++ b/imports/i18n/data/bg.i18n.json @@ -1404,7 +1404,70 @@ "back-to-settings": "Back to Settings", "board-id": "Board ID", "board-migration": "Board Migration", + "board-migrations": "Board Migrations", "card-show-lists-on-minicard": "Show Lists on Minicard", + "comprehensive-board-migration": "Comprehensive Board Migration", + "comprehensive-board-migration-description": "Performs comprehensive checks and fixes for board data integrity, including list ordering, card positions, and swimlane structure.", + "delete-duplicate-empty-lists-migration": "Delete Duplicate Empty Lists", + "delete-duplicate-empty-lists-migration-description": "Safely deletes empty duplicate lists. Only removes lists that have no cards AND have another list with the same title that contains cards.", + "lost-cards": "Lost Cards", + "lost-cards-list": "Restored Items", + "restore-lost-cards-migration": "Restore Lost Cards", + "restore-lost-cards-migration-description": "Finds and restores cards and lists with missing swimlaneId or listId. Creates a 'Lost Cards' swimlane to make all lost items visible again.", + "restore-all-archived-migration": "Restore All Archived", + "restore-all-archived-migration-description": "Restores all archived swimlanes, lists, and cards. Automatically fixes any missing swimlaneId or listId to make items visible.", + "fix-missing-lists-migration": "Fix Missing Lists", + "fix-missing-lists-migration-description": "Detects and repairs missing or corrupted lists in the board structure.", + "fix-avatar-urls-migration": "Fix Avatar URLs", + "fix-avatar-urls-migration-description": "Updates avatar URLs for board members to use the correct storage backend and fixes broken avatar references.", + "fix-all-file-urls-migration": "Fix All File URLs", + "fix-all-file-urls-migration-description": "Updates all file attachment URLs on this board to use the correct storage backend and fixes broken file references.", + "migration-needed": "Migration Needed", + "migration-complete": "Complete", + "migration-running": "Running...", + "migration-successful": "Migration completed successfully", + "migration-failed": "Migration failed", + "migrations": "Migrations", + "migrations-admin-only": "Only board administrators can run migrations", + "migrations-description": "Run data integrity checks and repairs for this board. Each migration can be executed individually.", + "no-issues-found": "No issues found", + "run-migration": "Run Migration", + "run-comprehensive-migration-confirm": "This will perform a comprehensive migration to check and fix board data integrity. This may take a few moments. Continue?", + "run-delete-duplicate-empty-lists-migration-confirm": "This will first convert any shared lists to per-swimlane lists, then delete empty lists that have a duplicate list with the same title containing cards. Only truly redundant empty lists will be removed. Continue?", + "run-restore-lost-cards-migration-confirm": "This will create a 'Lost Cards' swimlane and restore all cards and lists with missing swimlaneId or listId. This only affects non-archived items. Continue?", + "run-restore-all-archived-migration-confirm": "This will restore ALL archived swimlanes, lists, and cards, making them visible again. Any items with missing IDs will be automatically fixed. This cannot be easily undone. Continue?", + "run-fix-missing-lists-migration-confirm": "This will detect and repair missing or corrupted lists in the board structure. Continue?", + "run-fix-avatar-urls-migration-confirm": "This will update avatar URLs for board members to use the correct storage backend. Continue?", + "run-fix-all-file-urls-migration-confirm": "This will update all file attachment URLs on this board to use the correct storage backend. Continue?", + "restore-lost-cards-nothing-to-restore": "No lost swimlanes, lists, or cards to restore", + + "migration-progress-title": "Board Migration in Progress", + "migration-progress-overall": "Overall Progress", + "migration-progress-current-step": "Current Step", + "migration-progress-status": "Състояние", + "migration-progress-details": "Details", + "migration-progress-note": "Please wait while we migrate your board to the latest structure...", + + "step-analyze-board-structure": "Analyze Board Structure", + "step-fix-orphaned-cards": "Fix Orphaned Cards", + "step-convert-shared-lists": "Convert Shared Lists", + "step-ensure-per-swimlane-lists": "Ensure Per-Swimlane Lists", + "step-validate-migration": "Validate Migration", + "step-fix-avatar-urls": "Fix Avatar URLs", + "step-fix-attachment-urls": "Fix Attachment URLs", + "step-analyze-lists": "Analyze Lists", + "step-create-missing-lists": "Create Missing Lists", + "step-update-cards": "Update Cards", + "step-finalize": "Finalize", + "step-delete-duplicate-empty-lists": "Delete Duplicate Empty Lists", + "step-ensure-lost-cards-swimlane": "Ensure Lost Cards Swimlane", + "step-restore-lists": "Restore Lists", + "step-restore-cards": "Restore Cards", + "step-restore-swimlanes": "Restore Swimlanes", + "step-fix-missing-ids": "Fix Missing IDs", + "step-scan-users": "Checking board member avatars", + "step-scan-files": "Checking board file attachments", + "step-fix-file-urls": "Fixing file URLs", "cleanup": "Cleanup", "cleanup-old-jobs": "Cleanup Old Jobs", "completed": "Завършено", diff --git a/imports/i18n/data/br.i18n.json b/imports/i18n/data/br.i18n.json index 899f9dbb2..0402c7064 100644 --- a/imports/i18n/data/br.i18n.json +++ b/imports/i18n/data/br.i18n.json @@ -1404,7 +1404,70 @@ "back-to-settings": "Back to Settings", "board-id": "Board ID", "board-migration": "Board Migration", + "board-migrations": "Board Migrations", "card-show-lists-on-minicard": "Show Lists on Minicard", + "comprehensive-board-migration": "Comprehensive Board Migration", + "comprehensive-board-migration-description": "Performs comprehensive checks and fixes for board data integrity, including list ordering, card positions, and swimlane structure.", + "delete-duplicate-empty-lists-migration": "Delete Duplicate Empty Lists", + "delete-duplicate-empty-lists-migration-description": "Safely deletes empty duplicate lists. Only removes lists that have no cards AND have another list with the same title that contains cards.", + "lost-cards": "Lost Cards", + "lost-cards-list": "Restored Items", + "restore-lost-cards-migration": "Restore Lost Cards", + "restore-lost-cards-migration-description": "Finds and restores cards and lists with missing swimlaneId or listId. Creates a 'Lost Cards' swimlane to make all lost items visible again.", + "restore-all-archived-migration": "Restore All Archived", + "restore-all-archived-migration-description": "Restores all archived swimlanes, lists, and cards. Automatically fixes any missing swimlaneId or listId to make items visible.", + "fix-missing-lists-migration": "Fix Missing Lists", + "fix-missing-lists-migration-description": "Detects and repairs missing or corrupted lists in the board structure.", + "fix-avatar-urls-migration": "Fix Avatar URLs", + "fix-avatar-urls-migration-description": "Updates avatar URLs for board members to use the correct storage backend and fixes broken avatar references.", + "fix-all-file-urls-migration": "Fix All File URLs", + "fix-all-file-urls-migration-description": "Updates all file attachment URLs on this board to use the correct storage backend and fixes broken file references.", + "migration-needed": "Migration Needed", + "migration-complete": "Complete", + "migration-running": "Running...", + "migration-successful": "Migration completed successfully", + "migration-failed": "Migration failed", + "migrations": "Migrations", + "migrations-admin-only": "Only board administrators can run migrations", + "migrations-description": "Run data integrity checks and repairs for this board. Each migration can be executed individually.", + "no-issues-found": "No issues found", + "run-migration": "Run Migration", + "run-comprehensive-migration-confirm": "This will perform a comprehensive migration to check and fix board data integrity. This may take a few moments. Continue?", + "run-delete-duplicate-empty-lists-migration-confirm": "This will first convert any shared lists to per-swimlane lists, then delete empty lists that have a duplicate list with the same title containing cards. Only truly redundant empty lists will be removed. Continue?", + "run-restore-lost-cards-migration-confirm": "This will create a 'Lost Cards' swimlane and restore all cards and lists with missing swimlaneId or listId. This only affects non-archived items. Continue?", + "run-restore-all-archived-migration-confirm": "This will restore ALL archived swimlanes, lists, and cards, making them visible again. Any items with missing IDs will be automatically fixed. This cannot be easily undone. Continue?", + "run-fix-missing-lists-migration-confirm": "This will detect and repair missing or corrupted lists in the board structure. Continue?", + "run-fix-avatar-urls-migration-confirm": "This will update avatar URLs for board members to use the correct storage backend. Continue?", + "run-fix-all-file-urls-migration-confirm": "This will update all file attachment URLs on this board to use the correct storage backend. Continue?", + "restore-lost-cards-nothing-to-restore": "No lost swimlanes, lists, or cards to restore", + + "migration-progress-title": "Board Migration in Progress", + "migration-progress-overall": "Overall Progress", + "migration-progress-current-step": "Current Step", + "migration-progress-status": "Status", + "migration-progress-details": "Details", + "migration-progress-note": "Please wait while we migrate your board to the latest structure...", + + "step-analyze-board-structure": "Analyze Board Structure", + "step-fix-orphaned-cards": "Fix Orphaned Cards", + "step-convert-shared-lists": "Convert Shared Lists", + "step-ensure-per-swimlane-lists": "Ensure Per-Swimlane Lists", + "step-validate-migration": "Validate Migration", + "step-fix-avatar-urls": "Fix Avatar URLs", + "step-fix-attachment-urls": "Fix Attachment URLs", + "step-analyze-lists": "Analyze Lists", + "step-create-missing-lists": "Create Missing Lists", + "step-update-cards": "Update Cards", + "step-finalize": "Finalize", + "step-delete-duplicate-empty-lists": "Delete Duplicate Empty Lists", + "step-ensure-lost-cards-swimlane": "Ensure Lost Cards Swimlane", + "step-restore-lists": "Restore Lists", + "step-restore-cards": "Restore Cards", + "step-restore-swimlanes": "Restore Swimlanes", + "step-fix-missing-ids": "Fix Missing IDs", + "step-scan-users": "Checking board member avatars", + "step-scan-files": "Checking board file attachments", + "step-fix-file-urls": "Fixing file URLs", "cleanup": "Cleanup", "cleanup-old-jobs": "Cleanup Old Jobs", "completed": "Completed", diff --git a/imports/i18n/data/ca.i18n.json b/imports/i18n/data/ca.i18n.json index 019a5eacd..dddf9afc4 100644 --- a/imports/i18n/data/ca.i18n.json +++ b/imports/i18n/data/ca.i18n.json @@ -1404,7 +1404,70 @@ "back-to-settings": "Back to Settings", "board-id": "Board ID", "board-migration": "Board Migration", + "board-migrations": "Board Migrations", "card-show-lists-on-minicard": "Show Lists on Minicard", + "comprehensive-board-migration": "Comprehensive Board Migration", + "comprehensive-board-migration-description": "Performs comprehensive checks and fixes for board data integrity, including list ordering, card positions, and swimlane structure.", + "delete-duplicate-empty-lists-migration": "Delete Duplicate Empty Lists", + "delete-duplicate-empty-lists-migration-description": "Safely deletes empty duplicate lists. Only removes lists that have no cards AND have another list with the same title that contains cards.", + "lost-cards": "Lost Cards", + "lost-cards-list": "Restored Items", + "restore-lost-cards-migration": "Restore Lost Cards", + "restore-lost-cards-migration-description": "Finds and restores cards and lists with missing swimlaneId or listId. Creates a 'Lost Cards' swimlane to make all lost items visible again.", + "restore-all-archived-migration": "Restore All Archived", + "restore-all-archived-migration-description": "Restores all archived swimlanes, lists, and cards. Automatically fixes any missing swimlaneId or listId to make items visible.", + "fix-missing-lists-migration": "Fix Missing Lists", + "fix-missing-lists-migration-description": "Detects and repairs missing or corrupted lists in the board structure.", + "fix-avatar-urls-migration": "Fix Avatar URLs", + "fix-avatar-urls-migration-description": "Updates avatar URLs for board members to use the correct storage backend and fixes broken avatar references.", + "fix-all-file-urls-migration": "Fix All File URLs", + "fix-all-file-urls-migration-description": "Updates all file attachment URLs on this board to use the correct storage backend and fixes broken file references.", + "migration-needed": "Migration Needed", + "migration-complete": "Complete", + "migration-running": "Running...", + "migration-successful": "Migration completed successfully", + "migration-failed": "Migration failed", + "migrations": "Migrations", + "migrations-admin-only": "Only board administrators can run migrations", + "migrations-description": "Run data integrity checks and repairs for this board. Each migration can be executed individually.", + "no-issues-found": "No issues found", + "run-migration": "Run Migration", + "run-comprehensive-migration-confirm": "This will perform a comprehensive migration to check and fix board data integrity. This may take a few moments. Continue?", + "run-delete-duplicate-empty-lists-migration-confirm": "This will first convert any shared lists to per-swimlane lists, then delete empty lists that have a duplicate list with the same title containing cards. Only truly redundant empty lists will be removed. Continue?", + "run-restore-lost-cards-migration-confirm": "This will create a 'Lost Cards' swimlane and restore all cards and lists with missing swimlaneId or listId. This only affects non-archived items. Continue?", + "run-restore-all-archived-migration-confirm": "This will restore ALL archived swimlanes, lists, and cards, making them visible again. Any items with missing IDs will be automatically fixed. This cannot be easily undone. Continue?", + "run-fix-missing-lists-migration-confirm": "This will detect and repair missing or corrupted lists in the board structure. Continue?", + "run-fix-avatar-urls-migration-confirm": "This will update avatar URLs for board members to use the correct storage backend. Continue?", + "run-fix-all-file-urls-migration-confirm": "This will update all file attachment URLs on this board to use the correct storage backend. Continue?", + "restore-lost-cards-nothing-to-restore": "No lost swimlanes, lists, or cards to restore", + + "migration-progress-title": "Board Migration in Progress", + "migration-progress-overall": "Overall Progress", + "migration-progress-current-step": "Current Step", + "migration-progress-status": "Estat", + "migration-progress-details": "Detalls", + "migration-progress-note": "Please wait while we migrate your board to the latest structure...", + + "step-analyze-board-structure": "Analyze Board Structure", + "step-fix-orphaned-cards": "Fix Orphaned Cards", + "step-convert-shared-lists": "Convert Shared Lists", + "step-ensure-per-swimlane-lists": "Ensure Per-Swimlane Lists", + "step-validate-migration": "Validate Migration", + "step-fix-avatar-urls": "Fix Avatar URLs", + "step-fix-attachment-urls": "Fix Attachment URLs", + "step-analyze-lists": "Analyze Lists", + "step-create-missing-lists": "Create Missing Lists", + "step-update-cards": "Update Cards", + "step-finalize": "Finalize", + "step-delete-duplicate-empty-lists": "Delete Duplicate Empty Lists", + "step-ensure-lost-cards-swimlane": "Ensure Lost Cards Swimlane", + "step-restore-lists": "Restore Lists", + "step-restore-cards": "Restore Cards", + "step-restore-swimlanes": "Restore Swimlanes", + "step-fix-missing-ids": "Fix Missing IDs", + "step-scan-users": "Checking board member avatars", + "step-scan-files": "Checking board file attachments", + "step-fix-file-urls": "Fixing file URLs", "cleanup": "Cleanup", "cleanup-old-jobs": "Cleanup Old Jobs", "completed": "Completat", diff --git a/imports/i18n/data/ca@valencia.i18n.json b/imports/i18n/data/ca@valencia.i18n.json index ab9a378b1..9af93bcf3 100644 --- a/imports/i18n/data/ca@valencia.i18n.json +++ b/imports/i18n/data/ca@valencia.i18n.json @@ -1404,7 +1404,70 @@ "back-to-settings": "Back to Settings", "board-id": "Board ID", "board-migration": "Board Migration", + "board-migrations": "Board Migrations", "card-show-lists-on-minicard": "Show Lists on Minicard", + "comprehensive-board-migration": "Comprehensive Board Migration", + "comprehensive-board-migration-description": "Performs comprehensive checks and fixes for board data integrity, including list ordering, card positions, and swimlane structure.", + "delete-duplicate-empty-lists-migration": "Delete Duplicate Empty Lists", + "delete-duplicate-empty-lists-migration-description": "Safely deletes empty duplicate lists. Only removes lists that have no cards AND have another list with the same title that contains cards.", + "lost-cards": "Lost Cards", + "lost-cards-list": "Restored Items", + "restore-lost-cards-migration": "Restore Lost Cards", + "restore-lost-cards-migration-description": "Finds and restores cards and lists with missing swimlaneId or listId. Creates a 'Lost Cards' swimlane to make all lost items visible again.", + "restore-all-archived-migration": "Restore All Archived", + "restore-all-archived-migration-description": "Restores all archived swimlanes, lists, and cards. Automatically fixes any missing swimlaneId or listId to make items visible.", + "fix-missing-lists-migration": "Fix Missing Lists", + "fix-missing-lists-migration-description": "Detects and repairs missing or corrupted lists in the board structure.", + "fix-avatar-urls-migration": "Fix Avatar URLs", + "fix-avatar-urls-migration-description": "Updates avatar URLs for board members to use the correct storage backend and fixes broken avatar references.", + "fix-all-file-urls-migration": "Fix All File URLs", + "fix-all-file-urls-migration-description": "Updates all file attachment URLs on this board to use the correct storage backend and fixes broken file references.", + "migration-needed": "Migration Needed", + "migration-complete": "Complete", + "migration-running": "Running...", + "migration-successful": "Migration completed successfully", + "migration-failed": "Migration failed", + "migrations": "Migrations", + "migrations-admin-only": "Only board administrators can run migrations", + "migrations-description": "Run data integrity checks and repairs for this board. Each migration can be executed individually.", + "no-issues-found": "No issues found", + "run-migration": "Run Migration", + "run-comprehensive-migration-confirm": "This will perform a comprehensive migration to check and fix board data integrity. This may take a few moments. Continue?", + "run-delete-duplicate-empty-lists-migration-confirm": "This will first convert any shared lists to per-swimlane lists, then delete empty lists that have a duplicate list with the same title containing cards. Only truly redundant empty lists will be removed. Continue?", + "run-restore-lost-cards-migration-confirm": "This will create a 'Lost Cards' swimlane and restore all cards and lists with missing swimlaneId or listId. This only affects non-archived items. Continue?", + "run-restore-all-archived-migration-confirm": "This will restore ALL archived swimlanes, lists, and cards, making them visible again. Any items with missing IDs will be automatically fixed. This cannot be easily undone. Continue?", + "run-fix-missing-lists-migration-confirm": "This will detect and repair missing or corrupted lists in the board structure. Continue?", + "run-fix-avatar-urls-migration-confirm": "This will update avatar URLs for board members to use the correct storage backend. Continue?", + "run-fix-all-file-urls-migration-confirm": "This will update all file attachment URLs on this board to use the correct storage backend. Continue?", + "restore-lost-cards-nothing-to-restore": "No lost swimlanes, lists, or cards to restore", + + "migration-progress-title": "Board Migration in Progress", + "migration-progress-overall": "Overall Progress", + "migration-progress-current-step": "Current Step", + "migration-progress-status": "Status", + "migration-progress-details": "Details", + "migration-progress-note": "Please wait while we migrate your board to the latest structure...", + + "step-analyze-board-structure": "Analyze Board Structure", + "step-fix-orphaned-cards": "Fix Orphaned Cards", + "step-convert-shared-lists": "Convert Shared Lists", + "step-ensure-per-swimlane-lists": "Ensure Per-Swimlane Lists", + "step-validate-migration": "Validate Migration", + "step-fix-avatar-urls": "Fix Avatar URLs", + "step-fix-attachment-urls": "Fix Attachment URLs", + "step-analyze-lists": "Analyze Lists", + "step-create-missing-lists": "Create Missing Lists", + "step-update-cards": "Update Cards", + "step-finalize": "Finalize", + "step-delete-duplicate-empty-lists": "Delete Duplicate Empty Lists", + "step-ensure-lost-cards-swimlane": "Ensure Lost Cards Swimlane", + "step-restore-lists": "Restore Lists", + "step-restore-cards": "Restore Cards", + "step-restore-swimlanes": "Restore Swimlanes", + "step-fix-missing-ids": "Fix Missing IDs", + "step-scan-users": "Checking board member avatars", + "step-scan-files": "Checking board file attachments", + "step-fix-file-urls": "Fixing file URLs", "cleanup": "Cleanup", "cleanup-old-jobs": "Cleanup Old Jobs", "completed": "Completed", diff --git a/imports/i18n/data/ca_ES.i18n.json b/imports/i18n/data/ca_ES.i18n.json index 7e2ba5ba7..9f4e10cb1 100644 --- a/imports/i18n/data/ca_ES.i18n.json +++ b/imports/i18n/data/ca_ES.i18n.json @@ -1404,7 +1404,70 @@ "back-to-settings": "Back to Settings", "board-id": "Board ID", "board-migration": "Board Migration", + "board-migrations": "Board Migrations", "card-show-lists-on-minicard": "Show Lists on Minicard", + "comprehensive-board-migration": "Comprehensive Board Migration", + "comprehensive-board-migration-description": "Performs comprehensive checks and fixes for board data integrity, including list ordering, card positions, and swimlane structure.", + "delete-duplicate-empty-lists-migration": "Delete Duplicate Empty Lists", + "delete-duplicate-empty-lists-migration-description": "Safely deletes empty duplicate lists. Only removes lists that have no cards AND have another list with the same title that contains cards.", + "lost-cards": "Lost Cards", + "lost-cards-list": "Restored Items", + "restore-lost-cards-migration": "Restore Lost Cards", + "restore-lost-cards-migration-description": "Finds and restores cards and lists with missing swimlaneId or listId. Creates a 'Lost Cards' swimlane to make all lost items visible again.", + "restore-all-archived-migration": "Restore All Archived", + "restore-all-archived-migration-description": "Restores all archived swimlanes, lists, and cards. Automatically fixes any missing swimlaneId or listId to make items visible.", + "fix-missing-lists-migration": "Fix Missing Lists", + "fix-missing-lists-migration-description": "Detects and repairs missing or corrupted lists in the board structure.", + "fix-avatar-urls-migration": "Fix Avatar URLs", + "fix-avatar-urls-migration-description": "Updates avatar URLs for board members to use the correct storage backend and fixes broken avatar references.", + "fix-all-file-urls-migration": "Fix All File URLs", + "fix-all-file-urls-migration-description": "Updates all file attachment URLs on this board to use the correct storage backend and fixes broken file references.", + "migration-needed": "Migration Needed", + "migration-complete": "Complete", + "migration-running": "Running...", + "migration-successful": "Migration completed successfully", + "migration-failed": "Migration failed", + "migrations": "Migrations", + "migrations-admin-only": "Only board administrators can run migrations", + "migrations-description": "Run data integrity checks and repairs for this board. Each migration can be executed individually.", + "no-issues-found": "No issues found", + "run-migration": "Run Migration", + "run-comprehensive-migration-confirm": "This will perform a comprehensive migration to check and fix board data integrity. This may take a few moments. Continue?", + "run-delete-duplicate-empty-lists-migration-confirm": "This will first convert any shared lists to per-swimlane lists, then delete empty lists that have a duplicate list with the same title containing cards. Only truly redundant empty lists will be removed. Continue?", + "run-restore-lost-cards-migration-confirm": "This will create a 'Lost Cards' swimlane and restore all cards and lists with missing swimlaneId or listId. This only affects non-archived items. Continue?", + "run-restore-all-archived-migration-confirm": "This will restore ALL archived swimlanes, lists, and cards, making them visible again. Any items with missing IDs will be automatically fixed. This cannot be easily undone. Continue?", + "run-fix-missing-lists-migration-confirm": "This will detect and repair missing or corrupted lists in the board structure. Continue?", + "run-fix-avatar-urls-migration-confirm": "This will update avatar URLs for board members to use the correct storage backend. Continue?", + "run-fix-all-file-urls-migration-confirm": "This will update all file attachment URLs on this board to use the correct storage backend. Continue?", + "restore-lost-cards-nothing-to-restore": "No lost swimlanes, lists, or cards to restore", + + "migration-progress-title": "Board Migration in Progress", + "migration-progress-overall": "Overall Progress", + "migration-progress-current-step": "Current Step", + "migration-progress-status": "Status", + "migration-progress-details": "Details", + "migration-progress-note": "Please wait while we migrate your board to the latest structure...", + + "step-analyze-board-structure": "Analyze Board Structure", + "step-fix-orphaned-cards": "Fix Orphaned Cards", + "step-convert-shared-lists": "Convert Shared Lists", + "step-ensure-per-swimlane-lists": "Ensure Per-Swimlane Lists", + "step-validate-migration": "Validate Migration", + "step-fix-avatar-urls": "Fix Avatar URLs", + "step-fix-attachment-urls": "Fix Attachment URLs", + "step-analyze-lists": "Analyze Lists", + "step-create-missing-lists": "Create Missing Lists", + "step-update-cards": "Update Cards", + "step-finalize": "Finalize", + "step-delete-duplicate-empty-lists": "Delete Duplicate Empty Lists", + "step-ensure-lost-cards-swimlane": "Ensure Lost Cards Swimlane", + "step-restore-lists": "Restore Lists", + "step-restore-cards": "Restore Cards", + "step-restore-swimlanes": "Restore Swimlanes", + "step-fix-missing-ids": "Fix Missing IDs", + "step-scan-users": "Checking board member avatars", + "step-scan-files": "Checking board file attachments", + "step-fix-file-urls": "Fixing file URLs", "cleanup": "Cleanup", "cleanup-old-jobs": "Cleanup Old Jobs", "completed": "Completed", diff --git a/imports/i18n/data/cmn.i18n.json b/imports/i18n/data/cmn.i18n.json index f99bd548e..5f18926f7 100644 --- a/imports/i18n/data/cmn.i18n.json +++ b/imports/i18n/data/cmn.i18n.json @@ -1404,7 +1404,70 @@ "back-to-settings": "Back to Settings", "board-id": "Board ID", "board-migration": "Board Migration", + "board-migrations": "Board Migrations", "card-show-lists-on-minicard": "Show Lists on Minicard", + "comprehensive-board-migration": "Comprehensive Board Migration", + "comprehensive-board-migration-description": "Performs comprehensive checks and fixes for board data integrity, including list ordering, card positions, and swimlane structure.", + "delete-duplicate-empty-lists-migration": "Delete Duplicate Empty Lists", + "delete-duplicate-empty-lists-migration-description": "Safely deletes empty duplicate lists. Only removes lists that have no cards AND have another list with the same title that contains cards.", + "lost-cards": "Lost Cards", + "lost-cards-list": "Restored Items", + "restore-lost-cards-migration": "Restore Lost Cards", + "restore-lost-cards-migration-description": "Finds and restores cards and lists with missing swimlaneId or listId. Creates a 'Lost Cards' swimlane to make all lost items visible again.", + "restore-all-archived-migration": "Restore All Archived", + "restore-all-archived-migration-description": "Restores all archived swimlanes, lists, and cards. Automatically fixes any missing swimlaneId or listId to make items visible.", + "fix-missing-lists-migration": "Fix Missing Lists", + "fix-missing-lists-migration-description": "Detects and repairs missing or corrupted lists in the board structure.", + "fix-avatar-urls-migration": "Fix Avatar URLs", + "fix-avatar-urls-migration-description": "Updates avatar URLs for board members to use the correct storage backend and fixes broken avatar references.", + "fix-all-file-urls-migration": "Fix All File URLs", + "fix-all-file-urls-migration-description": "Updates all file attachment URLs on this board to use the correct storage backend and fixes broken file references.", + "migration-needed": "Migration Needed", + "migration-complete": "Complete", + "migration-running": "Running...", + "migration-successful": "Migration completed successfully", + "migration-failed": "Migration failed", + "migrations": "Migrations", + "migrations-admin-only": "Only board administrators can run migrations", + "migrations-description": "Run data integrity checks and repairs for this board. Each migration can be executed individually.", + "no-issues-found": "No issues found", + "run-migration": "Run Migration", + "run-comprehensive-migration-confirm": "This will perform a comprehensive migration to check and fix board data integrity. This may take a few moments. Continue?", + "run-delete-duplicate-empty-lists-migration-confirm": "This will first convert any shared lists to per-swimlane lists, then delete empty lists that have a duplicate list with the same title containing cards. Only truly redundant empty lists will be removed. Continue?", + "run-restore-lost-cards-migration-confirm": "This will create a 'Lost Cards' swimlane and restore all cards and lists with missing swimlaneId or listId. This only affects non-archived items. Continue?", + "run-restore-all-archived-migration-confirm": "This will restore ALL archived swimlanes, lists, and cards, making them visible again. Any items with missing IDs will be automatically fixed. This cannot be easily undone. Continue?", + "run-fix-missing-lists-migration-confirm": "This will detect and repair missing or corrupted lists in the board structure. Continue?", + "run-fix-avatar-urls-migration-confirm": "This will update avatar URLs for board members to use the correct storage backend. Continue?", + "run-fix-all-file-urls-migration-confirm": "This will update all file attachment URLs on this board to use the correct storage backend. Continue?", + "restore-lost-cards-nothing-to-restore": "No lost swimlanes, lists, or cards to restore", + + "migration-progress-title": "Board Migration in Progress", + "migration-progress-overall": "Overall Progress", + "migration-progress-current-step": "Current Step", + "migration-progress-status": "Status", + "migration-progress-details": "Details", + "migration-progress-note": "Please wait while we migrate your board to the latest structure...", + + "step-analyze-board-structure": "Analyze Board Structure", + "step-fix-orphaned-cards": "Fix Orphaned Cards", + "step-convert-shared-lists": "Convert Shared Lists", + "step-ensure-per-swimlane-lists": "Ensure Per-Swimlane Lists", + "step-validate-migration": "Validate Migration", + "step-fix-avatar-urls": "Fix Avatar URLs", + "step-fix-attachment-urls": "Fix Attachment URLs", + "step-analyze-lists": "Analyze Lists", + "step-create-missing-lists": "Create Missing Lists", + "step-update-cards": "Update Cards", + "step-finalize": "Finalize", + "step-delete-duplicate-empty-lists": "Delete Duplicate Empty Lists", + "step-ensure-lost-cards-swimlane": "Ensure Lost Cards Swimlane", + "step-restore-lists": "Restore Lists", + "step-restore-cards": "Restore Cards", + "step-restore-swimlanes": "Restore Swimlanes", + "step-fix-missing-ids": "Fix Missing IDs", + "step-scan-users": "Checking board member avatars", + "step-scan-files": "Checking board file attachments", + "step-fix-file-urls": "Fixing file URLs", "cleanup": "Cleanup", "cleanup-old-jobs": "Cleanup Old Jobs", "completed": "Completed", diff --git a/imports/i18n/data/cs-CZ.i18n.json b/imports/i18n/data/cs-CZ.i18n.json index 0e915cedf..6a5ce87d7 100644 --- a/imports/i18n/data/cs-CZ.i18n.json +++ b/imports/i18n/data/cs-CZ.i18n.json @@ -1404,7 +1404,70 @@ "back-to-settings": "Back to Settings", "board-id": "Board ID", "board-migration": "Board Migration", + "board-migrations": "Board Migrations", "card-show-lists-on-minicard": "Show Lists on Minicard", + "comprehensive-board-migration": "Comprehensive Board Migration", + "comprehensive-board-migration-description": "Performs comprehensive checks and fixes for board data integrity, including list ordering, card positions, and swimlane structure.", + "delete-duplicate-empty-lists-migration": "Delete Duplicate Empty Lists", + "delete-duplicate-empty-lists-migration-description": "Safely deletes empty duplicate lists. Only removes lists that have no cards AND have another list with the same title that contains cards.", + "lost-cards": "Lost Cards", + "lost-cards-list": "Restored Items", + "restore-lost-cards-migration": "Restore Lost Cards", + "restore-lost-cards-migration-description": "Finds and restores cards and lists with missing swimlaneId or listId. Creates a 'Lost Cards' swimlane to make all lost items visible again.", + "restore-all-archived-migration": "Restore All Archived", + "restore-all-archived-migration-description": "Restores all archived swimlanes, lists, and cards. Automatically fixes any missing swimlaneId or listId to make items visible.", + "fix-missing-lists-migration": "Fix Missing Lists", + "fix-missing-lists-migration-description": "Detects and repairs missing or corrupted lists in the board structure.", + "fix-avatar-urls-migration": "Fix Avatar URLs", + "fix-avatar-urls-migration-description": "Updates avatar URLs for board members to use the correct storage backend and fixes broken avatar references.", + "fix-all-file-urls-migration": "Fix All File URLs", + "fix-all-file-urls-migration-description": "Updates all file attachment URLs on this board to use the correct storage backend and fixes broken file references.", + "migration-needed": "Migration Needed", + "migration-complete": "Complete", + "migration-running": "Running...", + "migration-successful": "Migration completed successfully", + "migration-failed": "Migration failed", + "migrations": "Migrations", + "migrations-admin-only": "Only board administrators can run migrations", + "migrations-description": "Run data integrity checks and repairs for this board. Each migration can be executed individually.", + "no-issues-found": "No issues found", + "run-migration": "Run Migration", + "run-comprehensive-migration-confirm": "This will perform a comprehensive migration to check and fix board data integrity. This may take a few moments. Continue?", + "run-delete-duplicate-empty-lists-migration-confirm": "This will first convert any shared lists to per-swimlane lists, then delete empty lists that have a duplicate list with the same title containing cards. Only truly redundant empty lists will be removed. Continue?", + "run-restore-lost-cards-migration-confirm": "This will create a 'Lost Cards' swimlane and restore all cards and lists with missing swimlaneId or listId. This only affects non-archived items. Continue?", + "run-restore-all-archived-migration-confirm": "This will restore ALL archived swimlanes, lists, and cards, making them visible again. Any items with missing IDs will be automatically fixed. This cannot be easily undone. Continue?", + "run-fix-missing-lists-migration-confirm": "This will detect and repair missing or corrupted lists in the board structure. Continue?", + "run-fix-avatar-urls-migration-confirm": "This will update avatar URLs for board members to use the correct storage backend. Continue?", + "run-fix-all-file-urls-migration-confirm": "This will update all file attachment URLs on this board to use the correct storage backend. Continue?", + "restore-lost-cards-nothing-to-restore": "No lost swimlanes, lists, or cards to restore", + + "migration-progress-title": "Board Migration in Progress", + "migration-progress-overall": "Overall Progress", + "migration-progress-current-step": "Current Step", + "migration-progress-status": "Stav", + "migration-progress-details": "Details", + "migration-progress-note": "Please wait while we migrate your board to the latest structure...", + + "step-analyze-board-structure": "Analyze Board Structure", + "step-fix-orphaned-cards": "Fix Orphaned Cards", + "step-convert-shared-lists": "Convert Shared Lists", + "step-ensure-per-swimlane-lists": "Ensure Per-Swimlane Lists", + "step-validate-migration": "Validate Migration", + "step-fix-avatar-urls": "Fix Avatar URLs", + "step-fix-attachment-urls": "Fix Attachment URLs", + "step-analyze-lists": "Analyze Lists", + "step-create-missing-lists": "Create Missing Lists", + "step-update-cards": "Update Cards", + "step-finalize": "Finalize", + "step-delete-duplicate-empty-lists": "Delete Duplicate Empty Lists", + "step-ensure-lost-cards-swimlane": "Ensure Lost Cards Swimlane", + "step-restore-lists": "Restore Lists", + "step-restore-cards": "Restore Cards", + "step-restore-swimlanes": "Restore Swimlanes", + "step-fix-missing-ids": "Fix Missing IDs", + "step-scan-users": "Checking board member avatars", + "step-scan-files": "Checking board file attachments", + "step-fix-file-urls": "Fixing file URLs", "cleanup": "Cleanup", "cleanup-old-jobs": "Cleanup Old Jobs", "completed": "Dokončeno", diff --git a/imports/i18n/data/cs.i18n.json b/imports/i18n/data/cs.i18n.json index 58abbc60d..60f7db17b 100644 --- a/imports/i18n/data/cs.i18n.json +++ b/imports/i18n/data/cs.i18n.json @@ -1404,7 +1404,70 @@ "back-to-settings": "Back to Settings", "board-id": "Board ID", "board-migration": "Board Migration", + "board-migrations": "Board Migrations", "card-show-lists-on-minicard": "Show Lists on Minicard", + "comprehensive-board-migration": "Comprehensive Board Migration", + "comprehensive-board-migration-description": "Performs comprehensive checks and fixes for board data integrity, including list ordering, card positions, and swimlane structure.", + "delete-duplicate-empty-lists-migration": "Delete Duplicate Empty Lists", + "delete-duplicate-empty-lists-migration-description": "Safely deletes empty duplicate lists. Only removes lists that have no cards AND have another list with the same title that contains cards.", + "lost-cards": "Lost Cards", + "lost-cards-list": "Restored Items", + "restore-lost-cards-migration": "Restore Lost Cards", + "restore-lost-cards-migration-description": "Finds and restores cards and lists with missing swimlaneId or listId. Creates a 'Lost Cards' swimlane to make all lost items visible again.", + "restore-all-archived-migration": "Restore All Archived", + "restore-all-archived-migration-description": "Restores all archived swimlanes, lists, and cards. Automatically fixes any missing swimlaneId or listId to make items visible.", + "fix-missing-lists-migration": "Fix Missing Lists", + "fix-missing-lists-migration-description": "Detects and repairs missing or corrupted lists in the board structure.", + "fix-avatar-urls-migration": "Fix Avatar URLs", + "fix-avatar-urls-migration-description": "Updates avatar URLs for board members to use the correct storage backend and fixes broken avatar references.", + "fix-all-file-urls-migration": "Fix All File URLs", + "fix-all-file-urls-migration-description": "Updates all file attachment URLs on this board to use the correct storage backend and fixes broken file references.", + "migration-needed": "Migration Needed", + "migration-complete": "Complete", + "migration-running": "Running...", + "migration-successful": "Migration completed successfully", + "migration-failed": "Migration failed", + "migrations": "Migrations", + "migrations-admin-only": "Only board administrators can run migrations", + "migrations-description": "Run data integrity checks and repairs for this board. Each migration can be executed individually.", + "no-issues-found": "No issues found", + "run-migration": "Run Migration", + "run-comprehensive-migration-confirm": "This will perform a comprehensive migration to check and fix board data integrity. This may take a few moments. Continue?", + "run-delete-duplicate-empty-lists-migration-confirm": "This will first convert any shared lists to per-swimlane lists, then delete empty lists that have a duplicate list with the same title containing cards. Only truly redundant empty lists will be removed. Continue?", + "run-restore-lost-cards-migration-confirm": "This will create a 'Lost Cards' swimlane and restore all cards and lists with missing swimlaneId or listId. This only affects non-archived items. Continue?", + "run-restore-all-archived-migration-confirm": "This will restore ALL archived swimlanes, lists, and cards, making them visible again. Any items with missing IDs will be automatically fixed. This cannot be easily undone. Continue?", + "run-fix-missing-lists-migration-confirm": "This will detect and repair missing or corrupted lists in the board structure. Continue?", + "run-fix-avatar-urls-migration-confirm": "This will update avatar URLs for board members to use the correct storage backend. Continue?", + "run-fix-all-file-urls-migration-confirm": "This will update all file attachment URLs on this board to use the correct storage backend. Continue?", + "restore-lost-cards-nothing-to-restore": "No lost swimlanes, lists, or cards to restore", + + "migration-progress-title": "Board Migration in Progress", + "migration-progress-overall": "Overall Progress", + "migration-progress-current-step": "Current Step", + "migration-progress-status": "Stav", + "migration-progress-details": "Podrobnosti", + "migration-progress-note": "Please wait while we migrate your board to the latest structure...", + + "step-analyze-board-structure": "Analyze Board Structure", + "step-fix-orphaned-cards": "Fix Orphaned Cards", + "step-convert-shared-lists": "Convert Shared Lists", + "step-ensure-per-swimlane-lists": "Ensure Per-Swimlane Lists", + "step-validate-migration": "Validate Migration", + "step-fix-avatar-urls": "Fix Avatar URLs", + "step-fix-attachment-urls": "Fix Attachment URLs", + "step-analyze-lists": "Analyze Lists", + "step-create-missing-lists": "Create Missing Lists", + "step-update-cards": "Update Cards", + "step-finalize": "Finalize", + "step-delete-duplicate-empty-lists": "Delete Duplicate Empty Lists", + "step-ensure-lost-cards-swimlane": "Ensure Lost Cards Swimlane", + "step-restore-lists": "Restore Lists", + "step-restore-cards": "Restore Cards", + "step-restore-swimlanes": "Restore Swimlanes", + "step-fix-missing-ids": "Fix Missing IDs", + "step-scan-users": "Checking board member avatars", + "step-scan-files": "Checking board file attachments", + "step-fix-file-urls": "Fixing file URLs", "cleanup": "Cleanup", "cleanup-old-jobs": "Cleanup Old Jobs", "completed": "Dokončeno", diff --git a/imports/i18n/data/cy-GB.i18n.json b/imports/i18n/data/cy-GB.i18n.json index 1ab8eeaa8..4804e6dff 100644 --- a/imports/i18n/data/cy-GB.i18n.json +++ b/imports/i18n/data/cy-GB.i18n.json @@ -1404,7 +1404,70 @@ "back-to-settings": "Back to Settings", "board-id": "Board ID", "board-migration": "Board Migration", + "board-migrations": "Board Migrations", "card-show-lists-on-minicard": "Show Lists on Minicard", + "comprehensive-board-migration": "Comprehensive Board Migration", + "comprehensive-board-migration-description": "Performs comprehensive checks and fixes for board data integrity, including list ordering, card positions, and swimlane structure.", + "delete-duplicate-empty-lists-migration": "Delete Duplicate Empty Lists", + "delete-duplicate-empty-lists-migration-description": "Safely deletes empty duplicate lists. Only removes lists that have no cards AND have another list with the same title that contains cards.", + "lost-cards": "Lost Cards", + "lost-cards-list": "Restored Items", + "restore-lost-cards-migration": "Restore Lost Cards", + "restore-lost-cards-migration-description": "Finds and restores cards and lists with missing swimlaneId or listId. Creates a 'Lost Cards' swimlane to make all lost items visible again.", + "restore-all-archived-migration": "Restore All Archived", + "restore-all-archived-migration-description": "Restores all archived swimlanes, lists, and cards. Automatically fixes any missing swimlaneId or listId to make items visible.", + "fix-missing-lists-migration": "Fix Missing Lists", + "fix-missing-lists-migration-description": "Detects and repairs missing or corrupted lists in the board structure.", + "fix-avatar-urls-migration": "Fix Avatar URLs", + "fix-avatar-urls-migration-description": "Updates avatar URLs for board members to use the correct storage backend and fixes broken avatar references.", + "fix-all-file-urls-migration": "Fix All File URLs", + "fix-all-file-urls-migration-description": "Updates all file attachment URLs on this board to use the correct storage backend and fixes broken file references.", + "migration-needed": "Migration Needed", + "migration-complete": "Complete", + "migration-running": "Running...", + "migration-successful": "Migration completed successfully", + "migration-failed": "Migration failed", + "migrations": "Migrations", + "migrations-admin-only": "Only board administrators can run migrations", + "migrations-description": "Run data integrity checks and repairs for this board. Each migration can be executed individually.", + "no-issues-found": "No issues found", + "run-migration": "Run Migration", + "run-comprehensive-migration-confirm": "This will perform a comprehensive migration to check and fix board data integrity. This may take a few moments. Continue?", + "run-delete-duplicate-empty-lists-migration-confirm": "This will first convert any shared lists to per-swimlane lists, then delete empty lists that have a duplicate list with the same title containing cards. Only truly redundant empty lists will be removed. Continue?", + "run-restore-lost-cards-migration-confirm": "This will create a 'Lost Cards' swimlane and restore all cards and lists with missing swimlaneId or listId. This only affects non-archived items. Continue?", + "run-restore-all-archived-migration-confirm": "This will restore ALL archived swimlanes, lists, and cards, making them visible again. Any items with missing IDs will be automatically fixed. This cannot be easily undone. Continue?", + "run-fix-missing-lists-migration-confirm": "This will detect and repair missing or corrupted lists in the board structure. Continue?", + "run-fix-avatar-urls-migration-confirm": "This will update avatar URLs for board members to use the correct storage backend. Continue?", + "run-fix-all-file-urls-migration-confirm": "This will update all file attachment URLs on this board to use the correct storage backend. Continue?", + "restore-lost-cards-nothing-to-restore": "No lost swimlanes, lists, or cards to restore", + + "migration-progress-title": "Board Migration in Progress", + "migration-progress-overall": "Overall Progress", + "migration-progress-current-step": "Current Step", + "migration-progress-status": "Status", + "migration-progress-details": "Details", + "migration-progress-note": "Please wait while we migrate your board to the latest structure...", + + "step-analyze-board-structure": "Analyze Board Structure", + "step-fix-orphaned-cards": "Fix Orphaned Cards", + "step-convert-shared-lists": "Convert Shared Lists", + "step-ensure-per-swimlane-lists": "Ensure Per-Swimlane Lists", + "step-validate-migration": "Validate Migration", + "step-fix-avatar-urls": "Fix Avatar URLs", + "step-fix-attachment-urls": "Fix Attachment URLs", + "step-analyze-lists": "Analyze Lists", + "step-create-missing-lists": "Create Missing Lists", + "step-update-cards": "Update Cards", + "step-finalize": "Finalize", + "step-delete-duplicate-empty-lists": "Delete Duplicate Empty Lists", + "step-ensure-lost-cards-swimlane": "Ensure Lost Cards Swimlane", + "step-restore-lists": "Restore Lists", + "step-restore-cards": "Restore Cards", + "step-restore-swimlanes": "Restore Swimlanes", + "step-fix-missing-ids": "Fix Missing IDs", + "step-scan-users": "Checking board member avatars", + "step-scan-files": "Checking board file attachments", + "step-fix-file-urls": "Fixing file URLs", "cleanup": "Cleanup", "cleanup-old-jobs": "Cleanup Old Jobs", "completed": "Completed", diff --git a/imports/i18n/data/cy.i18n.json b/imports/i18n/data/cy.i18n.json index 1ab8eeaa8..4804e6dff 100644 --- a/imports/i18n/data/cy.i18n.json +++ b/imports/i18n/data/cy.i18n.json @@ -1404,7 +1404,70 @@ "back-to-settings": "Back to Settings", "board-id": "Board ID", "board-migration": "Board Migration", + "board-migrations": "Board Migrations", "card-show-lists-on-minicard": "Show Lists on Minicard", + "comprehensive-board-migration": "Comprehensive Board Migration", + "comprehensive-board-migration-description": "Performs comprehensive checks and fixes for board data integrity, including list ordering, card positions, and swimlane structure.", + "delete-duplicate-empty-lists-migration": "Delete Duplicate Empty Lists", + "delete-duplicate-empty-lists-migration-description": "Safely deletes empty duplicate lists. Only removes lists that have no cards AND have another list with the same title that contains cards.", + "lost-cards": "Lost Cards", + "lost-cards-list": "Restored Items", + "restore-lost-cards-migration": "Restore Lost Cards", + "restore-lost-cards-migration-description": "Finds and restores cards and lists with missing swimlaneId or listId. Creates a 'Lost Cards' swimlane to make all lost items visible again.", + "restore-all-archived-migration": "Restore All Archived", + "restore-all-archived-migration-description": "Restores all archived swimlanes, lists, and cards. Automatically fixes any missing swimlaneId or listId to make items visible.", + "fix-missing-lists-migration": "Fix Missing Lists", + "fix-missing-lists-migration-description": "Detects and repairs missing or corrupted lists in the board structure.", + "fix-avatar-urls-migration": "Fix Avatar URLs", + "fix-avatar-urls-migration-description": "Updates avatar URLs for board members to use the correct storage backend and fixes broken avatar references.", + "fix-all-file-urls-migration": "Fix All File URLs", + "fix-all-file-urls-migration-description": "Updates all file attachment URLs on this board to use the correct storage backend and fixes broken file references.", + "migration-needed": "Migration Needed", + "migration-complete": "Complete", + "migration-running": "Running...", + "migration-successful": "Migration completed successfully", + "migration-failed": "Migration failed", + "migrations": "Migrations", + "migrations-admin-only": "Only board administrators can run migrations", + "migrations-description": "Run data integrity checks and repairs for this board. Each migration can be executed individually.", + "no-issues-found": "No issues found", + "run-migration": "Run Migration", + "run-comprehensive-migration-confirm": "This will perform a comprehensive migration to check and fix board data integrity. This may take a few moments. Continue?", + "run-delete-duplicate-empty-lists-migration-confirm": "This will first convert any shared lists to per-swimlane lists, then delete empty lists that have a duplicate list with the same title containing cards. Only truly redundant empty lists will be removed. Continue?", + "run-restore-lost-cards-migration-confirm": "This will create a 'Lost Cards' swimlane and restore all cards and lists with missing swimlaneId or listId. This only affects non-archived items. Continue?", + "run-restore-all-archived-migration-confirm": "This will restore ALL archived swimlanes, lists, and cards, making them visible again. Any items with missing IDs will be automatically fixed. This cannot be easily undone. Continue?", + "run-fix-missing-lists-migration-confirm": "This will detect and repair missing or corrupted lists in the board structure. Continue?", + "run-fix-avatar-urls-migration-confirm": "This will update avatar URLs for board members to use the correct storage backend. Continue?", + "run-fix-all-file-urls-migration-confirm": "This will update all file attachment URLs on this board to use the correct storage backend. Continue?", + "restore-lost-cards-nothing-to-restore": "No lost swimlanes, lists, or cards to restore", + + "migration-progress-title": "Board Migration in Progress", + "migration-progress-overall": "Overall Progress", + "migration-progress-current-step": "Current Step", + "migration-progress-status": "Status", + "migration-progress-details": "Details", + "migration-progress-note": "Please wait while we migrate your board to the latest structure...", + + "step-analyze-board-structure": "Analyze Board Structure", + "step-fix-orphaned-cards": "Fix Orphaned Cards", + "step-convert-shared-lists": "Convert Shared Lists", + "step-ensure-per-swimlane-lists": "Ensure Per-Swimlane Lists", + "step-validate-migration": "Validate Migration", + "step-fix-avatar-urls": "Fix Avatar URLs", + "step-fix-attachment-urls": "Fix Attachment URLs", + "step-analyze-lists": "Analyze Lists", + "step-create-missing-lists": "Create Missing Lists", + "step-update-cards": "Update Cards", + "step-finalize": "Finalize", + "step-delete-duplicate-empty-lists": "Delete Duplicate Empty Lists", + "step-ensure-lost-cards-swimlane": "Ensure Lost Cards Swimlane", + "step-restore-lists": "Restore Lists", + "step-restore-cards": "Restore Cards", + "step-restore-swimlanes": "Restore Swimlanes", + "step-fix-missing-ids": "Fix Missing IDs", + "step-scan-users": "Checking board member avatars", + "step-scan-files": "Checking board file attachments", + "step-fix-file-urls": "Fixing file URLs", "cleanup": "Cleanup", "cleanup-old-jobs": "Cleanup Old Jobs", "completed": "Completed", diff --git a/imports/i18n/data/da.i18n.json b/imports/i18n/data/da.i18n.json index bfe07883d..d6a3e0e15 100644 --- a/imports/i18n/data/da.i18n.json +++ b/imports/i18n/data/da.i18n.json @@ -1404,7 +1404,70 @@ "back-to-settings": "Back to Settings", "board-id": "Board ID", "board-migration": "Board Migration", + "board-migrations": "Board Migrations", "card-show-lists-on-minicard": "Show Lists on Minicard", + "comprehensive-board-migration": "Comprehensive Board Migration", + "comprehensive-board-migration-description": "Performs comprehensive checks and fixes for board data integrity, including list ordering, card positions, and swimlane structure.", + "delete-duplicate-empty-lists-migration": "Delete Duplicate Empty Lists", + "delete-duplicate-empty-lists-migration-description": "Safely deletes empty duplicate lists. Only removes lists that have no cards AND have another list with the same title that contains cards.", + "lost-cards": "Lost Cards", + "lost-cards-list": "Restored Items", + "restore-lost-cards-migration": "Restore Lost Cards", + "restore-lost-cards-migration-description": "Finds and restores cards and lists with missing swimlaneId or listId. Creates a 'Lost Cards' swimlane to make all lost items visible again.", + "restore-all-archived-migration": "Restore All Archived", + "restore-all-archived-migration-description": "Restores all archived swimlanes, lists, and cards. Automatically fixes any missing swimlaneId or listId to make items visible.", + "fix-missing-lists-migration": "Fix Missing Lists", + "fix-missing-lists-migration-description": "Detects and repairs missing or corrupted lists in the board structure.", + "fix-avatar-urls-migration": "Fix Avatar URLs", + "fix-avatar-urls-migration-description": "Updates avatar URLs for board members to use the correct storage backend and fixes broken avatar references.", + "fix-all-file-urls-migration": "Fix All File URLs", + "fix-all-file-urls-migration-description": "Updates all file attachment URLs on this board to use the correct storage backend and fixes broken file references.", + "migration-needed": "Migration Needed", + "migration-complete": "Complete", + "migration-running": "Running...", + "migration-successful": "Migration completed successfully", + "migration-failed": "Migration failed", + "migrations": "Migrations", + "migrations-admin-only": "Only board administrators can run migrations", + "migrations-description": "Run data integrity checks and repairs for this board. Each migration can be executed individually.", + "no-issues-found": "No issues found", + "run-migration": "Run Migration", + "run-comprehensive-migration-confirm": "This will perform a comprehensive migration to check and fix board data integrity. This may take a few moments. Continue?", + "run-delete-duplicate-empty-lists-migration-confirm": "This will first convert any shared lists to per-swimlane lists, then delete empty lists that have a duplicate list with the same title containing cards. Only truly redundant empty lists will be removed. Continue?", + "run-restore-lost-cards-migration-confirm": "This will create a 'Lost Cards' swimlane and restore all cards and lists with missing swimlaneId or listId. This only affects non-archived items. Continue?", + "run-restore-all-archived-migration-confirm": "This will restore ALL archived swimlanes, lists, and cards, making them visible again. Any items with missing IDs will be automatically fixed. This cannot be easily undone. Continue?", + "run-fix-missing-lists-migration-confirm": "This will detect and repair missing or corrupted lists in the board structure. Continue?", + "run-fix-avatar-urls-migration-confirm": "This will update avatar URLs for board members to use the correct storage backend. Continue?", + "run-fix-all-file-urls-migration-confirm": "This will update all file attachment URLs on this board to use the correct storage backend. Continue?", + "restore-lost-cards-nothing-to-restore": "No lost swimlanes, lists, or cards to restore", + + "migration-progress-title": "Board Migration in Progress", + "migration-progress-overall": "Overall Progress", + "migration-progress-current-step": "Current Step", + "migration-progress-status": "Status", + "migration-progress-details": "Details", + "migration-progress-note": "Please wait while we migrate your board to the latest structure...", + + "step-analyze-board-structure": "Analyze Board Structure", + "step-fix-orphaned-cards": "Fix Orphaned Cards", + "step-convert-shared-lists": "Convert Shared Lists", + "step-ensure-per-swimlane-lists": "Ensure Per-Swimlane Lists", + "step-validate-migration": "Validate Migration", + "step-fix-avatar-urls": "Fix Avatar URLs", + "step-fix-attachment-urls": "Fix Attachment URLs", + "step-analyze-lists": "Analyze Lists", + "step-create-missing-lists": "Create Missing Lists", + "step-update-cards": "Update Cards", + "step-finalize": "Finalize", + "step-delete-duplicate-empty-lists": "Delete Duplicate Empty Lists", + "step-ensure-lost-cards-swimlane": "Ensure Lost Cards Swimlane", + "step-restore-lists": "Restore Lists", + "step-restore-cards": "Restore Cards", + "step-restore-swimlanes": "Restore Swimlanes", + "step-fix-missing-ids": "Fix Missing IDs", + "step-scan-users": "Checking board member avatars", + "step-scan-files": "Checking board file attachments", + "step-fix-file-urls": "Fixing file URLs", "cleanup": "Cleanup", "cleanup-old-jobs": "Cleanup Old Jobs", "completed": "Fuldført", diff --git a/imports/i18n/data/de-AT.i18n.json b/imports/i18n/data/de-AT.i18n.json index 625ad5655..03616ba4c 100644 --- a/imports/i18n/data/de-AT.i18n.json +++ b/imports/i18n/data/de-AT.i18n.json @@ -1404,7 +1404,70 @@ "back-to-settings": "Back to Settings", "board-id": "Board ID", "board-migration": "Board Migration", + "board-migrations": "Board Migrations", "card-show-lists-on-minicard": "Show Lists on Minicard", + "comprehensive-board-migration": "Comprehensive Board Migration", + "comprehensive-board-migration-description": "Performs comprehensive checks and fixes for board data integrity, including list ordering, card positions, and swimlane structure.", + "delete-duplicate-empty-lists-migration": "Delete Duplicate Empty Lists", + "delete-duplicate-empty-lists-migration-description": "Safely deletes empty duplicate lists. Only removes lists that have no cards AND have another list with the same title that contains cards.", + "lost-cards": "Lost Cards", + "lost-cards-list": "Restored Items", + "restore-lost-cards-migration": "Restore Lost Cards", + "restore-lost-cards-migration-description": "Finds and restores cards and lists with missing swimlaneId or listId. Creates a 'Lost Cards' swimlane to make all lost items visible again.", + "restore-all-archived-migration": "Restore All Archived", + "restore-all-archived-migration-description": "Restores all archived swimlanes, lists, and cards. Automatically fixes any missing swimlaneId or listId to make items visible.", + "fix-missing-lists-migration": "Fix Missing Lists", + "fix-missing-lists-migration-description": "Detects and repairs missing or corrupted lists in the board structure.", + "fix-avatar-urls-migration": "Fix Avatar URLs", + "fix-avatar-urls-migration-description": "Updates avatar URLs for board members to use the correct storage backend and fixes broken avatar references.", + "fix-all-file-urls-migration": "Fix All File URLs", + "fix-all-file-urls-migration-description": "Updates all file attachment URLs on this board to use the correct storage backend and fixes broken file references.", + "migration-needed": "Migration Needed", + "migration-complete": "Complete", + "migration-running": "Running...", + "migration-successful": "Migration completed successfully", + "migration-failed": "Migration failed", + "migrations": "Migrations", + "migrations-admin-only": "Only board administrators can run migrations", + "migrations-description": "Run data integrity checks and repairs for this board. Each migration can be executed individually.", + "no-issues-found": "No issues found", + "run-migration": "Run Migration", + "run-comprehensive-migration-confirm": "This will perform a comprehensive migration to check and fix board data integrity. This may take a few moments. Continue?", + "run-delete-duplicate-empty-lists-migration-confirm": "This will first convert any shared lists to per-swimlane lists, then delete empty lists that have a duplicate list with the same title containing cards. Only truly redundant empty lists will be removed. Continue?", + "run-restore-lost-cards-migration-confirm": "This will create a 'Lost Cards' swimlane and restore all cards and lists with missing swimlaneId or listId. This only affects non-archived items. Continue?", + "run-restore-all-archived-migration-confirm": "This will restore ALL archived swimlanes, lists, and cards, making them visible again. Any items with missing IDs will be automatically fixed. This cannot be easily undone. Continue?", + "run-fix-missing-lists-migration-confirm": "This will detect and repair missing or corrupted lists in the board structure. Continue?", + "run-fix-avatar-urls-migration-confirm": "This will update avatar URLs for board members to use the correct storage backend. Continue?", + "run-fix-all-file-urls-migration-confirm": "This will update all file attachment URLs on this board to use the correct storage backend. Continue?", + "restore-lost-cards-nothing-to-restore": "No lost swimlanes, lists, or cards to restore", + + "migration-progress-title": "Board Migration in Progress", + "migration-progress-overall": "Overall Progress", + "migration-progress-current-step": "Current Step", + "migration-progress-status": "Status", + "migration-progress-details": "Details", + "migration-progress-note": "Please wait while we migrate your board to the latest structure...", + + "step-analyze-board-structure": "Analyze Board Structure", + "step-fix-orphaned-cards": "Fix Orphaned Cards", + "step-convert-shared-lists": "Convert Shared Lists", + "step-ensure-per-swimlane-lists": "Ensure Per-Swimlane Lists", + "step-validate-migration": "Validate Migration", + "step-fix-avatar-urls": "Fix Avatar URLs", + "step-fix-attachment-urls": "Fix Attachment URLs", + "step-analyze-lists": "Analyze Lists", + "step-create-missing-lists": "Create Missing Lists", + "step-update-cards": "Update Cards", + "step-finalize": "Finalize", + "step-delete-duplicate-empty-lists": "Delete Duplicate Empty Lists", + "step-ensure-lost-cards-swimlane": "Ensure Lost Cards Swimlane", + "step-restore-lists": "Restore Lists", + "step-restore-cards": "Restore Cards", + "step-restore-swimlanes": "Restore Swimlanes", + "step-fix-missing-ids": "Fix Missing IDs", + "step-scan-users": "Checking board member avatars", + "step-scan-files": "Checking board file attachments", + "step-fix-file-urls": "Fixing file URLs", "cleanup": "Cleanup", "cleanup-old-jobs": "Cleanup Old Jobs", "completed": "abgeschlossen", diff --git a/imports/i18n/data/de-CH.i18n.json b/imports/i18n/data/de-CH.i18n.json index 4c9682382..16a06886c 100644 --- a/imports/i18n/data/de-CH.i18n.json +++ b/imports/i18n/data/de-CH.i18n.json @@ -1404,7 +1404,70 @@ "back-to-settings": "Back to Settings", "board-id": "Board ID", "board-migration": "Board Migration", + "board-migrations": "Board Migrations", "card-show-lists-on-minicard": "Show Lists on Minicard", + "comprehensive-board-migration": "Comprehensive Board Migration", + "comprehensive-board-migration-description": "Performs comprehensive checks and fixes for board data integrity, including list ordering, card positions, and swimlane structure.", + "delete-duplicate-empty-lists-migration": "Delete Duplicate Empty Lists", + "delete-duplicate-empty-lists-migration-description": "Safely deletes empty duplicate lists. Only removes lists that have no cards AND have another list with the same title that contains cards.", + "lost-cards": "Lost Cards", + "lost-cards-list": "Restored Items", + "restore-lost-cards-migration": "Restore Lost Cards", + "restore-lost-cards-migration-description": "Finds and restores cards and lists with missing swimlaneId or listId. Creates a 'Lost Cards' swimlane to make all lost items visible again.", + "restore-all-archived-migration": "Restore All Archived", + "restore-all-archived-migration-description": "Restores all archived swimlanes, lists, and cards. Automatically fixes any missing swimlaneId or listId to make items visible.", + "fix-missing-lists-migration": "Fix Missing Lists", + "fix-missing-lists-migration-description": "Detects and repairs missing or corrupted lists in the board structure.", + "fix-avatar-urls-migration": "Fix Avatar URLs", + "fix-avatar-urls-migration-description": "Updates avatar URLs for board members to use the correct storage backend and fixes broken avatar references.", + "fix-all-file-urls-migration": "Fix All File URLs", + "fix-all-file-urls-migration-description": "Updates all file attachment URLs on this board to use the correct storage backend and fixes broken file references.", + "migration-needed": "Migration Needed", + "migration-complete": "Complete", + "migration-running": "Running...", + "migration-successful": "Migration completed successfully", + "migration-failed": "Migration failed", + "migrations": "Migrations", + "migrations-admin-only": "Only board administrators can run migrations", + "migrations-description": "Run data integrity checks and repairs for this board. Each migration can be executed individually.", + "no-issues-found": "No issues found", + "run-migration": "Run Migration", + "run-comprehensive-migration-confirm": "This will perform a comprehensive migration to check and fix board data integrity. This may take a few moments. Continue?", + "run-delete-duplicate-empty-lists-migration-confirm": "This will first convert any shared lists to per-swimlane lists, then delete empty lists that have a duplicate list with the same title containing cards. Only truly redundant empty lists will be removed. Continue?", + "run-restore-lost-cards-migration-confirm": "This will create a 'Lost Cards' swimlane and restore all cards and lists with missing swimlaneId or listId. This only affects non-archived items. Continue?", + "run-restore-all-archived-migration-confirm": "This will restore ALL archived swimlanes, lists, and cards, making them visible again. Any items with missing IDs will be automatically fixed. This cannot be easily undone. Continue?", + "run-fix-missing-lists-migration-confirm": "This will detect and repair missing or corrupted lists in the board structure. Continue?", + "run-fix-avatar-urls-migration-confirm": "This will update avatar URLs for board members to use the correct storage backend. Continue?", + "run-fix-all-file-urls-migration-confirm": "This will update all file attachment URLs on this board to use the correct storage backend. Continue?", + "restore-lost-cards-nothing-to-restore": "No lost swimlanes, lists, or cards to restore", + + "migration-progress-title": "Board Migration in Progress", + "migration-progress-overall": "Overall Progress", + "migration-progress-current-step": "Current Step", + "migration-progress-status": "Status", + "migration-progress-details": "Details", + "migration-progress-note": "Please wait while we migrate your board to the latest structure...", + + "step-analyze-board-structure": "Analyze Board Structure", + "step-fix-orphaned-cards": "Fix Orphaned Cards", + "step-convert-shared-lists": "Convert Shared Lists", + "step-ensure-per-swimlane-lists": "Ensure Per-Swimlane Lists", + "step-validate-migration": "Validate Migration", + "step-fix-avatar-urls": "Fix Avatar URLs", + "step-fix-attachment-urls": "Fix Attachment URLs", + "step-analyze-lists": "Analyze Lists", + "step-create-missing-lists": "Create Missing Lists", + "step-update-cards": "Update Cards", + "step-finalize": "Finalize", + "step-delete-duplicate-empty-lists": "Delete Duplicate Empty Lists", + "step-ensure-lost-cards-swimlane": "Ensure Lost Cards Swimlane", + "step-restore-lists": "Restore Lists", + "step-restore-cards": "Restore Cards", + "step-restore-swimlanes": "Restore Swimlanes", + "step-fix-missing-ids": "Fix Missing IDs", + "step-scan-users": "Checking board member avatars", + "step-scan-files": "Checking board file attachments", + "step-fix-file-urls": "Fixing file URLs", "cleanup": "Cleanup", "cleanup-old-jobs": "Cleanup Old Jobs", "completed": "abgeschlossen", diff --git a/imports/i18n/data/de.i18n.json b/imports/i18n/data/de.i18n.json index f138b8d6c..0ca4846d8 100644 --- a/imports/i18n/data/de.i18n.json +++ b/imports/i18n/data/de.i18n.json @@ -1404,7 +1404,70 @@ "back-to-settings": "Back to Settings", "board-id": "Board ID", "board-migration": "Board Migration", + "board-migrations": "Board Migrations", "card-show-lists-on-minicard": "Show Lists on Minicard", + "comprehensive-board-migration": "Comprehensive Board Migration", + "comprehensive-board-migration-description": "Performs comprehensive checks and fixes for board data integrity, including list ordering, card positions, and swimlane structure.", + "delete-duplicate-empty-lists-migration": "Delete Duplicate Empty Lists", + "delete-duplicate-empty-lists-migration-description": "Safely deletes empty duplicate lists. Only removes lists that have no cards AND have another list with the same title that contains cards.", + "lost-cards": "Lost Cards", + "lost-cards-list": "Restored Items", + "restore-lost-cards-migration": "Restore Lost Cards", + "restore-lost-cards-migration-description": "Finds and restores cards and lists with missing swimlaneId or listId. Creates a 'Lost Cards' swimlane to make all lost items visible again.", + "restore-all-archived-migration": "Restore All Archived", + "restore-all-archived-migration-description": "Restores all archived swimlanes, lists, and cards. Automatically fixes any missing swimlaneId or listId to make items visible.", + "fix-missing-lists-migration": "Fix Missing Lists", + "fix-missing-lists-migration-description": "Detects and repairs missing or corrupted lists in the board structure.", + "fix-avatar-urls-migration": "Fix Avatar URLs", + "fix-avatar-urls-migration-description": "Updates avatar URLs for board members to use the correct storage backend and fixes broken avatar references.", + "fix-all-file-urls-migration": "Fix All File URLs", + "fix-all-file-urls-migration-description": "Updates all file attachment URLs on this board to use the correct storage backend and fixes broken file references.", + "migration-needed": "Migration Needed", + "migration-complete": "Abgeschlossen", + "migration-running": "Running...", + "migration-successful": "Migration completed successfully", + "migration-failed": "Migration failed", + "migrations": "Migrations", + "migrations-admin-only": "Only board administrators can run migrations", + "migrations-description": "Run data integrity checks and repairs for this board. Each migration can be executed individually.", + "no-issues-found": "No issues found", + "run-migration": "Run Migration", + "run-comprehensive-migration-confirm": "This will perform a comprehensive migration to check and fix board data integrity. This may take a few moments. Continue?", + "run-delete-duplicate-empty-lists-migration-confirm": "This will first convert any shared lists to per-swimlane lists, then delete empty lists that have a duplicate list with the same title containing cards. Only truly redundant empty lists will be removed. Continue?", + "run-restore-lost-cards-migration-confirm": "This will create a 'Lost Cards' swimlane and restore all cards and lists with missing swimlaneId or listId. This only affects non-archived items. Continue?", + "run-restore-all-archived-migration-confirm": "This will restore ALL archived swimlanes, lists, and cards, making them visible again. Any items with missing IDs will be automatically fixed. This cannot be easily undone. Continue?", + "run-fix-missing-lists-migration-confirm": "This will detect and repair missing or corrupted lists in the board structure. Continue?", + "run-fix-avatar-urls-migration-confirm": "This will update avatar URLs for board members to use the correct storage backend. Continue?", + "run-fix-all-file-urls-migration-confirm": "This will update all file attachment URLs on this board to use the correct storage backend. Continue?", + "restore-lost-cards-nothing-to-restore": "No lost swimlanes, lists, or cards to restore", + + "migration-progress-title": "Board Migration in Progress", + "migration-progress-overall": "Overall Progress", + "migration-progress-current-step": "Current Step", + "migration-progress-status": "Status", + "migration-progress-details": "Details", + "migration-progress-note": "Please wait while we migrate your board to the latest structure...", + + "step-analyze-board-structure": "Analyze Board Structure", + "step-fix-orphaned-cards": "Fix Orphaned Cards", + "step-convert-shared-lists": "Convert Shared Lists", + "step-ensure-per-swimlane-lists": "Ensure Per-Swimlane Lists", + "step-validate-migration": "Validate Migration", + "step-fix-avatar-urls": "Fix Avatar URLs", + "step-fix-attachment-urls": "Fix Attachment URLs", + "step-analyze-lists": "Analyze Lists", + "step-create-missing-lists": "Create Missing Lists", + "step-update-cards": "Update Cards", + "step-finalize": "Finalize", + "step-delete-duplicate-empty-lists": "Delete Duplicate Empty Lists", + "step-ensure-lost-cards-swimlane": "Ensure Lost Cards Swimlane", + "step-restore-lists": "Restore Lists", + "step-restore-cards": "Restore Cards", + "step-restore-swimlanes": "Restore Swimlanes", + "step-fix-missing-ids": "Fix Missing IDs", + "step-scan-users": "Checking board member avatars", + "step-scan-files": "Checking board file attachments", + "step-fix-file-urls": "Fixing file URLs", "cleanup": "Cleanup", "cleanup-old-jobs": "Cleanup Old Jobs", "completed": "abgeschlossen", diff --git a/imports/i18n/data/de_DE.i18n.json b/imports/i18n/data/de_DE.i18n.json index 621bbaae0..cc1b326ae 100644 --- a/imports/i18n/data/de_DE.i18n.json +++ b/imports/i18n/data/de_DE.i18n.json @@ -1404,7 +1404,70 @@ "back-to-settings": "Zurück zu den Einstellungen", "board-id": "Brett ID", "board-migration": "Brettmigration", + "board-migrations": "Board Migrations", "card-show-lists-on-minicard": "Zeige Listen auf der Minikarte", + "comprehensive-board-migration": "Comprehensive Board Migration", + "comprehensive-board-migration-description": "Performs comprehensive checks and fixes for board data integrity, including list ordering, card positions, and swimlane structure.", + "delete-duplicate-empty-lists-migration": "Delete Duplicate Empty Lists", + "delete-duplicate-empty-lists-migration-description": "Safely deletes empty duplicate lists. Only removes lists that have no cards AND have another list with the same title that contains cards.", + "lost-cards": "Lost Cards", + "lost-cards-list": "Restored Items", + "restore-lost-cards-migration": "Restore Lost Cards", + "restore-lost-cards-migration-description": "Finds and restores cards and lists with missing swimlaneId or listId. Creates a 'Lost Cards' swimlane to make all lost items visible again.", + "restore-all-archived-migration": "Restore All Archived", + "restore-all-archived-migration-description": "Restores all archived swimlanes, lists, and cards. Automatically fixes any missing swimlaneId or listId to make items visible.", + "fix-missing-lists-migration": "Fix Missing Lists", + "fix-missing-lists-migration-description": "Detects and repairs missing or corrupted lists in the board structure.", + "fix-avatar-urls-migration": "Fix Avatar URLs", + "fix-avatar-urls-migration-description": "Updates avatar URLs for board members to use the correct storage backend and fixes broken avatar references.", + "fix-all-file-urls-migration": "Fix All File URLs", + "fix-all-file-urls-migration-description": "Updates all file attachment URLs on this board to use the correct storage backend and fixes broken file references.", + "migration-needed": "Migration Needed", + "migration-complete": "Vollständig", + "migration-running": "Running...", + "migration-successful": "Migration completed successfully", + "migration-failed": "Migration failed", + "migrations": "Migrations", + "migrations-admin-only": "Only board administrators can run migrations", + "migrations-description": "Run data integrity checks and repairs for this board. Each migration can be executed individually.", + "no-issues-found": "No issues found", + "run-migration": "Run Migration", + "run-comprehensive-migration-confirm": "This will perform a comprehensive migration to check and fix board data integrity. This may take a few moments. Continue?", + "run-delete-duplicate-empty-lists-migration-confirm": "This will first convert any shared lists to per-swimlane lists, then delete empty lists that have a duplicate list with the same title containing cards. Only truly redundant empty lists will be removed. Continue?", + "run-restore-lost-cards-migration-confirm": "This will create a 'Lost Cards' swimlane and restore all cards and lists with missing swimlaneId or listId. This only affects non-archived items. Continue?", + "run-restore-all-archived-migration-confirm": "This will restore ALL archived swimlanes, lists, and cards, making them visible again. Any items with missing IDs will be automatically fixed. This cannot be easily undone. Continue?", + "run-fix-missing-lists-migration-confirm": "This will detect and repair missing or corrupted lists in the board structure. Continue?", + "run-fix-avatar-urls-migration-confirm": "This will update avatar URLs for board members to use the correct storage backend. Continue?", + "run-fix-all-file-urls-migration-confirm": "This will update all file attachment URLs on this board to use the correct storage backend. Continue?", + "restore-lost-cards-nothing-to-restore": "No lost swimlanes, lists, or cards to restore", + + "migration-progress-title": "Board Migration in Progress", + "migration-progress-overall": "Gesamtfortschritt", + "migration-progress-current-step": "Current Step", + "migration-progress-status": "Status", + "migration-progress-details": "Details", + "migration-progress-note": "Please wait while we migrate your board to the latest structure...", + + "step-analyze-board-structure": "Analyze Board Structure", + "step-fix-orphaned-cards": "Fix Orphaned Cards", + "step-convert-shared-lists": "Convert Shared Lists", + "step-ensure-per-swimlane-lists": "Ensure Per-Swimlane Lists", + "step-validate-migration": "Validate Migration", + "step-fix-avatar-urls": "Fix Avatar URLs", + "step-fix-attachment-urls": "Fix Attachment URLs", + "step-analyze-lists": "Analyze Lists", + "step-create-missing-lists": "Create Missing Lists", + "step-update-cards": "Update Cards", + "step-finalize": "Finalize", + "step-delete-duplicate-empty-lists": "Delete Duplicate Empty Lists", + "step-ensure-lost-cards-swimlane": "Ensure Lost Cards Swimlane", + "step-restore-lists": "Restore Lists", + "step-restore-cards": "Restore Cards", + "step-restore-swimlanes": "Restore Swimlanes", + "step-fix-missing-ids": "Fix Missing IDs", + "step-scan-users": "Checking board member avatars", + "step-scan-files": "Checking board file attachments", + "step-fix-file-urls": "Fixing file URLs", "cleanup": "Aufräumen", "cleanup-old-jobs": "Alte Aufgaben aufräumen", "completed": "abgeschlossen", diff --git a/imports/i18n/data/el-GR.i18n.json b/imports/i18n/data/el-GR.i18n.json index 48fcea53a..17367197b 100644 --- a/imports/i18n/data/el-GR.i18n.json +++ b/imports/i18n/data/el-GR.i18n.json @@ -1404,7 +1404,70 @@ "back-to-settings": "Back to Settings", "board-id": "Board ID", "board-migration": "Board Migration", + "board-migrations": "Board Migrations", "card-show-lists-on-minicard": "Show Lists on Minicard", + "comprehensive-board-migration": "Comprehensive Board Migration", + "comprehensive-board-migration-description": "Performs comprehensive checks and fixes for board data integrity, including list ordering, card positions, and swimlane structure.", + "delete-duplicate-empty-lists-migration": "Delete Duplicate Empty Lists", + "delete-duplicate-empty-lists-migration-description": "Safely deletes empty duplicate lists. Only removes lists that have no cards AND have another list with the same title that contains cards.", + "lost-cards": "Lost Cards", + "lost-cards-list": "Restored Items", + "restore-lost-cards-migration": "Restore Lost Cards", + "restore-lost-cards-migration-description": "Finds and restores cards and lists with missing swimlaneId or listId. Creates a 'Lost Cards' swimlane to make all lost items visible again.", + "restore-all-archived-migration": "Restore All Archived", + "restore-all-archived-migration-description": "Restores all archived swimlanes, lists, and cards. Automatically fixes any missing swimlaneId or listId to make items visible.", + "fix-missing-lists-migration": "Fix Missing Lists", + "fix-missing-lists-migration-description": "Detects and repairs missing or corrupted lists in the board structure.", + "fix-avatar-urls-migration": "Fix Avatar URLs", + "fix-avatar-urls-migration-description": "Updates avatar URLs for board members to use the correct storage backend and fixes broken avatar references.", + "fix-all-file-urls-migration": "Fix All File URLs", + "fix-all-file-urls-migration-description": "Updates all file attachment URLs on this board to use the correct storage backend and fixes broken file references.", + "migration-needed": "Migration Needed", + "migration-complete": "Complete", + "migration-running": "Running...", + "migration-successful": "Migration completed successfully", + "migration-failed": "Migration failed", + "migrations": "Migrations", + "migrations-admin-only": "Only board administrators can run migrations", + "migrations-description": "Run data integrity checks and repairs for this board. Each migration can be executed individually.", + "no-issues-found": "No issues found", + "run-migration": "Run Migration", + "run-comprehensive-migration-confirm": "This will perform a comprehensive migration to check and fix board data integrity. This may take a few moments. Continue?", + "run-delete-duplicate-empty-lists-migration-confirm": "This will first convert any shared lists to per-swimlane lists, then delete empty lists that have a duplicate list with the same title containing cards. Only truly redundant empty lists will be removed. Continue?", + "run-restore-lost-cards-migration-confirm": "This will create a 'Lost Cards' swimlane and restore all cards and lists with missing swimlaneId or listId. This only affects non-archived items. Continue?", + "run-restore-all-archived-migration-confirm": "This will restore ALL archived swimlanes, lists, and cards, making them visible again. Any items with missing IDs will be automatically fixed. This cannot be easily undone. Continue?", + "run-fix-missing-lists-migration-confirm": "This will detect and repair missing or corrupted lists in the board structure. Continue?", + "run-fix-avatar-urls-migration-confirm": "This will update avatar URLs for board members to use the correct storage backend. Continue?", + "run-fix-all-file-urls-migration-confirm": "This will update all file attachment URLs on this board to use the correct storage backend. Continue?", + "restore-lost-cards-nothing-to-restore": "No lost swimlanes, lists, or cards to restore", + + "migration-progress-title": "Board Migration in Progress", + "migration-progress-overall": "Overall Progress", + "migration-progress-current-step": "Current Step", + "migration-progress-status": "Status", + "migration-progress-details": "Details", + "migration-progress-note": "Please wait while we migrate your board to the latest structure...", + + "step-analyze-board-structure": "Analyze Board Structure", + "step-fix-orphaned-cards": "Fix Orphaned Cards", + "step-convert-shared-lists": "Convert Shared Lists", + "step-ensure-per-swimlane-lists": "Ensure Per-Swimlane Lists", + "step-validate-migration": "Validate Migration", + "step-fix-avatar-urls": "Fix Avatar URLs", + "step-fix-attachment-urls": "Fix Attachment URLs", + "step-analyze-lists": "Analyze Lists", + "step-create-missing-lists": "Create Missing Lists", + "step-update-cards": "Update Cards", + "step-finalize": "Finalize", + "step-delete-duplicate-empty-lists": "Delete Duplicate Empty Lists", + "step-ensure-lost-cards-swimlane": "Ensure Lost Cards Swimlane", + "step-restore-lists": "Restore Lists", + "step-restore-cards": "Restore Cards", + "step-restore-swimlanes": "Restore Swimlanes", + "step-fix-missing-ids": "Fix Missing IDs", + "step-scan-users": "Checking board member avatars", + "step-scan-files": "Checking board file attachments", + "step-fix-file-urls": "Fixing file URLs", "cleanup": "Cleanup", "cleanup-old-jobs": "Cleanup Old Jobs", "completed": "Completed", diff --git a/imports/i18n/data/el.i18n.json b/imports/i18n/data/el.i18n.json index c4da6359d..3bd3440e6 100644 --- a/imports/i18n/data/el.i18n.json +++ b/imports/i18n/data/el.i18n.json @@ -1404,7 +1404,70 @@ "back-to-settings": "Back to Settings", "board-id": "Board ID", "board-migration": "Board Migration", + "board-migrations": "Board Migrations", "card-show-lists-on-minicard": "Show Lists on Minicard", + "comprehensive-board-migration": "Comprehensive Board Migration", + "comprehensive-board-migration-description": "Performs comprehensive checks and fixes for board data integrity, including list ordering, card positions, and swimlane structure.", + "delete-duplicate-empty-lists-migration": "Delete Duplicate Empty Lists", + "delete-duplicate-empty-lists-migration-description": "Safely deletes empty duplicate lists. Only removes lists that have no cards AND have another list with the same title that contains cards.", + "lost-cards": "Lost Cards", + "lost-cards-list": "Restored Items", + "restore-lost-cards-migration": "Restore Lost Cards", + "restore-lost-cards-migration-description": "Finds and restores cards and lists with missing swimlaneId or listId. Creates a 'Lost Cards' swimlane to make all lost items visible again.", + "restore-all-archived-migration": "Restore All Archived", + "restore-all-archived-migration-description": "Restores all archived swimlanes, lists, and cards. Automatically fixes any missing swimlaneId or listId to make items visible.", + "fix-missing-lists-migration": "Fix Missing Lists", + "fix-missing-lists-migration-description": "Detects and repairs missing or corrupted lists in the board structure.", + "fix-avatar-urls-migration": "Fix Avatar URLs", + "fix-avatar-urls-migration-description": "Updates avatar URLs for board members to use the correct storage backend and fixes broken avatar references.", + "fix-all-file-urls-migration": "Fix All File URLs", + "fix-all-file-urls-migration-description": "Updates all file attachment URLs on this board to use the correct storage backend and fixes broken file references.", + "migration-needed": "Migration Needed", + "migration-complete": "Complete", + "migration-running": "Running...", + "migration-successful": "Migration completed successfully", + "migration-failed": "Migration failed", + "migrations": "Migrations", + "migrations-admin-only": "Only board administrators can run migrations", + "migrations-description": "Run data integrity checks and repairs for this board. Each migration can be executed individually.", + "no-issues-found": "No issues found", + "run-migration": "Run Migration", + "run-comprehensive-migration-confirm": "This will perform a comprehensive migration to check and fix board data integrity. This may take a few moments. Continue?", + "run-delete-duplicate-empty-lists-migration-confirm": "This will first convert any shared lists to per-swimlane lists, then delete empty lists that have a duplicate list with the same title containing cards. Only truly redundant empty lists will be removed. Continue?", + "run-restore-lost-cards-migration-confirm": "This will create a 'Lost Cards' swimlane and restore all cards and lists with missing swimlaneId or listId. This only affects non-archived items. Continue?", + "run-restore-all-archived-migration-confirm": "This will restore ALL archived swimlanes, lists, and cards, making them visible again. Any items with missing IDs will be automatically fixed. This cannot be easily undone. Continue?", + "run-fix-missing-lists-migration-confirm": "This will detect and repair missing or corrupted lists in the board structure. Continue?", + "run-fix-avatar-urls-migration-confirm": "This will update avatar URLs for board members to use the correct storage backend. Continue?", + "run-fix-all-file-urls-migration-confirm": "This will update all file attachment URLs on this board to use the correct storage backend. Continue?", + "restore-lost-cards-nothing-to-restore": "No lost swimlanes, lists, or cards to restore", + + "migration-progress-title": "Board Migration in Progress", + "migration-progress-overall": "Overall Progress", + "migration-progress-current-step": "Current Step", + "migration-progress-status": "Status", + "migration-progress-details": "Details", + "migration-progress-note": "Please wait while we migrate your board to the latest structure...", + + "step-analyze-board-structure": "Analyze Board Structure", + "step-fix-orphaned-cards": "Fix Orphaned Cards", + "step-convert-shared-lists": "Convert Shared Lists", + "step-ensure-per-swimlane-lists": "Ensure Per-Swimlane Lists", + "step-validate-migration": "Validate Migration", + "step-fix-avatar-urls": "Fix Avatar URLs", + "step-fix-attachment-urls": "Fix Attachment URLs", + "step-analyze-lists": "Analyze Lists", + "step-create-missing-lists": "Create Missing Lists", + "step-update-cards": "Update Cards", + "step-finalize": "Finalize", + "step-delete-duplicate-empty-lists": "Delete Duplicate Empty Lists", + "step-ensure-lost-cards-swimlane": "Ensure Lost Cards Swimlane", + "step-restore-lists": "Restore Lists", + "step-restore-cards": "Restore Cards", + "step-restore-swimlanes": "Restore Swimlanes", + "step-fix-missing-ids": "Fix Missing IDs", + "step-scan-users": "Checking board member avatars", + "step-scan-files": "Checking board file attachments", + "step-fix-file-urls": "Fixing file URLs", "cleanup": "Cleanup", "cleanup-old-jobs": "Cleanup Old Jobs", "completed": "Completed", diff --git a/imports/i18n/data/en-BR.i18n.json b/imports/i18n/data/en-BR.i18n.json index 1ab8eeaa8..4804e6dff 100644 --- a/imports/i18n/data/en-BR.i18n.json +++ b/imports/i18n/data/en-BR.i18n.json @@ -1404,7 +1404,70 @@ "back-to-settings": "Back to Settings", "board-id": "Board ID", "board-migration": "Board Migration", + "board-migrations": "Board Migrations", "card-show-lists-on-minicard": "Show Lists on Minicard", + "comprehensive-board-migration": "Comprehensive Board Migration", + "comprehensive-board-migration-description": "Performs comprehensive checks and fixes for board data integrity, including list ordering, card positions, and swimlane structure.", + "delete-duplicate-empty-lists-migration": "Delete Duplicate Empty Lists", + "delete-duplicate-empty-lists-migration-description": "Safely deletes empty duplicate lists. Only removes lists that have no cards AND have another list with the same title that contains cards.", + "lost-cards": "Lost Cards", + "lost-cards-list": "Restored Items", + "restore-lost-cards-migration": "Restore Lost Cards", + "restore-lost-cards-migration-description": "Finds and restores cards and lists with missing swimlaneId or listId. Creates a 'Lost Cards' swimlane to make all lost items visible again.", + "restore-all-archived-migration": "Restore All Archived", + "restore-all-archived-migration-description": "Restores all archived swimlanes, lists, and cards. Automatically fixes any missing swimlaneId or listId to make items visible.", + "fix-missing-lists-migration": "Fix Missing Lists", + "fix-missing-lists-migration-description": "Detects and repairs missing or corrupted lists in the board structure.", + "fix-avatar-urls-migration": "Fix Avatar URLs", + "fix-avatar-urls-migration-description": "Updates avatar URLs for board members to use the correct storage backend and fixes broken avatar references.", + "fix-all-file-urls-migration": "Fix All File URLs", + "fix-all-file-urls-migration-description": "Updates all file attachment URLs on this board to use the correct storage backend and fixes broken file references.", + "migration-needed": "Migration Needed", + "migration-complete": "Complete", + "migration-running": "Running...", + "migration-successful": "Migration completed successfully", + "migration-failed": "Migration failed", + "migrations": "Migrations", + "migrations-admin-only": "Only board administrators can run migrations", + "migrations-description": "Run data integrity checks and repairs for this board. Each migration can be executed individually.", + "no-issues-found": "No issues found", + "run-migration": "Run Migration", + "run-comprehensive-migration-confirm": "This will perform a comprehensive migration to check and fix board data integrity. This may take a few moments. Continue?", + "run-delete-duplicate-empty-lists-migration-confirm": "This will first convert any shared lists to per-swimlane lists, then delete empty lists that have a duplicate list with the same title containing cards. Only truly redundant empty lists will be removed. Continue?", + "run-restore-lost-cards-migration-confirm": "This will create a 'Lost Cards' swimlane and restore all cards and lists with missing swimlaneId or listId. This only affects non-archived items. Continue?", + "run-restore-all-archived-migration-confirm": "This will restore ALL archived swimlanes, lists, and cards, making them visible again. Any items with missing IDs will be automatically fixed. This cannot be easily undone. Continue?", + "run-fix-missing-lists-migration-confirm": "This will detect and repair missing or corrupted lists in the board structure. Continue?", + "run-fix-avatar-urls-migration-confirm": "This will update avatar URLs for board members to use the correct storage backend. Continue?", + "run-fix-all-file-urls-migration-confirm": "This will update all file attachment URLs on this board to use the correct storage backend. Continue?", + "restore-lost-cards-nothing-to-restore": "No lost swimlanes, lists, or cards to restore", + + "migration-progress-title": "Board Migration in Progress", + "migration-progress-overall": "Overall Progress", + "migration-progress-current-step": "Current Step", + "migration-progress-status": "Status", + "migration-progress-details": "Details", + "migration-progress-note": "Please wait while we migrate your board to the latest structure...", + + "step-analyze-board-structure": "Analyze Board Structure", + "step-fix-orphaned-cards": "Fix Orphaned Cards", + "step-convert-shared-lists": "Convert Shared Lists", + "step-ensure-per-swimlane-lists": "Ensure Per-Swimlane Lists", + "step-validate-migration": "Validate Migration", + "step-fix-avatar-urls": "Fix Avatar URLs", + "step-fix-attachment-urls": "Fix Attachment URLs", + "step-analyze-lists": "Analyze Lists", + "step-create-missing-lists": "Create Missing Lists", + "step-update-cards": "Update Cards", + "step-finalize": "Finalize", + "step-delete-duplicate-empty-lists": "Delete Duplicate Empty Lists", + "step-ensure-lost-cards-swimlane": "Ensure Lost Cards Swimlane", + "step-restore-lists": "Restore Lists", + "step-restore-cards": "Restore Cards", + "step-restore-swimlanes": "Restore Swimlanes", + "step-fix-missing-ids": "Fix Missing IDs", + "step-scan-users": "Checking board member avatars", + "step-scan-files": "Checking board file attachments", + "step-fix-file-urls": "Fixing file URLs", "cleanup": "Cleanup", "cleanup-old-jobs": "Cleanup Old Jobs", "completed": "Completed", diff --git a/imports/i18n/data/en-DE.i18n.json b/imports/i18n/data/en-DE.i18n.json index 50ddae1d2..6f376a14c 100644 --- a/imports/i18n/data/en-DE.i18n.json +++ b/imports/i18n/data/en-DE.i18n.json @@ -1404,7 +1404,70 @@ "back-to-settings": "Back to Settings", "board-id": "Board ID", "board-migration": "Board Migration", + "board-migrations": "Board Migrations", "card-show-lists-on-minicard": "Show Lists on Minicard", + "comprehensive-board-migration": "Comprehensive Board Migration", + "comprehensive-board-migration-description": "Performs comprehensive checks and fixes for board data integrity, including list ordering, card positions, and swimlane structure.", + "delete-duplicate-empty-lists-migration": "Delete Duplicate Empty Lists", + "delete-duplicate-empty-lists-migration-description": "Safely deletes empty duplicate lists. Only removes lists that have no cards AND have another list with the same title that contains cards.", + "lost-cards": "Lost Cards", + "lost-cards-list": "Restored Items", + "restore-lost-cards-migration": "Restore Lost Cards", + "restore-lost-cards-migration-description": "Finds and restores cards and lists with missing swimlaneId or listId. Creates a 'Lost Cards' swimlane to make all lost items visible again.", + "restore-all-archived-migration": "Restore All Archived", + "restore-all-archived-migration-description": "Restores all archived swimlanes, lists, and cards. Automatically fixes any missing swimlaneId or listId to make items visible.", + "fix-missing-lists-migration": "Fix Missing Lists", + "fix-missing-lists-migration-description": "Detects and repairs missing or corrupted lists in the board structure.", + "fix-avatar-urls-migration": "Fix Avatar URLs", + "fix-avatar-urls-migration-description": "Updates avatar URLs for board members to use the correct storage backend and fixes broken avatar references.", + "fix-all-file-urls-migration": "Fix All File URLs", + "fix-all-file-urls-migration-description": "Updates all file attachment URLs on this board to use the correct storage backend and fixes broken file references.", + "migration-needed": "Migration Needed", + "migration-complete": "Complete", + "migration-running": "Running...", + "migration-successful": "Migration completed successfully", + "migration-failed": "Migration failed", + "migrations": "Migrations", + "migrations-admin-only": "Only board administrators can run migrations", + "migrations-description": "Run data integrity checks and repairs for this board. Each migration can be executed individually.", + "no-issues-found": "No issues found", + "run-migration": "Run Migration", + "run-comprehensive-migration-confirm": "This will perform a comprehensive migration to check and fix board data integrity. This may take a few moments. Continue?", + "run-delete-duplicate-empty-lists-migration-confirm": "This will first convert any shared lists to per-swimlane lists, then delete empty lists that have a duplicate list with the same title containing cards. Only truly redundant empty lists will be removed. Continue?", + "run-restore-lost-cards-migration-confirm": "This will create a 'Lost Cards' swimlane and restore all cards and lists with missing swimlaneId or listId. This only affects non-archived items. Continue?", + "run-restore-all-archived-migration-confirm": "This will restore ALL archived swimlanes, lists, and cards, making them visible again. Any items with missing IDs will be automatically fixed. This cannot be easily undone. Continue?", + "run-fix-missing-lists-migration-confirm": "This will detect and repair missing or corrupted lists in the board structure. Continue?", + "run-fix-avatar-urls-migration-confirm": "This will update avatar URLs for board members to use the correct storage backend. Continue?", + "run-fix-all-file-urls-migration-confirm": "This will update all file attachment URLs on this board to use the correct storage backend. Continue?", + "restore-lost-cards-nothing-to-restore": "No lost swimlanes, lists, or cards to restore", + + "migration-progress-title": "Board Migration in Progress", + "migration-progress-overall": "Overall Progress", + "migration-progress-current-step": "Current Step", + "migration-progress-status": "Status", + "migration-progress-details": "Details", + "migration-progress-note": "Please wait while we migrate your board to the latest structure...", + + "step-analyze-board-structure": "Analyze Board Structure", + "step-fix-orphaned-cards": "Fix Orphaned Cards", + "step-convert-shared-lists": "Convert Shared Lists", + "step-ensure-per-swimlane-lists": "Ensure Per-Swimlane Lists", + "step-validate-migration": "Validate Migration", + "step-fix-avatar-urls": "Fix Avatar URLs", + "step-fix-attachment-urls": "Fix Attachment URLs", + "step-analyze-lists": "Analyze Lists", + "step-create-missing-lists": "Create Missing Lists", + "step-update-cards": "Update Cards", + "step-finalize": "Finalize", + "step-delete-duplicate-empty-lists": "Delete Duplicate Empty Lists", + "step-ensure-lost-cards-swimlane": "Ensure Lost Cards Swimlane", + "step-restore-lists": "Restore Lists", + "step-restore-cards": "Restore Cards", + "step-restore-swimlanes": "Restore Swimlanes", + "step-fix-missing-ids": "Fix Missing IDs", + "step-scan-users": "Checking board member avatars", + "step-scan-files": "Checking board file attachments", + "step-fix-file-urls": "Fixing file URLs", "cleanup": "Cleanup", "cleanup-old-jobs": "Cleanup Old Jobs", "completed": "Completed", diff --git a/imports/i18n/data/en-GB.i18n.json b/imports/i18n/data/en-GB.i18n.json index eef27e1fe..958d8313a 100644 --- a/imports/i18n/data/en-GB.i18n.json +++ b/imports/i18n/data/en-GB.i18n.json @@ -1404,7 +1404,70 @@ "back-to-settings": "Back to Settings", "board-id": "Board ID", "board-migration": "Board Migration", + "board-migrations": "Board Migrations", "card-show-lists-on-minicard": "Show Lists on Minicard", + "comprehensive-board-migration": "Comprehensive Board Migration", + "comprehensive-board-migration-description": "Performs comprehensive checks and fixes for board data integrity, including list ordering, card positions, and swimlane structure.", + "delete-duplicate-empty-lists-migration": "Delete Duplicate Empty Lists", + "delete-duplicate-empty-lists-migration-description": "Safely deletes empty duplicate lists. Only removes lists that have no cards AND have another list with the same title that contains cards.", + "lost-cards": "Lost Cards", + "lost-cards-list": "Restored Items", + "restore-lost-cards-migration": "Restore Lost Cards", + "restore-lost-cards-migration-description": "Finds and restores cards and lists with missing swimlaneId or listId. Creates a 'Lost Cards' swimlane to make all lost items visible again.", + "restore-all-archived-migration": "Restore All Archived", + "restore-all-archived-migration-description": "Restores all archived swimlanes, lists, and cards. Automatically fixes any missing swimlaneId or listId to make items visible.", + "fix-missing-lists-migration": "Fix Missing Lists", + "fix-missing-lists-migration-description": "Detects and repairs missing or corrupted lists in the board structure.", + "fix-avatar-urls-migration": "Fix Avatar URLs", + "fix-avatar-urls-migration-description": "Updates avatar URLs for board members to use the correct storage backend and fixes broken avatar references.", + "fix-all-file-urls-migration": "Fix All File URLs", + "fix-all-file-urls-migration-description": "Updates all file attachment URLs on this board to use the correct storage backend and fixes broken file references.", + "migration-needed": "Migration Needed", + "migration-complete": "Complete", + "migration-running": "Running...", + "migration-successful": "Migration completed successfully", + "migration-failed": "Migration failed", + "migrations": "Migrations", + "migrations-admin-only": "Only board administrators can run migrations", + "migrations-description": "Run data integrity checks and repairs for this board. Each migration can be executed individually.", + "no-issues-found": "No issues found", + "run-migration": "Run Migration", + "run-comprehensive-migration-confirm": "This will perform a comprehensive migration to check and fix board data integrity. This may take a few moments. Continue?", + "run-delete-duplicate-empty-lists-migration-confirm": "This will first convert any shared lists to per-swimlane lists, then delete empty lists that have a duplicate list with the same title containing cards. Only truly redundant empty lists will be removed. Continue?", + "run-restore-lost-cards-migration-confirm": "This will create a 'Lost Cards' swimlane and restore all cards and lists with missing swimlaneId or listId. This only affects non-archived items. Continue?", + "run-restore-all-archived-migration-confirm": "This will restore ALL archived swimlanes, lists, and cards, making them visible again. Any items with missing IDs will be automatically fixed. This cannot be easily undone. Continue?", + "run-fix-missing-lists-migration-confirm": "This will detect and repair missing or corrupted lists in the board structure. Continue?", + "run-fix-avatar-urls-migration-confirm": "This will update avatar URLs for board members to use the correct storage backend. Continue?", + "run-fix-all-file-urls-migration-confirm": "This will update all file attachment URLs on this board to use the correct storage backend. Continue?", + "restore-lost-cards-nothing-to-restore": "No lost swimlanes, lists, or cards to restore", + + "migration-progress-title": "Board Migration in Progress", + "migration-progress-overall": "Overall Progress", + "migration-progress-current-step": "Current Step", + "migration-progress-status": "Status", + "migration-progress-details": "Details", + "migration-progress-note": "Please wait while we migrate your board to the latest structure...", + + "step-analyze-board-structure": "Analyze Board Structure", + "step-fix-orphaned-cards": "Fix Orphaned Cards", + "step-convert-shared-lists": "Convert Shared Lists", + "step-ensure-per-swimlane-lists": "Ensure Per-Swimlane Lists", + "step-validate-migration": "Validate Migration", + "step-fix-avatar-urls": "Fix Avatar URLs", + "step-fix-attachment-urls": "Fix Attachment URLs", + "step-analyze-lists": "Analyze Lists", + "step-create-missing-lists": "Create Missing Lists", + "step-update-cards": "Update Cards", + "step-finalize": "Finalize", + "step-delete-duplicate-empty-lists": "Delete Duplicate Empty Lists", + "step-ensure-lost-cards-swimlane": "Ensure Lost Cards Swimlane", + "step-restore-lists": "Restore Lists", + "step-restore-cards": "Restore Cards", + "step-restore-swimlanes": "Restore Swimlanes", + "step-fix-missing-ids": "Fix Missing IDs", + "step-scan-users": "Checking board member avatars", + "step-scan-files": "Checking board file attachments", + "step-fix-file-urls": "Fixing file URLs", "cleanup": "Cleanup", "cleanup-old-jobs": "Cleanup Old Jobs", "completed": "Completed", diff --git a/imports/i18n/data/en-IT.i18n.json b/imports/i18n/data/en-IT.i18n.json index 1ab8eeaa8..4804e6dff 100644 --- a/imports/i18n/data/en-IT.i18n.json +++ b/imports/i18n/data/en-IT.i18n.json @@ -1404,7 +1404,70 @@ "back-to-settings": "Back to Settings", "board-id": "Board ID", "board-migration": "Board Migration", + "board-migrations": "Board Migrations", "card-show-lists-on-minicard": "Show Lists on Minicard", + "comprehensive-board-migration": "Comprehensive Board Migration", + "comprehensive-board-migration-description": "Performs comprehensive checks and fixes for board data integrity, including list ordering, card positions, and swimlane structure.", + "delete-duplicate-empty-lists-migration": "Delete Duplicate Empty Lists", + "delete-duplicate-empty-lists-migration-description": "Safely deletes empty duplicate lists. Only removes lists that have no cards AND have another list with the same title that contains cards.", + "lost-cards": "Lost Cards", + "lost-cards-list": "Restored Items", + "restore-lost-cards-migration": "Restore Lost Cards", + "restore-lost-cards-migration-description": "Finds and restores cards and lists with missing swimlaneId or listId. Creates a 'Lost Cards' swimlane to make all lost items visible again.", + "restore-all-archived-migration": "Restore All Archived", + "restore-all-archived-migration-description": "Restores all archived swimlanes, lists, and cards. Automatically fixes any missing swimlaneId or listId to make items visible.", + "fix-missing-lists-migration": "Fix Missing Lists", + "fix-missing-lists-migration-description": "Detects and repairs missing or corrupted lists in the board structure.", + "fix-avatar-urls-migration": "Fix Avatar URLs", + "fix-avatar-urls-migration-description": "Updates avatar URLs for board members to use the correct storage backend and fixes broken avatar references.", + "fix-all-file-urls-migration": "Fix All File URLs", + "fix-all-file-urls-migration-description": "Updates all file attachment URLs on this board to use the correct storage backend and fixes broken file references.", + "migration-needed": "Migration Needed", + "migration-complete": "Complete", + "migration-running": "Running...", + "migration-successful": "Migration completed successfully", + "migration-failed": "Migration failed", + "migrations": "Migrations", + "migrations-admin-only": "Only board administrators can run migrations", + "migrations-description": "Run data integrity checks and repairs for this board. Each migration can be executed individually.", + "no-issues-found": "No issues found", + "run-migration": "Run Migration", + "run-comprehensive-migration-confirm": "This will perform a comprehensive migration to check and fix board data integrity. This may take a few moments. Continue?", + "run-delete-duplicate-empty-lists-migration-confirm": "This will first convert any shared lists to per-swimlane lists, then delete empty lists that have a duplicate list with the same title containing cards. Only truly redundant empty lists will be removed. Continue?", + "run-restore-lost-cards-migration-confirm": "This will create a 'Lost Cards' swimlane and restore all cards and lists with missing swimlaneId or listId. This only affects non-archived items. Continue?", + "run-restore-all-archived-migration-confirm": "This will restore ALL archived swimlanes, lists, and cards, making them visible again. Any items with missing IDs will be automatically fixed. This cannot be easily undone. Continue?", + "run-fix-missing-lists-migration-confirm": "This will detect and repair missing or corrupted lists in the board structure. Continue?", + "run-fix-avatar-urls-migration-confirm": "This will update avatar URLs for board members to use the correct storage backend. Continue?", + "run-fix-all-file-urls-migration-confirm": "This will update all file attachment URLs on this board to use the correct storage backend. Continue?", + "restore-lost-cards-nothing-to-restore": "No lost swimlanes, lists, or cards to restore", + + "migration-progress-title": "Board Migration in Progress", + "migration-progress-overall": "Overall Progress", + "migration-progress-current-step": "Current Step", + "migration-progress-status": "Status", + "migration-progress-details": "Details", + "migration-progress-note": "Please wait while we migrate your board to the latest structure...", + + "step-analyze-board-structure": "Analyze Board Structure", + "step-fix-orphaned-cards": "Fix Orphaned Cards", + "step-convert-shared-lists": "Convert Shared Lists", + "step-ensure-per-swimlane-lists": "Ensure Per-Swimlane Lists", + "step-validate-migration": "Validate Migration", + "step-fix-avatar-urls": "Fix Avatar URLs", + "step-fix-attachment-urls": "Fix Attachment URLs", + "step-analyze-lists": "Analyze Lists", + "step-create-missing-lists": "Create Missing Lists", + "step-update-cards": "Update Cards", + "step-finalize": "Finalize", + "step-delete-duplicate-empty-lists": "Delete Duplicate Empty Lists", + "step-ensure-lost-cards-swimlane": "Ensure Lost Cards Swimlane", + "step-restore-lists": "Restore Lists", + "step-restore-cards": "Restore Cards", + "step-restore-swimlanes": "Restore Swimlanes", + "step-fix-missing-ids": "Fix Missing IDs", + "step-scan-users": "Checking board member avatars", + "step-scan-files": "Checking board file attachments", + "step-fix-file-urls": "Fixing file URLs", "cleanup": "Cleanup", "cleanup-old-jobs": "Cleanup Old Jobs", "completed": "Completed", diff --git a/imports/i18n/data/en-MY.i18n.json b/imports/i18n/data/en-MY.i18n.json index 1ab8eeaa8..4804e6dff 100644 --- a/imports/i18n/data/en-MY.i18n.json +++ b/imports/i18n/data/en-MY.i18n.json @@ -1404,7 +1404,70 @@ "back-to-settings": "Back to Settings", "board-id": "Board ID", "board-migration": "Board Migration", + "board-migrations": "Board Migrations", "card-show-lists-on-minicard": "Show Lists on Minicard", + "comprehensive-board-migration": "Comprehensive Board Migration", + "comprehensive-board-migration-description": "Performs comprehensive checks and fixes for board data integrity, including list ordering, card positions, and swimlane structure.", + "delete-duplicate-empty-lists-migration": "Delete Duplicate Empty Lists", + "delete-duplicate-empty-lists-migration-description": "Safely deletes empty duplicate lists. Only removes lists that have no cards AND have another list with the same title that contains cards.", + "lost-cards": "Lost Cards", + "lost-cards-list": "Restored Items", + "restore-lost-cards-migration": "Restore Lost Cards", + "restore-lost-cards-migration-description": "Finds and restores cards and lists with missing swimlaneId or listId. Creates a 'Lost Cards' swimlane to make all lost items visible again.", + "restore-all-archived-migration": "Restore All Archived", + "restore-all-archived-migration-description": "Restores all archived swimlanes, lists, and cards. Automatically fixes any missing swimlaneId or listId to make items visible.", + "fix-missing-lists-migration": "Fix Missing Lists", + "fix-missing-lists-migration-description": "Detects and repairs missing or corrupted lists in the board structure.", + "fix-avatar-urls-migration": "Fix Avatar URLs", + "fix-avatar-urls-migration-description": "Updates avatar URLs for board members to use the correct storage backend and fixes broken avatar references.", + "fix-all-file-urls-migration": "Fix All File URLs", + "fix-all-file-urls-migration-description": "Updates all file attachment URLs on this board to use the correct storage backend and fixes broken file references.", + "migration-needed": "Migration Needed", + "migration-complete": "Complete", + "migration-running": "Running...", + "migration-successful": "Migration completed successfully", + "migration-failed": "Migration failed", + "migrations": "Migrations", + "migrations-admin-only": "Only board administrators can run migrations", + "migrations-description": "Run data integrity checks and repairs for this board. Each migration can be executed individually.", + "no-issues-found": "No issues found", + "run-migration": "Run Migration", + "run-comprehensive-migration-confirm": "This will perform a comprehensive migration to check and fix board data integrity. This may take a few moments. Continue?", + "run-delete-duplicate-empty-lists-migration-confirm": "This will first convert any shared lists to per-swimlane lists, then delete empty lists that have a duplicate list with the same title containing cards. Only truly redundant empty lists will be removed. Continue?", + "run-restore-lost-cards-migration-confirm": "This will create a 'Lost Cards' swimlane and restore all cards and lists with missing swimlaneId or listId. This only affects non-archived items. Continue?", + "run-restore-all-archived-migration-confirm": "This will restore ALL archived swimlanes, lists, and cards, making them visible again. Any items with missing IDs will be automatically fixed. This cannot be easily undone. Continue?", + "run-fix-missing-lists-migration-confirm": "This will detect and repair missing or corrupted lists in the board structure. Continue?", + "run-fix-avatar-urls-migration-confirm": "This will update avatar URLs for board members to use the correct storage backend. Continue?", + "run-fix-all-file-urls-migration-confirm": "This will update all file attachment URLs on this board to use the correct storage backend. Continue?", + "restore-lost-cards-nothing-to-restore": "No lost swimlanes, lists, or cards to restore", + + "migration-progress-title": "Board Migration in Progress", + "migration-progress-overall": "Overall Progress", + "migration-progress-current-step": "Current Step", + "migration-progress-status": "Status", + "migration-progress-details": "Details", + "migration-progress-note": "Please wait while we migrate your board to the latest structure...", + + "step-analyze-board-structure": "Analyze Board Structure", + "step-fix-orphaned-cards": "Fix Orphaned Cards", + "step-convert-shared-lists": "Convert Shared Lists", + "step-ensure-per-swimlane-lists": "Ensure Per-Swimlane Lists", + "step-validate-migration": "Validate Migration", + "step-fix-avatar-urls": "Fix Avatar URLs", + "step-fix-attachment-urls": "Fix Attachment URLs", + "step-analyze-lists": "Analyze Lists", + "step-create-missing-lists": "Create Missing Lists", + "step-update-cards": "Update Cards", + "step-finalize": "Finalize", + "step-delete-duplicate-empty-lists": "Delete Duplicate Empty Lists", + "step-ensure-lost-cards-swimlane": "Ensure Lost Cards Swimlane", + "step-restore-lists": "Restore Lists", + "step-restore-cards": "Restore Cards", + "step-restore-swimlanes": "Restore Swimlanes", + "step-fix-missing-ids": "Fix Missing IDs", + "step-scan-users": "Checking board member avatars", + "step-scan-files": "Checking board file attachments", + "step-fix-file-urls": "Fixing file URLs", "cleanup": "Cleanup", "cleanup-old-jobs": "Cleanup Old Jobs", "completed": "Completed", diff --git a/imports/i18n/data/en-YS.i18n.json b/imports/i18n/data/en-YS.i18n.json index 1ab8eeaa8..4804e6dff 100644 --- a/imports/i18n/data/en-YS.i18n.json +++ b/imports/i18n/data/en-YS.i18n.json @@ -1404,7 +1404,70 @@ "back-to-settings": "Back to Settings", "board-id": "Board ID", "board-migration": "Board Migration", + "board-migrations": "Board Migrations", "card-show-lists-on-minicard": "Show Lists on Minicard", + "comprehensive-board-migration": "Comprehensive Board Migration", + "comprehensive-board-migration-description": "Performs comprehensive checks and fixes for board data integrity, including list ordering, card positions, and swimlane structure.", + "delete-duplicate-empty-lists-migration": "Delete Duplicate Empty Lists", + "delete-duplicate-empty-lists-migration-description": "Safely deletes empty duplicate lists. Only removes lists that have no cards AND have another list with the same title that contains cards.", + "lost-cards": "Lost Cards", + "lost-cards-list": "Restored Items", + "restore-lost-cards-migration": "Restore Lost Cards", + "restore-lost-cards-migration-description": "Finds and restores cards and lists with missing swimlaneId or listId. Creates a 'Lost Cards' swimlane to make all lost items visible again.", + "restore-all-archived-migration": "Restore All Archived", + "restore-all-archived-migration-description": "Restores all archived swimlanes, lists, and cards. Automatically fixes any missing swimlaneId or listId to make items visible.", + "fix-missing-lists-migration": "Fix Missing Lists", + "fix-missing-lists-migration-description": "Detects and repairs missing or corrupted lists in the board structure.", + "fix-avatar-urls-migration": "Fix Avatar URLs", + "fix-avatar-urls-migration-description": "Updates avatar URLs for board members to use the correct storage backend and fixes broken avatar references.", + "fix-all-file-urls-migration": "Fix All File URLs", + "fix-all-file-urls-migration-description": "Updates all file attachment URLs on this board to use the correct storage backend and fixes broken file references.", + "migration-needed": "Migration Needed", + "migration-complete": "Complete", + "migration-running": "Running...", + "migration-successful": "Migration completed successfully", + "migration-failed": "Migration failed", + "migrations": "Migrations", + "migrations-admin-only": "Only board administrators can run migrations", + "migrations-description": "Run data integrity checks and repairs for this board. Each migration can be executed individually.", + "no-issues-found": "No issues found", + "run-migration": "Run Migration", + "run-comprehensive-migration-confirm": "This will perform a comprehensive migration to check and fix board data integrity. This may take a few moments. Continue?", + "run-delete-duplicate-empty-lists-migration-confirm": "This will first convert any shared lists to per-swimlane lists, then delete empty lists that have a duplicate list with the same title containing cards. Only truly redundant empty lists will be removed. Continue?", + "run-restore-lost-cards-migration-confirm": "This will create a 'Lost Cards' swimlane and restore all cards and lists with missing swimlaneId or listId. This only affects non-archived items. Continue?", + "run-restore-all-archived-migration-confirm": "This will restore ALL archived swimlanes, lists, and cards, making them visible again. Any items with missing IDs will be automatically fixed. This cannot be easily undone. Continue?", + "run-fix-missing-lists-migration-confirm": "This will detect and repair missing or corrupted lists in the board structure. Continue?", + "run-fix-avatar-urls-migration-confirm": "This will update avatar URLs for board members to use the correct storage backend. Continue?", + "run-fix-all-file-urls-migration-confirm": "This will update all file attachment URLs on this board to use the correct storage backend. Continue?", + "restore-lost-cards-nothing-to-restore": "No lost swimlanes, lists, or cards to restore", + + "migration-progress-title": "Board Migration in Progress", + "migration-progress-overall": "Overall Progress", + "migration-progress-current-step": "Current Step", + "migration-progress-status": "Status", + "migration-progress-details": "Details", + "migration-progress-note": "Please wait while we migrate your board to the latest structure...", + + "step-analyze-board-structure": "Analyze Board Structure", + "step-fix-orphaned-cards": "Fix Orphaned Cards", + "step-convert-shared-lists": "Convert Shared Lists", + "step-ensure-per-swimlane-lists": "Ensure Per-Swimlane Lists", + "step-validate-migration": "Validate Migration", + "step-fix-avatar-urls": "Fix Avatar URLs", + "step-fix-attachment-urls": "Fix Attachment URLs", + "step-analyze-lists": "Analyze Lists", + "step-create-missing-lists": "Create Missing Lists", + "step-update-cards": "Update Cards", + "step-finalize": "Finalize", + "step-delete-duplicate-empty-lists": "Delete Duplicate Empty Lists", + "step-ensure-lost-cards-swimlane": "Ensure Lost Cards Swimlane", + "step-restore-lists": "Restore Lists", + "step-restore-cards": "Restore Cards", + "step-restore-swimlanes": "Restore Swimlanes", + "step-fix-missing-ids": "Fix Missing IDs", + "step-scan-users": "Checking board member avatars", + "step-scan-files": "Checking board file attachments", + "step-fix-file-urls": "Fixing file URLs", "cleanup": "Cleanup", "cleanup-old-jobs": "Cleanup Old Jobs", "completed": "Completed", diff --git a/imports/i18n/data/en_AU.i18n.json b/imports/i18n/data/en_AU.i18n.json index eef27e1fe..958d8313a 100644 --- a/imports/i18n/data/en_AU.i18n.json +++ b/imports/i18n/data/en_AU.i18n.json @@ -1404,7 +1404,70 @@ "back-to-settings": "Back to Settings", "board-id": "Board ID", "board-migration": "Board Migration", + "board-migrations": "Board Migrations", "card-show-lists-on-minicard": "Show Lists on Minicard", + "comprehensive-board-migration": "Comprehensive Board Migration", + "comprehensive-board-migration-description": "Performs comprehensive checks and fixes for board data integrity, including list ordering, card positions, and swimlane structure.", + "delete-duplicate-empty-lists-migration": "Delete Duplicate Empty Lists", + "delete-duplicate-empty-lists-migration-description": "Safely deletes empty duplicate lists. Only removes lists that have no cards AND have another list with the same title that contains cards.", + "lost-cards": "Lost Cards", + "lost-cards-list": "Restored Items", + "restore-lost-cards-migration": "Restore Lost Cards", + "restore-lost-cards-migration-description": "Finds and restores cards and lists with missing swimlaneId or listId. Creates a 'Lost Cards' swimlane to make all lost items visible again.", + "restore-all-archived-migration": "Restore All Archived", + "restore-all-archived-migration-description": "Restores all archived swimlanes, lists, and cards. Automatically fixes any missing swimlaneId or listId to make items visible.", + "fix-missing-lists-migration": "Fix Missing Lists", + "fix-missing-lists-migration-description": "Detects and repairs missing or corrupted lists in the board structure.", + "fix-avatar-urls-migration": "Fix Avatar URLs", + "fix-avatar-urls-migration-description": "Updates avatar URLs for board members to use the correct storage backend and fixes broken avatar references.", + "fix-all-file-urls-migration": "Fix All File URLs", + "fix-all-file-urls-migration-description": "Updates all file attachment URLs on this board to use the correct storage backend and fixes broken file references.", + "migration-needed": "Migration Needed", + "migration-complete": "Complete", + "migration-running": "Running...", + "migration-successful": "Migration completed successfully", + "migration-failed": "Migration failed", + "migrations": "Migrations", + "migrations-admin-only": "Only board administrators can run migrations", + "migrations-description": "Run data integrity checks and repairs for this board. Each migration can be executed individually.", + "no-issues-found": "No issues found", + "run-migration": "Run Migration", + "run-comprehensive-migration-confirm": "This will perform a comprehensive migration to check and fix board data integrity. This may take a few moments. Continue?", + "run-delete-duplicate-empty-lists-migration-confirm": "This will first convert any shared lists to per-swimlane lists, then delete empty lists that have a duplicate list with the same title containing cards. Only truly redundant empty lists will be removed. Continue?", + "run-restore-lost-cards-migration-confirm": "This will create a 'Lost Cards' swimlane and restore all cards and lists with missing swimlaneId or listId. This only affects non-archived items. Continue?", + "run-restore-all-archived-migration-confirm": "This will restore ALL archived swimlanes, lists, and cards, making them visible again. Any items with missing IDs will be automatically fixed. This cannot be easily undone. Continue?", + "run-fix-missing-lists-migration-confirm": "This will detect and repair missing or corrupted lists in the board structure. Continue?", + "run-fix-avatar-urls-migration-confirm": "This will update avatar URLs for board members to use the correct storage backend. Continue?", + "run-fix-all-file-urls-migration-confirm": "This will update all file attachment URLs on this board to use the correct storage backend. Continue?", + "restore-lost-cards-nothing-to-restore": "No lost swimlanes, lists, or cards to restore", + + "migration-progress-title": "Board Migration in Progress", + "migration-progress-overall": "Overall Progress", + "migration-progress-current-step": "Current Step", + "migration-progress-status": "Status", + "migration-progress-details": "Details", + "migration-progress-note": "Please wait while we migrate your board to the latest structure...", + + "step-analyze-board-structure": "Analyze Board Structure", + "step-fix-orphaned-cards": "Fix Orphaned Cards", + "step-convert-shared-lists": "Convert Shared Lists", + "step-ensure-per-swimlane-lists": "Ensure Per-Swimlane Lists", + "step-validate-migration": "Validate Migration", + "step-fix-avatar-urls": "Fix Avatar URLs", + "step-fix-attachment-urls": "Fix Attachment URLs", + "step-analyze-lists": "Analyze Lists", + "step-create-missing-lists": "Create Missing Lists", + "step-update-cards": "Update Cards", + "step-finalize": "Finalize", + "step-delete-duplicate-empty-lists": "Delete Duplicate Empty Lists", + "step-ensure-lost-cards-swimlane": "Ensure Lost Cards Swimlane", + "step-restore-lists": "Restore Lists", + "step-restore-cards": "Restore Cards", + "step-restore-swimlanes": "Restore Swimlanes", + "step-fix-missing-ids": "Fix Missing IDs", + "step-scan-users": "Checking board member avatars", + "step-scan-files": "Checking board file attachments", + "step-fix-file-urls": "Fixing file URLs", "cleanup": "Cleanup", "cleanup-old-jobs": "Cleanup Old Jobs", "completed": "Completed", diff --git a/imports/i18n/data/en_ID.i18n.json b/imports/i18n/data/en_ID.i18n.json index eef27e1fe..958d8313a 100644 --- a/imports/i18n/data/en_ID.i18n.json +++ b/imports/i18n/data/en_ID.i18n.json @@ -1404,7 +1404,70 @@ "back-to-settings": "Back to Settings", "board-id": "Board ID", "board-migration": "Board Migration", + "board-migrations": "Board Migrations", "card-show-lists-on-minicard": "Show Lists on Minicard", + "comprehensive-board-migration": "Comprehensive Board Migration", + "comprehensive-board-migration-description": "Performs comprehensive checks and fixes for board data integrity, including list ordering, card positions, and swimlane structure.", + "delete-duplicate-empty-lists-migration": "Delete Duplicate Empty Lists", + "delete-duplicate-empty-lists-migration-description": "Safely deletes empty duplicate lists. Only removes lists that have no cards AND have another list with the same title that contains cards.", + "lost-cards": "Lost Cards", + "lost-cards-list": "Restored Items", + "restore-lost-cards-migration": "Restore Lost Cards", + "restore-lost-cards-migration-description": "Finds and restores cards and lists with missing swimlaneId or listId. Creates a 'Lost Cards' swimlane to make all lost items visible again.", + "restore-all-archived-migration": "Restore All Archived", + "restore-all-archived-migration-description": "Restores all archived swimlanes, lists, and cards. Automatically fixes any missing swimlaneId or listId to make items visible.", + "fix-missing-lists-migration": "Fix Missing Lists", + "fix-missing-lists-migration-description": "Detects and repairs missing or corrupted lists in the board structure.", + "fix-avatar-urls-migration": "Fix Avatar URLs", + "fix-avatar-urls-migration-description": "Updates avatar URLs for board members to use the correct storage backend and fixes broken avatar references.", + "fix-all-file-urls-migration": "Fix All File URLs", + "fix-all-file-urls-migration-description": "Updates all file attachment URLs on this board to use the correct storage backend and fixes broken file references.", + "migration-needed": "Migration Needed", + "migration-complete": "Complete", + "migration-running": "Running...", + "migration-successful": "Migration completed successfully", + "migration-failed": "Migration failed", + "migrations": "Migrations", + "migrations-admin-only": "Only board administrators can run migrations", + "migrations-description": "Run data integrity checks and repairs for this board. Each migration can be executed individually.", + "no-issues-found": "No issues found", + "run-migration": "Run Migration", + "run-comprehensive-migration-confirm": "This will perform a comprehensive migration to check and fix board data integrity. This may take a few moments. Continue?", + "run-delete-duplicate-empty-lists-migration-confirm": "This will first convert any shared lists to per-swimlane lists, then delete empty lists that have a duplicate list with the same title containing cards. Only truly redundant empty lists will be removed. Continue?", + "run-restore-lost-cards-migration-confirm": "This will create a 'Lost Cards' swimlane and restore all cards and lists with missing swimlaneId or listId. This only affects non-archived items. Continue?", + "run-restore-all-archived-migration-confirm": "This will restore ALL archived swimlanes, lists, and cards, making them visible again. Any items with missing IDs will be automatically fixed. This cannot be easily undone. Continue?", + "run-fix-missing-lists-migration-confirm": "This will detect and repair missing or corrupted lists in the board structure. Continue?", + "run-fix-avatar-urls-migration-confirm": "This will update avatar URLs for board members to use the correct storage backend. Continue?", + "run-fix-all-file-urls-migration-confirm": "This will update all file attachment URLs on this board to use the correct storage backend. Continue?", + "restore-lost-cards-nothing-to-restore": "No lost swimlanes, lists, or cards to restore", + + "migration-progress-title": "Board Migration in Progress", + "migration-progress-overall": "Overall Progress", + "migration-progress-current-step": "Current Step", + "migration-progress-status": "Status", + "migration-progress-details": "Details", + "migration-progress-note": "Please wait while we migrate your board to the latest structure...", + + "step-analyze-board-structure": "Analyze Board Structure", + "step-fix-orphaned-cards": "Fix Orphaned Cards", + "step-convert-shared-lists": "Convert Shared Lists", + "step-ensure-per-swimlane-lists": "Ensure Per-Swimlane Lists", + "step-validate-migration": "Validate Migration", + "step-fix-avatar-urls": "Fix Avatar URLs", + "step-fix-attachment-urls": "Fix Attachment URLs", + "step-analyze-lists": "Analyze Lists", + "step-create-missing-lists": "Create Missing Lists", + "step-update-cards": "Update Cards", + "step-finalize": "Finalize", + "step-delete-duplicate-empty-lists": "Delete Duplicate Empty Lists", + "step-ensure-lost-cards-swimlane": "Ensure Lost Cards Swimlane", + "step-restore-lists": "Restore Lists", + "step-restore-cards": "Restore Cards", + "step-restore-swimlanes": "Restore Swimlanes", + "step-fix-missing-ids": "Fix Missing IDs", + "step-scan-users": "Checking board member avatars", + "step-scan-files": "Checking board file attachments", + "step-fix-file-urls": "Fixing file URLs", "cleanup": "Cleanup", "cleanup-old-jobs": "Cleanup Old Jobs", "completed": "Completed", diff --git a/imports/i18n/data/en_SG.i18n.json b/imports/i18n/data/en_SG.i18n.json index eef27e1fe..958d8313a 100644 --- a/imports/i18n/data/en_SG.i18n.json +++ b/imports/i18n/data/en_SG.i18n.json @@ -1404,7 +1404,70 @@ "back-to-settings": "Back to Settings", "board-id": "Board ID", "board-migration": "Board Migration", + "board-migrations": "Board Migrations", "card-show-lists-on-minicard": "Show Lists on Minicard", + "comprehensive-board-migration": "Comprehensive Board Migration", + "comprehensive-board-migration-description": "Performs comprehensive checks and fixes for board data integrity, including list ordering, card positions, and swimlane structure.", + "delete-duplicate-empty-lists-migration": "Delete Duplicate Empty Lists", + "delete-duplicate-empty-lists-migration-description": "Safely deletes empty duplicate lists. Only removes lists that have no cards AND have another list with the same title that contains cards.", + "lost-cards": "Lost Cards", + "lost-cards-list": "Restored Items", + "restore-lost-cards-migration": "Restore Lost Cards", + "restore-lost-cards-migration-description": "Finds and restores cards and lists with missing swimlaneId or listId. Creates a 'Lost Cards' swimlane to make all lost items visible again.", + "restore-all-archived-migration": "Restore All Archived", + "restore-all-archived-migration-description": "Restores all archived swimlanes, lists, and cards. Automatically fixes any missing swimlaneId or listId to make items visible.", + "fix-missing-lists-migration": "Fix Missing Lists", + "fix-missing-lists-migration-description": "Detects and repairs missing or corrupted lists in the board structure.", + "fix-avatar-urls-migration": "Fix Avatar URLs", + "fix-avatar-urls-migration-description": "Updates avatar URLs for board members to use the correct storage backend and fixes broken avatar references.", + "fix-all-file-urls-migration": "Fix All File URLs", + "fix-all-file-urls-migration-description": "Updates all file attachment URLs on this board to use the correct storage backend and fixes broken file references.", + "migration-needed": "Migration Needed", + "migration-complete": "Complete", + "migration-running": "Running...", + "migration-successful": "Migration completed successfully", + "migration-failed": "Migration failed", + "migrations": "Migrations", + "migrations-admin-only": "Only board administrators can run migrations", + "migrations-description": "Run data integrity checks and repairs for this board. Each migration can be executed individually.", + "no-issues-found": "No issues found", + "run-migration": "Run Migration", + "run-comprehensive-migration-confirm": "This will perform a comprehensive migration to check and fix board data integrity. This may take a few moments. Continue?", + "run-delete-duplicate-empty-lists-migration-confirm": "This will first convert any shared lists to per-swimlane lists, then delete empty lists that have a duplicate list with the same title containing cards. Only truly redundant empty lists will be removed. Continue?", + "run-restore-lost-cards-migration-confirm": "This will create a 'Lost Cards' swimlane and restore all cards and lists with missing swimlaneId or listId. This only affects non-archived items. Continue?", + "run-restore-all-archived-migration-confirm": "This will restore ALL archived swimlanes, lists, and cards, making them visible again. Any items with missing IDs will be automatically fixed. This cannot be easily undone. Continue?", + "run-fix-missing-lists-migration-confirm": "This will detect and repair missing or corrupted lists in the board structure. Continue?", + "run-fix-avatar-urls-migration-confirm": "This will update avatar URLs for board members to use the correct storage backend. Continue?", + "run-fix-all-file-urls-migration-confirm": "This will update all file attachment URLs on this board to use the correct storage backend. Continue?", + "restore-lost-cards-nothing-to-restore": "No lost swimlanes, lists, or cards to restore", + + "migration-progress-title": "Board Migration in Progress", + "migration-progress-overall": "Overall Progress", + "migration-progress-current-step": "Current Step", + "migration-progress-status": "Status", + "migration-progress-details": "Details", + "migration-progress-note": "Please wait while we migrate your board to the latest structure...", + + "step-analyze-board-structure": "Analyze Board Structure", + "step-fix-orphaned-cards": "Fix Orphaned Cards", + "step-convert-shared-lists": "Convert Shared Lists", + "step-ensure-per-swimlane-lists": "Ensure Per-Swimlane Lists", + "step-validate-migration": "Validate Migration", + "step-fix-avatar-urls": "Fix Avatar URLs", + "step-fix-attachment-urls": "Fix Attachment URLs", + "step-analyze-lists": "Analyze Lists", + "step-create-missing-lists": "Create Missing Lists", + "step-update-cards": "Update Cards", + "step-finalize": "Finalize", + "step-delete-duplicate-empty-lists": "Delete Duplicate Empty Lists", + "step-ensure-lost-cards-swimlane": "Ensure Lost Cards Swimlane", + "step-restore-lists": "Restore Lists", + "step-restore-cards": "Restore Cards", + "step-restore-swimlanes": "Restore Swimlanes", + "step-fix-missing-ids": "Fix Missing IDs", + "step-scan-users": "Checking board member avatars", + "step-scan-files": "Checking board file attachments", + "step-fix-file-urls": "Fixing file URLs", "cleanup": "Cleanup", "cleanup-old-jobs": "Cleanup Old Jobs", "completed": "Completed", diff --git a/imports/i18n/data/en_TR.i18n.json b/imports/i18n/data/en_TR.i18n.json index eef27e1fe..958d8313a 100644 --- a/imports/i18n/data/en_TR.i18n.json +++ b/imports/i18n/data/en_TR.i18n.json @@ -1404,7 +1404,70 @@ "back-to-settings": "Back to Settings", "board-id": "Board ID", "board-migration": "Board Migration", + "board-migrations": "Board Migrations", "card-show-lists-on-minicard": "Show Lists on Minicard", + "comprehensive-board-migration": "Comprehensive Board Migration", + "comprehensive-board-migration-description": "Performs comprehensive checks and fixes for board data integrity, including list ordering, card positions, and swimlane structure.", + "delete-duplicate-empty-lists-migration": "Delete Duplicate Empty Lists", + "delete-duplicate-empty-lists-migration-description": "Safely deletes empty duplicate lists. Only removes lists that have no cards AND have another list with the same title that contains cards.", + "lost-cards": "Lost Cards", + "lost-cards-list": "Restored Items", + "restore-lost-cards-migration": "Restore Lost Cards", + "restore-lost-cards-migration-description": "Finds and restores cards and lists with missing swimlaneId or listId. Creates a 'Lost Cards' swimlane to make all lost items visible again.", + "restore-all-archived-migration": "Restore All Archived", + "restore-all-archived-migration-description": "Restores all archived swimlanes, lists, and cards. Automatically fixes any missing swimlaneId or listId to make items visible.", + "fix-missing-lists-migration": "Fix Missing Lists", + "fix-missing-lists-migration-description": "Detects and repairs missing or corrupted lists in the board structure.", + "fix-avatar-urls-migration": "Fix Avatar URLs", + "fix-avatar-urls-migration-description": "Updates avatar URLs for board members to use the correct storage backend and fixes broken avatar references.", + "fix-all-file-urls-migration": "Fix All File URLs", + "fix-all-file-urls-migration-description": "Updates all file attachment URLs on this board to use the correct storage backend and fixes broken file references.", + "migration-needed": "Migration Needed", + "migration-complete": "Complete", + "migration-running": "Running...", + "migration-successful": "Migration completed successfully", + "migration-failed": "Migration failed", + "migrations": "Migrations", + "migrations-admin-only": "Only board administrators can run migrations", + "migrations-description": "Run data integrity checks and repairs for this board. Each migration can be executed individually.", + "no-issues-found": "No issues found", + "run-migration": "Run Migration", + "run-comprehensive-migration-confirm": "This will perform a comprehensive migration to check and fix board data integrity. This may take a few moments. Continue?", + "run-delete-duplicate-empty-lists-migration-confirm": "This will first convert any shared lists to per-swimlane lists, then delete empty lists that have a duplicate list with the same title containing cards. Only truly redundant empty lists will be removed. Continue?", + "run-restore-lost-cards-migration-confirm": "This will create a 'Lost Cards' swimlane and restore all cards and lists with missing swimlaneId or listId. This only affects non-archived items. Continue?", + "run-restore-all-archived-migration-confirm": "This will restore ALL archived swimlanes, lists, and cards, making them visible again. Any items with missing IDs will be automatically fixed. This cannot be easily undone. Continue?", + "run-fix-missing-lists-migration-confirm": "This will detect and repair missing or corrupted lists in the board structure. Continue?", + "run-fix-avatar-urls-migration-confirm": "This will update avatar URLs for board members to use the correct storage backend. Continue?", + "run-fix-all-file-urls-migration-confirm": "This will update all file attachment URLs on this board to use the correct storage backend. Continue?", + "restore-lost-cards-nothing-to-restore": "No lost swimlanes, lists, or cards to restore", + + "migration-progress-title": "Board Migration in Progress", + "migration-progress-overall": "Overall Progress", + "migration-progress-current-step": "Current Step", + "migration-progress-status": "Status", + "migration-progress-details": "Details", + "migration-progress-note": "Please wait while we migrate your board to the latest structure...", + + "step-analyze-board-structure": "Analyze Board Structure", + "step-fix-orphaned-cards": "Fix Orphaned Cards", + "step-convert-shared-lists": "Convert Shared Lists", + "step-ensure-per-swimlane-lists": "Ensure Per-Swimlane Lists", + "step-validate-migration": "Validate Migration", + "step-fix-avatar-urls": "Fix Avatar URLs", + "step-fix-attachment-urls": "Fix Attachment URLs", + "step-analyze-lists": "Analyze Lists", + "step-create-missing-lists": "Create Missing Lists", + "step-update-cards": "Update Cards", + "step-finalize": "Finalize", + "step-delete-duplicate-empty-lists": "Delete Duplicate Empty Lists", + "step-ensure-lost-cards-swimlane": "Ensure Lost Cards Swimlane", + "step-restore-lists": "Restore Lists", + "step-restore-cards": "Restore Cards", + "step-restore-swimlanes": "Restore Swimlanes", + "step-fix-missing-ids": "Fix Missing IDs", + "step-scan-users": "Checking board member avatars", + "step-scan-files": "Checking board file attachments", + "step-fix-file-urls": "Fixing file URLs", "cleanup": "Cleanup", "cleanup-old-jobs": "Cleanup Old Jobs", "completed": "Completed", diff --git a/imports/i18n/data/en_ZA.i18n.json b/imports/i18n/data/en_ZA.i18n.json index eef27e1fe..958d8313a 100644 --- a/imports/i18n/data/en_ZA.i18n.json +++ b/imports/i18n/data/en_ZA.i18n.json @@ -1404,7 +1404,70 @@ "back-to-settings": "Back to Settings", "board-id": "Board ID", "board-migration": "Board Migration", + "board-migrations": "Board Migrations", "card-show-lists-on-minicard": "Show Lists on Minicard", + "comprehensive-board-migration": "Comprehensive Board Migration", + "comprehensive-board-migration-description": "Performs comprehensive checks and fixes for board data integrity, including list ordering, card positions, and swimlane structure.", + "delete-duplicate-empty-lists-migration": "Delete Duplicate Empty Lists", + "delete-duplicate-empty-lists-migration-description": "Safely deletes empty duplicate lists. Only removes lists that have no cards AND have another list with the same title that contains cards.", + "lost-cards": "Lost Cards", + "lost-cards-list": "Restored Items", + "restore-lost-cards-migration": "Restore Lost Cards", + "restore-lost-cards-migration-description": "Finds and restores cards and lists with missing swimlaneId or listId. Creates a 'Lost Cards' swimlane to make all lost items visible again.", + "restore-all-archived-migration": "Restore All Archived", + "restore-all-archived-migration-description": "Restores all archived swimlanes, lists, and cards. Automatically fixes any missing swimlaneId or listId to make items visible.", + "fix-missing-lists-migration": "Fix Missing Lists", + "fix-missing-lists-migration-description": "Detects and repairs missing or corrupted lists in the board structure.", + "fix-avatar-urls-migration": "Fix Avatar URLs", + "fix-avatar-urls-migration-description": "Updates avatar URLs for board members to use the correct storage backend and fixes broken avatar references.", + "fix-all-file-urls-migration": "Fix All File URLs", + "fix-all-file-urls-migration-description": "Updates all file attachment URLs on this board to use the correct storage backend and fixes broken file references.", + "migration-needed": "Migration Needed", + "migration-complete": "Complete", + "migration-running": "Running...", + "migration-successful": "Migration completed successfully", + "migration-failed": "Migration failed", + "migrations": "Migrations", + "migrations-admin-only": "Only board administrators can run migrations", + "migrations-description": "Run data integrity checks and repairs for this board. Each migration can be executed individually.", + "no-issues-found": "No issues found", + "run-migration": "Run Migration", + "run-comprehensive-migration-confirm": "This will perform a comprehensive migration to check and fix board data integrity. This may take a few moments. Continue?", + "run-delete-duplicate-empty-lists-migration-confirm": "This will first convert any shared lists to per-swimlane lists, then delete empty lists that have a duplicate list with the same title containing cards. Only truly redundant empty lists will be removed. Continue?", + "run-restore-lost-cards-migration-confirm": "This will create a 'Lost Cards' swimlane and restore all cards and lists with missing swimlaneId or listId. This only affects non-archived items. Continue?", + "run-restore-all-archived-migration-confirm": "This will restore ALL archived swimlanes, lists, and cards, making them visible again. Any items with missing IDs will be automatically fixed. This cannot be easily undone. Continue?", + "run-fix-missing-lists-migration-confirm": "This will detect and repair missing or corrupted lists in the board structure. Continue?", + "run-fix-avatar-urls-migration-confirm": "This will update avatar URLs for board members to use the correct storage backend. Continue?", + "run-fix-all-file-urls-migration-confirm": "This will update all file attachment URLs on this board to use the correct storage backend. Continue?", + "restore-lost-cards-nothing-to-restore": "No lost swimlanes, lists, or cards to restore", + + "migration-progress-title": "Board Migration in Progress", + "migration-progress-overall": "Overall Progress", + "migration-progress-current-step": "Current Step", + "migration-progress-status": "Status", + "migration-progress-details": "Details", + "migration-progress-note": "Please wait while we migrate your board to the latest structure...", + + "step-analyze-board-structure": "Analyze Board Structure", + "step-fix-orphaned-cards": "Fix Orphaned Cards", + "step-convert-shared-lists": "Convert Shared Lists", + "step-ensure-per-swimlane-lists": "Ensure Per-Swimlane Lists", + "step-validate-migration": "Validate Migration", + "step-fix-avatar-urls": "Fix Avatar URLs", + "step-fix-attachment-urls": "Fix Attachment URLs", + "step-analyze-lists": "Analyze Lists", + "step-create-missing-lists": "Create Missing Lists", + "step-update-cards": "Update Cards", + "step-finalize": "Finalize", + "step-delete-duplicate-empty-lists": "Delete Duplicate Empty Lists", + "step-ensure-lost-cards-swimlane": "Ensure Lost Cards Swimlane", + "step-restore-lists": "Restore Lists", + "step-restore-cards": "Restore Cards", + "step-restore-swimlanes": "Restore Swimlanes", + "step-fix-missing-ids": "Fix Missing IDs", + "step-scan-users": "Checking board member avatars", + "step-scan-files": "Checking board file attachments", + "step-fix-file-urls": "Fixing file URLs", "cleanup": "Cleanup", "cleanup-old-jobs": "Cleanup Old Jobs", "completed": "Completed", diff --git a/imports/i18n/data/eo.i18n.json b/imports/i18n/data/eo.i18n.json index 20a85f9f6..f26ef6c5e 100644 --- a/imports/i18n/data/eo.i18n.json +++ b/imports/i18n/data/eo.i18n.json @@ -1404,7 +1404,70 @@ "back-to-settings": "Back to Settings", "board-id": "Board ID", "board-migration": "Board Migration", + "board-migrations": "Board Migrations", "card-show-lists-on-minicard": "Show Lists on Minicard", + "comprehensive-board-migration": "Comprehensive Board Migration", + "comprehensive-board-migration-description": "Performs comprehensive checks and fixes for board data integrity, including list ordering, card positions, and swimlane structure.", + "delete-duplicate-empty-lists-migration": "Delete Duplicate Empty Lists", + "delete-duplicate-empty-lists-migration-description": "Safely deletes empty duplicate lists. Only removes lists that have no cards AND have another list with the same title that contains cards.", + "lost-cards": "Lost Cards", + "lost-cards-list": "Restored Items", + "restore-lost-cards-migration": "Restore Lost Cards", + "restore-lost-cards-migration-description": "Finds and restores cards and lists with missing swimlaneId or listId. Creates a 'Lost Cards' swimlane to make all lost items visible again.", + "restore-all-archived-migration": "Restore All Archived", + "restore-all-archived-migration-description": "Restores all archived swimlanes, lists, and cards. Automatically fixes any missing swimlaneId or listId to make items visible.", + "fix-missing-lists-migration": "Fix Missing Lists", + "fix-missing-lists-migration-description": "Detects and repairs missing or corrupted lists in the board structure.", + "fix-avatar-urls-migration": "Fix Avatar URLs", + "fix-avatar-urls-migration-description": "Updates avatar URLs for board members to use the correct storage backend and fixes broken avatar references.", + "fix-all-file-urls-migration": "Fix All File URLs", + "fix-all-file-urls-migration-description": "Updates all file attachment URLs on this board to use the correct storage backend and fixes broken file references.", + "migration-needed": "Migration Needed", + "migration-complete": "Complete", + "migration-running": "Running...", + "migration-successful": "Migration completed successfully", + "migration-failed": "Migration failed", + "migrations": "Migrations", + "migrations-admin-only": "Only board administrators can run migrations", + "migrations-description": "Run data integrity checks and repairs for this board. Each migration can be executed individually.", + "no-issues-found": "No issues found", + "run-migration": "Run Migration", + "run-comprehensive-migration-confirm": "This will perform a comprehensive migration to check and fix board data integrity. This may take a few moments. Continue?", + "run-delete-duplicate-empty-lists-migration-confirm": "This will first convert any shared lists to per-swimlane lists, then delete empty lists that have a duplicate list with the same title containing cards. Only truly redundant empty lists will be removed. Continue?", + "run-restore-lost-cards-migration-confirm": "This will create a 'Lost Cards' swimlane and restore all cards and lists with missing swimlaneId or listId. This only affects non-archived items. Continue?", + "run-restore-all-archived-migration-confirm": "This will restore ALL archived swimlanes, lists, and cards, making them visible again. Any items with missing IDs will be automatically fixed. This cannot be easily undone. Continue?", + "run-fix-missing-lists-migration-confirm": "This will detect and repair missing or corrupted lists in the board structure. Continue?", + "run-fix-avatar-urls-migration-confirm": "This will update avatar URLs for board members to use the correct storage backend. Continue?", + "run-fix-all-file-urls-migration-confirm": "This will update all file attachment URLs on this board to use the correct storage backend. Continue?", + "restore-lost-cards-nothing-to-restore": "No lost swimlanes, lists, or cards to restore", + + "migration-progress-title": "Board Migration in Progress", + "migration-progress-overall": "Overall Progress", + "migration-progress-current-step": "Current Step", + "migration-progress-status": "Status", + "migration-progress-details": "Details", + "migration-progress-note": "Please wait while we migrate your board to the latest structure...", + + "step-analyze-board-structure": "Analyze Board Structure", + "step-fix-orphaned-cards": "Fix Orphaned Cards", + "step-convert-shared-lists": "Convert Shared Lists", + "step-ensure-per-swimlane-lists": "Ensure Per-Swimlane Lists", + "step-validate-migration": "Validate Migration", + "step-fix-avatar-urls": "Fix Avatar URLs", + "step-fix-attachment-urls": "Fix Attachment URLs", + "step-analyze-lists": "Analyze Lists", + "step-create-missing-lists": "Create Missing Lists", + "step-update-cards": "Update Cards", + "step-finalize": "Finalize", + "step-delete-duplicate-empty-lists": "Delete Duplicate Empty Lists", + "step-ensure-lost-cards-swimlane": "Ensure Lost Cards Swimlane", + "step-restore-lists": "Restore Lists", + "step-restore-cards": "Restore Cards", + "step-restore-swimlanes": "Restore Swimlanes", + "step-fix-missing-ids": "Fix Missing IDs", + "step-scan-users": "Checking board member avatars", + "step-scan-files": "Checking board file attachments", + "step-fix-file-urls": "Fixing file URLs", "cleanup": "Cleanup", "cleanup-old-jobs": "Cleanup Old Jobs", "completed": "Completed", diff --git a/imports/i18n/data/es-AR.i18n.json b/imports/i18n/data/es-AR.i18n.json index 874d60c15..9b87e09ab 100644 --- a/imports/i18n/data/es-AR.i18n.json +++ b/imports/i18n/data/es-AR.i18n.json @@ -1404,7 +1404,70 @@ "back-to-settings": "Back to Settings", "board-id": "Board ID", "board-migration": "Board Migration", + "board-migrations": "Board Migrations", "card-show-lists-on-minicard": "Show Lists on Minicard", + "comprehensive-board-migration": "Comprehensive Board Migration", + "comprehensive-board-migration-description": "Performs comprehensive checks and fixes for board data integrity, including list ordering, card positions, and swimlane structure.", + "delete-duplicate-empty-lists-migration": "Delete Duplicate Empty Lists", + "delete-duplicate-empty-lists-migration-description": "Safely deletes empty duplicate lists. Only removes lists that have no cards AND have another list with the same title that contains cards.", + "lost-cards": "Lost Cards", + "lost-cards-list": "Restored Items", + "restore-lost-cards-migration": "Restore Lost Cards", + "restore-lost-cards-migration-description": "Finds and restores cards and lists with missing swimlaneId or listId. Creates a 'Lost Cards' swimlane to make all lost items visible again.", + "restore-all-archived-migration": "Restore All Archived", + "restore-all-archived-migration-description": "Restores all archived swimlanes, lists, and cards. Automatically fixes any missing swimlaneId or listId to make items visible.", + "fix-missing-lists-migration": "Fix Missing Lists", + "fix-missing-lists-migration-description": "Detects and repairs missing or corrupted lists in the board structure.", + "fix-avatar-urls-migration": "Fix Avatar URLs", + "fix-avatar-urls-migration-description": "Updates avatar URLs for board members to use the correct storage backend and fixes broken avatar references.", + "fix-all-file-urls-migration": "Fix All File URLs", + "fix-all-file-urls-migration-description": "Updates all file attachment URLs on this board to use the correct storage backend and fixes broken file references.", + "migration-needed": "Migration Needed", + "migration-complete": "Complete", + "migration-running": "Running...", + "migration-successful": "Migration completed successfully", + "migration-failed": "Migration failed", + "migrations": "Migrations", + "migrations-admin-only": "Only board administrators can run migrations", + "migrations-description": "Run data integrity checks and repairs for this board. Each migration can be executed individually.", + "no-issues-found": "No issues found", + "run-migration": "Run Migration", + "run-comprehensive-migration-confirm": "This will perform a comprehensive migration to check and fix board data integrity. This may take a few moments. Continue?", + "run-delete-duplicate-empty-lists-migration-confirm": "This will first convert any shared lists to per-swimlane lists, then delete empty lists that have a duplicate list with the same title containing cards. Only truly redundant empty lists will be removed. Continue?", + "run-restore-lost-cards-migration-confirm": "This will create a 'Lost Cards' swimlane and restore all cards and lists with missing swimlaneId or listId. This only affects non-archived items. Continue?", + "run-restore-all-archived-migration-confirm": "This will restore ALL archived swimlanes, lists, and cards, making them visible again. Any items with missing IDs will be automatically fixed. This cannot be easily undone. Continue?", + "run-fix-missing-lists-migration-confirm": "This will detect and repair missing or corrupted lists in the board structure. Continue?", + "run-fix-avatar-urls-migration-confirm": "This will update avatar URLs for board members to use the correct storage backend. Continue?", + "run-fix-all-file-urls-migration-confirm": "This will update all file attachment URLs on this board to use the correct storage backend. Continue?", + "restore-lost-cards-nothing-to-restore": "No lost swimlanes, lists, or cards to restore", + + "migration-progress-title": "Board Migration in Progress", + "migration-progress-overall": "Overall Progress", + "migration-progress-current-step": "Current Step", + "migration-progress-status": "Status", + "migration-progress-details": "Details", + "migration-progress-note": "Please wait while we migrate your board to the latest structure...", + + "step-analyze-board-structure": "Analyze Board Structure", + "step-fix-orphaned-cards": "Fix Orphaned Cards", + "step-convert-shared-lists": "Convert Shared Lists", + "step-ensure-per-swimlane-lists": "Ensure Per-Swimlane Lists", + "step-validate-migration": "Validate Migration", + "step-fix-avatar-urls": "Fix Avatar URLs", + "step-fix-attachment-urls": "Fix Attachment URLs", + "step-analyze-lists": "Analyze Lists", + "step-create-missing-lists": "Create Missing Lists", + "step-update-cards": "Update Cards", + "step-finalize": "Finalize", + "step-delete-duplicate-empty-lists": "Delete Duplicate Empty Lists", + "step-ensure-lost-cards-swimlane": "Ensure Lost Cards Swimlane", + "step-restore-lists": "Restore Lists", + "step-restore-cards": "Restore Cards", + "step-restore-swimlanes": "Restore Swimlanes", + "step-fix-missing-ids": "Fix Missing IDs", + "step-scan-users": "Checking board member avatars", + "step-scan-files": "Checking board file attachments", + "step-fix-file-urls": "Fixing file URLs", "cleanup": "Cleanup", "cleanup-old-jobs": "Cleanup Old Jobs", "completed": "Completed", diff --git a/imports/i18n/data/es-CL.i18n.json b/imports/i18n/data/es-CL.i18n.json index 6b167dc52..6adcd95b6 100644 --- a/imports/i18n/data/es-CL.i18n.json +++ b/imports/i18n/data/es-CL.i18n.json @@ -1404,7 +1404,70 @@ "back-to-settings": "Back to Settings", "board-id": "Board ID", "board-migration": "Board Migration", + "board-migrations": "Board Migrations", "card-show-lists-on-minicard": "Show Lists on Minicard", + "comprehensive-board-migration": "Comprehensive Board Migration", + "comprehensive-board-migration-description": "Performs comprehensive checks and fixes for board data integrity, including list ordering, card positions, and swimlane structure.", + "delete-duplicate-empty-lists-migration": "Delete Duplicate Empty Lists", + "delete-duplicate-empty-lists-migration-description": "Safely deletes empty duplicate lists. Only removes lists that have no cards AND have another list with the same title that contains cards.", + "lost-cards": "Lost Cards", + "lost-cards-list": "Restored Items", + "restore-lost-cards-migration": "Restore Lost Cards", + "restore-lost-cards-migration-description": "Finds and restores cards and lists with missing swimlaneId or listId. Creates a 'Lost Cards' swimlane to make all lost items visible again.", + "restore-all-archived-migration": "Restore All Archived", + "restore-all-archived-migration-description": "Restores all archived swimlanes, lists, and cards. Automatically fixes any missing swimlaneId or listId to make items visible.", + "fix-missing-lists-migration": "Fix Missing Lists", + "fix-missing-lists-migration-description": "Detects and repairs missing or corrupted lists in the board structure.", + "fix-avatar-urls-migration": "Fix Avatar URLs", + "fix-avatar-urls-migration-description": "Updates avatar URLs for board members to use the correct storage backend and fixes broken avatar references.", + "fix-all-file-urls-migration": "Fix All File URLs", + "fix-all-file-urls-migration-description": "Updates all file attachment URLs on this board to use the correct storage backend and fixes broken file references.", + "migration-needed": "Migration Needed", + "migration-complete": "Complete", + "migration-running": "Running...", + "migration-successful": "Migration completed successfully", + "migration-failed": "Migration failed", + "migrations": "Migrations", + "migrations-admin-only": "Only board administrators can run migrations", + "migrations-description": "Run data integrity checks and repairs for this board. Each migration can be executed individually.", + "no-issues-found": "No issues found", + "run-migration": "Run Migration", + "run-comprehensive-migration-confirm": "This will perform a comprehensive migration to check and fix board data integrity. This may take a few moments. Continue?", + "run-delete-duplicate-empty-lists-migration-confirm": "This will first convert any shared lists to per-swimlane lists, then delete empty lists that have a duplicate list with the same title containing cards. Only truly redundant empty lists will be removed. Continue?", + "run-restore-lost-cards-migration-confirm": "This will create a 'Lost Cards' swimlane and restore all cards and lists with missing swimlaneId or listId. This only affects non-archived items. Continue?", + "run-restore-all-archived-migration-confirm": "This will restore ALL archived swimlanes, lists, and cards, making them visible again. Any items with missing IDs will be automatically fixed. This cannot be easily undone. Continue?", + "run-fix-missing-lists-migration-confirm": "This will detect and repair missing or corrupted lists in the board structure. Continue?", + "run-fix-avatar-urls-migration-confirm": "This will update avatar URLs for board members to use the correct storage backend. Continue?", + "run-fix-all-file-urls-migration-confirm": "This will update all file attachment URLs on this board to use the correct storage backend. Continue?", + "restore-lost-cards-nothing-to-restore": "No lost swimlanes, lists, or cards to restore", + + "migration-progress-title": "Board Migration in Progress", + "migration-progress-overall": "Overall Progress", + "migration-progress-current-step": "Current Step", + "migration-progress-status": "Status", + "migration-progress-details": "Details", + "migration-progress-note": "Please wait while we migrate your board to the latest structure...", + + "step-analyze-board-structure": "Analyze Board Structure", + "step-fix-orphaned-cards": "Fix Orphaned Cards", + "step-convert-shared-lists": "Convert Shared Lists", + "step-ensure-per-swimlane-lists": "Ensure Per-Swimlane Lists", + "step-validate-migration": "Validate Migration", + "step-fix-avatar-urls": "Fix Avatar URLs", + "step-fix-attachment-urls": "Fix Attachment URLs", + "step-analyze-lists": "Analyze Lists", + "step-create-missing-lists": "Create Missing Lists", + "step-update-cards": "Update Cards", + "step-finalize": "Finalize", + "step-delete-duplicate-empty-lists": "Delete Duplicate Empty Lists", + "step-ensure-lost-cards-swimlane": "Ensure Lost Cards Swimlane", + "step-restore-lists": "Restore Lists", + "step-restore-cards": "Restore Cards", + "step-restore-swimlanes": "Restore Swimlanes", + "step-fix-missing-ids": "Fix Missing IDs", + "step-scan-users": "Checking board member avatars", + "step-scan-files": "Checking board file attachments", + "step-fix-file-urls": "Fixing file URLs", "cleanup": "Cleanup", "cleanup-old-jobs": "Cleanup Old Jobs", "completed": "Completada", diff --git a/imports/i18n/data/es-LA.i18n.json b/imports/i18n/data/es-LA.i18n.json index eef27e1fe..958d8313a 100644 --- a/imports/i18n/data/es-LA.i18n.json +++ b/imports/i18n/data/es-LA.i18n.json @@ -1404,7 +1404,70 @@ "back-to-settings": "Back to Settings", "board-id": "Board ID", "board-migration": "Board Migration", + "board-migrations": "Board Migrations", "card-show-lists-on-minicard": "Show Lists on Minicard", + "comprehensive-board-migration": "Comprehensive Board Migration", + "comprehensive-board-migration-description": "Performs comprehensive checks and fixes for board data integrity, including list ordering, card positions, and swimlane structure.", + "delete-duplicate-empty-lists-migration": "Delete Duplicate Empty Lists", + "delete-duplicate-empty-lists-migration-description": "Safely deletes empty duplicate lists. Only removes lists that have no cards AND have another list with the same title that contains cards.", + "lost-cards": "Lost Cards", + "lost-cards-list": "Restored Items", + "restore-lost-cards-migration": "Restore Lost Cards", + "restore-lost-cards-migration-description": "Finds and restores cards and lists with missing swimlaneId or listId. Creates a 'Lost Cards' swimlane to make all lost items visible again.", + "restore-all-archived-migration": "Restore All Archived", + "restore-all-archived-migration-description": "Restores all archived swimlanes, lists, and cards. Automatically fixes any missing swimlaneId or listId to make items visible.", + "fix-missing-lists-migration": "Fix Missing Lists", + "fix-missing-lists-migration-description": "Detects and repairs missing or corrupted lists in the board structure.", + "fix-avatar-urls-migration": "Fix Avatar URLs", + "fix-avatar-urls-migration-description": "Updates avatar URLs for board members to use the correct storage backend and fixes broken avatar references.", + "fix-all-file-urls-migration": "Fix All File URLs", + "fix-all-file-urls-migration-description": "Updates all file attachment URLs on this board to use the correct storage backend and fixes broken file references.", + "migration-needed": "Migration Needed", + "migration-complete": "Complete", + "migration-running": "Running...", + "migration-successful": "Migration completed successfully", + "migration-failed": "Migration failed", + "migrations": "Migrations", + "migrations-admin-only": "Only board administrators can run migrations", + "migrations-description": "Run data integrity checks and repairs for this board. Each migration can be executed individually.", + "no-issues-found": "No issues found", + "run-migration": "Run Migration", + "run-comprehensive-migration-confirm": "This will perform a comprehensive migration to check and fix board data integrity. This may take a few moments. Continue?", + "run-delete-duplicate-empty-lists-migration-confirm": "This will first convert any shared lists to per-swimlane lists, then delete empty lists that have a duplicate list with the same title containing cards. Only truly redundant empty lists will be removed. Continue?", + "run-restore-lost-cards-migration-confirm": "This will create a 'Lost Cards' swimlane and restore all cards and lists with missing swimlaneId or listId. This only affects non-archived items. Continue?", + "run-restore-all-archived-migration-confirm": "This will restore ALL archived swimlanes, lists, and cards, making them visible again. Any items with missing IDs will be automatically fixed. This cannot be easily undone. Continue?", + "run-fix-missing-lists-migration-confirm": "This will detect and repair missing or corrupted lists in the board structure. Continue?", + "run-fix-avatar-urls-migration-confirm": "This will update avatar URLs for board members to use the correct storage backend. Continue?", + "run-fix-all-file-urls-migration-confirm": "This will update all file attachment URLs on this board to use the correct storage backend. Continue?", + "restore-lost-cards-nothing-to-restore": "No lost swimlanes, lists, or cards to restore", + + "migration-progress-title": "Board Migration in Progress", + "migration-progress-overall": "Overall Progress", + "migration-progress-current-step": "Current Step", + "migration-progress-status": "Status", + "migration-progress-details": "Details", + "migration-progress-note": "Please wait while we migrate your board to the latest structure...", + + "step-analyze-board-structure": "Analyze Board Structure", + "step-fix-orphaned-cards": "Fix Orphaned Cards", + "step-convert-shared-lists": "Convert Shared Lists", + "step-ensure-per-swimlane-lists": "Ensure Per-Swimlane Lists", + "step-validate-migration": "Validate Migration", + "step-fix-avatar-urls": "Fix Avatar URLs", + "step-fix-attachment-urls": "Fix Attachment URLs", + "step-analyze-lists": "Analyze Lists", + "step-create-missing-lists": "Create Missing Lists", + "step-update-cards": "Update Cards", + "step-finalize": "Finalize", + "step-delete-duplicate-empty-lists": "Delete Duplicate Empty Lists", + "step-ensure-lost-cards-swimlane": "Ensure Lost Cards Swimlane", + "step-restore-lists": "Restore Lists", + "step-restore-cards": "Restore Cards", + "step-restore-swimlanes": "Restore Swimlanes", + "step-fix-missing-ids": "Fix Missing IDs", + "step-scan-users": "Checking board member avatars", + "step-scan-files": "Checking board file attachments", + "step-fix-file-urls": "Fixing file URLs", "cleanup": "Cleanup", "cleanup-old-jobs": "Cleanup Old Jobs", "completed": "Completed", diff --git a/imports/i18n/data/es-MX.i18n.json b/imports/i18n/data/es-MX.i18n.json index 858a623c9..97a451610 100644 --- a/imports/i18n/data/es-MX.i18n.json +++ b/imports/i18n/data/es-MX.i18n.json @@ -1404,7 +1404,70 @@ "back-to-settings": "Back to Settings", "board-id": "Board ID", "board-migration": "Board Migration", + "board-migrations": "Board Migrations", "card-show-lists-on-minicard": "Show Lists on Minicard", + "comprehensive-board-migration": "Comprehensive Board Migration", + "comprehensive-board-migration-description": "Performs comprehensive checks and fixes for board data integrity, including list ordering, card positions, and swimlane structure.", + "delete-duplicate-empty-lists-migration": "Delete Duplicate Empty Lists", + "delete-duplicate-empty-lists-migration-description": "Safely deletes empty duplicate lists. Only removes lists that have no cards AND have another list with the same title that contains cards.", + "lost-cards": "Lost Cards", + "lost-cards-list": "Restored Items", + "restore-lost-cards-migration": "Restore Lost Cards", + "restore-lost-cards-migration-description": "Finds and restores cards and lists with missing swimlaneId or listId. Creates a 'Lost Cards' swimlane to make all lost items visible again.", + "restore-all-archived-migration": "Restore All Archived", + "restore-all-archived-migration-description": "Restores all archived swimlanes, lists, and cards. Automatically fixes any missing swimlaneId or listId to make items visible.", + "fix-missing-lists-migration": "Fix Missing Lists", + "fix-missing-lists-migration-description": "Detects and repairs missing or corrupted lists in the board structure.", + "fix-avatar-urls-migration": "Fix Avatar URLs", + "fix-avatar-urls-migration-description": "Updates avatar URLs for board members to use the correct storage backend and fixes broken avatar references.", + "fix-all-file-urls-migration": "Fix All File URLs", + "fix-all-file-urls-migration-description": "Updates all file attachment URLs on this board to use the correct storage backend and fixes broken file references.", + "migration-needed": "Migration Needed", + "migration-complete": "Complete", + "migration-running": "Running...", + "migration-successful": "Migration completed successfully", + "migration-failed": "Migration failed", + "migrations": "Migrations", + "migrations-admin-only": "Only board administrators can run migrations", + "migrations-description": "Run data integrity checks and repairs for this board. Each migration can be executed individually.", + "no-issues-found": "No issues found", + "run-migration": "Run Migration", + "run-comprehensive-migration-confirm": "This will perform a comprehensive migration to check and fix board data integrity. This may take a few moments. Continue?", + "run-delete-duplicate-empty-lists-migration-confirm": "This will first convert any shared lists to per-swimlane lists, then delete empty lists that have a duplicate list with the same title containing cards. Only truly redundant empty lists will be removed. Continue?", + "run-restore-lost-cards-migration-confirm": "This will create a 'Lost Cards' swimlane and restore all cards and lists with missing swimlaneId or listId. This only affects non-archived items. Continue?", + "run-restore-all-archived-migration-confirm": "This will restore ALL archived swimlanes, lists, and cards, making them visible again. Any items with missing IDs will be automatically fixed. This cannot be easily undone. Continue?", + "run-fix-missing-lists-migration-confirm": "This will detect and repair missing or corrupted lists in the board structure. Continue?", + "run-fix-avatar-urls-migration-confirm": "This will update avatar URLs for board members to use the correct storage backend. Continue?", + "run-fix-all-file-urls-migration-confirm": "This will update all file attachment URLs on this board to use the correct storage backend. Continue?", + "restore-lost-cards-nothing-to-restore": "No lost swimlanes, lists, or cards to restore", + + "migration-progress-title": "Board Migration in Progress", + "migration-progress-overall": "Overall Progress", + "migration-progress-current-step": "Current Step", + "migration-progress-status": "Status", + "migration-progress-details": "Details", + "migration-progress-note": "Please wait while we migrate your board to the latest structure...", + + "step-analyze-board-structure": "Analyze Board Structure", + "step-fix-orphaned-cards": "Fix Orphaned Cards", + "step-convert-shared-lists": "Convert Shared Lists", + "step-ensure-per-swimlane-lists": "Ensure Per-Swimlane Lists", + "step-validate-migration": "Validate Migration", + "step-fix-avatar-urls": "Fix Avatar URLs", + "step-fix-attachment-urls": "Fix Attachment URLs", + "step-analyze-lists": "Analyze Lists", + "step-create-missing-lists": "Create Missing Lists", + "step-update-cards": "Update Cards", + "step-finalize": "Finalize", + "step-delete-duplicate-empty-lists": "Delete Duplicate Empty Lists", + "step-ensure-lost-cards-swimlane": "Ensure Lost Cards Swimlane", + "step-restore-lists": "Restore Lists", + "step-restore-cards": "Restore Cards", + "step-restore-swimlanes": "Restore Swimlanes", + "step-fix-missing-ids": "Fix Missing IDs", + "step-scan-users": "Checking board member avatars", + "step-scan-files": "Checking board file attachments", + "step-fix-file-urls": "Fixing file URLs", "cleanup": "Cleanup", "cleanup-old-jobs": "Cleanup Old Jobs", "completed": "Completed", diff --git a/imports/i18n/data/es-PE.i18n.json b/imports/i18n/data/es-PE.i18n.json index 276b1d368..4deb3a96d 100644 --- a/imports/i18n/data/es-PE.i18n.json +++ b/imports/i18n/data/es-PE.i18n.json @@ -1404,7 +1404,70 @@ "back-to-settings": "Back to Settings", "board-id": "Board ID", "board-migration": "Board Migration", + "board-migrations": "Board Migrations", "card-show-lists-on-minicard": "Show Lists on Minicard", + "comprehensive-board-migration": "Comprehensive Board Migration", + "comprehensive-board-migration-description": "Performs comprehensive checks and fixes for board data integrity, including list ordering, card positions, and swimlane structure.", + "delete-duplicate-empty-lists-migration": "Delete Duplicate Empty Lists", + "delete-duplicate-empty-lists-migration-description": "Safely deletes empty duplicate lists. Only removes lists that have no cards AND have another list with the same title that contains cards.", + "lost-cards": "Lost Cards", + "lost-cards-list": "Restored Items", + "restore-lost-cards-migration": "Restore Lost Cards", + "restore-lost-cards-migration-description": "Finds and restores cards and lists with missing swimlaneId or listId. Creates a 'Lost Cards' swimlane to make all lost items visible again.", + "restore-all-archived-migration": "Restore All Archived", + "restore-all-archived-migration-description": "Restores all archived swimlanes, lists, and cards. Automatically fixes any missing swimlaneId or listId to make items visible.", + "fix-missing-lists-migration": "Fix Missing Lists", + "fix-missing-lists-migration-description": "Detects and repairs missing or corrupted lists in the board structure.", + "fix-avatar-urls-migration": "Fix Avatar URLs", + "fix-avatar-urls-migration-description": "Updates avatar URLs for board members to use the correct storage backend and fixes broken avatar references.", + "fix-all-file-urls-migration": "Fix All File URLs", + "fix-all-file-urls-migration-description": "Updates all file attachment URLs on this board to use the correct storage backend and fixes broken file references.", + "migration-needed": "Migration Needed", + "migration-complete": "Complete", + "migration-running": "Running...", + "migration-successful": "Migration completed successfully", + "migration-failed": "Migration failed", + "migrations": "Migrations", + "migrations-admin-only": "Only board administrators can run migrations", + "migrations-description": "Run data integrity checks and repairs for this board. Each migration can be executed individually.", + "no-issues-found": "No issues found", + "run-migration": "Run Migration", + "run-comprehensive-migration-confirm": "This will perform a comprehensive migration to check and fix board data integrity. This may take a few moments. Continue?", + "run-delete-duplicate-empty-lists-migration-confirm": "This will first convert any shared lists to per-swimlane lists, then delete empty lists that have a duplicate list with the same title containing cards. Only truly redundant empty lists will be removed. Continue?", + "run-restore-lost-cards-migration-confirm": "This will create a 'Lost Cards' swimlane and restore all cards and lists with missing swimlaneId or listId. This only affects non-archived items. Continue?", + "run-restore-all-archived-migration-confirm": "This will restore ALL archived swimlanes, lists, and cards, making them visible again. Any items with missing IDs will be automatically fixed. This cannot be easily undone. Continue?", + "run-fix-missing-lists-migration-confirm": "This will detect and repair missing or corrupted lists in the board structure. Continue?", + "run-fix-avatar-urls-migration-confirm": "This will update avatar URLs for board members to use the correct storage backend. Continue?", + "run-fix-all-file-urls-migration-confirm": "This will update all file attachment URLs on this board to use the correct storage backend. Continue?", + "restore-lost-cards-nothing-to-restore": "No lost swimlanes, lists, or cards to restore", + + "migration-progress-title": "Board Migration in Progress", + "migration-progress-overall": "Overall Progress", + "migration-progress-current-step": "Current Step", + "migration-progress-status": "Estado", + "migration-progress-details": "Details", + "migration-progress-note": "Please wait while we migrate your board to the latest structure...", + + "step-analyze-board-structure": "Analyze Board Structure", + "step-fix-orphaned-cards": "Fix Orphaned Cards", + "step-convert-shared-lists": "Convert Shared Lists", + "step-ensure-per-swimlane-lists": "Ensure Per-Swimlane Lists", + "step-validate-migration": "Validate Migration", + "step-fix-avatar-urls": "Fix Avatar URLs", + "step-fix-attachment-urls": "Fix Attachment URLs", + "step-analyze-lists": "Analyze Lists", + "step-create-missing-lists": "Create Missing Lists", + "step-update-cards": "Update Cards", + "step-finalize": "Finalize", + "step-delete-duplicate-empty-lists": "Delete Duplicate Empty Lists", + "step-ensure-lost-cards-swimlane": "Ensure Lost Cards Swimlane", + "step-restore-lists": "Restore Lists", + "step-restore-cards": "Restore Cards", + "step-restore-swimlanes": "Restore Swimlanes", + "step-fix-missing-ids": "Fix Missing IDs", + "step-scan-users": "Checking board member avatars", + "step-scan-files": "Checking board file attachments", + "step-fix-file-urls": "Fixing file URLs", "cleanup": "Cleanup", "cleanup-old-jobs": "Cleanup Old Jobs", "completed": "Completada", diff --git a/imports/i18n/data/es-PY.i18n.json b/imports/i18n/data/es-PY.i18n.json index eef27e1fe..958d8313a 100644 --- a/imports/i18n/data/es-PY.i18n.json +++ b/imports/i18n/data/es-PY.i18n.json @@ -1404,7 +1404,70 @@ "back-to-settings": "Back to Settings", "board-id": "Board ID", "board-migration": "Board Migration", + "board-migrations": "Board Migrations", "card-show-lists-on-minicard": "Show Lists on Minicard", + "comprehensive-board-migration": "Comprehensive Board Migration", + "comprehensive-board-migration-description": "Performs comprehensive checks and fixes for board data integrity, including list ordering, card positions, and swimlane structure.", + "delete-duplicate-empty-lists-migration": "Delete Duplicate Empty Lists", + "delete-duplicate-empty-lists-migration-description": "Safely deletes empty duplicate lists. Only removes lists that have no cards AND have another list with the same title that contains cards.", + "lost-cards": "Lost Cards", + "lost-cards-list": "Restored Items", + "restore-lost-cards-migration": "Restore Lost Cards", + "restore-lost-cards-migration-description": "Finds and restores cards and lists with missing swimlaneId or listId. Creates a 'Lost Cards' swimlane to make all lost items visible again.", + "restore-all-archived-migration": "Restore All Archived", + "restore-all-archived-migration-description": "Restores all archived swimlanes, lists, and cards. Automatically fixes any missing swimlaneId or listId to make items visible.", + "fix-missing-lists-migration": "Fix Missing Lists", + "fix-missing-lists-migration-description": "Detects and repairs missing or corrupted lists in the board structure.", + "fix-avatar-urls-migration": "Fix Avatar URLs", + "fix-avatar-urls-migration-description": "Updates avatar URLs for board members to use the correct storage backend and fixes broken avatar references.", + "fix-all-file-urls-migration": "Fix All File URLs", + "fix-all-file-urls-migration-description": "Updates all file attachment URLs on this board to use the correct storage backend and fixes broken file references.", + "migration-needed": "Migration Needed", + "migration-complete": "Complete", + "migration-running": "Running...", + "migration-successful": "Migration completed successfully", + "migration-failed": "Migration failed", + "migrations": "Migrations", + "migrations-admin-only": "Only board administrators can run migrations", + "migrations-description": "Run data integrity checks and repairs for this board. Each migration can be executed individually.", + "no-issues-found": "No issues found", + "run-migration": "Run Migration", + "run-comprehensive-migration-confirm": "This will perform a comprehensive migration to check and fix board data integrity. This may take a few moments. Continue?", + "run-delete-duplicate-empty-lists-migration-confirm": "This will first convert any shared lists to per-swimlane lists, then delete empty lists that have a duplicate list with the same title containing cards. Only truly redundant empty lists will be removed. Continue?", + "run-restore-lost-cards-migration-confirm": "This will create a 'Lost Cards' swimlane and restore all cards and lists with missing swimlaneId or listId. This only affects non-archived items. Continue?", + "run-restore-all-archived-migration-confirm": "This will restore ALL archived swimlanes, lists, and cards, making them visible again. Any items with missing IDs will be automatically fixed. This cannot be easily undone. Continue?", + "run-fix-missing-lists-migration-confirm": "This will detect and repair missing or corrupted lists in the board structure. Continue?", + "run-fix-avatar-urls-migration-confirm": "This will update avatar URLs for board members to use the correct storage backend. Continue?", + "run-fix-all-file-urls-migration-confirm": "This will update all file attachment URLs on this board to use the correct storage backend. Continue?", + "restore-lost-cards-nothing-to-restore": "No lost swimlanes, lists, or cards to restore", + + "migration-progress-title": "Board Migration in Progress", + "migration-progress-overall": "Overall Progress", + "migration-progress-current-step": "Current Step", + "migration-progress-status": "Status", + "migration-progress-details": "Details", + "migration-progress-note": "Please wait while we migrate your board to the latest structure...", + + "step-analyze-board-structure": "Analyze Board Structure", + "step-fix-orphaned-cards": "Fix Orphaned Cards", + "step-convert-shared-lists": "Convert Shared Lists", + "step-ensure-per-swimlane-lists": "Ensure Per-Swimlane Lists", + "step-validate-migration": "Validate Migration", + "step-fix-avatar-urls": "Fix Avatar URLs", + "step-fix-attachment-urls": "Fix Attachment URLs", + "step-analyze-lists": "Analyze Lists", + "step-create-missing-lists": "Create Missing Lists", + "step-update-cards": "Update Cards", + "step-finalize": "Finalize", + "step-delete-duplicate-empty-lists": "Delete Duplicate Empty Lists", + "step-ensure-lost-cards-swimlane": "Ensure Lost Cards Swimlane", + "step-restore-lists": "Restore Lists", + "step-restore-cards": "Restore Cards", + "step-restore-swimlanes": "Restore Swimlanes", + "step-fix-missing-ids": "Fix Missing IDs", + "step-scan-users": "Checking board member avatars", + "step-scan-files": "Checking board file attachments", + "step-fix-file-urls": "Fixing file URLs", "cleanup": "Cleanup", "cleanup-old-jobs": "Cleanup Old Jobs", "completed": "Completed", diff --git a/imports/i18n/data/es.i18n.json b/imports/i18n/data/es.i18n.json index c0ea65cd5..ad4d63edb 100644 --- a/imports/i18n/data/es.i18n.json +++ b/imports/i18n/data/es.i18n.json @@ -1404,7 +1404,70 @@ "back-to-settings": "Back to Settings", "board-id": "Board ID", "board-migration": "Board Migration", + "board-migrations": "Board Migrations", "card-show-lists-on-minicard": "Show Lists on Minicard", + "comprehensive-board-migration": "Comprehensive Board Migration", + "comprehensive-board-migration-description": "Performs comprehensive checks and fixes for board data integrity, including list ordering, card positions, and swimlane structure.", + "delete-duplicate-empty-lists-migration": "Delete Duplicate Empty Lists", + "delete-duplicate-empty-lists-migration-description": "Safely deletes empty duplicate lists. Only removes lists that have no cards AND have another list with the same title that contains cards.", + "lost-cards": "Lost Cards", + "lost-cards-list": "Restored Items", + "restore-lost-cards-migration": "Restore Lost Cards", + "restore-lost-cards-migration-description": "Finds and restores cards and lists with missing swimlaneId or listId. Creates a 'Lost Cards' swimlane to make all lost items visible again.", + "restore-all-archived-migration": "Restore All Archived", + "restore-all-archived-migration-description": "Restores all archived swimlanes, lists, and cards. Automatically fixes any missing swimlaneId or listId to make items visible.", + "fix-missing-lists-migration": "Fix Missing Lists", + "fix-missing-lists-migration-description": "Detects and repairs missing or corrupted lists in the board structure.", + "fix-avatar-urls-migration": "Fix Avatar URLs", + "fix-avatar-urls-migration-description": "Updates avatar URLs for board members to use the correct storage backend and fixes broken avatar references.", + "fix-all-file-urls-migration": "Fix All File URLs", + "fix-all-file-urls-migration-description": "Updates all file attachment URLs on this board to use the correct storage backend and fixes broken file references.", + "migration-needed": "Migration Needed", + "migration-complete": "Completado", + "migration-running": "Running...", + "migration-successful": "Migration completed successfully", + "migration-failed": "Migration failed", + "migrations": "Migrations", + "migrations-admin-only": "Only board administrators can run migrations", + "migrations-description": "Run data integrity checks and repairs for this board. Each migration can be executed individually.", + "no-issues-found": "No issues found", + "run-migration": "Run Migration", + "run-comprehensive-migration-confirm": "This will perform a comprehensive migration to check and fix board data integrity. This may take a few moments. Continue?", + "run-delete-duplicate-empty-lists-migration-confirm": "This will first convert any shared lists to per-swimlane lists, then delete empty lists that have a duplicate list with the same title containing cards. Only truly redundant empty lists will be removed. Continue?", + "run-restore-lost-cards-migration-confirm": "This will create a 'Lost Cards' swimlane and restore all cards and lists with missing swimlaneId or listId. This only affects non-archived items. Continue?", + "run-restore-all-archived-migration-confirm": "This will restore ALL archived swimlanes, lists, and cards, making them visible again. Any items with missing IDs will be automatically fixed. This cannot be easily undone. Continue?", + "run-fix-missing-lists-migration-confirm": "This will detect and repair missing or corrupted lists in the board structure. Continue?", + "run-fix-avatar-urls-migration-confirm": "This will update avatar URLs for board members to use the correct storage backend. Continue?", + "run-fix-all-file-urls-migration-confirm": "This will update all file attachment URLs on this board to use the correct storage backend. Continue?", + "restore-lost-cards-nothing-to-restore": "No lost swimlanes, lists, or cards to restore", + + "migration-progress-title": "Board Migration in Progress", + "migration-progress-overall": "Overall Progress", + "migration-progress-current-step": "Current Step", + "migration-progress-status": "Estado", + "migration-progress-details": "Detalles", + "migration-progress-note": "Please wait while we migrate your board to the latest structure...", + + "step-analyze-board-structure": "Analyze Board Structure", + "step-fix-orphaned-cards": "Fix Orphaned Cards", + "step-convert-shared-lists": "Convert Shared Lists", + "step-ensure-per-swimlane-lists": "Ensure Per-Swimlane Lists", + "step-validate-migration": "Validate Migration", + "step-fix-avatar-urls": "Fix Avatar URLs", + "step-fix-attachment-urls": "Fix Attachment URLs", + "step-analyze-lists": "Analyze Lists", + "step-create-missing-lists": "Create Missing Lists", + "step-update-cards": "Update Cards", + "step-finalize": "Finalize", + "step-delete-duplicate-empty-lists": "Delete Duplicate Empty Lists", + "step-ensure-lost-cards-swimlane": "Ensure Lost Cards Swimlane", + "step-restore-lists": "Restore Lists", + "step-restore-cards": "Restore Cards", + "step-restore-swimlanes": "Restore Swimlanes", + "step-fix-missing-ids": "Fix Missing IDs", + "step-scan-users": "Checking board member avatars", + "step-scan-files": "Checking board file attachments", + "step-fix-file-urls": "Fixing file URLs", "cleanup": "Cleanup", "cleanup-old-jobs": "Cleanup Old Jobs", "completed": "Completada", diff --git a/imports/i18n/data/es_CO.i18n.json b/imports/i18n/data/es_CO.i18n.json index eef27e1fe..958d8313a 100644 --- a/imports/i18n/data/es_CO.i18n.json +++ b/imports/i18n/data/es_CO.i18n.json @@ -1404,7 +1404,70 @@ "back-to-settings": "Back to Settings", "board-id": "Board ID", "board-migration": "Board Migration", + "board-migrations": "Board Migrations", "card-show-lists-on-minicard": "Show Lists on Minicard", + "comprehensive-board-migration": "Comprehensive Board Migration", + "comprehensive-board-migration-description": "Performs comprehensive checks and fixes for board data integrity, including list ordering, card positions, and swimlane structure.", + "delete-duplicate-empty-lists-migration": "Delete Duplicate Empty Lists", + "delete-duplicate-empty-lists-migration-description": "Safely deletes empty duplicate lists. Only removes lists that have no cards AND have another list with the same title that contains cards.", + "lost-cards": "Lost Cards", + "lost-cards-list": "Restored Items", + "restore-lost-cards-migration": "Restore Lost Cards", + "restore-lost-cards-migration-description": "Finds and restores cards and lists with missing swimlaneId or listId. Creates a 'Lost Cards' swimlane to make all lost items visible again.", + "restore-all-archived-migration": "Restore All Archived", + "restore-all-archived-migration-description": "Restores all archived swimlanes, lists, and cards. Automatically fixes any missing swimlaneId or listId to make items visible.", + "fix-missing-lists-migration": "Fix Missing Lists", + "fix-missing-lists-migration-description": "Detects and repairs missing or corrupted lists in the board structure.", + "fix-avatar-urls-migration": "Fix Avatar URLs", + "fix-avatar-urls-migration-description": "Updates avatar URLs for board members to use the correct storage backend and fixes broken avatar references.", + "fix-all-file-urls-migration": "Fix All File URLs", + "fix-all-file-urls-migration-description": "Updates all file attachment URLs on this board to use the correct storage backend and fixes broken file references.", + "migration-needed": "Migration Needed", + "migration-complete": "Complete", + "migration-running": "Running...", + "migration-successful": "Migration completed successfully", + "migration-failed": "Migration failed", + "migrations": "Migrations", + "migrations-admin-only": "Only board administrators can run migrations", + "migrations-description": "Run data integrity checks and repairs for this board. Each migration can be executed individually.", + "no-issues-found": "No issues found", + "run-migration": "Run Migration", + "run-comprehensive-migration-confirm": "This will perform a comprehensive migration to check and fix board data integrity. This may take a few moments. Continue?", + "run-delete-duplicate-empty-lists-migration-confirm": "This will first convert any shared lists to per-swimlane lists, then delete empty lists that have a duplicate list with the same title containing cards. Only truly redundant empty lists will be removed. Continue?", + "run-restore-lost-cards-migration-confirm": "This will create a 'Lost Cards' swimlane and restore all cards and lists with missing swimlaneId or listId. This only affects non-archived items. Continue?", + "run-restore-all-archived-migration-confirm": "This will restore ALL archived swimlanes, lists, and cards, making them visible again. Any items with missing IDs will be automatically fixed. This cannot be easily undone. Continue?", + "run-fix-missing-lists-migration-confirm": "This will detect and repair missing or corrupted lists in the board structure. Continue?", + "run-fix-avatar-urls-migration-confirm": "This will update avatar URLs for board members to use the correct storage backend. Continue?", + "run-fix-all-file-urls-migration-confirm": "This will update all file attachment URLs on this board to use the correct storage backend. Continue?", + "restore-lost-cards-nothing-to-restore": "No lost swimlanes, lists, or cards to restore", + + "migration-progress-title": "Board Migration in Progress", + "migration-progress-overall": "Overall Progress", + "migration-progress-current-step": "Current Step", + "migration-progress-status": "Status", + "migration-progress-details": "Details", + "migration-progress-note": "Please wait while we migrate your board to the latest structure...", + + "step-analyze-board-structure": "Analyze Board Structure", + "step-fix-orphaned-cards": "Fix Orphaned Cards", + "step-convert-shared-lists": "Convert Shared Lists", + "step-ensure-per-swimlane-lists": "Ensure Per-Swimlane Lists", + "step-validate-migration": "Validate Migration", + "step-fix-avatar-urls": "Fix Avatar URLs", + "step-fix-attachment-urls": "Fix Attachment URLs", + "step-analyze-lists": "Analyze Lists", + "step-create-missing-lists": "Create Missing Lists", + "step-update-cards": "Update Cards", + "step-finalize": "Finalize", + "step-delete-duplicate-empty-lists": "Delete Duplicate Empty Lists", + "step-ensure-lost-cards-swimlane": "Ensure Lost Cards Swimlane", + "step-restore-lists": "Restore Lists", + "step-restore-cards": "Restore Cards", + "step-restore-swimlanes": "Restore Swimlanes", + "step-fix-missing-ids": "Fix Missing IDs", + "step-scan-users": "Checking board member avatars", + "step-scan-files": "Checking board file attachments", + "step-fix-file-urls": "Fixing file URLs", "cleanup": "Cleanup", "cleanup-old-jobs": "Cleanup Old Jobs", "completed": "Completed", diff --git a/imports/i18n/data/et-EE.i18n.json b/imports/i18n/data/et-EE.i18n.json index 40bf924c4..64bb83a99 100644 --- a/imports/i18n/data/et-EE.i18n.json +++ b/imports/i18n/data/et-EE.i18n.json @@ -1404,7 +1404,70 @@ "back-to-settings": "Back to Settings", "board-id": "Board ID", "board-migration": "Board Migration", + "board-migrations": "Board Migrations", "card-show-lists-on-minicard": "Show Lists on Minicard", + "comprehensive-board-migration": "Comprehensive Board Migration", + "comprehensive-board-migration-description": "Performs comprehensive checks and fixes for board data integrity, including list ordering, card positions, and swimlane structure.", + "delete-duplicate-empty-lists-migration": "Delete Duplicate Empty Lists", + "delete-duplicate-empty-lists-migration-description": "Safely deletes empty duplicate lists. Only removes lists that have no cards AND have another list with the same title that contains cards.", + "lost-cards": "Lost Cards", + "lost-cards-list": "Restored Items", + "restore-lost-cards-migration": "Restore Lost Cards", + "restore-lost-cards-migration-description": "Finds and restores cards and lists with missing swimlaneId or listId. Creates a 'Lost Cards' swimlane to make all lost items visible again.", + "restore-all-archived-migration": "Restore All Archived", + "restore-all-archived-migration-description": "Restores all archived swimlanes, lists, and cards. Automatically fixes any missing swimlaneId or listId to make items visible.", + "fix-missing-lists-migration": "Fix Missing Lists", + "fix-missing-lists-migration-description": "Detects and repairs missing or corrupted lists in the board structure.", + "fix-avatar-urls-migration": "Fix Avatar URLs", + "fix-avatar-urls-migration-description": "Updates avatar URLs for board members to use the correct storage backend and fixes broken avatar references.", + "fix-all-file-urls-migration": "Fix All File URLs", + "fix-all-file-urls-migration-description": "Updates all file attachment URLs on this board to use the correct storage backend and fixes broken file references.", + "migration-needed": "Migration Needed", + "migration-complete": "Complete", + "migration-running": "Running...", + "migration-successful": "Migration completed successfully", + "migration-failed": "Migration failed", + "migrations": "Migrations", + "migrations-admin-only": "Only board administrators can run migrations", + "migrations-description": "Run data integrity checks and repairs for this board. Each migration can be executed individually.", + "no-issues-found": "No issues found", + "run-migration": "Run Migration", + "run-comprehensive-migration-confirm": "This will perform a comprehensive migration to check and fix board data integrity. This may take a few moments. Continue?", + "run-delete-duplicate-empty-lists-migration-confirm": "This will first convert any shared lists to per-swimlane lists, then delete empty lists that have a duplicate list with the same title containing cards. Only truly redundant empty lists will be removed. Continue?", + "run-restore-lost-cards-migration-confirm": "This will create a 'Lost Cards' swimlane and restore all cards and lists with missing swimlaneId or listId. This only affects non-archived items. Continue?", + "run-restore-all-archived-migration-confirm": "This will restore ALL archived swimlanes, lists, and cards, making them visible again. Any items with missing IDs will be automatically fixed. This cannot be easily undone. Continue?", + "run-fix-missing-lists-migration-confirm": "This will detect and repair missing or corrupted lists in the board structure. Continue?", + "run-fix-avatar-urls-migration-confirm": "This will update avatar URLs for board members to use the correct storage backend. Continue?", + "run-fix-all-file-urls-migration-confirm": "This will update all file attachment URLs on this board to use the correct storage backend. Continue?", + "restore-lost-cards-nothing-to-restore": "No lost swimlanes, lists, or cards to restore", + + "migration-progress-title": "Board Migration in Progress", + "migration-progress-overall": "Overall Progress", + "migration-progress-current-step": "Current Step", + "migration-progress-status": "Staatus", + "migration-progress-details": "Üksikasjad", + "migration-progress-note": "Please wait while we migrate your board to the latest structure...", + + "step-analyze-board-structure": "Analyze Board Structure", + "step-fix-orphaned-cards": "Fix Orphaned Cards", + "step-convert-shared-lists": "Convert Shared Lists", + "step-ensure-per-swimlane-lists": "Ensure Per-Swimlane Lists", + "step-validate-migration": "Validate Migration", + "step-fix-avatar-urls": "Fix Avatar URLs", + "step-fix-attachment-urls": "Fix Attachment URLs", + "step-analyze-lists": "Analyze Lists", + "step-create-missing-lists": "Create Missing Lists", + "step-update-cards": "Update Cards", + "step-finalize": "Finalize", + "step-delete-duplicate-empty-lists": "Delete Duplicate Empty Lists", + "step-ensure-lost-cards-swimlane": "Ensure Lost Cards Swimlane", + "step-restore-lists": "Restore Lists", + "step-restore-cards": "Restore Cards", + "step-restore-swimlanes": "Restore Swimlanes", + "step-fix-missing-ids": "Fix Missing IDs", + "step-scan-users": "Checking board member avatars", + "step-scan-files": "Checking board file attachments", + "step-fix-file-urls": "Fixing file URLs", "cleanup": "Cleanup", "cleanup-old-jobs": "Cleanup Old Jobs", "completed": "Lõpetatud", diff --git a/imports/i18n/data/eu.i18n.json b/imports/i18n/data/eu.i18n.json index 84dd9440d..91e716031 100644 --- a/imports/i18n/data/eu.i18n.json +++ b/imports/i18n/data/eu.i18n.json @@ -1404,7 +1404,70 @@ "back-to-settings": "Back to Settings", "board-id": "Board ID", "board-migration": "Board Migration", + "board-migrations": "Board Migrations", "card-show-lists-on-minicard": "Show Lists on Minicard", + "comprehensive-board-migration": "Comprehensive Board Migration", + "comprehensive-board-migration-description": "Performs comprehensive checks and fixes for board data integrity, including list ordering, card positions, and swimlane structure.", + "delete-duplicate-empty-lists-migration": "Delete Duplicate Empty Lists", + "delete-duplicate-empty-lists-migration-description": "Safely deletes empty duplicate lists. Only removes lists that have no cards AND have another list with the same title that contains cards.", + "lost-cards": "Lost Cards", + "lost-cards-list": "Restored Items", + "restore-lost-cards-migration": "Restore Lost Cards", + "restore-lost-cards-migration-description": "Finds and restores cards and lists with missing swimlaneId or listId. Creates a 'Lost Cards' swimlane to make all lost items visible again.", + "restore-all-archived-migration": "Restore All Archived", + "restore-all-archived-migration-description": "Restores all archived swimlanes, lists, and cards. Automatically fixes any missing swimlaneId or listId to make items visible.", + "fix-missing-lists-migration": "Fix Missing Lists", + "fix-missing-lists-migration-description": "Detects and repairs missing or corrupted lists in the board structure.", + "fix-avatar-urls-migration": "Fix Avatar URLs", + "fix-avatar-urls-migration-description": "Updates avatar URLs for board members to use the correct storage backend and fixes broken avatar references.", + "fix-all-file-urls-migration": "Fix All File URLs", + "fix-all-file-urls-migration-description": "Updates all file attachment URLs on this board to use the correct storage backend and fixes broken file references.", + "migration-needed": "Migration Needed", + "migration-complete": "Complete", + "migration-running": "Running...", + "migration-successful": "Migration completed successfully", + "migration-failed": "Migration failed", + "migrations": "Migrations", + "migrations-admin-only": "Only board administrators can run migrations", + "migrations-description": "Run data integrity checks and repairs for this board. Each migration can be executed individually.", + "no-issues-found": "No issues found", + "run-migration": "Run Migration", + "run-comprehensive-migration-confirm": "This will perform a comprehensive migration to check and fix board data integrity. This may take a few moments. Continue?", + "run-delete-duplicate-empty-lists-migration-confirm": "This will first convert any shared lists to per-swimlane lists, then delete empty lists that have a duplicate list with the same title containing cards. Only truly redundant empty lists will be removed. Continue?", + "run-restore-lost-cards-migration-confirm": "This will create a 'Lost Cards' swimlane and restore all cards and lists with missing swimlaneId or listId. This only affects non-archived items. Continue?", + "run-restore-all-archived-migration-confirm": "This will restore ALL archived swimlanes, lists, and cards, making them visible again. Any items with missing IDs will be automatically fixed. This cannot be easily undone. Continue?", + "run-fix-missing-lists-migration-confirm": "This will detect and repair missing or corrupted lists in the board structure. Continue?", + "run-fix-avatar-urls-migration-confirm": "This will update avatar URLs for board members to use the correct storage backend. Continue?", + "run-fix-all-file-urls-migration-confirm": "This will update all file attachment URLs on this board to use the correct storage backend. Continue?", + "restore-lost-cards-nothing-to-restore": "No lost swimlanes, lists, or cards to restore", + + "migration-progress-title": "Board Migration in Progress", + "migration-progress-overall": "Overall Progress", + "migration-progress-current-step": "Current Step", + "migration-progress-status": "Status", + "migration-progress-details": "Details", + "migration-progress-note": "Please wait while we migrate your board to the latest structure...", + + "step-analyze-board-structure": "Analyze Board Structure", + "step-fix-orphaned-cards": "Fix Orphaned Cards", + "step-convert-shared-lists": "Convert Shared Lists", + "step-ensure-per-swimlane-lists": "Ensure Per-Swimlane Lists", + "step-validate-migration": "Validate Migration", + "step-fix-avatar-urls": "Fix Avatar URLs", + "step-fix-attachment-urls": "Fix Attachment URLs", + "step-analyze-lists": "Analyze Lists", + "step-create-missing-lists": "Create Missing Lists", + "step-update-cards": "Update Cards", + "step-finalize": "Finalize", + "step-delete-duplicate-empty-lists": "Delete Duplicate Empty Lists", + "step-ensure-lost-cards-swimlane": "Ensure Lost Cards Swimlane", + "step-restore-lists": "Restore Lists", + "step-restore-cards": "Restore Cards", + "step-restore-swimlanes": "Restore Swimlanes", + "step-fix-missing-ids": "Fix Missing IDs", + "step-scan-users": "Checking board member avatars", + "step-scan-files": "Checking board file attachments", + "step-fix-file-urls": "Fixing file URLs", "cleanup": "Cleanup", "cleanup-old-jobs": "Cleanup Old Jobs", "completed": "Completed", diff --git a/imports/i18n/data/fa-IR.i18n.json b/imports/i18n/data/fa-IR.i18n.json index ddfb6a36d..8662becce 100644 --- a/imports/i18n/data/fa-IR.i18n.json +++ b/imports/i18n/data/fa-IR.i18n.json @@ -1404,7 +1404,70 @@ "back-to-settings": "Back to Settings", "board-id": "Board ID", "board-migration": "Board Migration", + "board-migrations": "Board Migrations", "card-show-lists-on-minicard": "Show Lists on Minicard", + "comprehensive-board-migration": "Comprehensive Board Migration", + "comprehensive-board-migration-description": "Performs comprehensive checks and fixes for board data integrity, including list ordering, card positions, and swimlane structure.", + "delete-duplicate-empty-lists-migration": "Delete Duplicate Empty Lists", + "delete-duplicate-empty-lists-migration-description": "Safely deletes empty duplicate lists. Only removes lists that have no cards AND have another list with the same title that contains cards.", + "lost-cards": "Lost Cards", + "lost-cards-list": "Restored Items", + "restore-lost-cards-migration": "Restore Lost Cards", + "restore-lost-cards-migration-description": "Finds and restores cards and lists with missing swimlaneId or listId. Creates a 'Lost Cards' swimlane to make all lost items visible again.", + "restore-all-archived-migration": "Restore All Archived", + "restore-all-archived-migration-description": "Restores all archived swimlanes, lists, and cards. Automatically fixes any missing swimlaneId or listId to make items visible.", + "fix-missing-lists-migration": "Fix Missing Lists", + "fix-missing-lists-migration-description": "Detects and repairs missing or corrupted lists in the board structure.", + "fix-avatar-urls-migration": "Fix Avatar URLs", + "fix-avatar-urls-migration-description": "Updates avatar URLs for board members to use the correct storage backend and fixes broken avatar references.", + "fix-all-file-urls-migration": "Fix All File URLs", + "fix-all-file-urls-migration-description": "Updates all file attachment URLs on this board to use the correct storage backend and fixes broken file references.", + "migration-needed": "Migration Needed", + "migration-complete": "Complete", + "migration-running": "Running...", + "migration-successful": "Migration completed successfully", + "migration-failed": "Migration failed", + "migrations": "Migrations", + "migrations-admin-only": "Only board administrators can run migrations", + "migrations-description": "Run data integrity checks and repairs for this board. Each migration can be executed individually.", + "no-issues-found": "No issues found", + "run-migration": "Run Migration", + "run-comprehensive-migration-confirm": "This will perform a comprehensive migration to check and fix board data integrity. This may take a few moments. Continue?", + "run-delete-duplicate-empty-lists-migration-confirm": "This will first convert any shared lists to per-swimlane lists, then delete empty lists that have a duplicate list with the same title containing cards. Only truly redundant empty lists will be removed. Continue?", + "run-restore-lost-cards-migration-confirm": "This will create a 'Lost Cards' swimlane and restore all cards and lists with missing swimlaneId or listId. This only affects non-archived items. Continue?", + "run-restore-all-archived-migration-confirm": "This will restore ALL archived swimlanes, lists, and cards, making them visible again. Any items with missing IDs will be automatically fixed. This cannot be easily undone. Continue?", + "run-fix-missing-lists-migration-confirm": "This will detect and repair missing or corrupted lists in the board structure. Continue?", + "run-fix-avatar-urls-migration-confirm": "This will update avatar URLs for board members to use the correct storage backend. Continue?", + "run-fix-all-file-urls-migration-confirm": "This will update all file attachment URLs on this board to use the correct storage backend. Continue?", + "restore-lost-cards-nothing-to-restore": "No lost swimlanes, lists, or cards to restore", + + "migration-progress-title": "Board Migration in Progress", + "migration-progress-overall": "Overall Progress", + "migration-progress-current-step": "Current Step", + "migration-progress-status": "وضعیت", + "migration-progress-details": "Details", + "migration-progress-note": "Please wait while we migrate your board to the latest structure...", + + "step-analyze-board-structure": "Analyze Board Structure", + "step-fix-orphaned-cards": "Fix Orphaned Cards", + "step-convert-shared-lists": "Convert Shared Lists", + "step-ensure-per-swimlane-lists": "Ensure Per-Swimlane Lists", + "step-validate-migration": "Validate Migration", + "step-fix-avatar-urls": "Fix Avatar URLs", + "step-fix-attachment-urls": "Fix Attachment URLs", + "step-analyze-lists": "Analyze Lists", + "step-create-missing-lists": "Create Missing Lists", + "step-update-cards": "Update Cards", + "step-finalize": "Finalize", + "step-delete-duplicate-empty-lists": "Delete Duplicate Empty Lists", + "step-ensure-lost-cards-swimlane": "Ensure Lost Cards Swimlane", + "step-restore-lists": "Restore Lists", + "step-restore-cards": "Restore Cards", + "step-restore-swimlanes": "Restore Swimlanes", + "step-fix-missing-ids": "Fix Missing IDs", + "step-scan-users": "Checking board member avatars", + "step-scan-files": "Checking board file attachments", + "step-fix-file-urls": "Fixing file URLs", "cleanup": "Cleanup", "cleanup-old-jobs": "Cleanup Old Jobs", "completed": "تمام شده", diff --git a/imports/i18n/data/fa.i18n.json b/imports/i18n/data/fa.i18n.json index 38d280360..6c44ea6f2 100644 --- a/imports/i18n/data/fa.i18n.json +++ b/imports/i18n/data/fa.i18n.json @@ -1404,7 +1404,70 @@ "back-to-settings": "Back to Settings", "board-id": "Board ID", "board-migration": "Board Migration", + "board-migrations": "Board Migrations", "card-show-lists-on-minicard": "Show Lists on Minicard", + "comprehensive-board-migration": "Comprehensive Board Migration", + "comprehensive-board-migration-description": "Performs comprehensive checks and fixes for board data integrity, including list ordering, card positions, and swimlane structure.", + "delete-duplicate-empty-lists-migration": "Delete Duplicate Empty Lists", + "delete-duplicate-empty-lists-migration-description": "Safely deletes empty duplicate lists. Only removes lists that have no cards AND have another list with the same title that contains cards.", + "lost-cards": "Lost Cards", + "lost-cards-list": "Restored Items", + "restore-lost-cards-migration": "Restore Lost Cards", + "restore-lost-cards-migration-description": "Finds and restores cards and lists with missing swimlaneId or listId. Creates a 'Lost Cards' swimlane to make all lost items visible again.", + "restore-all-archived-migration": "Restore All Archived", + "restore-all-archived-migration-description": "Restores all archived swimlanes, lists, and cards. Automatically fixes any missing swimlaneId or listId to make items visible.", + "fix-missing-lists-migration": "Fix Missing Lists", + "fix-missing-lists-migration-description": "Detects and repairs missing or corrupted lists in the board structure.", + "fix-avatar-urls-migration": "Fix Avatar URLs", + "fix-avatar-urls-migration-description": "Updates avatar URLs for board members to use the correct storage backend and fixes broken avatar references.", + "fix-all-file-urls-migration": "Fix All File URLs", + "fix-all-file-urls-migration-description": "Updates all file attachment URLs on this board to use the correct storage backend and fixes broken file references.", + "migration-needed": "Migration Needed", + "migration-complete": "Complete", + "migration-running": "Running...", + "migration-successful": "Migration completed successfully", + "migration-failed": "Migration failed", + "migrations": "Migrations", + "migrations-admin-only": "Only board administrators can run migrations", + "migrations-description": "Run data integrity checks and repairs for this board. Each migration can be executed individually.", + "no-issues-found": "No issues found", + "run-migration": "Run Migration", + "run-comprehensive-migration-confirm": "This will perform a comprehensive migration to check and fix board data integrity. This may take a few moments. Continue?", + "run-delete-duplicate-empty-lists-migration-confirm": "This will first convert any shared lists to per-swimlane lists, then delete empty lists that have a duplicate list with the same title containing cards. Only truly redundant empty lists will be removed. Continue?", + "run-restore-lost-cards-migration-confirm": "This will create a 'Lost Cards' swimlane and restore all cards and lists with missing swimlaneId or listId. This only affects non-archived items. Continue?", + "run-restore-all-archived-migration-confirm": "This will restore ALL archived swimlanes, lists, and cards, making them visible again. Any items with missing IDs will be automatically fixed. This cannot be easily undone. Continue?", + "run-fix-missing-lists-migration-confirm": "This will detect and repair missing or corrupted lists in the board structure. Continue?", + "run-fix-avatar-urls-migration-confirm": "This will update avatar URLs for board members to use the correct storage backend. Continue?", + "run-fix-all-file-urls-migration-confirm": "This will update all file attachment URLs on this board to use the correct storage backend. Continue?", + "restore-lost-cards-nothing-to-restore": "No lost swimlanes, lists, or cards to restore", + + "migration-progress-title": "Board Migration in Progress", + "migration-progress-overall": "Overall Progress", + "migration-progress-current-step": "Current Step", + "migration-progress-status": "وضعیت", + "migration-progress-details": "Details", + "migration-progress-note": "Please wait while we migrate your board to the latest structure...", + + "step-analyze-board-structure": "Analyze Board Structure", + "step-fix-orphaned-cards": "Fix Orphaned Cards", + "step-convert-shared-lists": "Convert Shared Lists", + "step-ensure-per-swimlane-lists": "Ensure Per-Swimlane Lists", + "step-validate-migration": "Validate Migration", + "step-fix-avatar-urls": "Fix Avatar URLs", + "step-fix-attachment-urls": "Fix Attachment URLs", + "step-analyze-lists": "Analyze Lists", + "step-create-missing-lists": "Create Missing Lists", + "step-update-cards": "Update Cards", + "step-finalize": "Finalize", + "step-delete-duplicate-empty-lists": "Delete Duplicate Empty Lists", + "step-ensure-lost-cards-swimlane": "Ensure Lost Cards Swimlane", + "step-restore-lists": "Restore Lists", + "step-restore-cards": "Restore Cards", + "step-restore-swimlanes": "Restore Swimlanes", + "step-fix-missing-ids": "Fix Missing IDs", + "step-scan-users": "Checking board member avatars", + "step-scan-files": "Checking board file attachments", + "step-fix-file-urls": "Fixing file URLs", "cleanup": "Cleanup", "cleanup-old-jobs": "Cleanup Old Jobs", "completed": "تمام شده", diff --git a/imports/i18n/data/fi.i18n.json b/imports/i18n/data/fi.i18n.json index cdec49369..54b041ab3 100644 --- a/imports/i18n/data/fi.i18n.json +++ b/imports/i18n/data/fi.i18n.json @@ -1404,7 +1404,70 @@ "back-to-settings": "Takaisin asetuksiin", "board-id": "Taulun tunnus", "board-migration": "Taulun siirto", + "board-migrations": "Taulu migraatiot", "card-show-lists-on-minicard": "Näytä listat minikortilla", + "comprehensive-board-migration": "Perusteellinen taulu migraatio", + "comprehensive-board-migration-description": "Suorittaa kattavia tarkistuksia ja korjauksia taulun tietojen eheyden varmistamiseksi, mukaan lukien listajärjestyksen, korttien sijainnit ja uimaratarakenteen.", + "delete-duplicate-empty-lists-migration": "Poista kaksoiskappaleet tyhjistä listoista", + "delete-duplicate-empty-lists-migration-description": "Poistaa tyhjät kaksoiskappalelistat turvallisesti. Poistaa vain listat, joissa ei ole kortteja JA joilla on toinen samanniminen lista, joka sisältää kortteja.", + "lost-cards": "Kadonneet kortit", + "lost-cards-list": "Palautetut kohteet", + "restore-lost-cards-migration": "Palauta kadonneet kortit", + "restore-lost-cards-migration-description": "Etsii ja palauttaa kortit ja listat, joista puuttuu swimlaneId tai listId. Luo 'Kadonneet kortit' -uimaradan, jotta kaikki kadonneet ovat taas näkyvissä.", + "restore-all-archived-migration": "Palauta kaikki arkistoidut", + "restore-all-archived-migration-description": "Palauttaa kaikki arkistoidut uimaradat, listat ja kortit. Korjaa automaattisesti puuttuvat uimaratatunnukset tai listatunnukset, jotta kohteet ovat näkyvissä.", + "fix-missing-lists-migration": "Korjaa puuttuvat listat", + "fix-missing-lists-migration-description": "Havaitsee ja korjaa puuttuvat tai vioittuneet listat taulun rakenteessa.", + "fix-avatar-urls-migration": "Korjaa avatar-URL-osoitteet", + "fix-avatar-urls-migration-description": "Päivittää taulun jäsenten avatar-osoitteiden URL-osoitteet oikean tallennustilan käyttämiseksi ja korjaa rikkinäiset avatar-viittaukset.", + "fix-all-file-urls-migration": "Korjaa kaikki tiedostojen URL-osoitteet", + "fix-all-file-urls-migration-description": "Päivittää kaikkien tällä taululla olevien tiedostoliitteiden URL-osoitteet käyttämään oikeaa tallennuspalvelinta ja korjaa rikkinäiset tiedostoviittaukset.", + "migration-needed": "Migraatio tarvitaan", + "migration-complete": "Valmis", + "migration-running": "Suoritetaan...", + "migration-successful": "Migraatio valmistui onnistuneesti", + "migration-failed": "Migraatio epäonnistui", + "migrations": "Migraatiot", + "migrations-admin-only": "Vain taulu ylläpitäjät voivat suorittaa migraatioita", + "migrations-description": "Suorita tietojen eheystarkistukset ja korjaukset tälle taululle. Jokainen migraatio voidaan suorittaa erikseen.", + "no-issues-found": "Ei löytynyt ongelmia", + "run-migration": "Suorita migraatio", + "run-comprehensive-migration-confirm": "Tämä suorittaa kattavan migraation, jolla tarkistetaan ja korjataan taulun tietojen eheys. Tämä voi kestää hetken. Jatketaanko?", + "run-delete-duplicate-empty-lists-migration-confirm": "Tämä muuntaa ensin kaikki jaetut listat uimaratakohtaisiksi listoiksi ja poistaa sitten tyhjät listat, joissa on samanniminen kaksoiskappale, joka sisältää kortteja. Vain todella tarpeettomat tyhjät listat poistetaan. Jatketaanko?", + "run-restore-lost-cards-migration-confirm": "Tämä luo Kadonneet kortit -uimaradan ja palauttaa kaikki kortit ja listat, joista puuttuu uimaradan tunnus tai listan tunnus. Tämä vaikuttaa vain arkistoimattomiin kohteisiin. Jatketaanko?", + "run-restore-all-archived-migration-confirm": "Tämä palauttaa KAIKKI arkistoidut uintikaistat, listat ja kortit, jolloin ne näkyvät taas. Puuttuvista tunnuksista puuttuvat kohteet korjataan automaattisesti. Tätä ei voi helposti perua. Jatketaanko?", + "run-fix-missing-lists-migration-confirm": "Tämä havaitsee ja korjaa puuttuvat tai vioittuneet listat taulun rakenteessa. Jatketaanko?", + "run-fix-avatar-urls-migration-confirm": "Tämä päivittää taulun jäsenten avatar-URL-osoitteet käyttämään oikeaa tallennustilaa. Jatketaanko?", + "run-fix-all-file-urls-migration-confirm": "Tämä päivittää kaikkien tällä taululla olevien tiedostoliitteiden URL-osoitteet käyttämään oikeaa tallennuspalvelinta. Jatketaanko?", + "restore-lost-cards-nothing-to-restore": "Ei kadonneita uintikaistoja, listoja tai kortteja palautettavaksi", + + "migration-progress-title": "Taulu migraatio meneillään", + "migration-progress-overall": "Kokonaisedistyminen", + "migration-progress-current-step": "Nykyinen vaihe", + "migration-progress-status": "Tilanne", + "migration-progress-details": "Yksityiskohdat", + "migration-progress-note": "Odota hetki, siirrämme taulusi uusimpaan rakenteeseen...", + + "step-analyze-board-structure": "Analysoi taulun rakennetta", + "step-fix-orphaned-cards": "Korjaa orvot kortit", + "step-convert-shared-lists": "Muunna jaetut listat", + "step-ensure-per-swimlane-lists": "Varmista uimaratakohtaiset listat", + "step-validate-migration": "Varmistetaan migraatio", + "step-fix-avatar-urls": "Korjaa avatar-URL-osoitteet", + "step-fix-attachment-urls": "Korjaa liitetiedosto URLit", + "step-analyze-lists": "Analysoidaan listoja", + "step-create-missing-lists": "Luo puuttuvat listat", + "step-update-cards": "Päivitä kortit", + "step-finalize": "Viimeistellään", + "step-delete-duplicate-empty-lists": "Poista kaksoiskappaleet tyhjistä listoista", + "step-ensure-lost-cards-swimlane": "Varmistetaan hävinneiden korttien uimarata", + "step-restore-lists": "Palauta listat", + "step-restore-cards": "Palauta kortit", + "step-restore-swimlanes": "Palauta uimaradat", + "step-fix-missing-ids": "Korjaa puuttuvat ID:t", + "step-scan-users": "Tarkistetaan taulun jäsenten avatarit", + "step-scan-files": "Tarkistetaan taulun liitetiedostot", + "step-fix-file-urls": "Korjataan tiedosto URLit", "cleanup": "Siivous", "cleanup-old-jobs": "Siivoa vanhat työt", "completed": "Valmistunut", diff --git a/imports/i18n/data/fr-CH.i18n.json b/imports/i18n/data/fr-CH.i18n.json index ab5df8545..ad02429ed 100644 --- a/imports/i18n/data/fr-CH.i18n.json +++ b/imports/i18n/data/fr-CH.i18n.json @@ -1404,7 +1404,70 @@ "back-to-settings": "Back to Settings", "board-id": "Board ID", "board-migration": "Board Migration", + "board-migrations": "Board Migrations", "card-show-lists-on-minicard": "Show Lists on Minicard", + "comprehensive-board-migration": "Comprehensive Board Migration", + "comprehensive-board-migration-description": "Performs comprehensive checks and fixes for board data integrity, including list ordering, card positions, and swimlane structure.", + "delete-duplicate-empty-lists-migration": "Delete Duplicate Empty Lists", + "delete-duplicate-empty-lists-migration-description": "Safely deletes empty duplicate lists. Only removes lists that have no cards AND have another list with the same title that contains cards.", + "lost-cards": "Lost Cards", + "lost-cards-list": "Restored Items", + "restore-lost-cards-migration": "Restore Lost Cards", + "restore-lost-cards-migration-description": "Finds and restores cards and lists with missing swimlaneId or listId. Creates a 'Lost Cards' swimlane to make all lost items visible again.", + "restore-all-archived-migration": "Restore All Archived", + "restore-all-archived-migration-description": "Restores all archived swimlanes, lists, and cards. Automatically fixes any missing swimlaneId or listId to make items visible.", + "fix-missing-lists-migration": "Fix Missing Lists", + "fix-missing-lists-migration-description": "Detects and repairs missing or corrupted lists in the board structure.", + "fix-avatar-urls-migration": "Fix Avatar URLs", + "fix-avatar-urls-migration-description": "Updates avatar URLs for board members to use the correct storage backend and fixes broken avatar references.", + "fix-all-file-urls-migration": "Fix All File URLs", + "fix-all-file-urls-migration-description": "Updates all file attachment URLs on this board to use the correct storage backend and fixes broken file references.", + "migration-needed": "Migration Needed", + "migration-complete": "Complete", + "migration-running": "Running...", + "migration-successful": "Migration completed successfully", + "migration-failed": "Migration failed", + "migrations": "Migrations", + "migrations-admin-only": "Only board administrators can run migrations", + "migrations-description": "Run data integrity checks and repairs for this board. Each migration can be executed individually.", + "no-issues-found": "No issues found", + "run-migration": "Run Migration", + "run-comprehensive-migration-confirm": "This will perform a comprehensive migration to check and fix board data integrity. This may take a few moments. Continue?", + "run-delete-duplicate-empty-lists-migration-confirm": "This will first convert any shared lists to per-swimlane lists, then delete empty lists that have a duplicate list with the same title containing cards. Only truly redundant empty lists will be removed. Continue?", + "run-restore-lost-cards-migration-confirm": "This will create a 'Lost Cards' swimlane and restore all cards and lists with missing swimlaneId or listId. This only affects non-archived items. Continue?", + "run-restore-all-archived-migration-confirm": "This will restore ALL archived swimlanes, lists, and cards, making them visible again. Any items with missing IDs will be automatically fixed. This cannot be easily undone. Continue?", + "run-fix-missing-lists-migration-confirm": "This will detect and repair missing or corrupted lists in the board structure. Continue?", + "run-fix-avatar-urls-migration-confirm": "This will update avatar URLs for board members to use the correct storage backend. Continue?", + "run-fix-all-file-urls-migration-confirm": "This will update all file attachment URLs on this board to use the correct storage backend. Continue?", + "restore-lost-cards-nothing-to-restore": "No lost swimlanes, lists, or cards to restore", + + "migration-progress-title": "Board Migration in Progress", + "migration-progress-overall": "Overall Progress", + "migration-progress-current-step": "Current Step", + "migration-progress-status": "Status", + "migration-progress-details": "Details", + "migration-progress-note": "Please wait while we migrate your board to the latest structure...", + + "step-analyze-board-structure": "Analyze Board Structure", + "step-fix-orphaned-cards": "Fix Orphaned Cards", + "step-convert-shared-lists": "Convert Shared Lists", + "step-ensure-per-swimlane-lists": "Ensure Per-Swimlane Lists", + "step-validate-migration": "Validate Migration", + "step-fix-avatar-urls": "Fix Avatar URLs", + "step-fix-attachment-urls": "Fix Attachment URLs", + "step-analyze-lists": "Analyze Lists", + "step-create-missing-lists": "Create Missing Lists", + "step-update-cards": "Update Cards", + "step-finalize": "Finalize", + "step-delete-duplicate-empty-lists": "Delete Duplicate Empty Lists", + "step-ensure-lost-cards-swimlane": "Ensure Lost Cards Swimlane", + "step-restore-lists": "Restore Lists", + "step-restore-cards": "Restore Cards", + "step-restore-swimlanes": "Restore Swimlanes", + "step-fix-missing-ids": "Fix Missing IDs", + "step-scan-users": "Checking board member avatars", + "step-scan-files": "Checking board file attachments", + "step-fix-file-urls": "Fixing file URLs", "cleanup": "Cleanup", "cleanup-old-jobs": "Cleanup Old Jobs", "completed": "Completed", diff --git a/imports/i18n/data/fr-FR.i18n.json b/imports/i18n/data/fr-FR.i18n.json index 643f299b7..721c8be40 100644 --- a/imports/i18n/data/fr-FR.i18n.json +++ b/imports/i18n/data/fr-FR.i18n.json @@ -1404,7 +1404,70 @@ "back-to-settings": "Back to Settings", "board-id": "Board ID", "board-migration": "Board Migration", + "board-migrations": "Board Migrations", "card-show-lists-on-minicard": "Show Lists on Minicard", + "comprehensive-board-migration": "Comprehensive Board Migration", + "comprehensive-board-migration-description": "Performs comprehensive checks and fixes for board data integrity, including list ordering, card positions, and swimlane structure.", + "delete-duplicate-empty-lists-migration": "Delete Duplicate Empty Lists", + "delete-duplicate-empty-lists-migration-description": "Safely deletes empty duplicate lists. Only removes lists that have no cards AND have another list with the same title that contains cards.", + "lost-cards": "Lost Cards", + "lost-cards-list": "Restored Items", + "restore-lost-cards-migration": "Restore Lost Cards", + "restore-lost-cards-migration-description": "Finds and restores cards and lists with missing swimlaneId or listId. Creates a 'Lost Cards' swimlane to make all lost items visible again.", + "restore-all-archived-migration": "Restore All Archived", + "restore-all-archived-migration-description": "Restores all archived swimlanes, lists, and cards. Automatically fixes any missing swimlaneId or listId to make items visible.", + "fix-missing-lists-migration": "Fix Missing Lists", + "fix-missing-lists-migration-description": "Detects and repairs missing or corrupted lists in the board structure.", + "fix-avatar-urls-migration": "Fix Avatar URLs", + "fix-avatar-urls-migration-description": "Updates avatar URLs for board members to use the correct storage backend and fixes broken avatar references.", + "fix-all-file-urls-migration": "Fix All File URLs", + "fix-all-file-urls-migration-description": "Updates all file attachment URLs on this board to use the correct storage backend and fixes broken file references.", + "migration-needed": "Migration Needed", + "migration-complete": "Complete", + "migration-running": "Running...", + "migration-successful": "Migration completed successfully", + "migration-failed": "Migration failed", + "migrations": "Migrations", + "migrations-admin-only": "Only board administrators can run migrations", + "migrations-description": "Run data integrity checks and repairs for this board. Each migration can be executed individually.", + "no-issues-found": "No issues found", + "run-migration": "Run Migration", + "run-comprehensive-migration-confirm": "This will perform a comprehensive migration to check and fix board data integrity. This may take a few moments. Continue?", + "run-delete-duplicate-empty-lists-migration-confirm": "This will first convert any shared lists to per-swimlane lists, then delete empty lists that have a duplicate list with the same title containing cards. Only truly redundant empty lists will be removed. Continue?", + "run-restore-lost-cards-migration-confirm": "This will create a 'Lost Cards' swimlane and restore all cards and lists with missing swimlaneId or listId. This only affects non-archived items. Continue?", + "run-restore-all-archived-migration-confirm": "This will restore ALL archived swimlanes, lists, and cards, making them visible again. Any items with missing IDs will be automatically fixed. This cannot be easily undone. Continue?", + "run-fix-missing-lists-migration-confirm": "This will detect and repair missing or corrupted lists in the board structure. Continue?", + "run-fix-avatar-urls-migration-confirm": "This will update avatar URLs for board members to use the correct storage backend. Continue?", + "run-fix-all-file-urls-migration-confirm": "This will update all file attachment URLs on this board to use the correct storage backend. Continue?", + "restore-lost-cards-nothing-to-restore": "No lost swimlanes, lists, or cards to restore", + + "migration-progress-title": "Board Migration in Progress", + "migration-progress-overall": "Overall Progress", + "migration-progress-current-step": "Current Step", + "migration-progress-status": "Statut", + "migration-progress-details": "Détails", + "migration-progress-note": "Please wait while we migrate your board to the latest structure...", + + "step-analyze-board-structure": "Analyze Board Structure", + "step-fix-orphaned-cards": "Fix Orphaned Cards", + "step-convert-shared-lists": "Convert Shared Lists", + "step-ensure-per-swimlane-lists": "Ensure Per-Swimlane Lists", + "step-validate-migration": "Validate Migration", + "step-fix-avatar-urls": "Fix Avatar URLs", + "step-fix-attachment-urls": "Fix Attachment URLs", + "step-analyze-lists": "Analyze Lists", + "step-create-missing-lists": "Create Missing Lists", + "step-update-cards": "Update Cards", + "step-finalize": "Finalize", + "step-delete-duplicate-empty-lists": "Delete Duplicate Empty Lists", + "step-ensure-lost-cards-swimlane": "Ensure Lost Cards Swimlane", + "step-restore-lists": "Restore Lists", + "step-restore-cards": "Restore Cards", + "step-restore-swimlanes": "Restore Swimlanes", + "step-fix-missing-ids": "Fix Missing IDs", + "step-scan-users": "Checking board member avatars", + "step-scan-files": "Checking board file attachments", + "step-fix-file-urls": "Fixing file URLs", "cleanup": "Cleanup", "cleanup-old-jobs": "Cleanup Old Jobs", "completed": "Terminé", diff --git a/imports/i18n/data/fr.i18n.json b/imports/i18n/data/fr.i18n.json index 042abb941..cd3fdecf6 100644 --- a/imports/i18n/data/fr.i18n.json +++ b/imports/i18n/data/fr.i18n.json @@ -1404,7 +1404,70 @@ "back-to-settings": "Retour aux paramètres", "board-id": "ID du tableau", "board-migration": "Board Migration", + "board-migrations": "Board Migrations", "card-show-lists-on-minicard": "Afficher les listes sur la mini-carte", + "comprehensive-board-migration": "Comprehensive Board Migration", + "comprehensive-board-migration-description": "Performs comprehensive checks and fixes for board data integrity, including list ordering, card positions, and swimlane structure.", + "delete-duplicate-empty-lists-migration": "Delete Duplicate Empty Lists", + "delete-duplicate-empty-lists-migration-description": "Safely deletes empty duplicate lists. Only removes lists that have no cards AND have another list with the same title that contains cards.", + "lost-cards": "Lost Cards", + "lost-cards-list": "Restored Items", + "restore-lost-cards-migration": "Restore Lost Cards", + "restore-lost-cards-migration-description": "Finds and restores cards and lists with missing swimlaneId or listId. Creates a 'Lost Cards' swimlane to make all lost items visible again.", + "restore-all-archived-migration": "Restore All Archived", + "restore-all-archived-migration-description": "Restores all archived swimlanes, lists, and cards. Automatically fixes any missing swimlaneId or listId to make items visible.", + "fix-missing-lists-migration": "Fix Missing Lists", + "fix-missing-lists-migration-description": "Detects and repairs missing or corrupted lists in the board structure.", + "fix-avatar-urls-migration": "Fix Avatar URLs", + "fix-avatar-urls-migration-description": "Updates avatar URLs for board members to use the correct storage backend and fixes broken avatar references.", + "fix-all-file-urls-migration": "Fix All File URLs", + "fix-all-file-urls-migration-description": "Updates all file attachment URLs on this board to use the correct storage backend and fixes broken file references.", + "migration-needed": "Migration Needed", + "migration-complete": "Terminé", + "migration-running": "Running...", + "migration-successful": "Migration completed successfully", + "migration-failed": "Migration failed", + "migrations": "Migrations", + "migrations-admin-only": "Only board administrators can run migrations", + "migrations-description": "Run data integrity checks and repairs for this board. Each migration can be executed individually.", + "no-issues-found": "No issues found", + "run-migration": "Run Migration", + "run-comprehensive-migration-confirm": "This will perform a comprehensive migration to check and fix board data integrity. This may take a few moments. Continue?", + "run-delete-duplicate-empty-lists-migration-confirm": "This will first convert any shared lists to per-swimlane lists, then delete empty lists that have a duplicate list with the same title containing cards. Only truly redundant empty lists will be removed. Continue?", + "run-restore-lost-cards-migration-confirm": "This will create a 'Lost Cards' swimlane and restore all cards and lists with missing swimlaneId or listId. This only affects non-archived items. Continue?", + "run-restore-all-archived-migration-confirm": "This will restore ALL archived swimlanes, lists, and cards, making them visible again. Any items with missing IDs will be automatically fixed. This cannot be easily undone. Continue?", + "run-fix-missing-lists-migration-confirm": "This will detect and repair missing or corrupted lists in the board structure. Continue?", + "run-fix-avatar-urls-migration-confirm": "This will update avatar URLs for board members to use the correct storage backend. Continue?", + "run-fix-all-file-urls-migration-confirm": "This will update all file attachment URLs on this board to use the correct storage backend. Continue?", + "restore-lost-cards-nothing-to-restore": "No lost swimlanes, lists, or cards to restore", + + "migration-progress-title": "Board Migration in Progress", + "migration-progress-overall": "Overall Progress", + "migration-progress-current-step": "Current Step", + "migration-progress-status": "Statut", + "migration-progress-details": "Détails", + "migration-progress-note": "Please wait while we migrate your board to the latest structure...", + + "step-analyze-board-structure": "Analyze Board Structure", + "step-fix-orphaned-cards": "Fix Orphaned Cards", + "step-convert-shared-lists": "Convert Shared Lists", + "step-ensure-per-swimlane-lists": "Ensure Per-Swimlane Lists", + "step-validate-migration": "Validate Migration", + "step-fix-avatar-urls": "Fix Avatar URLs", + "step-fix-attachment-urls": "Fix Attachment URLs", + "step-analyze-lists": "Analyze Lists", + "step-create-missing-lists": "Create Missing Lists", + "step-update-cards": "Update Cards", + "step-finalize": "Finalize", + "step-delete-duplicate-empty-lists": "Delete Duplicate Empty Lists", + "step-ensure-lost-cards-swimlane": "Ensure Lost Cards Swimlane", + "step-restore-lists": "Restore Lists", + "step-restore-cards": "Restore Cards", + "step-restore-swimlanes": "Restore Swimlanes", + "step-fix-missing-ids": "Fix Missing IDs", + "step-scan-users": "Checking board member avatars", + "step-scan-files": "Checking board file attachments", + "step-fix-file-urls": "Fixing file URLs", "cleanup": "Cleanup", "cleanup-old-jobs": "Cleanup Old Jobs", "completed": "Terminé", diff --git a/imports/i18n/data/fy-NL.i18n.json b/imports/i18n/data/fy-NL.i18n.json index eef27e1fe..958d8313a 100644 --- a/imports/i18n/data/fy-NL.i18n.json +++ b/imports/i18n/data/fy-NL.i18n.json @@ -1404,7 +1404,70 @@ "back-to-settings": "Back to Settings", "board-id": "Board ID", "board-migration": "Board Migration", + "board-migrations": "Board Migrations", "card-show-lists-on-minicard": "Show Lists on Minicard", + "comprehensive-board-migration": "Comprehensive Board Migration", + "comprehensive-board-migration-description": "Performs comprehensive checks and fixes for board data integrity, including list ordering, card positions, and swimlane structure.", + "delete-duplicate-empty-lists-migration": "Delete Duplicate Empty Lists", + "delete-duplicate-empty-lists-migration-description": "Safely deletes empty duplicate lists. Only removes lists that have no cards AND have another list with the same title that contains cards.", + "lost-cards": "Lost Cards", + "lost-cards-list": "Restored Items", + "restore-lost-cards-migration": "Restore Lost Cards", + "restore-lost-cards-migration-description": "Finds and restores cards and lists with missing swimlaneId or listId. Creates a 'Lost Cards' swimlane to make all lost items visible again.", + "restore-all-archived-migration": "Restore All Archived", + "restore-all-archived-migration-description": "Restores all archived swimlanes, lists, and cards. Automatically fixes any missing swimlaneId or listId to make items visible.", + "fix-missing-lists-migration": "Fix Missing Lists", + "fix-missing-lists-migration-description": "Detects and repairs missing or corrupted lists in the board structure.", + "fix-avatar-urls-migration": "Fix Avatar URLs", + "fix-avatar-urls-migration-description": "Updates avatar URLs for board members to use the correct storage backend and fixes broken avatar references.", + "fix-all-file-urls-migration": "Fix All File URLs", + "fix-all-file-urls-migration-description": "Updates all file attachment URLs on this board to use the correct storage backend and fixes broken file references.", + "migration-needed": "Migration Needed", + "migration-complete": "Complete", + "migration-running": "Running...", + "migration-successful": "Migration completed successfully", + "migration-failed": "Migration failed", + "migrations": "Migrations", + "migrations-admin-only": "Only board administrators can run migrations", + "migrations-description": "Run data integrity checks and repairs for this board. Each migration can be executed individually.", + "no-issues-found": "No issues found", + "run-migration": "Run Migration", + "run-comprehensive-migration-confirm": "This will perform a comprehensive migration to check and fix board data integrity. This may take a few moments. Continue?", + "run-delete-duplicate-empty-lists-migration-confirm": "This will first convert any shared lists to per-swimlane lists, then delete empty lists that have a duplicate list with the same title containing cards. Only truly redundant empty lists will be removed. Continue?", + "run-restore-lost-cards-migration-confirm": "This will create a 'Lost Cards' swimlane and restore all cards and lists with missing swimlaneId or listId. This only affects non-archived items. Continue?", + "run-restore-all-archived-migration-confirm": "This will restore ALL archived swimlanes, lists, and cards, making them visible again. Any items with missing IDs will be automatically fixed. This cannot be easily undone. Continue?", + "run-fix-missing-lists-migration-confirm": "This will detect and repair missing or corrupted lists in the board structure. Continue?", + "run-fix-avatar-urls-migration-confirm": "This will update avatar URLs for board members to use the correct storage backend. Continue?", + "run-fix-all-file-urls-migration-confirm": "This will update all file attachment URLs on this board to use the correct storage backend. Continue?", + "restore-lost-cards-nothing-to-restore": "No lost swimlanes, lists, or cards to restore", + + "migration-progress-title": "Board Migration in Progress", + "migration-progress-overall": "Overall Progress", + "migration-progress-current-step": "Current Step", + "migration-progress-status": "Status", + "migration-progress-details": "Details", + "migration-progress-note": "Please wait while we migrate your board to the latest structure...", + + "step-analyze-board-structure": "Analyze Board Structure", + "step-fix-orphaned-cards": "Fix Orphaned Cards", + "step-convert-shared-lists": "Convert Shared Lists", + "step-ensure-per-swimlane-lists": "Ensure Per-Swimlane Lists", + "step-validate-migration": "Validate Migration", + "step-fix-avatar-urls": "Fix Avatar URLs", + "step-fix-attachment-urls": "Fix Attachment URLs", + "step-analyze-lists": "Analyze Lists", + "step-create-missing-lists": "Create Missing Lists", + "step-update-cards": "Update Cards", + "step-finalize": "Finalize", + "step-delete-duplicate-empty-lists": "Delete Duplicate Empty Lists", + "step-ensure-lost-cards-swimlane": "Ensure Lost Cards Swimlane", + "step-restore-lists": "Restore Lists", + "step-restore-cards": "Restore Cards", + "step-restore-swimlanes": "Restore Swimlanes", + "step-fix-missing-ids": "Fix Missing IDs", + "step-scan-users": "Checking board member avatars", + "step-scan-files": "Checking board file attachments", + "step-fix-file-urls": "Fixing file URLs", "cleanup": "Cleanup", "cleanup-old-jobs": "Cleanup Old Jobs", "completed": "Completed", diff --git a/imports/i18n/data/fy.i18n.json b/imports/i18n/data/fy.i18n.json index eef27e1fe..958d8313a 100644 --- a/imports/i18n/data/fy.i18n.json +++ b/imports/i18n/data/fy.i18n.json @@ -1404,7 +1404,70 @@ "back-to-settings": "Back to Settings", "board-id": "Board ID", "board-migration": "Board Migration", + "board-migrations": "Board Migrations", "card-show-lists-on-minicard": "Show Lists on Minicard", + "comprehensive-board-migration": "Comprehensive Board Migration", + "comprehensive-board-migration-description": "Performs comprehensive checks and fixes for board data integrity, including list ordering, card positions, and swimlane structure.", + "delete-duplicate-empty-lists-migration": "Delete Duplicate Empty Lists", + "delete-duplicate-empty-lists-migration-description": "Safely deletes empty duplicate lists. Only removes lists that have no cards AND have another list with the same title that contains cards.", + "lost-cards": "Lost Cards", + "lost-cards-list": "Restored Items", + "restore-lost-cards-migration": "Restore Lost Cards", + "restore-lost-cards-migration-description": "Finds and restores cards and lists with missing swimlaneId or listId. Creates a 'Lost Cards' swimlane to make all lost items visible again.", + "restore-all-archived-migration": "Restore All Archived", + "restore-all-archived-migration-description": "Restores all archived swimlanes, lists, and cards. Automatically fixes any missing swimlaneId or listId to make items visible.", + "fix-missing-lists-migration": "Fix Missing Lists", + "fix-missing-lists-migration-description": "Detects and repairs missing or corrupted lists in the board structure.", + "fix-avatar-urls-migration": "Fix Avatar URLs", + "fix-avatar-urls-migration-description": "Updates avatar URLs for board members to use the correct storage backend and fixes broken avatar references.", + "fix-all-file-urls-migration": "Fix All File URLs", + "fix-all-file-urls-migration-description": "Updates all file attachment URLs on this board to use the correct storage backend and fixes broken file references.", + "migration-needed": "Migration Needed", + "migration-complete": "Complete", + "migration-running": "Running...", + "migration-successful": "Migration completed successfully", + "migration-failed": "Migration failed", + "migrations": "Migrations", + "migrations-admin-only": "Only board administrators can run migrations", + "migrations-description": "Run data integrity checks and repairs for this board. Each migration can be executed individually.", + "no-issues-found": "No issues found", + "run-migration": "Run Migration", + "run-comprehensive-migration-confirm": "This will perform a comprehensive migration to check and fix board data integrity. This may take a few moments. Continue?", + "run-delete-duplicate-empty-lists-migration-confirm": "This will first convert any shared lists to per-swimlane lists, then delete empty lists that have a duplicate list with the same title containing cards. Only truly redundant empty lists will be removed. Continue?", + "run-restore-lost-cards-migration-confirm": "This will create a 'Lost Cards' swimlane and restore all cards and lists with missing swimlaneId or listId. This only affects non-archived items. Continue?", + "run-restore-all-archived-migration-confirm": "This will restore ALL archived swimlanes, lists, and cards, making them visible again. Any items with missing IDs will be automatically fixed. This cannot be easily undone. Continue?", + "run-fix-missing-lists-migration-confirm": "This will detect and repair missing or corrupted lists in the board structure. Continue?", + "run-fix-avatar-urls-migration-confirm": "This will update avatar URLs for board members to use the correct storage backend. Continue?", + "run-fix-all-file-urls-migration-confirm": "This will update all file attachment URLs on this board to use the correct storage backend. Continue?", + "restore-lost-cards-nothing-to-restore": "No lost swimlanes, lists, or cards to restore", + + "migration-progress-title": "Board Migration in Progress", + "migration-progress-overall": "Overall Progress", + "migration-progress-current-step": "Current Step", + "migration-progress-status": "Status", + "migration-progress-details": "Details", + "migration-progress-note": "Please wait while we migrate your board to the latest structure...", + + "step-analyze-board-structure": "Analyze Board Structure", + "step-fix-orphaned-cards": "Fix Orphaned Cards", + "step-convert-shared-lists": "Convert Shared Lists", + "step-ensure-per-swimlane-lists": "Ensure Per-Swimlane Lists", + "step-validate-migration": "Validate Migration", + "step-fix-avatar-urls": "Fix Avatar URLs", + "step-fix-attachment-urls": "Fix Attachment URLs", + "step-analyze-lists": "Analyze Lists", + "step-create-missing-lists": "Create Missing Lists", + "step-update-cards": "Update Cards", + "step-finalize": "Finalize", + "step-delete-duplicate-empty-lists": "Delete Duplicate Empty Lists", + "step-ensure-lost-cards-swimlane": "Ensure Lost Cards Swimlane", + "step-restore-lists": "Restore Lists", + "step-restore-cards": "Restore Cards", + "step-restore-swimlanes": "Restore Swimlanes", + "step-fix-missing-ids": "Fix Missing IDs", + "step-scan-users": "Checking board member avatars", + "step-scan-files": "Checking board file attachments", + "step-fix-file-urls": "Fixing file URLs", "cleanup": "Cleanup", "cleanup-old-jobs": "Cleanup Old Jobs", "completed": "Completed", diff --git a/imports/i18n/data/gl-ES.i18n.json b/imports/i18n/data/gl-ES.i18n.json index 8c523be4e..37d1e5efd 100644 --- a/imports/i18n/data/gl-ES.i18n.json +++ b/imports/i18n/data/gl-ES.i18n.json @@ -1404,7 +1404,70 @@ "back-to-settings": "Back to Settings", "board-id": "Board ID", "board-migration": "Board Migration", + "board-migrations": "Board Migrations", "card-show-lists-on-minicard": "Show Lists on Minicard", + "comprehensive-board-migration": "Comprehensive Board Migration", + "comprehensive-board-migration-description": "Performs comprehensive checks and fixes for board data integrity, including list ordering, card positions, and swimlane structure.", + "delete-duplicate-empty-lists-migration": "Delete Duplicate Empty Lists", + "delete-duplicate-empty-lists-migration-description": "Safely deletes empty duplicate lists. Only removes lists that have no cards AND have another list with the same title that contains cards.", + "lost-cards": "Lost Cards", + "lost-cards-list": "Restored Items", + "restore-lost-cards-migration": "Restore Lost Cards", + "restore-lost-cards-migration-description": "Finds and restores cards and lists with missing swimlaneId or listId. Creates a 'Lost Cards' swimlane to make all lost items visible again.", + "restore-all-archived-migration": "Restore All Archived", + "restore-all-archived-migration-description": "Restores all archived swimlanes, lists, and cards. Automatically fixes any missing swimlaneId or listId to make items visible.", + "fix-missing-lists-migration": "Fix Missing Lists", + "fix-missing-lists-migration-description": "Detects and repairs missing or corrupted lists in the board structure.", + "fix-avatar-urls-migration": "Fix Avatar URLs", + "fix-avatar-urls-migration-description": "Updates avatar URLs for board members to use the correct storage backend and fixes broken avatar references.", + "fix-all-file-urls-migration": "Fix All File URLs", + "fix-all-file-urls-migration-description": "Updates all file attachment URLs on this board to use the correct storage backend and fixes broken file references.", + "migration-needed": "Migration Needed", + "migration-complete": "Complete", + "migration-running": "Running...", + "migration-successful": "Migration completed successfully", + "migration-failed": "Migration failed", + "migrations": "Migrations", + "migrations-admin-only": "Only board administrators can run migrations", + "migrations-description": "Run data integrity checks and repairs for this board. Each migration can be executed individually.", + "no-issues-found": "No issues found", + "run-migration": "Run Migration", + "run-comprehensive-migration-confirm": "This will perform a comprehensive migration to check and fix board data integrity. This may take a few moments. Continue?", + "run-delete-duplicate-empty-lists-migration-confirm": "This will first convert any shared lists to per-swimlane lists, then delete empty lists that have a duplicate list with the same title containing cards. Only truly redundant empty lists will be removed. Continue?", + "run-restore-lost-cards-migration-confirm": "This will create a 'Lost Cards' swimlane and restore all cards and lists with missing swimlaneId or listId. This only affects non-archived items. Continue?", + "run-restore-all-archived-migration-confirm": "This will restore ALL archived swimlanes, lists, and cards, making them visible again. Any items with missing IDs will be automatically fixed. This cannot be easily undone. Continue?", + "run-fix-missing-lists-migration-confirm": "This will detect and repair missing or corrupted lists in the board structure. Continue?", + "run-fix-avatar-urls-migration-confirm": "This will update avatar URLs for board members to use the correct storage backend. Continue?", + "run-fix-all-file-urls-migration-confirm": "This will update all file attachment URLs on this board to use the correct storage backend. Continue?", + "restore-lost-cards-nothing-to-restore": "No lost swimlanes, lists, or cards to restore", + + "migration-progress-title": "Board Migration in Progress", + "migration-progress-overall": "Overall Progress", + "migration-progress-current-step": "Current Step", + "migration-progress-status": "Status", + "migration-progress-details": "Details", + "migration-progress-note": "Please wait while we migrate your board to the latest structure...", + + "step-analyze-board-structure": "Analyze Board Structure", + "step-fix-orphaned-cards": "Fix Orphaned Cards", + "step-convert-shared-lists": "Convert Shared Lists", + "step-ensure-per-swimlane-lists": "Ensure Per-Swimlane Lists", + "step-validate-migration": "Validate Migration", + "step-fix-avatar-urls": "Fix Avatar URLs", + "step-fix-attachment-urls": "Fix Attachment URLs", + "step-analyze-lists": "Analyze Lists", + "step-create-missing-lists": "Create Missing Lists", + "step-update-cards": "Update Cards", + "step-finalize": "Finalize", + "step-delete-duplicate-empty-lists": "Delete Duplicate Empty Lists", + "step-ensure-lost-cards-swimlane": "Ensure Lost Cards Swimlane", + "step-restore-lists": "Restore Lists", + "step-restore-cards": "Restore Cards", + "step-restore-swimlanes": "Restore Swimlanes", + "step-fix-missing-ids": "Fix Missing IDs", + "step-scan-users": "Checking board member avatars", + "step-scan-files": "Checking board file attachments", + "step-fix-file-urls": "Fixing file URLs", "cleanup": "Cleanup", "cleanup-old-jobs": "Cleanup Old Jobs", "completed": "Completed", diff --git a/imports/i18n/data/gl.i18n.json b/imports/i18n/data/gl.i18n.json index 9b9438f28..08f3aa450 100644 --- a/imports/i18n/data/gl.i18n.json +++ b/imports/i18n/data/gl.i18n.json @@ -1404,7 +1404,70 @@ "back-to-settings": "Back to Settings", "board-id": "Board ID", "board-migration": "Board Migration", + "board-migrations": "Board Migrations", "card-show-lists-on-minicard": "Show Lists on Minicard", + "comprehensive-board-migration": "Comprehensive Board Migration", + "comprehensive-board-migration-description": "Performs comprehensive checks and fixes for board data integrity, including list ordering, card positions, and swimlane structure.", + "delete-duplicate-empty-lists-migration": "Delete Duplicate Empty Lists", + "delete-duplicate-empty-lists-migration-description": "Safely deletes empty duplicate lists. Only removes lists that have no cards AND have another list with the same title that contains cards.", + "lost-cards": "Lost Cards", + "lost-cards-list": "Restored Items", + "restore-lost-cards-migration": "Restore Lost Cards", + "restore-lost-cards-migration-description": "Finds and restores cards and lists with missing swimlaneId or listId. Creates a 'Lost Cards' swimlane to make all lost items visible again.", + "restore-all-archived-migration": "Restore All Archived", + "restore-all-archived-migration-description": "Restores all archived swimlanes, lists, and cards. Automatically fixes any missing swimlaneId or listId to make items visible.", + "fix-missing-lists-migration": "Fix Missing Lists", + "fix-missing-lists-migration-description": "Detects and repairs missing or corrupted lists in the board structure.", + "fix-avatar-urls-migration": "Fix Avatar URLs", + "fix-avatar-urls-migration-description": "Updates avatar URLs for board members to use the correct storage backend and fixes broken avatar references.", + "fix-all-file-urls-migration": "Fix All File URLs", + "fix-all-file-urls-migration-description": "Updates all file attachment URLs on this board to use the correct storage backend and fixes broken file references.", + "migration-needed": "Migration Needed", + "migration-complete": "Complete", + "migration-running": "Running...", + "migration-successful": "Migration completed successfully", + "migration-failed": "Migration failed", + "migrations": "Migrations", + "migrations-admin-only": "Only board administrators can run migrations", + "migrations-description": "Run data integrity checks and repairs for this board. Each migration can be executed individually.", + "no-issues-found": "No issues found", + "run-migration": "Run Migration", + "run-comprehensive-migration-confirm": "This will perform a comprehensive migration to check and fix board data integrity. This may take a few moments. Continue?", + "run-delete-duplicate-empty-lists-migration-confirm": "This will first convert any shared lists to per-swimlane lists, then delete empty lists that have a duplicate list with the same title containing cards. Only truly redundant empty lists will be removed. Continue?", + "run-restore-lost-cards-migration-confirm": "This will create a 'Lost Cards' swimlane and restore all cards and lists with missing swimlaneId or listId. This only affects non-archived items. Continue?", + "run-restore-all-archived-migration-confirm": "This will restore ALL archived swimlanes, lists, and cards, making them visible again. Any items with missing IDs will be automatically fixed. This cannot be easily undone. Continue?", + "run-fix-missing-lists-migration-confirm": "This will detect and repair missing or corrupted lists in the board structure. Continue?", + "run-fix-avatar-urls-migration-confirm": "This will update avatar URLs for board members to use the correct storage backend. Continue?", + "run-fix-all-file-urls-migration-confirm": "This will update all file attachment URLs on this board to use the correct storage backend. Continue?", + "restore-lost-cards-nothing-to-restore": "No lost swimlanes, lists, or cards to restore", + + "migration-progress-title": "Board Migration in Progress", + "migration-progress-overall": "Overall Progress", + "migration-progress-current-step": "Current Step", + "migration-progress-status": "Status", + "migration-progress-details": "Details", + "migration-progress-note": "Please wait while we migrate your board to the latest structure...", + + "step-analyze-board-structure": "Analyze Board Structure", + "step-fix-orphaned-cards": "Fix Orphaned Cards", + "step-convert-shared-lists": "Convert Shared Lists", + "step-ensure-per-swimlane-lists": "Ensure Per-Swimlane Lists", + "step-validate-migration": "Validate Migration", + "step-fix-avatar-urls": "Fix Avatar URLs", + "step-fix-attachment-urls": "Fix Attachment URLs", + "step-analyze-lists": "Analyze Lists", + "step-create-missing-lists": "Create Missing Lists", + "step-update-cards": "Update Cards", + "step-finalize": "Finalize", + "step-delete-duplicate-empty-lists": "Delete Duplicate Empty Lists", + "step-ensure-lost-cards-swimlane": "Ensure Lost Cards Swimlane", + "step-restore-lists": "Restore Lists", + "step-restore-cards": "Restore Cards", + "step-restore-swimlanes": "Restore Swimlanes", + "step-fix-missing-ids": "Fix Missing IDs", + "step-scan-users": "Checking board member avatars", + "step-scan-files": "Checking board file attachments", + "step-fix-file-urls": "Fixing file URLs", "cleanup": "Cleanup", "cleanup-old-jobs": "Cleanup Old Jobs", "completed": "Completed", diff --git a/imports/i18n/data/gu-IN.i18n.json b/imports/i18n/data/gu-IN.i18n.json index 1ab8eeaa8..4804e6dff 100644 --- a/imports/i18n/data/gu-IN.i18n.json +++ b/imports/i18n/data/gu-IN.i18n.json @@ -1404,7 +1404,70 @@ "back-to-settings": "Back to Settings", "board-id": "Board ID", "board-migration": "Board Migration", + "board-migrations": "Board Migrations", "card-show-lists-on-minicard": "Show Lists on Minicard", + "comprehensive-board-migration": "Comprehensive Board Migration", + "comprehensive-board-migration-description": "Performs comprehensive checks and fixes for board data integrity, including list ordering, card positions, and swimlane structure.", + "delete-duplicate-empty-lists-migration": "Delete Duplicate Empty Lists", + "delete-duplicate-empty-lists-migration-description": "Safely deletes empty duplicate lists. Only removes lists that have no cards AND have another list with the same title that contains cards.", + "lost-cards": "Lost Cards", + "lost-cards-list": "Restored Items", + "restore-lost-cards-migration": "Restore Lost Cards", + "restore-lost-cards-migration-description": "Finds and restores cards and lists with missing swimlaneId or listId. Creates a 'Lost Cards' swimlane to make all lost items visible again.", + "restore-all-archived-migration": "Restore All Archived", + "restore-all-archived-migration-description": "Restores all archived swimlanes, lists, and cards. Automatically fixes any missing swimlaneId or listId to make items visible.", + "fix-missing-lists-migration": "Fix Missing Lists", + "fix-missing-lists-migration-description": "Detects and repairs missing or corrupted lists in the board structure.", + "fix-avatar-urls-migration": "Fix Avatar URLs", + "fix-avatar-urls-migration-description": "Updates avatar URLs for board members to use the correct storage backend and fixes broken avatar references.", + "fix-all-file-urls-migration": "Fix All File URLs", + "fix-all-file-urls-migration-description": "Updates all file attachment URLs on this board to use the correct storage backend and fixes broken file references.", + "migration-needed": "Migration Needed", + "migration-complete": "Complete", + "migration-running": "Running...", + "migration-successful": "Migration completed successfully", + "migration-failed": "Migration failed", + "migrations": "Migrations", + "migrations-admin-only": "Only board administrators can run migrations", + "migrations-description": "Run data integrity checks and repairs for this board. Each migration can be executed individually.", + "no-issues-found": "No issues found", + "run-migration": "Run Migration", + "run-comprehensive-migration-confirm": "This will perform a comprehensive migration to check and fix board data integrity. This may take a few moments. Continue?", + "run-delete-duplicate-empty-lists-migration-confirm": "This will first convert any shared lists to per-swimlane lists, then delete empty lists that have a duplicate list with the same title containing cards. Only truly redundant empty lists will be removed. Continue?", + "run-restore-lost-cards-migration-confirm": "This will create a 'Lost Cards' swimlane and restore all cards and lists with missing swimlaneId or listId. This only affects non-archived items. Continue?", + "run-restore-all-archived-migration-confirm": "This will restore ALL archived swimlanes, lists, and cards, making them visible again. Any items with missing IDs will be automatically fixed. This cannot be easily undone. Continue?", + "run-fix-missing-lists-migration-confirm": "This will detect and repair missing or corrupted lists in the board structure. Continue?", + "run-fix-avatar-urls-migration-confirm": "This will update avatar URLs for board members to use the correct storage backend. Continue?", + "run-fix-all-file-urls-migration-confirm": "This will update all file attachment URLs on this board to use the correct storage backend. Continue?", + "restore-lost-cards-nothing-to-restore": "No lost swimlanes, lists, or cards to restore", + + "migration-progress-title": "Board Migration in Progress", + "migration-progress-overall": "Overall Progress", + "migration-progress-current-step": "Current Step", + "migration-progress-status": "Status", + "migration-progress-details": "Details", + "migration-progress-note": "Please wait while we migrate your board to the latest structure...", + + "step-analyze-board-structure": "Analyze Board Structure", + "step-fix-orphaned-cards": "Fix Orphaned Cards", + "step-convert-shared-lists": "Convert Shared Lists", + "step-ensure-per-swimlane-lists": "Ensure Per-Swimlane Lists", + "step-validate-migration": "Validate Migration", + "step-fix-avatar-urls": "Fix Avatar URLs", + "step-fix-attachment-urls": "Fix Attachment URLs", + "step-analyze-lists": "Analyze Lists", + "step-create-missing-lists": "Create Missing Lists", + "step-update-cards": "Update Cards", + "step-finalize": "Finalize", + "step-delete-duplicate-empty-lists": "Delete Duplicate Empty Lists", + "step-ensure-lost-cards-swimlane": "Ensure Lost Cards Swimlane", + "step-restore-lists": "Restore Lists", + "step-restore-cards": "Restore Cards", + "step-restore-swimlanes": "Restore Swimlanes", + "step-fix-missing-ids": "Fix Missing IDs", + "step-scan-users": "Checking board member avatars", + "step-scan-files": "Checking board file attachments", + "step-fix-file-urls": "Fixing file URLs", "cleanup": "Cleanup", "cleanup-old-jobs": "Cleanup Old Jobs", "completed": "Completed", diff --git a/imports/i18n/data/he-IL.i18n.json b/imports/i18n/data/he-IL.i18n.json index b34277bac..2b1393267 100644 --- a/imports/i18n/data/he-IL.i18n.json +++ b/imports/i18n/data/he-IL.i18n.json @@ -1404,7 +1404,70 @@ "back-to-settings": "Back to Settings", "board-id": "Board ID", "board-migration": "Board Migration", + "board-migrations": "Board Migrations", "card-show-lists-on-minicard": "Show Lists on Minicard", + "comprehensive-board-migration": "Comprehensive Board Migration", + "comprehensive-board-migration-description": "Performs comprehensive checks and fixes for board data integrity, including list ordering, card positions, and swimlane structure.", + "delete-duplicate-empty-lists-migration": "Delete Duplicate Empty Lists", + "delete-duplicate-empty-lists-migration-description": "Safely deletes empty duplicate lists. Only removes lists that have no cards AND have another list with the same title that contains cards.", + "lost-cards": "Lost Cards", + "lost-cards-list": "Restored Items", + "restore-lost-cards-migration": "Restore Lost Cards", + "restore-lost-cards-migration-description": "Finds and restores cards and lists with missing swimlaneId or listId. Creates a 'Lost Cards' swimlane to make all lost items visible again.", + "restore-all-archived-migration": "Restore All Archived", + "restore-all-archived-migration-description": "Restores all archived swimlanes, lists, and cards. Automatically fixes any missing swimlaneId or listId to make items visible.", + "fix-missing-lists-migration": "Fix Missing Lists", + "fix-missing-lists-migration-description": "Detects and repairs missing or corrupted lists in the board structure.", + "fix-avatar-urls-migration": "Fix Avatar URLs", + "fix-avatar-urls-migration-description": "Updates avatar URLs for board members to use the correct storage backend and fixes broken avatar references.", + "fix-all-file-urls-migration": "Fix All File URLs", + "fix-all-file-urls-migration-description": "Updates all file attachment URLs on this board to use the correct storage backend and fixes broken file references.", + "migration-needed": "Migration Needed", + "migration-complete": "Complete", + "migration-running": "Running...", + "migration-successful": "Migration completed successfully", + "migration-failed": "Migration failed", + "migrations": "Migrations", + "migrations-admin-only": "Only board administrators can run migrations", + "migrations-description": "Run data integrity checks and repairs for this board. Each migration can be executed individually.", + "no-issues-found": "No issues found", + "run-migration": "Run Migration", + "run-comprehensive-migration-confirm": "This will perform a comprehensive migration to check and fix board data integrity. This may take a few moments. Continue?", + "run-delete-duplicate-empty-lists-migration-confirm": "This will first convert any shared lists to per-swimlane lists, then delete empty lists that have a duplicate list with the same title containing cards. Only truly redundant empty lists will be removed. Continue?", + "run-restore-lost-cards-migration-confirm": "This will create a 'Lost Cards' swimlane and restore all cards and lists with missing swimlaneId or listId. This only affects non-archived items. Continue?", + "run-restore-all-archived-migration-confirm": "This will restore ALL archived swimlanes, lists, and cards, making them visible again. Any items with missing IDs will be automatically fixed. This cannot be easily undone. Continue?", + "run-fix-missing-lists-migration-confirm": "This will detect and repair missing or corrupted lists in the board structure. Continue?", + "run-fix-avatar-urls-migration-confirm": "This will update avatar URLs for board members to use the correct storage backend. Continue?", + "run-fix-all-file-urls-migration-confirm": "This will update all file attachment URLs on this board to use the correct storage backend. Continue?", + "restore-lost-cards-nothing-to-restore": "No lost swimlanes, lists, or cards to restore", + + "migration-progress-title": "Board Migration in Progress", + "migration-progress-overall": "Overall Progress", + "migration-progress-current-step": "Current Step", + "migration-progress-status": "Status", + "migration-progress-details": "Details", + "migration-progress-note": "Please wait while we migrate your board to the latest structure...", + + "step-analyze-board-structure": "Analyze Board Structure", + "step-fix-orphaned-cards": "Fix Orphaned Cards", + "step-convert-shared-lists": "Convert Shared Lists", + "step-ensure-per-swimlane-lists": "Ensure Per-Swimlane Lists", + "step-validate-migration": "Validate Migration", + "step-fix-avatar-urls": "Fix Avatar URLs", + "step-fix-attachment-urls": "Fix Attachment URLs", + "step-analyze-lists": "Analyze Lists", + "step-create-missing-lists": "Create Missing Lists", + "step-update-cards": "Update Cards", + "step-finalize": "Finalize", + "step-delete-duplicate-empty-lists": "Delete Duplicate Empty Lists", + "step-ensure-lost-cards-swimlane": "Ensure Lost Cards Swimlane", + "step-restore-lists": "Restore Lists", + "step-restore-cards": "Restore Cards", + "step-restore-swimlanes": "Restore Swimlanes", + "step-fix-missing-ids": "Fix Missing IDs", + "step-scan-users": "Checking board member avatars", + "step-scan-files": "Checking board file attachments", + "step-fix-file-urls": "Fixing file URLs", "cleanup": "Cleanup", "cleanup-old-jobs": "Cleanup Old Jobs", "completed": "Completed", diff --git a/imports/i18n/data/he.i18n.json b/imports/i18n/data/he.i18n.json index 2b6d8f5ef..831768065 100644 --- a/imports/i18n/data/he.i18n.json +++ b/imports/i18n/data/he.i18n.json @@ -1404,7 +1404,70 @@ "back-to-settings": "חזרה להגדרות", "board-id": "מזהה לוח", "board-migration": "הסבת לוחות", + "board-migrations": "Board Migrations", "card-show-lists-on-minicard": "הצגת רשימות בכרטיסון", + "comprehensive-board-migration": "Comprehensive Board Migration", + "comprehensive-board-migration-description": "Performs comprehensive checks and fixes for board data integrity, including list ordering, card positions, and swimlane structure.", + "delete-duplicate-empty-lists-migration": "Delete Duplicate Empty Lists", + "delete-duplicate-empty-lists-migration-description": "Safely deletes empty duplicate lists. Only removes lists that have no cards AND have another list with the same title that contains cards.", + "lost-cards": "Lost Cards", + "lost-cards-list": "Restored Items", + "restore-lost-cards-migration": "Restore Lost Cards", + "restore-lost-cards-migration-description": "Finds and restores cards and lists with missing swimlaneId or listId. Creates a 'Lost Cards' swimlane to make all lost items visible again.", + "restore-all-archived-migration": "Restore All Archived", + "restore-all-archived-migration-description": "Restores all archived swimlanes, lists, and cards. Automatically fixes any missing swimlaneId or listId to make items visible.", + "fix-missing-lists-migration": "Fix Missing Lists", + "fix-missing-lists-migration-description": "Detects and repairs missing or corrupted lists in the board structure.", + "fix-avatar-urls-migration": "Fix Avatar URLs", + "fix-avatar-urls-migration-description": "Updates avatar URLs for board members to use the correct storage backend and fixes broken avatar references.", + "fix-all-file-urls-migration": "Fix All File URLs", + "fix-all-file-urls-migration-description": "Updates all file attachment URLs on this board to use the correct storage backend and fixes broken file references.", + "migration-needed": "Migration Needed", + "migration-complete": "הושלם", + "migration-running": "Running...", + "migration-successful": "Migration completed successfully", + "migration-failed": "Migration failed", + "migrations": "Migrations", + "migrations-admin-only": "Only board administrators can run migrations", + "migrations-description": "Run data integrity checks and repairs for this board. Each migration can be executed individually.", + "no-issues-found": "No issues found", + "run-migration": "Run Migration", + "run-comprehensive-migration-confirm": "This will perform a comprehensive migration to check and fix board data integrity. This may take a few moments. Continue?", + "run-delete-duplicate-empty-lists-migration-confirm": "This will first convert any shared lists to per-swimlane lists, then delete empty lists that have a duplicate list with the same title containing cards. Only truly redundant empty lists will be removed. Continue?", + "run-restore-lost-cards-migration-confirm": "This will create a 'Lost Cards' swimlane and restore all cards and lists with missing swimlaneId or listId. This only affects non-archived items. Continue?", + "run-restore-all-archived-migration-confirm": "This will restore ALL archived swimlanes, lists, and cards, making them visible again. Any items with missing IDs will be automatically fixed. This cannot be easily undone. Continue?", + "run-fix-missing-lists-migration-confirm": "This will detect and repair missing or corrupted lists in the board structure. Continue?", + "run-fix-avatar-urls-migration-confirm": "This will update avatar URLs for board members to use the correct storage backend. Continue?", + "run-fix-all-file-urls-migration-confirm": "This will update all file attachment URLs on this board to use the correct storage backend. Continue?", + "restore-lost-cards-nothing-to-restore": "No lost swimlanes, lists, or cards to restore", + + "migration-progress-title": "Board Migration in Progress", + "migration-progress-overall": "סך כל ההתקדמות", + "migration-progress-current-step": "Current Step", + "migration-progress-status": "מצב", + "migration-progress-details": "פרטים", + "migration-progress-note": "Please wait while we migrate your board to the latest structure...", + + "step-analyze-board-structure": "Analyze Board Structure", + "step-fix-orphaned-cards": "Fix Orphaned Cards", + "step-convert-shared-lists": "Convert Shared Lists", + "step-ensure-per-swimlane-lists": "Ensure Per-Swimlane Lists", + "step-validate-migration": "Validate Migration", + "step-fix-avatar-urls": "Fix Avatar URLs", + "step-fix-attachment-urls": "Fix Attachment URLs", + "step-analyze-lists": "Analyze Lists", + "step-create-missing-lists": "Create Missing Lists", + "step-update-cards": "Update Cards", + "step-finalize": "Finalize", + "step-delete-duplicate-empty-lists": "Delete Duplicate Empty Lists", + "step-ensure-lost-cards-swimlane": "Ensure Lost Cards Swimlane", + "step-restore-lists": "Restore Lists", + "step-restore-cards": "Restore Cards", + "step-restore-swimlanes": "Restore Swimlanes", + "step-fix-missing-ids": "Fix Missing IDs", + "step-scan-users": "Checking board member avatars", + "step-scan-files": "Checking board file attachments", + "step-fix-file-urls": "Fixing file URLs", "cleanup": "ניקיון", "cleanup-old-jobs": "ניקוי משימות ישנות", "completed": "הושלמה", diff --git a/imports/i18n/data/hi-IN.i18n.json b/imports/i18n/data/hi-IN.i18n.json index e65f94bf7..aa6f7f298 100644 --- a/imports/i18n/data/hi-IN.i18n.json +++ b/imports/i18n/data/hi-IN.i18n.json @@ -1404,7 +1404,70 @@ "back-to-settings": "Back to Settings", "board-id": "Board ID", "board-migration": "Board Migration", + "board-migrations": "Board Migrations", "card-show-lists-on-minicard": "Show Lists on Minicard", + "comprehensive-board-migration": "Comprehensive Board Migration", + "comprehensive-board-migration-description": "Performs comprehensive checks and fixes for board data integrity, including list ordering, card positions, and swimlane structure.", + "delete-duplicate-empty-lists-migration": "Delete Duplicate Empty Lists", + "delete-duplicate-empty-lists-migration-description": "Safely deletes empty duplicate lists. Only removes lists that have no cards AND have another list with the same title that contains cards.", + "lost-cards": "Lost Cards", + "lost-cards-list": "Restored Items", + "restore-lost-cards-migration": "Restore Lost Cards", + "restore-lost-cards-migration-description": "Finds and restores cards and lists with missing swimlaneId or listId. Creates a 'Lost Cards' swimlane to make all lost items visible again.", + "restore-all-archived-migration": "Restore All Archived", + "restore-all-archived-migration-description": "Restores all archived swimlanes, lists, and cards. Automatically fixes any missing swimlaneId or listId to make items visible.", + "fix-missing-lists-migration": "Fix Missing Lists", + "fix-missing-lists-migration-description": "Detects and repairs missing or corrupted lists in the board structure.", + "fix-avatar-urls-migration": "Fix Avatar URLs", + "fix-avatar-urls-migration-description": "Updates avatar URLs for board members to use the correct storage backend and fixes broken avatar references.", + "fix-all-file-urls-migration": "Fix All File URLs", + "fix-all-file-urls-migration-description": "Updates all file attachment URLs on this board to use the correct storage backend and fixes broken file references.", + "migration-needed": "Migration Needed", + "migration-complete": "Complete", + "migration-running": "Running...", + "migration-successful": "Migration completed successfully", + "migration-failed": "Migration failed", + "migrations": "Migrations", + "migrations-admin-only": "Only board administrators can run migrations", + "migrations-description": "Run data integrity checks and repairs for this board. Each migration can be executed individually.", + "no-issues-found": "No issues found", + "run-migration": "Run Migration", + "run-comprehensive-migration-confirm": "This will perform a comprehensive migration to check and fix board data integrity. This may take a few moments. Continue?", + "run-delete-duplicate-empty-lists-migration-confirm": "This will first convert any shared lists to per-swimlane lists, then delete empty lists that have a duplicate list with the same title containing cards. Only truly redundant empty lists will be removed. Continue?", + "run-restore-lost-cards-migration-confirm": "This will create a 'Lost Cards' swimlane and restore all cards and lists with missing swimlaneId or listId. This only affects non-archived items. Continue?", + "run-restore-all-archived-migration-confirm": "This will restore ALL archived swimlanes, lists, and cards, making them visible again. Any items with missing IDs will be automatically fixed. This cannot be easily undone. Continue?", + "run-fix-missing-lists-migration-confirm": "This will detect and repair missing or corrupted lists in the board structure. Continue?", + "run-fix-avatar-urls-migration-confirm": "This will update avatar URLs for board members to use the correct storage backend. Continue?", + "run-fix-all-file-urls-migration-confirm": "This will update all file attachment URLs on this board to use the correct storage backend. Continue?", + "restore-lost-cards-nothing-to-restore": "No lost swimlanes, lists, or cards to restore", + + "migration-progress-title": "Board Migration in Progress", + "migration-progress-overall": "Overall Progress", + "migration-progress-current-step": "Current Step", + "migration-progress-status": "Status", + "migration-progress-details": "Details", + "migration-progress-note": "Please wait while we migrate your board to the latest structure...", + + "step-analyze-board-structure": "Analyze Board Structure", + "step-fix-orphaned-cards": "Fix Orphaned Cards", + "step-convert-shared-lists": "Convert Shared Lists", + "step-ensure-per-swimlane-lists": "Ensure Per-Swimlane Lists", + "step-validate-migration": "Validate Migration", + "step-fix-avatar-urls": "Fix Avatar URLs", + "step-fix-attachment-urls": "Fix Attachment URLs", + "step-analyze-lists": "Analyze Lists", + "step-create-missing-lists": "Create Missing Lists", + "step-update-cards": "Update Cards", + "step-finalize": "Finalize", + "step-delete-duplicate-empty-lists": "Delete Duplicate Empty Lists", + "step-ensure-lost-cards-swimlane": "Ensure Lost Cards Swimlane", + "step-restore-lists": "Restore Lists", + "step-restore-cards": "Restore Cards", + "step-restore-swimlanes": "Restore Swimlanes", + "step-fix-missing-ids": "Fix Missing IDs", + "step-scan-users": "Checking board member avatars", + "step-scan-files": "Checking board file attachments", + "step-fix-file-urls": "Fixing file URLs", "cleanup": "Cleanup", "cleanup-old-jobs": "Cleanup Old Jobs", "completed": "Completed", diff --git a/imports/i18n/data/hi.i18n.json b/imports/i18n/data/hi.i18n.json index 8febb7ce7..4f7be7672 100644 --- a/imports/i18n/data/hi.i18n.json +++ b/imports/i18n/data/hi.i18n.json @@ -1404,7 +1404,70 @@ "back-to-settings": "Back to Settings", "board-id": "Board ID", "board-migration": "Board Migration", + "board-migrations": "Board Migrations", "card-show-lists-on-minicard": "Show Lists on Minicard", + "comprehensive-board-migration": "Comprehensive Board Migration", + "comprehensive-board-migration-description": "Performs comprehensive checks and fixes for board data integrity, including list ordering, card positions, and swimlane structure.", + "delete-duplicate-empty-lists-migration": "Delete Duplicate Empty Lists", + "delete-duplicate-empty-lists-migration-description": "Safely deletes empty duplicate lists. Only removes lists that have no cards AND have another list with the same title that contains cards.", + "lost-cards": "Lost Cards", + "lost-cards-list": "Restored Items", + "restore-lost-cards-migration": "Restore Lost Cards", + "restore-lost-cards-migration-description": "Finds and restores cards and lists with missing swimlaneId or listId. Creates a 'Lost Cards' swimlane to make all lost items visible again.", + "restore-all-archived-migration": "Restore All Archived", + "restore-all-archived-migration-description": "Restores all archived swimlanes, lists, and cards. Automatically fixes any missing swimlaneId or listId to make items visible.", + "fix-missing-lists-migration": "Fix Missing Lists", + "fix-missing-lists-migration-description": "Detects and repairs missing or corrupted lists in the board structure.", + "fix-avatar-urls-migration": "Fix Avatar URLs", + "fix-avatar-urls-migration-description": "Updates avatar URLs for board members to use the correct storage backend and fixes broken avatar references.", + "fix-all-file-urls-migration": "Fix All File URLs", + "fix-all-file-urls-migration-description": "Updates all file attachment URLs on this board to use the correct storage backend and fixes broken file references.", + "migration-needed": "Migration Needed", + "migration-complete": "Complete", + "migration-running": "Running...", + "migration-successful": "Migration completed successfully", + "migration-failed": "Migration failed", + "migrations": "Migrations", + "migrations-admin-only": "Only board administrators can run migrations", + "migrations-description": "Run data integrity checks and repairs for this board. Each migration can be executed individually.", + "no-issues-found": "No issues found", + "run-migration": "Run Migration", + "run-comprehensive-migration-confirm": "This will perform a comprehensive migration to check and fix board data integrity. This may take a few moments. Continue?", + "run-delete-duplicate-empty-lists-migration-confirm": "This will first convert any shared lists to per-swimlane lists, then delete empty lists that have a duplicate list with the same title containing cards. Only truly redundant empty lists will be removed. Continue?", + "run-restore-lost-cards-migration-confirm": "This will create a 'Lost Cards' swimlane and restore all cards and lists with missing swimlaneId or listId. This only affects non-archived items. Continue?", + "run-restore-all-archived-migration-confirm": "This will restore ALL archived swimlanes, lists, and cards, making them visible again. Any items with missing IDs will be automatically fixed. This cannot be easily undone. Continue?", + "run-fix-missing-lists-migration-confirm": "This will detect and repair missing or corrupted lists in the board structure. Continue?", + "run-fix-avatar-urls-migration-confirm": "This will update avatar URLs for board members to use the correct storage backend. Continue?", + "run-fix-all-file-urls-migration-confirm": "This will update all file attachment URLs on this board to use the correct storage backend. Continue?", + "restore-lost-cards-nothing-to-restore": "No lost swimlanes, lists, or cards to restore", + + "migration-progress-title": "Board Migration in Progress", + "migration-progress-overall": "Overall Progress", + "migration-progress-current-step": "Current Step", + "migration-progress-status": "Status", + "migration-progress-details": "Details", + "migration-progress-note": "Please wait while we migrate your board to the latest structure...", + + "step-analyze-board-structure": "Analyze Board Structure", + "step-fix-orphaned-cards": "Fix Orphaned Cards", + "step-convert-shared-lists": "Convert Shared Lists", + "step-ensure-per-swimlane-lists": "Ensure Per-Swimlane Lists", + "step-validate-migration": "Validate Migration", + "step-fix-avatar-urls": "Fix Avatar URLs", + "step-fix-attachment-urls": "Fix Attachment URLs", + "step-analyze-lists": "Analyze Lists", + "step-create-missing-lists": "Create Missing Lists", + "step-update-cards": "Update Cards", + "step-finalize": "Finalize", + "step-delete-duplicate-empty-lists": "Delete Duplicate Empty Lists", + "step-ensure-lost-cards-swimlane": "Ensure Lost Cards Swimlane", + "step-restore-lists": "Restore Lists", + "step-restore-cards": "Restore Cards", + "step-restore-swimlanes": "Restore Swimlanes", + "step-fix-missing-ids": "Fix Missing IDs", + "step-scan-users": "Checking board member avatars", + "step-scan-files": "Checking board file attachments", + "step-fix-file-urls": "Fixing file URLs", "cleanup": "Cleanup", "cleanup-old-jobs": "Cleanup Old Jobs", "completed": "Completed", diff --git a/imports/i18n/data/hr.i18n.json b/imports/i18n/data/hr.i18n.json index 5b66d06dd..8c1125a31 100644 --- a/imports/i18n/data/hr.i18n.json +++ b/imports/i18n/data/hr.i18n.json @@ -1404,7 +1404,70 @@ "back-to-settings": "Back to Settings", "board-id": "Board ID", "board-migration": "Board Migration", + "board-migrations": "Board Migrations", "card-show-lists-on-minicard": "Show Lists on Minicard", + "comprehensive-board-migration": "Comprehensive Board Migration", + "comprehensive-board-migration-description": "Performs comprehensive checks and fixes for board data integrity, including list ordering, card positions, and swimlane structure.", + "delete-duplicate-empty-lists-migration": "Delete Duplicate Empty Lists", + "delete-duplicate-empty-lists-migration-description": "Safely deletes empty duplicate lists. Only removes lists that have no cards AND have another list with the same title that contains cards.", + "lost-cards": "Lost Cards", + "lost-cards-list": "Restored Items", + "restore-lost-cards-migration": "Restore Lost Cards", + "restore-lost-cards-migration-description": "Finds and restores cards and lists with missing swimlaneId or listId. Creates a 'Lost Cards' swimlane to make all lost items visible again.", + "restore-all-archived-migration": "Restore All Archived", + "restore-all-archived-migration-description": "Restores all archived swimlanes, lists, and cards. Automatically fixes any missing swimlaneId or listId to make items visible.", + "fix-missing-lists-migration": "Fix Missing Lists", + "fix-missing-lists-migration-description": "Detects and repairs missing or corrupted lists in the board structure.", + "fix-avatar-urls-migration": "Fix Avatar URLs", + "fix-avatar-urls-migration-description": "Updates avatar URLs for board members to use the correct storage backend and fixes broken avatar references.", + "fix-all-file-urls-migration": "Fix All File URLs", + "fix-all-file-urls-migration-description": "Updates all file attachment URLs on this board to use the correct storage backend and fixes broken file references.", + "migration-needed": "Migration Needed", + "migration-complete": "Complete", + "migration-running": "Running...", + "migration-successful": "Migration completed successfully", + "migration-failed": "Migration failed", + "migrations": "Migrations", + "migrations-admin-only": "Only board administrators can run migrations", + "migrations-description": "Run data integrity checks and repairs for this board. Each migration can be executed individually.", + "no-issues-found": "No issues found", + "run-migration": "Run Migration", + "run-comprehensive-migration-confirm": "This will perform a comprehensive migration to check and fix board data integrity. This may take a few moments. Continue?", + "run-delete-duplicate-empty-lists-migration-confirm": "This will first convert any shared lists to per-swimlane lists, then delete empty lists that have a duplicate list with the same title containing cards. Only truly redundant empty lists will be removed. Continue?", + "run-restore-lost-cards-migration-confirm": "This will create a 'Lost Cards' swimlane and restore all cards and lists with missing swimlaneId or listId. This only affects non-archived items. Continue?", + "run-restore-all-archived-migration-confirm": "This will restore ALL archived swimlanes, lists, and cards, making them visible again. Any items with missing IDs will be automatically fixed. This cannot be easily undone. Continue?", + "run-fix-missing-lists-migration-confirm": "This will detect and repair missing or corrupted lists in the board structure. Continue?", + "run-fix-avatar-urls-migration-confirm": "This will update avatar URLs for board members to use the correct storage backend. Continue?", + "run-fix-all-file-urls-migration-confirm": "This will update all file attachment URLs on this board to use the correct storage backend. Continue?", + "restore-lost-cards-nothing-to-restore": "No lost swimlanes, lists, or cards to restore", + + "migration-progress-title": "Board Migration in Progress", + "migration-progress-overall": "Overall Progress", + "migration-progress-current-step": "Current Step", + "migration-progress-status": "Status", + "migration-progress-details": "Details", + "migration-progress-note": "Please wait while we migrate your board to the latest structure...", + + "step-analyze-board-structure": "Analyze Board Structure", + "step-fix-orphaned-cards": "Fix Orphaned Cards", + "step-convert-shared-lists": "Convert Shared Lists", + "step-ensure-per-swimlane-lists": "Ensure Per-Swimlane Lists", + "step-validate-migration": "Validate Migration", + "step-fix-avatar-urls": "Fix Avatar URLs", + "step-fix-attachment-urls": "Fix Attachment URLs", + "step-analyze-lists": "Analyze Lists", + "step-create-missing-lists": "Create Missing Lists", + "step-update-cards": "Update Cards", + "step-finalize": "Finalize", + "step-delete-duplicate-empty-lists": "Delete Duplicate Empty Lists", + "step-ensure-lost-cards-swimlane": "Ensure Lost Cards Swimlane", + "step-restore-lists": "Restore Lists", + "step-restore-cards": "Restore Cards", + "step-restore-swimlanes": "Restore Swimlanes", + "step-fix-missing-ids": "Fix Missing IDs", + "step-scan-users": "Checking board member avatars", + "step-scan-files": "Checking board file attachments", + "step-fix-file-urls": "Fixing file URLs", "cleanup": "Cleanup", "cleanup-old-jobs": "Cleanup Old Jobs", "completed": "Completed", diff --git a/imports/i18n/data/hu.i18n.json b/imports/i18n/data/hu.i18n.json index 82d25500a..cc54d40f4 100644 --- a/imports/i18n/data/hu.i18n.json +++ b/imports/i18n/data/hu.i18n.json @@ -1404,7 +1404,70 @@ "back-to-settings": "Back to Settings", "board-id": "Board ID", "board-migration": "Board Migration", + "board-migrations": "Board Migrations", "card-show-lists-on-minicard": "Show Lists on Minicard", + "comprehensive-board-migration": "Comprehensive Board Migration", + "comprehensive-board-migration-description": "Performs comprehensive checks and fixes for board data integrity, including list ordering, card positions, and swimlane structure.", + "delete-duplicate-empty-lists-migration": "Delete Duplicate Empty Lists", + "delete-duplicate-empty-lists-migration-description": "Safely deletes empty duplicate lists. Only removes lists that have no cards AND have another list with the same title that contains cards.", + "lost-cards": "Lost Cards", + "lost-cards-list": "Restored Items", + "restore-lost-cards-migration": "Restore Lost Cards", + "restore-lost-cards-migration-description": "Finds and restores cards and lists with missing swimlaneId or listId. Creates a 'Lost Cards' swimlane to make all lost items visible again.", + "restore-all-archived-migration": "Restore All Archived", + "restore-all-archived-migration-description": "Restores all archived swimlanes, lists, and cards. Automatically fixes any missing swimlaneId or listId to make items visible.", + "fix-missing-lists-migration": "Fix Missing Lists", + "fix-missing-lists-migration-description": "Detects and repairs missing or corrupted lists in the board structure.", + "fix-avatar-urls-migration": "Fix Avatar URLs", + "fix-avatar-urls-migration-description": "Updates avatar URLs for board members to use the correct storage backend and fixes broken avatar references.", + "fix-all-file-urls-migration": "Fix All File URLs", + "fix-all-file-urls-migration-description": "Updates all file attachment URLs on this board to use the correct storage backend and fixes broken file references.", + "migration-needed": "Migration Needed", + "migration-complete": "Complete", + "migration-running": "Running...", + "migration-successful": "Migration completed successfully", + "migration-failed": "Migration failed", + "migrations": "Migrations", + "migrations-admin-only": "Only board administrators can run migrations", + "migrations-description": "Run data integrity checks and repairs for this board. Each migration can be executed individually.", + "no-issues-found": "No issues found", + "run-migration": "Run Migration", + "run-comprehensive-migration-confirm": "This will perform a comprehensive migration to check and fix board data integrity. This may take a few moments. Continue?", + "run-delete-duplicate-empty-lists-migration-confirm": "This will first convert any shared lists to per-swimlane lists, then delete empty lists that have a duplicate list with the same title containing cards. Only truly redundant empty lists will be removed. Continue?", + "run-restore-lost-cards-migration-confirm": "This will create a 'Lost Cards' swimlane and restore all cards and lists with missing swimlaneId or listId. This only affects non-archived items. Continue?", + "run-restore-all-archived-migration-confirm": "This will restore ALL archived swimlanes, lists, and cards, making them visible again. Any items with missing IDs will be automatically fixed. This cannot be easily undone. Continue?", + "run-fix-missing-lists-migration-confirm": "This will detect and repair missing or corrupted lists in the board structure. Continue?", + "run-fix-avatar-urls-migration-confirm": "This will update avatar URLs for board members to use the correct storage backend. Continue?", + "run-fix-all-file-urls-migration-confirm": "This will update all file attachment URLs on this board to use the correct storage backend. Continue?", + "restore-lost-cards-nothing-to-restore": "No lost swimlanes, lists, or cards to restore", + + "migration-progress-title": "Board Migration in Progress", + "migration-progress-overall": "Overall Progress", + "migration-progress-current-step": "Current Step", + "migration-progress-status": "Állapot", + "migration-progress-details": "Részletek", + "migration-progress-note": "Please wait while we migrate your board to the latest structure...", + + "step-analyze-board-structure": "Analyze Board Structure", + "step-fix-orphaned-cards": "Fix Orphaned Cards", + "step-convert-shared-lists": "Convert Shared Lists", + "step-ensure-per-swimlane-lists": "Ensure Per-Swimlane Lists", + "step-validate-migration": "Validate Migration", + "step-fix-avatar-urls": "Fix Avatar URLs", + "step-fix-attachment-urls": "Fix Attachment URLs", + "step-analyze-lists": "Analyze Lists", + "step-create-missing-lists": "Create Missing Lists", + "step-update-cards": "Update Cards", + "step-finalize": "Finalize", + "step-delete-duplicate-empty-lists": "Delete Duplicate Empty Lists", + "step-ensure-lost-cards-swimlane": "Ensure Lost Cards Swimlane", + "step-restore-lists": "Restore Lists", + "step-restore-cards": "Restore Cards", + "step-restore-swimlanes": "Restore Swimlanes", + "step-fix-missing-ids": "Fix Missing IDs", + "step-scan-users": "Checking board member avatars", + "step-scan-files": "Checking board file attachments", + "step-fix-file-urls": "Fixing file URLs", "cleanup": "Cleanup", "cleanup-old-jobs": "Cleanup Old Jobs", "completed": "Kész", diff --git a/imports/i18n/data/hy.i18n.json b/imports/i18n/data/hy.i18n.json index 78f216ac6..3ac06cc65 100644 --- a/imports/i18n/data/hy.i18n.json +++ b/imports/i18n/data/hy.i18n.json @@ -1404,7 +1404,70 @@ "back-to-settings": "Back to Settings", "board-id": "Board ID", "board-migration": "Board Migration", + "board-migrations": "Board Migrations", "card-show-lists-on-minicard": "Show Lists on Minicard", + "comprehensive-board-migration": "Comprehensive Board Migration", + "comprehensive-board-migration-description": "Performs comprehensive checks and fixes for board data integrity, including list ordering, card positions, and swimlane structure.", + "delete-duplicate-empty-lists-migration": "Delete Duplicate Empty Lists", + "delete-duplicate-empty-lists-migration-description": "Safely deletes empty duplicate lists. Only removes lists that have no cards AND have another list with the same title that contains cards.", + "lost-cards": "Lost Cards", + "lost-cards-list": "Restored Items", + "restore-lost-cards-migration": "Restore Lost Cards", + "restore-lost-cards-migration-description": "Finds and restores cards and lists with missing swimlaneId or listId. Creates a 'Lost Cards' swimlane to make all lost items visible again.", + "restore-all-archived-migration": "Restore All Archived", + "restore-all-archived-migration-description": "Restores all archived swimlanes, lists, and cards. Automatically fixes any missing swimlaneId or listId to make items visible.", + "fix-missing-lists-migration": "Fix Missing Lists", + "fix-missing-lists-migration-description": "Detects and repairs missing or corrupted lists in the board structure.", + "fix-avatar-urls-migration": "Fix Avatar URLs", + "fix-avatar-urls-migration-description": "Updates avatar URLs for board members to use the correct storage backend and fixes broken avatar references.", + "fix-all-file-urls-migration": "Fix All File URLs", + "fix-all-file-urls-migration-description": "Updates all file attachment URLs on this board to use the correct storage backend and fixes broken file references.", + "migration-needed": "Migration Needed", + "migration-complete": "Complete", + "migration-running": "Running...", + "migration-successful": "Migration completed successfully", + "migration-failed": "Migration failed", + "migrations": "Migrations", + "migrations-admin-only": "Only board administrators can run migrations", + "migrations-description": "Run data integrity checks and repairs for this board. Each migration can be executed individually.", + "no-issues-found": "No issues found", + "run-migration": "Run Migration", + "run-comprehensive-migration-confirm": "This will perform a comprehensive migration to check and fix board data integrity. This may take a few moments. Continue?", + "run-delete-duplicate-empty-lists-migration-confirm": "This will first convert any shared lists to per-swimlane lists, then delete empty lists that have a duplicate list with the same title containing cards. Only truly redundant empty lists will be removed. Continue?", + "run-restore-lost-cards-migration-confirm": "This will create a 'Lost Cards' swimlane and restore all cards and lists with missing swimlaneId or listId. This only affects non-archived items. Continue?", + "run-restore-all-archived-migration-confirm": "This will restore ALL archived swimlanes, lists, and cards, making them visible again. Any items with missing IDs will be automatically fixed. This cannot be easily undone. Continue?", + "run-fix-missing-lists-migration-confirm": "This will detect and repair missing or corrupted lists in the board structure. Continue?", + "run-fix-avatar-urls-migration-confirm": "This will update avatar URLs for board members to use the correct storage backend. Continue?", + "run-fix-all-file-urls-migration-confirm": "This will update all file attachment URLs on this board to use the correct storage backend. Continue?", + "restore-lost-cards-nothing-to-restore": "No lost swimlanes, lists, or cards to restore", + + "migration-progress-title": "Board Migration in Progress", + "migration-progress-overall": "Overall Progress", + "migration-progress-current-step": "Current Step", + "migration-progress-status": "Status", + "migration-progress-details": "Details", + "migration-progress-note": "Please wait while we migrate your board to the latest structure...", + + "step-analyze-board-structure": "Analyze Board Structure", + "step-fix-orphaned-cards": "Fix Orphaned Cards", + "step-convert-shared-lists": "Convert Shared Lists", + "step-ensure-per-swimlane-lists": "Ensure Per-Swimlane Lists", + "step-validate-migration": "Validate Migration", + "step-fix-avatar-urls": "Fix Avatar URLs", + "step-fix-attachment-urls": "Fix Attachment URLs", + "step-analyze-lists": "Analyze Lists", + "step-create-missing-lists": "Create Missing Lists", + "step-update-cards": "Update Cards", + "step-finalize": "Finalize", + "step-delete-duplicate-empty-lists": "Delete Duplicate Empty Lists", + "step-ensure-lost-cards-swimlane": "Ensure Lost Cards Swimlane", + "step-restore-lists": "Restore Lists", + "step-restore-cards": "Restore Cards", + "step-restore-swimlanes": "Restore Swimlanes", + "step-fix-missing-ids": "Fix Missing IDs", + "step-scan-users": "Checking board member avatars", + "step-scan-files": "Checking board file attachments", + "step-fix-file-urls": "Fixing file URLs", "cleanup": "Cleanup", "cleanup-old-jobs": "Cleanup Old Jobs", "completed": "Completed", diff --git a/imports/i18n/data/id.i18n.json b/imports/i18n/data/id.i18n.json index e5646003c..d49641ae7 100644 --- a/imports/i18n/data/id.i18n.json +++ b/imports/i18n/data/id.i18n.json @@ -1404,7 +1404,70 @@ "back-to-settings": "Back to Settings", "board-id": "Board ID", "board-migration": "Board Migration", + "board-migrations": "Board Migrations", "card-show-lists-on-minicard": "Show Lists on Minicard", + "comprehensive-board-migration": "Comprehensive Board Migration", + "comprehensive-board-migration-description": "Performs comprehensive checks and fixes for board data integrity, including list ordering, card positions, and swimlane structure.", + "delete-duplicate-empty-lists-migration": "Delete Duplicate Empty Lists", + "delete-duplicate-empty-lists-migration-description": "Safely deletes empty duplicate lists. Only removes lists that have no cards AND have another list with the same title that contains cards.", + "lost-cards": "Lost Cards", + "lost-cards-list": "Restored Items", + "restore-lost-cards-migration": "Restore Lost Cards", + "restore-lost-cards-migration-description": "Finds and restores cards and lists with missing swimlaneId or listId. Creates a 'Lost Cards' swimlane to make all lost items visible again.", + "restore-all-archived-migration": "Restore All Archived", + "restore-all-archived-migration-description": "Restores all archived swimlanes, lists, and cards. Automatically fixes any missing swimlaneId or listId to make items visible.", + "fix-missing-lists-migration": "Fix Missing Lists", + "fix-missing-lists-migration-description": "Detects and repairs missing or corrupted lists in the board structure.", + "fix-avatar-urls-migration": "Fix Avatar URLs", + "fix-avatar-urls-migration-description": "Updates avatar URLs for board members to use the correct storage backend and fixes broken avatar references.", + "fix-all-file-urls-migration": "Fix All File URLs", + "fix-all-file-urls-migration-description": "Updates all file attachment URLs on this board to use the correct storage backend and fixes broken file references.", + "migration-needed": "Migration Needed", + "migration-complete": "Complete", + "migration-running": "Running...", + "migration-successful": "Migration completed successfully", + "migration-failed": "Migration failed", + "migrations": "Migrations", + "migrations-admin-only": "Only board administrators can run migrations", + "migrations-description": "Run data integrity checks and repairs for this board. Each migration can be executed individually.", + "no-issues-found": "No issues found", + "run-migration": "Run Migration", + "run-comprehensive-migration-confirm": "This will perform a comprehensive migration to check and fix board data integrity. This may take a few moments. Continue?", + "run-delete-duplicate-empty-lists-migration-confirm": "This will first convert any shared lists to per-swimlane lists, then delete empty lists that have a duplicate list with the same title containing cards. Only truly redundant empty lists will be removed. Continue?", + "run-restore-lost-cards-migration-confirm": "This will create a 'Lost Cards' swimlane and restore all cards and lists with missing swimlaneId or listId. This only affects non-archived items. Continue?", + "run-restore-all-archived-migration-confirm": "This will restore ALL archived swimlanes, lists, and cards, making them visible again. Any items with missing IDs will be automatically fixed. This cannot be easily undone. Continue?", + "run-fix-missing-lists-migration-confirm": "This will detect and repair missing or corrupted lists in the board structure. Continue?", + "run-fix-avatar-urls-migration-confirm": "This will update avatar URLs for board members to use the correct storage backend. Continue?", + "run-fix-all-file-urls-migration-confirm": "This will update all file attachment URLs on this board to use the correct storage backend. Continue?", + "restore-lost-cards-nothing-to-restore": "No lost swimlanes, lists, or cards to restore", + + "migration-progress-title": "Board Migration in Progress", + "migration-progress-overall": "Overall Progress", + "migration-progress-current-step": "Current Step", + "migration-progress-status": "Status", + "migration-progress-details": "Details", + "migration-progress-note": "Please wait while we migrate your board to the latest structure...", + + "step-analyze-board-structure": "Analyze Board Structure", + "step-fix-orphaned-cards": "Fix Orphaned Cards", + "step-convert-shared-lists": "Convert Shared Lists", + "step-ensure-per-swimlane-lists": "Ensure Per-Swimlane Lists", + "step-validate-migration": "Validate Migration", + "step-fix-avatar-urls": "Fix Avatar URLs", + "step-fix-attachment-urls": "Fix Attachment URLs", + "step-analyze-lists": "Analyze Lists", + "step-create-missing-lists": "Create Missing Lists", + "step-update-cards": "Update Cards", + "step-finalize": "Finalize", + "step-delete-duplicate-empty-lists": "Delete Duplicate Empty Lists", + "step-ensure-lost-cards-swimlane": "Ensure Lost Cards Swimlane", + "step-restore-lists": "Restore Lists", + "step-restore-cards": "Restore Cards", + "step-restore-swimlanes": "Restore Swimlanes", + "step-fix-missing-ids": "Fix Missing IDs", + "step-scan-users": "Checking board member avatars", + "step-scan-files": "Checking board file attachments", + "step-fix-file-urls": "Fixing file URLs", "cleanup": "Cleanup", "cleanup-old-jobs": "Cleanup Old Jobs", "completed": "Completed", diff --git a/imports/i18n/data/ig.i18n.json b/imports/i18n/data/ig.i18n.json index 38e4d0620..acce99747 100644 --- a/imports/i18n/data/ig.i18n.json +++ b/imports/i18n/data/ig.i18n.json @@ -1404,7 +1404,70 @@ "back-to-settings": "Back to Settings", "board-id": "Board ID", "board-migration": "Board Migration", + "board-migrations": "Board Migrations", "card-show-lists-on-minicard": "Show Lists on Minicard", + "comprehensive-board-migration": "Comprehensive Board Migration", + "comprehensive-board-migration-description": "Performs comprehensive checks and fixes for board data integrity, including list ordering, card positions, and swimlane structure.", + "delete-duplicate-empty-lists-migration": "Delete Duplicate Empty Lists", + "delete-duplicate-empty-lists-migration-description": "Safely deletes empty duplicate lists. Only removes lists that have no cards AND have another list with the same title that contains cards.", + "lost-cards": "Lost Cards", + "lost-cards-list": "Restored Items", + "restore-lost-cards-migration": "Restore Lost Cards", + "restore-lost-cards-migration-description": "Finds and restores cards and lists with missing swimlaneId or listId. Creates a 'Lost Cards' swimlane to make all lost items visible again.", + "restore-all-archived-migration": "Restore All Archived", + "restore-all-archived-migration-description": "Restores all archived swimlanes, lists, and cards. Automatically fixes any missing swimlaneId or listId to make items visible.", + "fix-missing-lists-migration": "Fix Missing Lists", + "fix-missing-lists-migration-description": "Detects and repairs missing or corrupted lists in the board structure.", + "fix-avatar-urls-migration": "Fix Avatar URLs", + "fix-avatar-urls-migration-description": "Updates avatar URLs for board members to use the correct storage backend and fixes broken avatar references.", + "fix-all-file-urls-migration": "Fix All File URLs", + "fix-all-file-urls-migration-description": "Updates all file attachment URLs on this board to use the correct storage backend and fixes broken file references.", + "migration-needed": "Migration Needed", + "migration-complete": "Complete", + "migration-running": "Running...", + "migration-successful": "Migration completed successfully", + "migration-failed": "Migration failed", + "migrations": "Migrations", + "migrations-admin-only": "Only board administrators can run migrations", + "migrations-description": "Run data integrity checks and repairs for this board. Each migration can be executed individually.", + "no-issues-found": "No issues found", + "run-migration": "Run Migration", + "run-comprehensive-migration-confirm": "This will perform a comprehensive migration to check and fix board data integrity. This may take a few moments. Continue?", + "run-delete-duplicate-empty-lists-migration-confirm": "This will first convert any shared lists to per-swimlane lists, then delete empty lists that have a duplicate list with the same title containing cards. Only truly redundant empty lists will be removed. Continue?", + "run-restore-lost-cards-migration-confirm": "This will create a 'Lost Cards' swimlane and restore all cards and lists with missing swimlaneId or listId. This only affects non-archived items. Continue?", + "run-restore-all-archived-migration-confirm": "This will restore ALL archived swimlanes, lists, and cards, making them visible again. Any items with missing IDs will be automatically fixed. This cannot be easily undone. Continue?", + "run-fix-missing-lists-migration-confirm": "This will detect and repair missing or corrupted lists in the board structure. Continue?", + "run-fix-avatar-urls-migration-confirm": "This will update avatar URLs for board members to use the correct storage backend. Continue?", + "run-fix-all-file-urls-migration-confirm": "This will update all file attachment URLs on this board to use the correct storage backend. Continue?", + "restore-lost-cards-nothing-to-restore": "No lost swimlanes, lists, or cards to restore", + + "migration-progress-title": "Board Migration in Progress", + "migration-progress-overall": "Overall Progress", + "migration-progress-current-step": "Current Step", + "migration-progress-status": "Status", + "migration-progress-details": "Details", + "migration-progress-note": "Please wait while we migrate your board to the latest structure...", + + "step-analyze-board-structure": "Analyze Board Structure", + "step-fix-orphaned-cards": "Fix Orphaned Cards", + "step-convert-shared-lists": "Convert Shared Lists", + "step-ensure-per-swimlane-lists": "Ensure Per-Swimlane Lists", + "step-validate-migration": "Validate Migration", + "step-fix-avatar-urls": "Fix Avatar URLs", + "step-fix-attachment-urls": "Fix Attachment URLs", + "step-analyze-lists": "Analyze Lists", + "step-create-missing-lists": "Create Missing Lists", + "step-update-cards": "Update Cards", + "step-finalize": "Finalize", + "step-delete-duplicate-empty-lists": "Delete Duplicate Empty Lists", + "step-ensure-lost-cards-swimlane": "Ensure Lost Cards Swimlane", + "step-restore-lists": "Restore Lists", + "step-restore-cards": "Restore Cards", + "step-restore-swimlanes": "Restore Swimlanes", + "step-fix-missing-ids": "Fix Missing IDs", + "step-scan-users": "Checking board member avatars", + "step-scan-files": "Checking board file attachments", + "step-fix-file-urls": "Fixing file URLs", "cleanup": "Cleanup", "cleanup-old-jobs": "Cleanup Old Jobs", "completed": "Completed", diff --git a/imports/i18n/data/it.i18n.json b/imports/i18n/data/it.i18n.json index a6046dd35..e6f9f1f62 100644 --- a/imports/i18n/data/it.i18n.json +++ b/imports/i18n/data/it.i18n.json @@ -1404,7 +1404,70 @@ "back-to-settings": "Back to Settings", "board-id": "Board ID", "board-migration": "Board Migration", + "board-migrations": "Board Migrations", "card-show-lists-on-minicard": "Show Lists on Minicard", + "comprehensive-board-migration": "Comprehensive Board Migration", + "comprehensive-board-migration-description": "Performs comprehensive checks and fixes for board data integrity, including list ordering, card positions, and swimlane structure.", + "delete-duplicate-empty-lists-migration": "Delete Duplicate Empty Lists", + "delete-duplicate-empty-lists-migration-description": "Safely deletes empty duplicate lists. Only removes lists that have no cards AND have another list with the same title that contains cards.", + "lost-cards": "Lost Cards", + "lost-cards-list": "Restored Items", + "restore-lost-cards-migration": "Restore Lost Cards", + "restore-lost-cards-migration-description": "Finds and restores cards and lists with missing swimlaneId or listId. Creates a 'Lost Cards' swimlane to make all lost items visible again.", + "restore-all-archived-migration": "Restore All Archived", + "restore-all-archived-migration-description": "Restores all archived swimlanes, lists, and cards. Automatically fixes any missing swimlaneId or listId to make items visible.", + "fix-missing-lists-migration": "Fix Missing Lists", + "fix-missing-lists-migration-description": "Detects and repairs missing or corrupted lists in the board structure.", + "fix-avatar-urls-migration": "Fix Avatar URLs", + "fix-avatar-urls-migration-description": "Updates avatar URLs for board members to use the correct storage backend and fixes broken avatar references.", + "fix-all-file-urls-migration": "Fix All File URLs", + "fix-all-file-urls-migration-description": "Updates all file attachment URLs on this board to use the correct storage backend and fixes broken file references.", + "migration-needed": "Migration Needed", + "migration-complete": "Completato", + "migration-running": "Running...", + "migration-successful": "Migration completed successfully", + "migration-failed": "Migration failed", + "migrations": "Migrations", + "migrations-admin-only": "Only board administrators can run migrations", + "migrations-description": "Run data integrity checks and repairs for this board. Each migration can be executed individually.", + "no-issues-found": "No issues found", + "run-migration": "Run Migration", + "run-comprehensive-migration-confirm": "This will perform a comprehensive migration to check and fix board data integrity. This may take a few moments. Continue?", + "run-delete-duplicate-empty-lists-migration-confirm": "This will first convert any shared lists to per-swimlane lists, then delete empty lists that have a duplicate list with the same title containing cards. Only truly redundant empty lists will be removed. Continue?", + "run-restore-lost-cards-migration-confirm": "This will create a 'Lost Cards' swimlane and restore all cards and lists with missing swimlaneId or listId. This only affects non-archived items. Continue?", + "run-restore-all-archived-migration-confirm": "This will restore ALL archived swimlanes, lists, and cards, making them visible again. Any items with missing IDs will be automatically fixed. This cannot be easily undone. Continue?", + "run-fix-missing-lists-migration-confirm": "This will detect and repair missing or corrupted lists in the board structure. Continue?", + "run-fix-avatar-urls-migration-confirm": "This will update avatar URLs for board members to use the correct storage backend. Continue?", + "run-fix-all-file-urls-migration-confirm": "This will update all file attachment URLs on this board to use the correct storage backend. Continue?", + "restore-lost-cards-nothing-to-restore": "No lost swimlanes, lists, or cards to restore", + + "migration-progress-title": "Board Migration in Progress", + "migration-progress-overall": "Overall Progress", + "migration-progress-current-step": "Current Step", + "migration-progress-status": "Stato", + "migration-progress-details": "Dettagli", + "migration-progress-note": "Please wait while we migrate your board to the latest structure...", + + "step-analyze-board-structure": "Analyze Board Structure", + "step-fix-orphaned-cards": "Fix Orphaned Cards", + "step-convert-shared-lists": "Convert Shared Lists", + "step-ensure-per-swimlane-lists": "Ensure Per-Swimlane Lists", + "step-validate-migration": "Validate Migration", + "step-fix-avatar-urls": "Fix Avatar URLs", + "step-fix-attachment-urls": "Fix Attachment URLs", + "step-analyze-lists": "Analyze Lists", + "step-create-missing-lists": "Create Missing Lists", + "step-update-cards": "Update Cards", + "step-finalize": "Finalize", + "step-delete-duplicate-empty-lists": "Delete Duplicate Empty Lists", + "step-ensure-lost-cards-swimlane": "Ensure Lost Cards Swimlane", + "step-restore-lists": "Restore Lists", + "step-restore-cards": "Restore Cards", + "step-restore-swimlanes": "Restore Swimlanes", + "step-fix-missing-ids": "Fix Missing IDs", + "step-scan-users": "Checking board member avatars", + "step-scan-files": "Checking board file attachments", + "step-fix-file-urls": "Fixing file URLs", "cleanup": "Cleanup", "cleanup-old-jobs": "Cleanup Old Jobs", "completed": "Completato/a", diff --git a/imports/i18n/data/ja-HI.i18n.json b/imports/i18n/data/ja-HI.i18n.json index db428e93a..4ca8bbb16 100644 --- a/imports/i18n/data/ja-HI.i18n.json +++ b/imports/i18n/data/ja-HI.i18n.json @@ -1404,7 +1404,70 @@ "back-to-settings": "Back to Settings", "board-id": "Board ID", "board-migration": "Board Migration", + "board-migrations": "Board Migrations", "card-show-lists-on-minicard": "Show Lists on Minicard", + "comprehensive-board-migration": "Comprehensive Board Migration", + "comprehensive-board-migration-description": "Performs comprehensive checks and fixes for board data integrity, including list ordering, card positions, and swimlane structure.", + "delete-duplicate-empty-lists-migration": "Delete Duplicate Empty Lists", + "delete-duplicate-empty-lists-migration-description": "Safely deletes empty duplicate lists. Only removes lists that have no cards AND have another list with the same title that contains cards.", + "lost-cards": "Lost Cards", + "lost-cards-list": "Restored Items", + "restore-lost-cards-migration": "Restore Lost Cards", + "restore-lost-cards-migration-description": "Finds and restores cards and lists with missing swimlaneId or listId. Creates a 'Lost Cards' swimlane to make all lost items visible again.", + "restore-all-archived-migration": "Restore All Archived", + "restore-all-archived-migration-description": "Restores all archived swimlanes, lists, and cards. Automatically fixes any missing swimlaneId or listId to make items visible.", + "fix-missing-lists-migration": "Fix Missing Lists", + "fix-missing-lists-migration-description": "Detects and repairs missing or corrupted lists in the board structure.", + "fix-avatar-urls-migration": "Fix Avatar URLs", + "fix-avatar-urls-migration-description": "Updates avatar URLs for board members to use the correct storage backend and fixes broken avatar references.", + "fix-all-file-urls-migration": "Fix All File URLs", + "fix-all-file-urls-migration-description": "Updates all file attachment URLs on this board to use the correct storage backend and fixes broken file references.", + "migration-needed": "Migration Needed", + "migration-complete": "Complete", + "migration-running": "Running...", + "migration-successful": "Migration completed successfully", + "migration-failed": "Migration failed", + "migrations": "Migrations", + "migrations-admin-only": "Only board administrators can run migrations", + "migrations-description": "Run data integrity checks and repairs for this board. Each migration can be executed individually.", + "no-issues-found": "No issues found", + "run-migration": "Run Migration", + "run-comprehensive-migration-confirm": "This will perform a comprehensive migration to check and fix board data integrity. This may take a few moments. Continue?", + "run-delete-duplicate-empty-lists-migration-confirm": "This will first convert any shared lists to per-swimlane lists, then delete empty lists that have a duplicate list with the same title containing cards. Only truly redundant empty lists will be removed. Continue?", + "run-restore-lost-cards-migration-confirm": "This will create a 'Lost Cards' swimlane and restore all cards and lists with missing swimlaneId or listId. This only affects non-archived items. Continue?", + "run-restore-all-archived-migration-confirm": "This will restore ALL archived swimlanes, lists, and cards, making them visible again. Any items with missing IDs will be automatically fixed. This cannot be easily undone. Continue?", + "run-fix-missing-lists-migration-confirm": "This will detect and repair missing or corrupted lists in the board structure. Continue?", + "run-fix-avatar-urls-migration-confirm": "This will update avatar URLs for board members to use the correct storage backend. Continue?", + "run-fix-all-file-urls-migration-confirm": "This will update all file attachment URLs on this board to use the correct storage backend. Continue?", + "restore-lost-cards-nothing-to-restore": "No lost swimlanes, lists, or cards to restore", + + "migration-progress-title": "Board Migration in Progress", + "migration-progress-overall": "Overall Progress", + "migration-progress-current-step": "Current Step", + "migration-progress-status": "Status", + "migration-progress-details": "Details", + "migration-progress-note": "Please wait while we migrate your board to the latest structure...", + + "step-analyze-board-structure": "Analyze Board Structure", + "step-fix-orphaned-cards": "Fix Orphaned Cards", + "step-convert-shared-lists": "Convert Shared Lists", + "step-ensure-per-swimlane-lists": "Ensure Per-Swimlane Lists", + "step-validate-migration": "Validate Migration", + "step-fix-avatar-urls": "Fix Avatar URLs", + "step-fix-attachment-urls": "Fix Attachment URLs", + "step-analyze-lists": "Analyze Lists", + "step-create-missing-lists": "Create Missing Lists", + "step-update-cards": "Update Cards", + "step-finalize": "Finalize", + "step-delete-duplicate-empty-lists": "Delete Duplicate Empty Lists", + "step-ensure-lost-cards-swimlane": "Ensure Lost Cards Swimlane", + "step-restore-lists": "Restore Lists", + "step-restore-cards": "Restore Cards", + "step-restore-swimlanes": "Restore Swimlanes", + "step-fix-missing-ids": "Fix Missing IDs", + "step-scan-users": "Checking board member avatars", + "step-scan-files": "Checking board file attachments", + "step-fix-file-urls": "Fixing file URLs", "cleanup": "Cleanup", "cleanup-old-jobs": "Cleanup Old Jobs", "completed": "Completed", diff --git a/imports/i18n/data/ja.i18n.json b/imports/i18n/data/ja.i18n.json index 3f6a29f13..a6ec7dedb 100644 --- a/imports/i18n/data/ja.i18n.json +++ b/imports/i18n/data/ja.i18n.json @@ -1404,7 +1404,70 @@ "back-to-settings": "Back to Settings", "board-id": "Board ID", "board-migration": "Board Migration", + "board-migrations": "Board Migrations", "card-show-lists-on-minicard": "Show Lists on Minicard", + "comprehensive-board-migration": "Comprehensive Board Migration", + "comprehensive-board-migration-description": "Performs comprehensive checks and fixes for board data integrity, including list ordering, card positions, and swimlane structure.", + "delete-duplicate-empty-lists-migration": "Delete Duplicate Empty Lists", + "delete-duplicate-empty-lists-migration-description": "Safely deletes empty duplicate lists. Only removes lists that have no cards AND have another list with the same title that contains cards.", + "lost-cards": "Lost Cards", + "lost-cards-list": "Restored Items", + "restore-lost-cards-migration": "Restore Lost Cards", + "restore-lost-cards-migration-description": "Finds and restores cards and lists with missing swimlaneId or listId. Creates a 'Lost Cards' swimlane to make all lost items visible again.", + "restore-all-archived-migration": "Restore All Archived", + "restore-all-archived-migration-description": "Restores all archived swimlanes, lists, and cards. Automatically fixes any missing swimlaneId or listId to make items visible.", + "fix-missing-lists-migration": "Fix Missing Lists", + "fix-missing-lists-migration-description": "Detects and repairs missing or corrupted lists in the board structure.", + "fix-avatar-urls-migration": "Fix Avatar URLs", + "fix-avatar-urls-migration-description": "Updates avatar URLs for board members to use the correct storage backend and fixes broken avatar references.", + "fix-all-file-urls-migration": "Fix All File URLs", + "fix-all-file-urls-migration-description": "Updates all file attachment URLs on this board to use the correct storage backend and fixes broken file references.", + "migration-needed": "Migration Needed", + "migration-complete": "完了", + "migration-running": "Running...", + "migration-successful": "Migration completed successfully", + "migration-failed": "Migration failed", + "migrations": "Migrations", + "migrations-admin-only": "Only board administrators can run migrations", + "migrations-description": "Run data integrity checks and repairs for this board. Each migration can be executed individually.", + "no-issues-found": "No issues found", + "run-migration": "Run Migration", + "run-comprehensive-migration-confirm": "This will perform a comprehensive migration to check and fix board data integrity. This may take a few moments. Continue?", + "run-delete-duplicate-empty-lists-migration-confirm": "This will first convert any shared lists to per-swimlane lists, then delete empty lists that have a duplicate list with the same title containing cards. Only truly redundant empty lists will be removed. Continue?", + "run-restore-lost-cards-migration-confirm": "This will create a 'Lost Cards' swimlane and restore all cards and lists with missing swimlaneId or listId. This only affects non-archived items. Continue?", + "run-restore-all-archived-migration-confirm": "This will restore ALL archived swimlanes, lists, and cards, making them visible again. Any items with missing IDs will be automatically fixed. This cannot be easily undone. Continue?", + "run-fix-missing-lists-migration-confirm": "This will detect and repair missing or corrupted lists in the board structure. Continue?", + "run-fix-avatar-urls-migration-confirm": "This will update avatar URLs for board members to use the correct storage backend. Continue?", + "run-fix-all-file-urls-migration-confirm": "This will update all file attachment URLs on this board to use the correct storage backend. Continue?", + "restore-lost-cards-nothing-to-restore": "No lost swimlanes, lists, or cards to restore", + + "migration-progress-title": "Board Migration in Progress", + "migration-progress-overall": "Overall Progress", + "migration-progress-current-step": "Current Step", + "migration-progress-status": "ステータス", + "migration-progress-details": "詳細", + "migration-progress-note": "Please wait while we migrate your board to the latest structure...", + + "step-analyze-board-structure": "Analyze Board Structure", + "step-fix-orphaned-cards": "Fix Orphaned Cards", + "step-convert-shared-lists": "Convert Shared Lists", + "step-ensure-per-swimlane-lists": "Ensure Per-Swimlane Lists", + "step-validate-migration": "Validate Migration", + "step-fix-avatar-urls": "Fix Avatar URLs", + "step-fix-attachment-urls": "Fix Attachment URLs", + "step-analyze-lists": "Analyze Lists", + "step-create-missing-lists": "Create Missing Lists", + "step-update-cards": "Update Cards", + "step-finalize": "Finalize", + "step-delete-duplicate-empty-lists": "Delete Duplicate Empty Lists", + "step-ensure-lost-cards-swimlane": "Ensure Lost Cards Swimlane", + "step-restore-lists": "Restore Lists", + "step-restore-cards": "Restore Cards", + "step-restore-swimlanes": "Restore Swimlanes", + "step-fix-missing-ids": "Fix Missing IDs", + "step-scan-users": "Checking board member avatars", + "step-scan-files": "Checking board file attachments", + "step-fix-file-urls": "Fixing file URLs", "cleanup": "Cleanup", "cleanup-old-jobs": "Cleanup Old Jobs", "completed": "完了した時", diff --git a/imports/i18n/data/ka.i18n.json b/imports/i18n/data/ka.i18n.json index 0e41f2671..ea24aa1ef 100644 --- a/imports/i18n/data/ka.i18n.json +++ b/imports/i18n/data/ka.i18n.json @@ -1404,7 +1404,70 @@ "back-to-settings": "Back to Settings", "board-id": "Board ID", "board-migration": "Board Migration", + "board-migrations": "Board Migrations", "card-show-lists-on-minicard": "Show Lists on Minicard", + "comprehensive-board-migration": "Comprehensive Board Migration", + "comprehensive-board-migration-description": "Performs comprehensive checks and fixes for board data integrity, including list ordering, card positions, and swimlane structure.", + "delete-duplicate-empty-lists-migration": "Delete Duplicate Empty Lists", + "delete-duplicate-empty-lists-migration-description": "Safely deletes empty duplicate lists. Only removes lists that have no cards AND have another list with the same title that contains cards.", + "lost-cards": "Lost Cards", + "lost-cards-list": "Restored Items", + "restore-lost-cards-migration": "Restore Lost Cards", + "restore-lost-cards-migration-description": "Finds and restores cards and lists with missing swimlaneId or listId. Creates a 'Lost Cards' swimlane to make all lost items visible again.", + "restore-all-archived-migration": "Restore All Archived", + "restore-all-archived-migration-description": "Restores all archived swimlanes, lists, and cards. Automatically fixes any missing swimlaneId or listId to make items visible.", + "fix-missing-lists-migration": "Fix Missing Lists", + "fix-missing-lists-migration-description": "Detects and repairs missing or corrupted lists in the board structure.", + "fix-avatar-urls-migration": "Fix Avatar URLs", + "fix-avatar-urls-migration-description": "Updates avatar URLs for board members to use the correct storage backend and fixes broken avatar references.", + "fix-all-file-urls-migration": "Fix All File URLs", + "fix-all-file-urls-migration-description": "Updates all file attachment URLs on this board to use the correct storage backend and fixes broken file references.", + "migration-needed": "Migration Needed", + "migration-complete": "Complete", + "migration-running": "Running...", + "migration-successful": "Migration completed successfully", + "migration-failed": "Migration failed", + "migrations": "Migrations", + "migrations-admin-only": "Only board administrators can run migrations", + "migrations-description": "Run data integrity checks and repairs for this board. Each migration can be executed individually.", + "no-issues-found": "No issues found", + "run-migration": "Run Migration", + "run-comprehensive-migration-confirm": "This will perform a comprehensive migration to check and fix board data integrity. This may take a few moments. Continue?", + "run-delete-duplicate-empty-lists-migration-confirm": "This will first convert any shared lists to per-swimlane lists, then delete empty lists that have a duplicate list with the same title containing cards. Only truly redundant empty lists will be removed. Continue?", + "run-restore-lost-cards-migration-confirm": "This will create a 'Lost Cards' swimlane and restore all cards and lists with missing swimlaneId or listId. This only affects non-archived items. Continue?", + "run-restore-all-archived-migration-confirm": "This will restore ALL archived swimlanes, lists, and cards, making them visible again. Any items with missing IDs will be automatically fixed. This cannot be easily undone. Continue?", + "run-fix-missing-lists-migration-confirm": "This will detect and repair missing or corrupted lists in the board structure. Continue?", + "run-fix-avatar-urls-migration-confirm": "This will update avatar URLs for board members to use the correct storage backend. Continue?", + "run-fix-all-file-urls-migration-confirm": "This will update all file attachment URLs on this board to use the correct storage backend. Continue?", + "restore-lost-cards-nothing-to-restore": "No lost swimlanes, lists, or cards to restore", + + "migration-progress-title": "Board Migration in Progress", + "migration-progress-overall": "Overall Progress", + "migration-progress-current-step": "Current Step", + "migration-progress-status": "Status", + "migration-progress-details": "Details", + "migration-progress-note": "Please wait while we migrate your board to the latest structure...", + + "step-analyze-board-structure": "Analyze Board Structure", + "step-fix-orphaned-cards": "Fix Orphaned Cards", + "step-convert-shared-lists": "Convert Shared Lists", + "step-ensure-per-swimlane-lists": "Ensure Per-Swimlane Lists", + "step-validate-migration": "Validate Migration", + "step-fix-avatar-urls": "Fix Avatar URLs", + "step-fix-attachment-urls": "Fix Attachment URLs", + "step-analyze-lists": "Analyze Lists", + "step-create-missing-lists": "Create Missing Lists", + "step-update-cards": "Update Cards", + "step-finalize": "Finalize", + "step-delete-duplicate-empty-lists": "Delete Duplicate Empty Lists", + "step-ensure-lost-cards-swimlane": "Ensure Lost Cards Swimlane", + "step-restore-lists": "Restore Lists", + "step-restore-cards": "Restore Cards", + "step-restore-swimlanes": "Restore Swimlanes", + "step-fix-missing-ids": "Fix Missing IDs", + "step-scan-users": "Checking board member avatars", + "step-scan-files": "Checking board file attachments", + "step-fix-file-urls": "Fixing file URLs", "cleanup": "Cleanup", "cleanup-old-jobs": "Cleanup Old Jobs", "completed": "Completed", diff --git a/imports/i18n/data/km.i18n.json b/imports/i18n/data/km.i18n.json index 99d757f8b..ec5b3989c 100644 --- a/imports/i18n/data/km.i18n.json +++ b/imports/i18n/data/km.i18n.json @@ -1404,7 +1404,70 @@ "back-to-settings": "Back to Settings", "board-id": "Board ID", "board-migration": "Board Migration", + "board-migrations": "Board Migrations", "card-show-lists-on-minicard": "Show Lists on Minicard", + "comprehensive-board-migration": "Comprehensive Board Migration", + "comprehensive-board-migration-description": "Performs comprehensive checks and fixes for board data integrity, including list ordering, card positions, and swimlane structure.", + "delete-duplicate-empty-lists-migration": "Delete Duplicate Empty Lists", + "delete-duplicate-empty-lists-migration-description": "Safely deletes empty duplicate lists. Only removes lists that have no cards AND have another list with the same title that contains cards.", + "lost-cards": "Lost Cards", + "lost-cards-list": "Restored Items", + "restore-lost-cards-migration": "Restore Lost Cards", + "restore-lost-cards-migration-description": "Finds and restores cards and lists with missing swimlaneId or listId. Creates a 'Lost Cards' swimlane to make all lost items visible again.", + "restore-all-archived-migration": "Restore All Archived", + "restore-all-archived-migration-description": "Restores all archived swimlanes, lists, and cards. Automatically fixes any missing swimlaneId or listId to make items visible.", + "fix-missing-lists-migration": "Fix Missing Lists", + "fix-missing-lists-migration-description": "Detects and repairs missing or corrupted lists in the board structure.", + "fix-avatar-urls-migration": "Fix Avatar URLs", + "fix-avatar-urls-migration-description": "Updates avatar URLs for board members to use the correct storage backend and fixes broken avatar references.", + "fix-all-file-urls-migration": "Fix All File URLs", + "fix-all-file-urls-migration-description": "Updates all file attachment URLs on this board to use the correct storage backend and fixes broken file references.", + "migration-needed": "Migration Needed", + "migration-complete": "Complete", + "migration-running": "Running...", + "migration-successful": "Migration completed successfully", + "migration-failed": "Migration failed", + "migrations": "Migrations", + "migrations-admin-only": "Only board administrators can run migrations", + "migrations-description": "Run data integrity checks and repairs for this board. Each migration can be executed individually.", + "no-issues-found": "No issues found", + "run-migration": "Run Migration", + "run-comprehensive-migration-confirm": "This will perform a comprehensive migration to check and fix board data integrity. This may take a few moments. Continue?", + "run-delete-duplicate-empty-lists-migration-confirm": "This will first convert any shared lists to per-swimlane lists, then delete empty lists that have a duplicate list with the same title containing cards. Only truly redundant empty lists will be removed. Continue?", + "run-restore-lost-cards-migration-confirm": "This will create a 'Lost Cards' swimlane and restore all cards and lists with missing swimlaneId or listId. This only affects non-archived items. Continue?", + "run-restore-all-archived-migration-confirm": "This will restore ALL archived swimlanes, lists, and cards, making them visible again. Any items with missing IDs will be automatically fixed. This cannot be easily undone. Continue?", + "run-fix-missing-lists-migration-confirm": "This will detect and repair missing or corrupted lists in the board structure. Continue?", + "run-fix-avatar-urls-migration-confirm": "This will update avatar URLs for board members to use the correct storage backend. Continue?", + "run-fix-all-file-urls-migration-confirm": "This will update all file attachment URLs on this board to use the correct storage backend. Continue?", + "restore-lost-cards-nothing-to-restore": "No lost swimlanes, lists, or cards to restore", + + "migration-progress-title": "Board Migration in Progress", + "migration-progress-overall": "Overall Progress", + "migration-progress-current-step": "Current Step", + "migration-progress-status": "Status", + "migration-progress-details": "Details", + "migration-progress-note": "Please wait while we migrate your board to the latest structure...", + + "step-analyze-board-structure": "Analyze Board Structure", + "step-fix-orphaned-cards": "Fix Orphaned Cards", + "step-convert-shared-lists": "Convert Shared Lists", + "step-ensure-per-swimlane-lists": "Ensure Per-Swimlane Lists", + "step-validate-migration": "Validate Migration", + "step-fix-avatar-urls": "Fix Avatar URLs", + "step-fix-attachment-urls": "Fix Attachment URLs", + "step-analyze-lists": "Analyze Lists", + "step-create-missing-lists": "Create Missing Lists", + "step-update-cards": "Update Cards", + "step-finalize": "Finalize", + "step-delete-duplicate-empty-lists": "Delete Duplicate Empty Lists", + "step-ensure-lost-cards-swimlane": "Ensure Lost Cards Swimlane", + "step-restore-lists": "Restore Lists", + "step-restore-cards": "Restore Cards", + "step-restore-swimlanes": "Restore Swimlanes", + "step-fix-missing-ids": "Fix Missing IDs", + "step-scan-users": "Checking board member avatars", + "step-scan-files": "Checking board file attachments", + "step-fix-file-urls": "Fixing file URLs", "cleanup": "Cleanup", "cleanup-old-jobs": "Cleanup Old Jobs", "completed": "Completed", diff --git a/imports/i18n/data/ko-KR.i18n.json b/imports/i18n/data/ko-KR.i18n.json index 284d8e20b..e17b1dc3b 100644 --- a/imports/i18n/data/ko-KR.i18n.json +++ b/imports/i18n/data/ko-KR.i18n.json @@ -1404,7 +1404,70 @@ "back-to-settings": "Back to Settings", "board-id": "Board ID", "board-migration": "Board Migration", + "board-migrations": "Board Migrations", "card-show-lists-on-minicard": "Show Lists on Minicard", + "comprehensive-board-migration": "Comprehensive Board Migration", + "comprehensive-board-migration-description": "Performs comprehensive checks and fixes for board data integrity, including list ordering, card positions, and swimlane structure.", + "delete-duplicate-empty-lists-migration": "Delete Duplicate Empty Lists", + "delete-duplicate-empty-lists-migration-description": "Safely deletes empty duplicate lists. Only removes lists that have no cards AND have another list with the same title that contains cards.", + "lost-cards": "Lost Cards", + "lost-cards-list": "Restored Items", + "restore-lost-cards-migration": "Restore Lost Cards", + "restore-lost-cards-migration-description": "Finds and restores cards and lists with missing swimlaneId or listId. Creates a 'Lost Cards' swimlane to make all lost items visible again.", + "restore-all-archived-migration": "Restore All Archived", + "restore-all-archived-migration-description": "Restores all archived swimlanes, lists, and cards. Automatically fixes any missing swimlaneId or listId to make items visible.", + "fix-missing-lists-migration": "Fix Missing Lists", + "fix-missing-lists-migration-description": "Detects and repairs missing or corrupted lists in the board structure.", + "fix-avatar-urls-migration": "Fix Avatar URLs", + "fix-avatar-urls-migration-description": "Updates avatar URLs for board members to use the correct storage backend and fixes broken avatar references.", + "fix-all-file-urls-migration": "Fix All File URLs", + "fix-all-file-urls-migration-description": "Updates all file attachment URLs on this board to use the correct storage backend and fixes broken file references.", + "migration-needed": "Migration Needed", + "migration-complete": "Complete", + "migration-running": "Running...", + "migration-successful": "Migration completed successfully", + "migration-failed": "Migration failed", + "migrations": "Migrations", + "migrations-admin-only": "Only board administrators can run migrations", + "migrations-description": "Run data integrity checks and repairs for this board. Each migration can be executed individually.", + "no-issues-found": "No issues found", + "run-migration": "Run Migration", + "run-comprehensive-migration-confirm": "This will perform a comprehensive migration to check and fix board data integrity. This may take a few moments. Continue?", + "run-delete-duplicate-empty-lists-migration-confirm": "This will first convert any shared lists to per-swimlane lists, then delete empty lists that have a duplicate list with the same title containing cards. Only truly redundant empty lists will be removed. Continue?", + "run-restore-lost-cards-migration-confirm": "This will create a 'Lost Cards' swimlane and restore all cards and lists with missing swimlaneId or listId. This only affects non-archived items. Continue?", + "run-restore-all-archived-migration-confirm": "This will restore ALL archived swimlanes, lists, and cards, making them visible again. Any items with missing IDs will be automatically fixed. This cannot be easily undone. Continue?", + "run-fix-missing-lists-migration-confirm": "This will detect and repair missing or corrupted lists in the board structure. Continue?", + "run-fix-avatar-urls-migration-confirm": "This will update avatar URLs for board members to use the correct storage backend. Continue?", + "run-fix-all-file-urls-migration-confirm": "This will update all file attachment URLs on this board to use the correct storage backend. Continue?", + "restore-lost-cards-nothing-to-restore": "No lost swimlanes, lists, or cards to restore", + + "migration-progress-title": "Board Migration in Progress", + "migration-progress-overall": "Overall Progress", + "migration-progress-current-step": "Current Step", + "migration-progress-status": "Status", + "migration-progress-details": "Details", + "migration-progress-note": "Please wait while we migrate your board to the latest structure...", + + "step-analyze-board-structure": "Analyze Board Structure", + "step-fix-orphaned-cards": "Fix Orphaned Cards", + "step-convert-shared-lists": "Convert Shared Lists", + "step-ensure-per-swimlane-lists": "Ensure Per-Swimlane Lists", + "step-validate-migration": "Validate Migration", + "step-fix-avatar-urls": "Fix Avatar URLs", + "step-fix-attachment-urls": "Fix Attachment URLs", + "step-analyze-lists": "Analyze Lists", + "step-create-missing-lists": "Create Missing Lists", + "step-update-cards": "Update Cards", + "step-finalize": "Finalize", + "step-delete-duplicate-empty-lists": "Delete Duplicate Empty Lists", + "step-ensure-lost-cards-swimlane": "Ensure Lost Cards Swimlane", + "step-restore-lists": "Restore Lists", + "step-restore-cards": "Restore Cards", + "step-restore-swimlanes": "Restore Swimlanes", + "step-fix-missing-ids": "Fix Missing IDs", + "step-scan-users": "Checking board member avatars", + "step-scan-files": "Checking board file attachments", + "step-fix-file-urls": "Fixing file URLs", "cleanup": "Cleanup", "cleanup-old-jobs": "Cleanup Old Jobs", "completed": "Completed", diff --git a/imports/i18n/data/ko.i18n.json b/imports/i18n/data/ko.i18n.json index 1796716ba..501cd1d47 100644 --- a/imports/i18n/data/ko.i18n.json +++ b/imports/i18n/data/ko.i18n.json @@ -1404,7 +1404,70 @@ "back-to-settings": "Back to Settings", "board-id": "Board ID", "board-migration": "Board Migration", + "board-migrations": "Board Migrations", "card-show-lists-on-minicard": "Show Lists on Minicard", + "comprehensive-board-migration": "Comprehensive Board Migration", + "comprehensive-board-migration-description": "Performs comprehensive checks and fixes for board data integrity, including list ordering, card positions, and swimlane structure.", + "delete-duplicate-empty-lists-migration": "Delete Duplicate Empty Lists", + "delete-duplicate-empty-lists-migration-description": "Safely deletes empty duplicate lists. Only removes lists that have no cards AND have another list with the same title that contains cards.", + "lost-cards": "Lost Cards", + "lost-cards-list": "Restored Items", + "restore-lost-cards-migration": "Restore Lost Cards", + "restore-lost-cards-migration-description": "Finds and restores cards and lists with missing swimlaneId or listId. Creates a 'Lost Cards' swimlane to make all lost items visible again.", + "restore-all-archived-migration": "Restore All Archived", + "restore-all-archived-migration-description": "Restores all archived swimlanes, lists, and cards. Automatically fixes any missing swimlaneId or listId to make items visible.", + "fix-missing-lists-migration": "Fix Missing Lists", + "fix-missing-lists-migration-description": "Detects and repairs missing or corrupted lists in the board structure.", + "fix-avatar-urls-migration": "Fix Avatar URLs", + "fix-avatar-urls-migration-description": "Updates avatar URLs for board members to use the correct storage backend and fixes broken avatar references.", + "fix-all-file-urls-migration": "Fix All File URLs", + "fix-all-file-urls-migration-description": "Updates all file attachment URLs on this board to use the correct storage backend and fixes broken file references.", + "migration-needed": "Migration Needed", + "migration-complete": "완료", + "migration-running": "Running...", + "migration-successful": "Migration completed successfully", + "migration-failed": "Migration failed", + "migrations": "Migrations", + "migrations-admin-only": "Only board administrators can run migrations", + "migrations-description": "Run data integrity checks and repairs for this board. Each migration can be executed individually.", + "no-issues-found": "No issues found", + "run-migration": "Run Migration", + "run-comprehensive-migration-confirm": "This will perform a comprehensive migration to check and fix board data integrity. This may take a few moments. Continue?", + "run-delete-duplicate-empty-lists-migration-confirm": "This will first convert any shared lists to per-swimlane lists, then delete empty lists that have a duplicate list with the same title containing cards. Only truly redundant empty lists will be removed. Continue?", + "run-restore-lost-cards-migration-confirm": "This will create a 'Lost Cards' swimlane and restore all cards and lists with missing swimlaneId or listId. This only affects non-archived items. Continue?", + "run-restore-all-archived-migration-confirm": "This will restore ALL archived swimlanes, lists, and cards, making them visible again. Any items with missing IDs will be automatically fixed. This cannot be easily undone. Continue?", + "run-fix-missing-lists-migration-confirm": "This will detect and repair missing or corrupted lists in the board structure. Continue?", + "run-fix-avatar-urls-migration-confirm": "This will update avatar URLs for board members to use the correct storage backend. Continue?", + "run-fix-all-file-urls-migration-confirm": "This will update all file attachment URLs on this board to use the correct storage backend. Continue?", + "restore-lost-cards-nothing-to-restore": "No lost swimlanes, lists, or cards to restore", + + "migration-progress-title": "Board Migration in Progress", + "migration-progress-overall": "Overall Progress", + "migration-progress-current-step": "Current Step", + "migration-progress-status": "Status", + "migration-progress-details": "상세", + "migration-progress-note": "Please wait while we migrate your board to the latest structure...", + + "step-analyze-board-structure": "Analyze Board Structure", + "step-fix-orphaned-cards": "Fix Orphaned Cards", + "step-convert-shared-lists": "Convert Shared Lists", + "step-ensure-per-swimlane-lists": "Ensure Per-Swimlane Lists", + "step-validate-migration": "Validate Migration", + "step-fix-avatar-urls": "Fix Avatar URLs", + "step-fix-attachment-urls": "Fix Attachment URLs", + "step-analyze-lists": "Analyze Lists", + "step-create-missing-lists": "Create Missing Lists", + "step-update-cards": "Update Cards", + "step-finalize": "Finalize", + "step-delete-duplicate-empty-lists": "Delete Duplicate Empty Lists", + "step-ensure-lost-cards-swimlane": "Ensure Lost Cards Swimlane", + "step-restore-lists": "Restore Lists", + "step-restore-cards": "Restore Cards", + "step-restore-swimlanes": "Restore Swimlanes", + "step-fix-missing-ids": "Fix Missing IDs", + "step-scan-users": "Checking board member avatars", + "step-scan-files": "Checking board file attachments", + "step-fix-file-urls": "Fixing file URLs", "cleanup": "Cleanup", "cleanup-old-jobs": "Cleanup Old Jobs", "completed": "완료", diff --git a/imports/i18n/data/lt.i18n.json b/imports/i18n/data/lt.i18n.json index 1ab8eeaa8..4804e6dff 100644 --- a/imports/i18n/data/lt.i18n.json +++ b/imports/i18n/data/lt.i18n.json @@ -1404,7 +1404,70 @@ "back-to-settings": "Back to Settings", "board-id": "Board ID", "board-migration": "Board Migration", + "board-migrations": "Board Migrations", "card-show-lists-on-minicard": "Show Lists on Minicard", + "comprehensive-board-migration": "Comprehensive Board Migration", + "comprehensive-board-migration-description": "Performs comprehensive checks and fixes for board data integrity, including list ordering, card positions, and swimlane structure.", + "delete-duplicate-empty-lists-migration": "Delete Duplicate Empty Lists", + "delete-duplicate-empty-lists-migration-description": "Safely deletes empty duplicate lists. Only removes lists that have no cards AND have another list with the same title that contains cards.", + "lost-cards": "Lost Cards", + "lost-cards-list": "Restored Items", + "restore-lost-cards-migration": "Restore Lost Cards", + "restore-lost-cards-migration-description": "Finds and restores cards and lists with missing swimlaneId or listId. Creates a 'Lost Cards' swimlane to make all lost items visible again.", + "restore-all-archived-migration": "Restore All Archived", + "restore-all-archived-migration-description": "Restores all archived swimlanes, lists, and cards. Automatically fixes any missing swimlaneId or listId to make items visible.", + "fix-missing-lists-migration": "Fix Missing Lists", + "fix-missing-lists-migration-description": "Detects and repairs missing or corrupted lists in the board structure.", + "fix-avatar-urls-migration": "Fix Avatar URLs", + "fix-avatar-urls-migration-description": "Updates avatar URLs for board members to use the correct storage backend and fixes broken avatar references.", + "fix-all-file-urls-migration": "Fix All File URLs", + "fix-all-file-urls-migration-description": "Updates all file attachment URLs on this board to use the correct storage backend and fixes broken file references.", + "migration-needed": "Migration Needed", + "migration-complete": "Complete", + "migration-running": "Running...", + "migration-successful": "Migration completed successfully", + "migration-failed": "Migration failed", + "migrations": "Migrations", + "migrations-admin-only": "Only board administrators can run migrations", + "migrations-description": "Run data integrity checks and repairs for this board. Each migration can be executed individually.", + "no-issues-found": "No issues found", + "run-migration": "Run Migration", + "run-comprehensive-migration-confirm": "This will perform a comprehensive migration to check and fix board data integrity. This may take a few moments. Continue?", + "run-delete-duplicate-empty-lists-migration-confirm": "This will first convert any shared lists to per-swimlane lists, then delete empty lists that have a duplicate list with the same title containing cards. Only truly redundant empty lists will be removed. Continue?", + "run-restore-lost-cards-migration-confirm": "This will create a 'Lost Cards' swimlane and restore all cards and lists with missing swimlaneId or listId. This only affects non-archived items. Continue?", + "run-restore-all-archived-migration-confirm": "This will restore ALL archived swimlanes, lists, and cards, making them visible again. Any items with missing IDs will be automatically fixed. This cannot be easily undone. Continue?", + "run-fix-missing-lists-migration-confirm": "This will detect and repair missing or corrupted lists in the board structure. Continue?", + "run-fix-avatar-urls-migration-confirm": "This will update avatar URLs for board members to use the correct storage backend. Continue?", + "run-fix-all-file-urls-migration-confirm": "This will update all file attachment URLs on this board to use the correct storage backend. Continue?", + "restore-lost-cards-nothing-to-restore": "No lost swimlanes, lists, or cards to restore", + + "migration-progress-title": "Board Migration in Progress", + "migration-progress-overall": "Overall Progress", + "migration-progress-current-step": "Current Step", + "migration-progress-status": "Status", + "migration-progress-details": "Details", + "migration-progress-note": "Please wait while we migrate your board to the latest structure...", + + "step-analyze-board-structure": "Analyze Board Structure", + "step-fix-orphaned-cards": "Fix Orphaned Cards", + "step-convert-shared-lists": "Convert Shared Lists", + "step-ensure-per-swimlane-lists": "Ensure Per-Swimlane Lists", + "step-validate-migration": "Validate Migration", + "step-fix-avatar-urls": "Fix Avatar URLs", + "step-fix-attachment-urls": "Fix Attachment URLs", + "step-analyze-lists": "Analyze Lists", + "step-create-missing-lists": "Create Missing Lists", + "step-update-cards": "Update Cards", + "step-finalize": "Finalize", + "step-delete-duplicate-empty-lists": "Delete Duplicate Empty Lists", + "step-ensure-lost-cards-swimlane": "Ensure Lost Cards Swimlane", + "step-restore-lists": "Restore Lists", + "step-restore-cards": "Restore Cards", + "step-restore-swimlanes": "Restore Swimlanes", + "step-fix-missing-ids": "Fix Missing IDs", + "step-scan-users": "Checking board member avatars", + "step-scan-files": "Checking board file attachments", + "step-fix-file-urls": "Fixing file URLs", "cleanup": "Cleanup", "cleanup-old-jobs": "Cleanup Old Jobs", "completed": "Completed", diff --git a/imports/i18n/data/lv.i18n.json b/imports/i18n/data/lv.i18n.json index d051fb8a5..51e6984c0 100644 --- a/imports/i18n/data/lv.i18n.json +++ b/imports/i18n/data/lv.i18n.json @@ -1404,7 +1404,70 @@ "back-to-settings": "Back to Settings", "board-id": "Board ID", "board-migration": "Board Migration", + "board-migrations": "Board Migrations", "card-show-lists-on-minicard": "Show Lists on Minicard", + "comprehensive-board-migration": "Comprehensive Board Migration", + "comprehensive-board-migration-description": "Performs comprehensive checks and fixes for board data integrity, including list ordering, card positions, and swimlane structure.", + "delete-duplicate-empty-lists-migration": "Delete Duplicate Empty Lists", + "delete-duplicate-empty-lists-migration-description": "Safely deletes empty duplicate lists. Only removes lists that have no cards AND have another list with the same title that contains cards.", + "lost-cards": "Lost Cards", + "lost-cards-list": "Restored Items", + "restore-lost-cards-migration": "Restore Lost Cards", + "restore-lost-cards-migration-description": "Finds and restores cards and lists with missing swimlaneId or listId. Creates a 'Lost Cards' swimlane to make all lost items visible again.", + "restore-all-archived-migration": "Restore All Archived", + "restore-all-archived-migration-description": "Restores all archived swimlanes, lists, and cards. Automatically fixes any missing swimlaneId or listId to make items visible.", + "fix-missing-lists-migration": "Fix Missing Lists", + "fix-missing-lists-migration-description": "Detects and repairs missing or corrupted lists in the board structure.", + "fix-avatar-urls-migration": "Fix Avatar URLs", + "fix-avatar-urls-migration-description": "Updates avatar URLs for board members to use the correct storage backend and fixes broken avatar references.", + "fix-all-file-urls-migration": "Fix All File URLs", + "fix-all-file-urls-migration-description": "Updates all file attachment URLs on this board to use the correct storage backend and fixes broken file references.", + "migration-needed": "Migration Needed", + "migration-complete": "Complete", + "migration-running": "Running...", + "migration-successful": "Migration completed successfully", + "migration-failed": "Migration failed", + "migrations": "Migrations", + "migrations-admin-only": "Only board administrators can run migrations", + "migrations-description": "Run data integrity checks and repairs for this board. Each migration can be executed individually.", + "no-issues-found": "No issues found", + "run-migration": "Run Migration", + "run-comprehensive-migration-confirm": "This will perform a comprehensive migration to check and fix board data integrity. This may take a few moments. Continue?", + "run-delete-duplicate-empty-lists-migration-confirm": "This will first convert any shared lists to per-swimlane lists, then delete empty lists that have a duplicate list with the same title containing cards. Only truly redundant empty lists will be removed. Continue?", + "run-restore-lost-cards-migration-confirm": "This will create a 'Lost Cards' swimlane and restore all cards and lists with missing swimlaneId or listId. This only affects non-archived items. Continue?", + "run-restore-all-archived-migration-confirm": "This will restore ALL archived swimlanes, lists, and cards, making them visible again. Any items with missing IDs will be automatically fixed. This cannot be easily undone. Continue?", + "run-fix-missing-lists-migration-confirm": "This will detect and repair missing or corrupted lists in the board structure. Continue?", + "run-fix-avatar-urls-migration-confirm": "This will update avatar URLs for board members to use the correct storage backend. Continue?", + "run-fix-all-file-urls-migration-confirm": "This will update all file attachment URLs on this board to use the correct storage backend. Continue?", + "restore-lost-cards-nothing-to-restore": "No lost swimlanes, lists, or cards to restore", + + "migration-progress-title": "Board Migration in Progress", + "migration-progress-overall": "Overall Progress", + "migration-progress-current-step": "Current Step", + "migration-progress-status": "Statuss", + "migration-progress-details": "Details", + "migration-progress-note": "Please wait while we migrate your board to the latest structure...", + + "step-analyze-board-structure": "Analyze Board Structure", + "step-fix-orphaned-cards": "Fix Orphaned Cards", + "step-convert-shared-lists": "Convert Shared Lists", + "step-ensure-per-swimlane-lists": "Ensure Per-Swimlane Lists", + "step-validate-migration": "Validate Migration", + "step-fix-avatar-urls": "Fix Avatar URLs", + "step-fix-attachment-urls": "Fix Attachment URLs", + "step-analyze-lists": "Analyze Lists", + "step-create-missing-lists": "Create Missing Lists", + "step-update-cards": "Update Cards", + "step-finalize": "Finalize", + "step-delete-duplicate-empty-lists": "Delete Duplicate Empty Lists", + "step-ensure-lost-cards-swimlane": "Ensure Lost Cards Swimlane", + "step-restore-lists": "Restore Lists", + "step-restore-cards": "Restore Cards", + "step-restore-swimlanes": "Restore Swimlanes", + "step-fix-missing-ids": "Fix Missing IDs", + "step-scan-users": "Checking board member avatars", + "step-scan-files": "Checking board file attachments", + "step-fix-file-urls": "Fixing file URLs", "cleanup": "Cleanup", "cleanup-old-jobs": "Cleanup Old Jobs", "completed": "Pabeigts", diff --git a/imports/i18n/data/mk.i18n.json b/imports/i18n/data/mk.i18n.json index 491feeb6e..6cc517c8b 100644 --- a/imports/i18n/data/mk.i18n.json +++ b/imports/i18n/data/mk.i18n.json @@ -1404,7 +1404,70 @@ "back-to-settings": "Back to Settings", "board-id": "Board ID", "board-migration": "Board Migration", + "board-migrations": "Board Migrations", "card-show-lists-on-minicard": "Show Lists on Minicard", + "comprehensive-board-migration": "Comprehensive Board Migration", + "comprehensive-board-migration-description": "Performs comprehensive checks and fixes for board data integrity, including list ordering, card positions, and swimlane structure.", + "delete-duplicate-empty-lists-migration": "Delete Duplicate Empty Lists", + "delete-duplicate-empty-lists-migration-description": "Safely deletes empty duplicate lists. Only removes lists that have no cards AND have another list with the same title that contains cards.", + "lost-cards": "Lost Cards", + "lost-cards-list": "Restored Items", + "restore-lost-cards-migration": "Restore Lost Cards", + "restore-lost-cards-migration-description": "Finds and restores cards and lists with missing swimlaneId or listId. Creates a 'Lost Cards' swimlane to make all lost items visible again.", + "restore-all-archived-migration": "Restore All Archived", + "restore-all-archived-migration-description": "Restores all archived swimlanes, lists, and cards. Automatically fixes any missing swimlaneId or listId to make items visible.", + "fix-missing-lists-migration": "Fix Missing Lists", + "fix-missing-lists-migration-description": "Detects and repairs missing or corrupted lists in the board structure.", + "fix-avatar-urls-migration": "Fix Avatar URLs", + "fix-avatar-urls-migration-description": "Updates avatar URLs for board members to use the correct storage backend and fixes broken avatar references.", + "fix-all-file-urls-migration": "Fix All File URLs", + "fix-all-file-urls-migration-description": "Updates all file attachment URLs on this board to use the correct storage backend and fixes broken file references.", + "migration-needed": "Migration Needed", + "migration-complete": "Complete", + "migration-running": "Running...", + "migration-successful": "Migration completed successfully", + "migration-failed": "Migration failed", + "migrations": "Migrations", + "migrations-admin-only": "Only board administrators can run migrations", + "migrations-description": "Run data integrity checks and repairs for this board. Each migration can be executed individually.", + "no-issues-found": "No issues found", + "run-migration": "Run Migration", + "run-comprehensive-migration-confirm": "This will perform a comprehensive migration to check and fix board data integrity. This may take a few moments. Continue?", + "run-delete-duplicate-empty-lists-migration-confirm": "This will first convert any shared lists to per-swimlane lists, then delete empty lists that have a duplicate list with the same title containing cards. Only truly redundant empty lists will be removed. Continue?", + "run-restore-lost-cards-migration-confirm": "This will create a 'Lost Cards' swimlane and restore all cards and lists with missing swimlaneId or listId. This only affects non-archived items. Continue?", + "run-restore-all-archived-migration-confirm": "This will restore ALL archived swimlanes, lists, and cards, making them visible again. Any items with missing IDs will be automatically fixed. This cannot be easily undone. Continue?", + "run-fix-missing-lists-migration-confirm": "This will detect and repair missing or corrupted lists in the board structure. Continue?", + "run-fix-avatar-urls-migration-confirm": "This will update avatar URLs for board members to use the correct storage backend. Continue?", + "run-fix-all-file-urls-migration-confirm": "This will update all file attachment URLs on this board to use the correct storage backend. Continue?", + "restore-lost-cards-nothing-to-restore": "No lost swimlanes, lists, or cards to restore", + + "migration-progress-title": "Board Migration in Progress", + "migration-progress-overall": "Overall Progress", + "migration-progress-current-step": "Current Step", + "migration-progress-status": "Status", + "migration-progress-details": "Details", + "migration-progress-note": "Please wait while we migrate your board to the latest structure...", + + "step-analyze-board-structure": "Analyze Board Structure", + "step-fix-orphaned-cards": "Fix Orphaned Cards", + "step-convert-shared-lists": "Convert Shared Lists", + "step-ensure-per-swimlane-lists": "Ensure Per-Swimlane Lists", + "step-validate-migration": "Validate Migration", + "step-fix-avatar-urls": "Fix Avatar URLs", + "step-fix-attachment-urls": "Fix Attachment URLs", + "step-analyze-lists": "Analyze Lists", + "step-create-missing-lists": "Create Missing Lists", + "step-update-cards": "Update Cards", + "step-finalize": "Finalize", + "step-delete-duplicate-empty-lists": "Delete Duplicate Empty Lists", + "step-ensure-lost-cards-swimlane": "Ensure Lost Cards Swimlane", + "step-restore-lists": "Restore Lists", + "step-restore-cards": "Restore Cards", + "step-restore-swimlanes": "Restore Swimlanes", + "step-fix-missing-ids": "Fix Missing IDs", + "step-scan-users": "Checking board member avatars", + "step-scan-files": "Checking board file attachments", + "step-fix-file-urls": "Fixing file URLs", "cleanup": "Cleanup", "cleanup-old-jobs": "Cleanup Old Jobs", "completed": "Completed", diff --git a/imports/i18n/data/mn.i18n.json b/imports/i18n/data/mn.i18n.json index de4627964..ed21faac3 100644 --- a/imports/i18n/data/mn.i18n.json +++ b/imports/i18n/data/mn.i18n.json @@ -1404,7 +1404,70 @@ "back-to-settings": "Back to Settings", "board-id": "Board ID", "board-migration": "Board Migration", + "board-migrations": "Board Migrations", "card-show-lists-on-minicard": "Show Lists on Minicard", + "comprehensive-board-migration": "Comprehensive Board Migration", + "comprehensive-board-migration-description": "Performs comprehensive checks and fixes for board data integrity, including list ordering, card positions, and swimlane structure.", + "delete-duplicate-empty-lists-migration": "Delete Duplicate Empty Lists", + "delete-duplicate-empty-lists-migration-description": "Safely deletes empty duplicate lists. Only removes lists that have no cards AND have another list with the same title that contains cards.", + "lost-cards": "Lost Cards", + "lost-cards-list": "Restored Items", + "restore-lost-cards-migration": "Restore Lost Cards", + "restore-lost-cards-migration-description": "Finds and restores cards and lists with missing swimlaneId or listId. Creates a 'Lost Cards' swimlane to make all lost items visible again.", + "restore-all-archived-migration": "Restore All Archived", + "restore-all-archived-migration-description": "Restores all archived swimlanes, lists, and cards. Automatically fixes any missing swimlaneId or listId to make items visible.", + "fix-missing-lists-migration": "Fix Missing Lists", + "fix-missing-lists-migration-description": "Detects and repairs missing or corrupted lists in the board structure.", + "fix-avatar-urls-migration": "Fix Avatar URLs", + "fix-avatar-urls-migration-description": "Updates avatar URLs for board members to use the correct storage backend and fixes broken avatar references.", + "fix-all-file-urls-migration": "Fix All File URLs", + "fix-all-file-urls-migration-description": "Updates all file attachment URLs on this board to use the correct storage backend and fixes broken file references.", + "migration-needed": "Migration Needed", + "migration-complete": "Complete", + "migration-running": "Running...", + "migration-successful": "Migration completed successfully", + "migration-failed": "Migration failed", + "migrations": "Migrations", + "migrations-admin-only": "Only board administrators can run migrations", + "migrations-description": "Run data integrity checks and repairs for this board. Each migration can be executed individually.", + "no-issues-found": "No issues found", + "run-migration": "Run Migration", + "run-comprehensive-migration-confirm": "This will perform a comprehensive migration to check and fix board data integrity. This may take a few moments. Continue?", + "run-delete-duplicate-empty-lists-migration-confirm": "This will first convert any shared lists to per-swimlane lists, then delete empty lists that have a duplicate list with the same title containing cards. Only truly redundant empty lists will be removed. Continue?", + "run-restore-lost-cards-migration-confirm": "This will create a 'Lost Cards' swimlane and restore all cards and lists with missing swimlaneId or listId. This only affects non-archived items. Continue?", + "run-restore-all-archived-migration-confirm": "This will restore ALL archived swimlanes, lists, and cards, making them visible again. Any items with missing IDs will be automatically fixed. This cannot be easily undone. Continue?", + "run-fix-missing-lists-migration-confirm": "This will detect and repair missing or corrupted lists in the board structure. Continue?", + "run-fix-avatar-urls-migration-confirm": "This will update avatar URLs for board members to use the correct storage backend. Continue?", + "run-fix-all-file-urls-migration-confirm": "This will update all file attachment URLs on this board to use the correct storage backend. Continue?", + "restore-lost-cards-nothing-to-restore": "No lost swimlanes, lists, or cards to restore", + + "migration-progress-title": "Board Migration in Progress", + "migration-progress-overall": "Overall Progress", + "migration-progress-current-step": "Current Step", + "migration-progress-status": "Status", + "migration-progress-details": "Details", + "migration-progress-note": "Please wait while we migrate your board to the latest structure...", + + "step-analyze-board-structure": "Analyze Board Structure", + "step-fix-orphaned-cards": "Fix Orphaned Cards", + "step-convert-shared-lists": "Convert Shared Lists", + "step-ensure-per-swimlane-lists": "Ensure Per-Swimlane Lists", + "step-validate-migration": "Validate Migration", + "step-fix-avatar-urls": "Fix Avatar URLs", + "step-fix-attachment-urls": "Fix Attachment URLs", + "step-analyze-lists": "Analyze Lists", + "step-create-missing-lists": "Create Missing Lists", + "step-update-cards": "Update Cards", + "step-finalize": "Finalize", + "step-delete-duplicate-empty-lists": "Delete Duplicate Empty Lists", + "step-ensure-lost-cards-swimlane": "Ensure Lost Cards Swimlane", + "step-restore-lists": "Restore Lists", + "step-restore-cards": "Restore Cards", + "step-restore-swimlanes": "Restore Swimlanes", + "step-fix-missing-ids": "Fix Missing IDs", + "step-scan-users": "Checking board member avatars", + "step-scan-files": "Checking board file attachments", + "step-fix-file-urls": "Fixing file URLs", "cleanup": "Cleanup", "cleanup-old-jobs": "Cleanup Old Jobs", "completed": "Completed", diff --git a/imports/i18n/data/ms-MY.i18n.json b/imports/i18n/data/ms-MY.i18n.json index e1ea29b8a..c5a4d2586 100644 --- a/imports/i18n/data/ms-MY.i18n.json +++ b/imports/i18n/data/ms-MY.i18n.json @@ -1404,7 +1404,70 @@ "back-to-settings": "Back to Settings", "board-id": "Board ID", "board-migration": "Board Migration", + "board-migrations": "Board Migrations", "card-show-lists-on-minicard": "Show Lists on Minicard", + "comprehensive-board-migration": "Comprehensive Board Migration", + "comprehensive-board-migration-description": "Performs comprehensive checks and fixes for board data integrity, including list ordering, card positions, and swimlane structure.", + "delete-duplicate-empty-lists-migration": "Delete Duplicate Empty Lists", + "delete-duplicate-empty-lists-migration-description": "Safely deletes empty duplicate lists. Only removes lists that have no cards AND have another list with the same title that contains cards.", + "lost-cards": "Lost Cards", + "lost-cards-list": "Restored Items", + "restore-lost-cards-migration": "Restore Lost Cards", + "restore-lost-cards-migration-description": "Finds and restores cards and lists with missing swimlaneId or listId. Creates a 'Lost Cards' swimlane to make all lost items visible again.", + "restore-all-archived-migration": "Restore All Archived", + "restore-all-archived-migration-description": "Restores all archived swimlanes, lists, and cards. Automatically fixes any missing swimlaneId or listId to make items visible.", + "fix-missing-lists-migration": "Fix Missing Lists", + "fix-missing-lists-migration-description": "Detects and repairs missing or corrupted lists in the board structure.", + "fix-avatar-urls-migration": "Fix Avatar URLs", + "fix-avatar-urls-migration-description": "Updates avatar URLs for board members to use the correct storage backend and fixes broken avatar references.", + "fix-all-file-urls-migration": "Fix All File URLs", + "fix-all-file-urls-migration-description": "Updates all file attachment URLs on this board to use the correct storage backend and fixes broken file references.", + "migration-needed": "Migration Needed", + "migration-complete": "Complete", + "migration-running": "Running...", + "migration-successful": "Migration completed successfully", + "migration-failed": "Migration failed", + "migrations": "Migrations", + "migrations-admin-only": "Only board administrators can run migrations", + "migrations-description": "Run data integrity checks and repairs for this board. Each migration can be executed individually.", + "no-issues-found": "No issues found", + "run-migration": "Run Migration", + "run-comprehensive-migration-confirm": "This will perform a comprehensive migration to check and fix board data integrity. This may take a few moments. Continue?", + "run-delete-duplicate-empty-lists-migration-confirm": "This will first convert any shared lists to per-swimlane lists, then delete empty lists that have a duplicate list with the same title containing cards. Only truly redundant empty lists will be removed. Continue?", + "run-restore-lost-cards-migration-confirm": "This will create a 'Lost Cards' swimlane and restore all cards and lists with missing swimlaneId or listId. This only affects non-archived items. Continue?", + "run-restore-all-archived-migration-confirm": "This will restore ALL archived swimlanes, lists, and cards, making them visible again. Any items with missing IDs will be automatically fixed. This cannot be easily undone. Continue?", + "run-fix-missing-lists-migration-confirm": "This will detect and repair missing or corrupted lists in the board structure. Continue?", + "run-fix-avatar-urls-migration-confirm": "This will update avatar URLs for board members to use the correct storage backend. Continue?", + "run-fix-all-file-urls-migration-confirm": "This will update all file attachment URLs on this board to use the correct storage backend. Continue?", + "restore-lost-cards-nothing-to-restore": "No lost swimlanes, lists, or cards to restore", + + "migration-progress-title": "Board Migration in Progress", + "migration-progress-overall": "Overall Progress", + "migration-progress-current-step": "Current Step", + "migration-progress-status": "Status", + "migration-progress-details": "Maklumat", + "migration-progress-note": "Please wait while we migrate your board to the latest structure...", + + "step-analyze-board-structure": "Analyze Board Structure", + "step-fix-orphaned-cards": "Fix Orphaned Cards", + "step-convert-shared-lists": "Convert Shared Lists", + "step-ensure-per-swimlane-lists": "Ensure Per-Swimlane Lists", + "step-validate-migration": "Validate Migration", + "step-fix-avatar-urls": "Fix Avatar URLs", + "step-fix-attachment-urls": "Fix Attachment URLs", + "step-analyze-lists": "Analyze Lists", + "step-create-missing-lists": "Create Missing Lists", + "step-update-cards": "Update Cards", + "step-finalize": "Finalize", + "step-delete-duplicate-empty-lists": "Delete Duplicate Empty Lists", + "step-ensure-lost-cards-swimlane": "Ensure Lost Cards Swimlane", + "step-restore-lists": "Restore Lists", + "step-restore-cards": "Restore Cards", + "step-restore-swimlanes": "Restore Swimlanes", + "step-fix-missing-ids": "Fix Missing IDs", + "step-scan-users": "Checking board member avatars", + "step-scan-files": "Checking board file attachments", + "step-fix-file-urls": "Fixing file URLs", "cleanup": "Cleanup", "cleanup-old-jobs": "Cleanup Old Jobs", "completed": "Completed", diff --git a/imports/i18n/data/ms.i18n.json b/imports/i18n/data/ms.i18n.json index ad26e0f97..fc9e8b938 100644 --- a/imports/i18n/data/ms.i18n.json +++ b/imports/i18n/data/ms.i18n.json @@ -1404,7 +1404,70 @@ "back-to-settings": "Back to Settings", "board-id": "Board ID", "board-migration": "Board Migration", + "board-migrations": "Board Migrations", "card-show-lists-on-minicard": "Show Lists on Minicard", + "comprehensive-board-migration": "Comprehensive Board Migration", + "comprehensive-board-migration-description": "Performs comprehensive checks and fixes for board data integrity, including list ordering, card positions, and swimlane structure.", + "delete-duplicate-empty-lists-migration": "Delete Duplicate Empty Lists", + "delete-duplicate-empty-lists-migration-description": "Safely deletes empty duplicate lists. Only removes lists that have no cards AND have another list with the same title that contains cards.", + "lost-cards": "Lost Cards", + "lost-cards-list": "Restored Items", + "restore-lost-cards-migration": "Restore Lost Cards", + "restore-lost-cards-migration-description": "Finds and restores cards and lists with missing swimlaneId or listId. Creates a 'Lost Cards' swimlane to make all lost items visible again.", + "restore-all-archived-migration": "Restore All Archived", + "restore-all-archived-migration-description": "Restores all archived swimlanes, lists, and cards. Automatically fixes any missing swimlaneId or listId to make items visible.", + "fix-missing-lists-migration": "Fix Missing Lists", + "fix-missing-lists-migration-description": "Detects and repairs missing or corrupted lists in the board structure.", + "fix-avatar-urls-migration": "Fix Avatar URLs", + "fix-avatar-urls-migration-description": "Updates avatar URLs for board members to use the correct storage backend and fixes broken avatar references.", + "fix-all-file-urls-migration": "Fix All File URLs", + "fix-all-file-urls-migration-description": "Updates all file attachment URLs on this board to use the correct storage backend and fixes broken file references.", + "migration-needed": "Migration Needed", + "migration-complete": "Complete", + "migration-running": "Running...", + "migration-successful": "Migration completed successfully", + "migration-failed": "Migration failed", + "migrations": "Migrations", + "migrations-admin-only": "Only board administrators can run migrations", + "migrations-description": "Run data integrity checks and repairs for this board. Each migration can be executed individually.", + "no-issues-found": "No issues found", + "run-migration": "Run Migration", + "run-comprehensive-migration-confirm": "This will perform a comprehensive migration to check and fix board data integrity. This may take a few moments. Continue?", + "run-delete-duplicate-empty-lists-migration-confirm": "This will first convert any shared lists to per-swimlane lists, then delete empty lists that have a duplicate list with the same title containing cards. Only truly redundant empty lists will be removed. Continue?", + "run-restore-lost-cards-migration-confirm": "This will create a 'Lost Cards' swimlane and restore all cards and lists with missing swimlaneId or listId. This only affects non-archived items. Continue?", + "run-restore-all-archived-migration-confirm": "This will restore ALL archived swimlanes, lists, and cards, making them visible again. Any items with missing IDs will be automatically fixed. This cannot be easily undone. Continue?", + "run-fix-missing-lists-migration-confirm": "This will detect and repair missing or corrupted lists in the board structure. Continue?", + "run-fix-avatar-urls-migration-confirm": "This will update avatar URLs for board members to use the correct storage backend. Continue?", + "run-fix-all-file-urls-migration-confirm": "This will update all file attachment URLs on this board to use the correct storage backend. Continue?", + "restore-lost-cards-nothing-to-restore": "No lost swimlanes, lists, or cards to restore", + + "migration-progress-title": "Board Migration in Progress", + "migration-progress-overall": "Overall Progress", + "migration-progress-current-step": "Current Step", + "migration-progress-status": "Status", + "migration-progress-details": "Perincian", + "migration-progress-note": "Please wait while we migrate your board to the latest structure...", + + "step-analyze-board-structure": "Analyze Board Structure", + "step-fix-orphaned-cards": "Fix Orphaned Cards", + "step-convert-shared-lists": "Convert Shared Lists", + "step-ensure-per-swimlane-lists": "Ensure Per-Swimlane Lists", + "step-validate-migration": "Validate Migration", + "step-fix-avatar-urls": "Fix Avatar URLs", + "step-fix-attachment-urls": "Fix Attachment URLs", + "step-analyze-lists": "Analyze Lists", + "step-create-missing-lists": "Create Missing Lists", + "step-update-cards": "Update Cards", + "step-finalize": "Finalize", + "step-delete-duplicate-empty-lists": "Delete Duplicate Empty Lists", + "step-ensure-lost-cards-swimlane": "Ensure Lost Cards Swimlane", + "step-restore-lists": "Restore Lists", + "step-restore-cards": "Restore Cards", + "step-restore-swimlanes": "Restore Swimlanes", + "step-fix-missing-ids": "Fix Missing IDs", + "step-scan-users": "Checking board member avatars", + "step-scan-files": "Checking board file attachments", + "step-fix-file-urls": "Fixing file URLs", "cleanup": "Cleanup", "cleanup-old-jobs": "Cleanup Old Jobs", "completed": "Lengkap", diff --git a/imports/i18n/data/nb.i18n.json b/imports/i18n/data/nb.i18n.json index f3fb0f0ed..998b6973f 100644 --- a/imports/i18n/data/nb.i18n.json +++ b/imports/i18n/data/nb.i18n.json @@ -1404,7 +1404,70 @@ "back-to-settings": "Back to Settings", "board-id": "Board ID", "board-migration": "Board Migration", + "board-migrations": "Board Migrations", "card-show-lists-on-minicard": "Show Lists on Minicard", + "comprehensive-board-migration": "Comprehensive Board Migration", + "comprehensive-board-migration-description": "Performs comprehensive checks and fixes for board data integrity, including list ordering, card positions, and swimlane structure.", + "delete-duplicate-empty-lists-migration": "Delete Duplicate Empty Lists", + "delete-duplicate-empty-lists-migration-description": "Safely deletes empty duplicate lists. Only removes lists that have no cards AND have another list with the same title that contains cards.", + "lost-cards": "Lost Cards", + "lost-cards-list": "Restored Items", + "restore-lost-cards-migration": "Restore Lost Cards", + "restore-lost-cards-migration-description": "Finds and restores cards and lists with missing swimlaneId or listId. Creates a 'Lost Cards' swimlane to make all lost items visible again.", + "restore-all-archived-migration": "Restore All Archived", + "restore-all-archived-migration-description": "Restores all archived swimlanes, lists, and cards. Automatically fixes any missing swimlaneId or listId to make items visible.", + "fix-missing-lists-migration": "Fix Missing Lists", + "fix-missing-lists-migration-description": "Detects and repairs missing or corrupted lists in the board structure.", + "fix-avatar-urls-migration": "Fix Avatar URLs", + "fix-avatar-urls-migration-description": "Updates avatar URLs for board members to use the correct storage backend and fixes broken avatar references.", + "fix-all-file-urls-migration": "Fix All File URLs", + "fix-all-file-urls-migration-description": "Updates all file attachment URLs on this board to use the correct storage backend and fixes broken file references.", + "migration-needed": "Migration Needed", + "migration-complete": "Complete", + "migration-running": "Running...", + "migration-successful": "Migration completed successfully", + "migration-failed": "Migration failed", + "migrations": "Migrations", + "migrations-admin-only": "Only board administrators can run migrations", + "migrations-description": "Run data integrity checks and repairs for this board. Each migration can be executed individually.", + "no-issues-found": "No issues found", + "run-migration": "Run Migration", + "run-comprehensive-migration-confirm": "This will perform a comprehensive migration to check and fix board data integrity. This may take a few moments. Continue?", + "run-delete-duplicate-empty-lists-migration-confirm": "This will first convert any shared lists to per-swimlane lists, then delete empty lists that have a duplicate list with the same title containing cards. Only truly redundant empty lists will be removed. Continue?", + "run-restore-lost-cards-migration-confirm": "This will create a 'Lost Cards' swimlane and restore all cards and lists with missing swimlaneId or listId. This only affects non-archived items. Continue?", + "run-restore-all-archived-migration-confirm": "This will restore ALL archived swimlanes, lists, and cards, making them visible again. Any items with missing IDs will be automatically fixed. This cannot be easily undone. Continue?", + "run-fix-missing-lists-migration-confirm": "This will detect and repair missing or corrupted lists in the board structure. Continue?", + "run-fix-avatar-urls-migration-confirm": "This will update avatar URLs for board members to use the correct storage backend. Continue?", + "run-fix-all-file-urls-migration-confirm": "This will update all file attachment URLs on this board to use the correct storage backend. Continue?", + "restore-lost-cards-nothing-to-restore": "No lost swimlanes, lists, or cards to restore", + + "migration-progress-title": "Board Migration in Progress", + "migration-progress-overall": "Overall Progress", + "migration-progress-current-step": "Current Step", + "migration-progress-status": "Status", + "migration-progress-details": "Detaljer", + "migration-progress-note": "Please wait while we migrate your board to the latest structure...", + + "step-analyze-board-structure": "Analyze Board Structure", + "step-fix-orphaned-cards": "Fix Orphaned Cards", + "step-convert-shared-lists": "Convert Shared Lists", + "step-ensure-per-swimlane-lists": "Ensure Per-Swimlane Lists", + "step-validate-migration": "Validate Migration", + "step-fix-avatar-urls": "Fix Avatar URLs", + "step-fix-attachment-urls": "Fix Attachment URLs", + "step-analyze-lists": "Analyze Lists", + "step-create-missing-lists": "Create Missing Lists", + "step-update-cards": "Update Cards", + "step-finalize": "Finalize", + "step-delete-duplicate-empty-lists": "Delete Duplicate Empty Lists", + "step-ensure-lost-cards-swimlane": "Ensure Lost Cards Swimlane", + "step-restore-lists": "Restore Lists", + "step-restore-cards": "Restore Cards", + "step-restore-swimlanes": "Restore Swimlanes", + "step-fix-missing-ids": "Fix Missing IDs", + "step-scan-users": "Checking board member avatars", + "step-scan-files": "Checking board file attachments", + "step-fix-file-urls": "Fixing file URLs", "cleanup": "Cleanup", "cleanup-old-jobs": "Cleanup Old Jobs", "completed": "Gjennomført", diff --git a/imports/i18n/data/nl-NL.i18n.json b/imports/i18n/data/nl-NL.i18n.json index e1e932403..5ff6008ed 100644 --- a/imports/i18n/data/nl-NL.i18n.json +++ b/imports/i18n/data/nl-NL.i18n.json @@ -1404,7 +1404,70 @@ "back-to-settings": "Back to Settings", "board-id": "Board ID", "board-migration": "Board Migration", + "board-migrations": "Board Migrations", "card-show-lists-on-minicard": "Show Lists on Minicard", + "comprehensive-board-migration": "Comprehensive Board Migration", + "comprehensive-board-migration-description": "Performs comprehensive checks and fixes for board data integrity, including list ordering, card positions, and swimlane structure.", + "delete-duplicate-empty-lists-migration": "Delete Duplicate Empty Lists", + "delete-duplicate-empty-lists-migration-description": "Safely deletes empty duplicate lists. Only removes lists that have no cards AND have another list with the same title that contains cards.", + "lost-cards": "Lost Cards", + "lost-cards-list": "Restored Items", + "restore-lost-cards-migration": "Restore Lost Cards", + "restore-lost-cards-migration-description": "Finds and restores cards and lists with missing swimlaneId or listId. Creates a 'Lost Cards' swimlane to make all lost items visible again.", + "restore-all-archived-migration": "Restore All Archived", + "restore-all-archived-migration-description": "Restores all archived swimlanes, lists, and cards. Automatically fixes any missing swimlaneId or listId to make items visible.", + "fix-missing-lists-migration": "Fix Missing Lists", + "fix-missing-lists-migration-description": "Detects and repairs missing or corrupted lists in the board structure.", + "fix-avatar-urls-migration": "Fix Avatar URLs", + "fix-avatar-urls-migration-description": "Updates avatar URLs for board members to use the correct storage backend and fixes broken avatar references.", + "fix-all-file-urls-migration": "Fix All File URLs", + "fix-all-file-urls-migration-description": "Updates all file attachment URLs on this board to use the correct storage backend and fixes broken file references.", + "migration-needed": "Migration Needed", + "migration-complete": "Complete", + "migration-running": "Running...", + "migration-successful": "Migration completed successfully", + "migration-failed": "Migration failed", + "migrations": "Migrations", + "migrations-admin-only": "Only board administrators can run migrations", + "migrations-description": "Run data integrity checks and repairs for this board. Each migration can be executed individually.", + "no-issues-found": "No issues found", + "run-migration": "Run Migration", + "run-comprehensive-migration-confirm": "This will perform a comprehensive migration to check and fix board data integrity. This may take a few moments. Continue?", + "run-delete-duplicate-empty-lists-migration-confirm": "This will first convert any shared lists to per-swimlane lists, then delete empty lists that have a duplicate list with the same title containing cards. Only truly redundant empty lists will be removed. Continue?", + "run-restore-lost-cards-migration-confirm": "This will create a 'Lost Cards' swimlane and restore all cards and lists with missing swimlaneId or listId. This only affects non-archived items. Continue?", + "run-restore-all-archived-migration-confirm": "This will restore ALL archived swimlanes, lists, and cards, making them visible again. Any items with missing IDs will be automatically fixed. This cannot be easily undone. Continue?", + "run-fix-missing-lists-migration-confirm": "This will detect and repair missing or corrupted lists in the board structure. Continue?", + "run-fix-avatar-urls-migration-confirm": "This will update avatar URLs for board members to use the correct storage backend. Continue?", + "run-fix-all-file-urls-migration-confirm": "This will update all file attachment URLs on this board to use the correct storage backend. Continue?", + "restore-lost-cards-nothing-to-restore": "No lost swimlanes, lists, or cards to restore", + + "migration-progress-title": "Board Migration in Progress", + "migration-progress-overall": "Overall Progress", + "migration-progress-current-step": "Current Step", + "migration-progress-status": "Status", + "migration-progress-details": "Details", + "migration-progress-note": "Please wait while we migrate your board to the latest structure...", + + "step-analyze-board-structure": "Analyze Board Structure", + "step-fix-orphaned-cards": "Fix Orphaned Cards", + "step-convert-shared-lists": "Convert Shared Lists", + "step-ensure-per-swimlane-lists": "Ensure Per-Swimlane Lists", + "step-validate-migration": "Validate Migration", + "step-fix-avatar-urls": "Fix Avatar URLs", + "step-fix-attachment-urls": "Fix Attachment URLs", + "step-analyze-lists": "Analyze Lists", + "step-create-missing-lists": "Create Missing Lists", + "step-update-cards": "Update Cards", + "step-finalize": "Finalize", + "step-delete-duplicate-empty-lists": "Delete Duplicate Empty Lists", + "step-ensure-lost-cards-swimlane": "Ensure Lost Cards Swimlane", + "step-restore-lists": "Restore Lists", + "step-restore-cards": "Restore Cards", + "step-restore-swimlanes": "Restore Swimlanes", + "step-fix-missing-ids": "Fix Missing IDs", + "step-scan-users": "Checking board member avatars", + "step-scan-files": "Checking board file attachments", + "step-fix-file-urls": "Fixing file URLs", "cleanup": "Cleanup", "cleanup-old-jobs": "Cleanup Old Jobs", "completed": "Afgewerkt", diff --git a/imports/i18n/data/nl.i18n.json b/imports/i18n/data/nl.i18n.json index e854597fd..2d5624db5 100644 --- a/imports/i18n/data/nl.i18n.json +++ b/imports/i18n/data/nl.i18n.json @@ -1404,7 +1404,70 @@ "back-to-settings": "Terug naar Instellingen", "board-id": "Bord ID", "board-migration": "Bord Migratie", + "board-migrations": "Board Migrations", "card-show-lists-on-minicard": "Toon Lijsten op Minikaart", + "comprehensive-board-migration": "Comprehensive Board Migration", + "comprehensive-board-migration-description": "Performs comprehensive checks and fixes for board data integrity, including list ordering, card positions, and swimlane structure.", + "delete-duplicate-empty-lists-migration": "Delete Duplicate Empty Lists", + "delete-duplicate-empty-lists-migration-description": "Safely deletes empty duplicate lists. Only removes lists that have no cards AND have another list with the same title that contains cards.", + "lost-cards": "Lost Cards", + "lost-cards-list": "Restored Items", + "restore-lost-cards-migration": "Restore Lost Cards", + "restore-lost-cards-migration-description": "Finds and restores cards and lists with missing swimlaneId or listId. Creates a 'Lost Cards' swimlane to make all lost items visible again.", + "restore-all-archived-migration": "Restore All Archived", + "restore-all-archived-migration-description": "Restores all archived swimlanes, lists, and cards. Automatically fixes any missing swimlaneId or listId to make items visible.", + "fix-missing-lists-migration": "Fix Missing Lists", + "fix-missing-lists-migration-description": "Detects and repairs missing or corrupted lists in the board structure.", + "fix-avatar-urls-migration": "Fix Avatar URLs", + "fix-avatar-urls-migration-description": "Updates avatar URLs for board members to use the correct storage backend and fixes broken avatar references.", + "fix-all-file-urls-migration": "Fix All File URLs", + "fix-all-file-urls-migration-description": "Updates all file attachment URLs on this board to use the correct storage backend and fixes broken file references.", + "migration-needed": "Migration Needed", + "migration-complete": "Voltooid", + "migration-running": "Running...", + "migration-successful": "Migration completed successfully", + "migration-failed": "Migration failed", + "migrations": "Migrations", + "migrations-admin-only": "Only board administrators can run migrations", + "migrations-description": "Run data integrity checks and repairs for this board. Each migration can be executed individually.", + "no-issues-found": "No issues found", + "run-migration": "Run Migration", + "run-comprehensive-migration-confirm": "This will perform a comprehensive migration to check and fix board data integrity. This may take a few moments. Continue?", + "run-delete-duplicate-empty-lists-migration-confirm": "This will first convert any shared lists to per-swimlane lists, then delete empty lists that have a duplicate list with the same title containing cards. Only truly redundant empty lists will be removed. Continue?", + "run-restore-lost-cards-migration-confirm": "This will create a 'Lost Cards' swimlane and restore all cards and lists with missing swimlaneId or listId. This only affects non-archived items. Continue?", + "run-restore-all-archived-migration-confirm": "This will restore ALL archived swimlanes, lists, and cards, making them visible again. Any items with missing IDs will be automatically fixed. This cannot be easily undone. Continue?", + "run-fix-missing-lists-migration-confirm": "This will detect and repair missing or corrupted lists in the board structure. Continue?", + "run-fix-avatar-urls-migration-confirm": "This will update avatar URLs for board members to use the correct storage backend. Continue?", + "run-fix-all-file-urls-migration-confirm": "This will update all file attachment URLs on this board to use the correct storage backend. Continue?", + "restore-lost-cards-nothing-to-restore": "No lost swimlanes, lists, or cards to restore", + + "migration-progress-title": "Board Migration in Progress", + "migration-progress-overall": "Algehele Voortgang", + "migration-progress-current-step": "Current Step", + "migration-progress-status": "Status", + "migration-progress-details": "Details", + "migration-progress-note": "Please wait while we migrate your board to the latest structure...", + + "step-analyze-board-structure": "Analyze Board Structure", + "step-fix-orphaned-cards": "Fix Orphaned Cards", + "step-convert-shared-lists": "Convert Shared Lists", + "step-ensure-per-swimlane-lists": "Ensure Per-Swimlane Lists", + "step-validate-migration": "Validate Migration", + "step-fix-avatar-urls": "Fix Avatar URLs", + "step-fix-attachment-urls": "Fix Attachment URLs", + "step-analyze-lists": "Analyze Lists", + "step-create-missing-lists": "Create Missing Lists", + "step-update-cards": "Update Cards", + "step-finalize": "Finalize", + "step-delete-duplicate-empty-lists": "Delete Duplicate Empty Lists", + "step-ensure-lost-cards-swimlane": "Ensure Lost Cards Swimlane", + "step-restore-lists": "Restore Lists", + "step-restore-cards": "Restore Cards", + "step-restore-swimlanes": "Restore Swimlanes", + "step-fix-missing-ids": "Fix Missing IDs", + "step-scan-users": "Checking board member avatars", + "step-scan-files": "Checking board file attachments", + "step-fix-file-urls": "Fixing file URLs", "cleanup": "Opschonen", "cleanup-old-jobs": "Schoon Oude Taken Op", "completed": "Afgewerkt", diff --git a/imports/i18n/data/oc.i18n.json b/imports/i18n/data/oc.i18n.json index 5da5ae2bc..1acaa8d5d 100644 --- a/imports/i18n/data/oc.i18n.json +++ b/imports/i18n/data/oc.i18n.json @@ -1404,7 +1404,70 @@ "back-to-settings": "Back to Settings", "board-id": "Board ID", "board-migration": "Board Migration", + "board-migrations": "Board Migrations", "card-show-lists-on-minicard": "Show Lists on Minicard", + "comprehensive-board-migration": "Comprehensive Board Migration", + "comprehensive-board-migration-description": "Performs comprehensive checks and fixes for board data integrity, including list ordering, card positions, and swimlane structure.", + "delete-duplicate-empty-lists-migration": "Delete Duplicate Empty Lists", + "delete-duplicate-empty-lists-migration-description": "Safely deletes empty duplicate lists. Only removes lists that have no cards AND have another list with the same title that contains cards.", + "lost-cards": "Lost Cards", + "lost-cards-list": "Restored Items", + "restore-lost-cards-migration": "Restore Lost Cards", + "restore-lost-cards-migration-description": "Finds and restores cards and lists with missing swimlaneId or listId. Creates a 'Lost Cards' swimlane to make all lost items visible again.", + "restore-all-archived-migration": "Restore All Archived", + "restore-all-archived-migration-description": "Restores all archived swimlanes, lists, and cards. Automatically fixes any missing swimlaneId or listId to make items visible.", + "fix-missing-lists-migration": "Fix Missing Lists", + "fix-missing-lists-migration-description": "Detects and repairs missing or corrupted lists in the board structure.", + "fix-avatar-urls-migration": "Fix Avatar URLs", + "fix-avatar-urls-migration-description": "Updates avatar URLs for board members to use the correct storage backend and fixes broken avatar references.", + "fix-all-file-urls-migration": "Fix All File URLs", + "fix-all-file-urls-migration-description": "Updates all file attachment URLs on this board to use the correct storage backend and fixes broken file references.", + "migration-needed": "Migration Needed", + "migration-complete": "Complete", + "migration-running": "Running...", + "migration-successful": "Migration completed successfully", + "migration-failed": "Migration failed", + "migrations": "Migrations", + "migrations-admin-only": "Only board administrators can run migrations", + "migrations-description": "Run data integrity checks and repairs for this board. Each migration can be executed individually.", + "no-issues-found": "No issues found", + "run-migration": "Run Migration", + "run-comprehensive-migration-confirm": "This will perform a comprehensive migration to check and fix board data integrity. This may take a few moments. Continue?", + "run-delete-duplicate-empty-lists-migration-confirm": "This will first convert any shared lists to per-swimlane lists, then delete empty lists that have a duplicate list with the same title containing cards. Only truly redundant empty lists will be removed. Continue?", + "run-restore-lost-cards-migration-confirm": "This will create a 'Lost Cards' swimlane and restore all cards and lists with missing swimlaneId or listId. This only affects non-archived items. Continue?", + "run-restore-all-archived-migration-confirm": "This will restore ALL archived swimlanes, lists, and cards, making them visible again. Any items with missing IDs will be automatically fixed. This cannot be easily undone. Continue?", + "run-fix-missing-lists-migration-confirm": "This will detect and repair missing or corrupted lists in the board structure. Continue?", + "run-fix-avatar-urls-migration-confirm": "This will update avatar URLs for board members to use the correct storage backend. Continue?", + "run-fix-all-file-urls-migration-confirm": "This will update all file attachment URLs on this board to use the correct storage backend. Continue?", + "restore-lost-cards-nothing-to-restore": "No lost swimlanes, lists, or cards to restore", + + "migration-progress-title": "Board Migration in Progress", + "migration-progress-overall": "Overall Progress", + "migration-progress-current-step": "Current Step", + "migration-progress-status": "Status", + "migration-progress-details": "Details", + "migration-progress-note": "Please wait while we migrate your board to the latest structure...", + + "step-analyze-board-structure": "Analyze Board Structure", + "step-fix-orphaned-cards": "Fix Orphaned Cards", + "step-convert-shared-lists": "Convert Shared Lists", + "step-ensure-per-swimlane-lists": "Ensure Per-Swimlane Lists", + "step-validate-migration": "Validate Migration", + "step-fix-avatar-urls": "Fix Avatar URLs", + "step-fix-attachment-urls": "Fix Attachment URLs", + "step-analyze-lists": "Analyze Lists", + "step-create-missing-lists": "Create Missing Lists", + "step-update-cards": "Update Cards", + "step-finalize": "Finalize", + "step-delete-duplicate-empty-lists": "Delete Duplicate Empty Lists", + "step-ensure-lost-cards-swimlane": "Ensure Lost Cards Swimlane", + "step-restore-lists": "Restore Lists", + "step-restore-cards": "Restore Cards", + "step-restore-swimlanes": "Restore Swimlanes", + "step-fix-missing-ids": "Fix Missing IDs", + "step-scan-users": "Checking board member avatars", + "step-scan-files": "Checking board file attachments", + "step-fix-file-urls": "Fixing file URLs", "cleanup": "Cleanup", "cleanup-old-jobs": "Cleanup Old Jobs", "completed": "Completed", diff --git a/imports/i18n/data/or_IN.i18n.json b/imports/i18n/data/or_IN.i18n.json index eef27e1fe..958d8313a 100644 --- a/imports/i18n/data/or_IN.i18n.json +++ b/imports/i18n/data/or_IN.i18n.json @@ -1404,7 +1404,70 @@ "back-to-settings": "Back to Settings", "board-id": "Board ID", "board-migration": "Board Migration", + "board-migrations": "Board Migrations", "card-show-lists-on-minicard": "Show Lists on Minicard", + "comprehensive-board-migration": "Comprehensive Board Migration", + "comprehensive-board-migration-description": "Performs comprehensive checks and fixes for board data integrity, including list ordering, card positions, and swimlane structure.", + "delete-duplicate-empty-lists-migration": "Delete Duplicate Empty Lists", + "delete-duplicate-empty-lists-migration-description": "Safely deletes empty duplicate lists. Only removes lists that have no cards AND have another list with the same title that contains cards.", + "lost-cards": "Lost Cards", + "lost-cards-list": "Restored Items", + "restore-lost-cards-migration": "Restore Lost Cards", + "restore-lost-cards-migration-description": "Finds and restores cards and lists with missing swimlaneId or listId. Creates a 'Lost Cards' swimlane to make all lost items visible again.", + "restore-all-archived-migration": "Restore All Archived", + "restore-all-archived-migration-description": "Restores all archived swimlanes, lists, and cards. Automatically fixes any missing swimlaneId or listId to make items visible.", + "fix-missing-lists-migration": "Fix Missing Lists", + "fix-missing-lists-migration-description": "Detects and repairs missing or corrupted lists in the board structure.", + "fix-avatar-urls-migration": "Fix Avatar URLs", + "fix-avatar-urls-migration-description": "Updates avatar URLs for board members to use the correct storage backend and fixes broken avatar references.", + "fix-all-file-urls-migration": "Fix All File URLs", + "fix-all-file-urls-migration-description": "Updates all file attachment URLs on this board to use the correct storage backend and fixes broken file references.", + "migration-needed": "Migration Needed", + "migration-complete": "Complete", + "migration-running": "Running...", + "migration-successful": "Migration completed successfully", + "migration-failed": "Migration failed", + "migrations": "Migrations", + "migrations-admin-only": "Only board administrators can run migrations", + "migrations-description": "Run data integrity checks and repairs for this board. Each migration can be executed individually.", + "no-issues-found": "No issues found", + "run-migration": "Run Migration", + "run-comprehensive-migration-confirm": "This will perform a comprehensive migration to check and fix board data integrity. This may take a few moments. Continue?", + "run-delete-duplicate-empty-lists-migration-confirm": "This will first convert any shared lists to per-swimlane lists, then delete empty lists that have a duplicate list with the same title containing cards. Only truly redundant empty lists will be removed. Continue?", + "run-restore-lost-cards-migration-confirm": "This will create a 'Lost Cards' swimlane and restore all cards and lists with missing swimlaneId or listId. This only affects non-archived items. Continue?", + "run-restore-all-archived-migration-confirm": "This will restore ALL archived swimlanes, lists, and cards, making them visible again. Any items with missing IDs will be automatically fixed. This cannot be easily undone. Continue?", + "run-fix-missing-lists-migration-confirm": "This will detect and repair missing or corrupted lists in the board structure. Continue?", + "run-fix-avatar-urls-migration-confirm": "This will update avatar URLs for board members to use the correct storage backend. Continue?", + "run-fix-all-file-urls-migration-confirm": "This will update all file attachment URLs on this board to use the correct storage backend. Continue?", + "restore-lost-cards-nothing-to-restore": "No lost swimlanes, lists, or cards to restore", + + "migration-progress-title": "Board Migration in Progress", + "migration-progress-overall": "Overall Progress", + "migration-progress-current-step": "Current Step", + "migration-progress-status": "Status", + "migration-progress-details": "Details", + "migration-progress-note": "Please wait while we migrate your board to the latest structure...", + + "step-analyze-board-structure": "Analyze Board Structure", + "step-fix-orphaned-cards": "Fix Orphaned Cards", + "step-convert-shared-lists": "Convert Shared Lists", + "step-ensure-per-swimlane-lists": "Ensure Per-Swimlane Lists", + "step-validate-migration": "Validate Migration", + "step-fix-avatar-urls": "Fix Avatar URLs", + "step-fix-attachment-urls": "Fix Attachment URLs", + "step-analyze-lists": "Analyze Lists", + "step-create-missing-lists": "Create Missing Lists", + "step-update-cards": "Update Cards", + "step-finalize": "Finalize", + "step-delete-duplicate-empty-lists": "Delete Duplicate Empty Lists", + "step-ensure-lost-cards-swimlane": "Ensure Lost Cards Swimlane", + "step-restore-lists": "Restore Lists", + "step-restore-cards": "Restore Cards", + "step-restore-swimlanes": "Restore Swimlanes", + "step-fix-missing-ids": "Fix Missing IDs", + "step-scan-users": "Checking board member avatars", + "step-scan-files": "Checking board file attachments", + "step-fix-file-urls": "Fixing file URLs", "cleanup": "Cleanup", "cleanup-old-jobs": "Cleanup Old Jobs", "completed": "Completed", diff --git a/imports/i18n/data/pa.i18n.json b/imports/i18n/data/pa.i18n.json index 1ab8eeaa8..4804e6dff 100644 --- a/imports/i18n/data/pa.i18n.json +++ b/imports/i18n/data/pa.i18n.json @@ -1404,7 +1404,70 @@ "back-to-settings": "Back to Settings", "board-id": "Board ID", "board-migration": "Board Migration", + "board-migrations": "Board Migrations", "card-show-lists-on-minicard": "Show Lists on Minicard", + "comprehensive-board-migration": "Comprehensive Board Migration", + "comprehensive-board-migration-description": "Performs comprehensive checks and fixes for board data integrity, including list ordering, card positions, and swimlane structure.", + "delete-duplicate-empty-lists-migration": "Delete Duplicate Empty Lists", + "delete-duplicate-empty-lists-migration-description": "Safely deletes empty duplicate lists. Only removes lists that have no cards AND have another list with the same title that contains cards.", + "lost-cards": "Lost Cards", + "lost-cards-list": "Restored Items", + "restore-lost-cards-migration": "Restore Lost Cards", + "restore-lost-cards-migration-description": "Finds and restores cards and lists with missing swimlaneId or listId. Creates a 'Lost Cards' swimlane to make all lost items visible again.", + "restore-all-archived-migration": "Restore All Archived", + "restore-all-archived-migration-description": "Restores all archived swimlanes, lists, and cards. Automatically fixes any missing swimlaneId or listId to make items visible.", + "fix-missing-lists-migration": "Fix Missing Lists", + "fix-missing-lists-migration-description": "Detects and repairs missing or corrupted lists in the board structure.", + "fix-avatar-urls-migration": "Fix Avatar URLs", + "fix-avatar-urls-migration-description": "Updates avatar URLs for board members to use the correct storage backend and fixes broken avatar references.", + "fix-all-file-urls-migration": "Fix All File URLs", + "fix-all-file-urls-migration-description": "Updates all file attachment URLs on this board to use the correct storage backend and fixes broken file references.", + "migration-needed": "Migration Needed", + "migration-complete": "Complete", + "migration-running": "Running...", + "migration-successful": "Migration completed successfully", + "migration-failed": "Migration failed", + "migrations": "Migrations", + "migrations-admin-only": "Only board administrators can run migrations", + "migrations-description": "Run data integrity checks and repairs for this board. Each migration can be executed individually.", + "no-issues-found": "No issues found", + "run-migration": "Run Migration", + "run-comprehensive-migration-confirm": "This will perform a comprehensive migration to check and fix board data integrity. This may take a few moments. Continue?", + "run-delete-duplicate-empty-lists-migration-confirm": "This will first convert any shared lists to per-swimlane lists, then delete empty lists that have a duplicate list with the same title containing cards. Only truly redundant empty lists will be removed. Continue?", + "run-restore-lost-cards-migration-confirm": "This will create a 'Lost Cards' swimlane and restore all cards and lists with missing swimlaneId or listId. This only affects non-archived items. Continue?", + "run-restore-all-archived-migration-confirm": "This will restore ALL archived swimlanes, lists, and cards, making them visible again. Any items with missing IDs will be automatically fixed. This cannot be easily undone. Continue?", + "run-fix-missing-lists-migration-confirm": "This will detect and repair missing or corrupted lists in the board structure. Continue?", + "run-fix-avatar-urls-migration-confirm": "This will update avatar URLs for board members to use the correct storage backend. Continue?", + "run-fix-all-file-urls-migration-confirm": "This will update all file attachment URLs on this board to use the correct storage backend. Continue?", + "restore-lost-cards-nothing-to-restore": "No lost swimlanes, lists, or cards to restore", + + "migration-progress-title": "Board Migration in Progress", + "migration-progress-overall": "Overall Progress", + "migration-progress-current-step": "Current Step", + "migration-progress-status": "Status", + "migration-progress-details": "Details", + "migration-progress-note": "Please wait while we migrate your board to the latest structure...", + + "step-analyze-board-structure": "Analyze Board Structure", + "step-fix-orphaned-cards": "Fix Orphaned Cards", + "step-convert-shared-lists": "Convert Shared Lists", + "step-ensure-per-swimlane-lists": "Ensure Per-Swimlane Lists", + "step-validate-migration": "Validate Migration", + "step-fix-avatar-urls": "Fix Avatar URLs", + "step-fix-attachment-urls": "Fix Attachment URLs", + "step-analyze-lists": "Analyze Lists", + "step-create-missing-lists": "Create Missing Lists", + "step-update-cards": "Update Cards", + "step-finalize": "Finalize", + "step-delete-duplicate-empty-lists": "Delete Duplicate Empty Lists", + "step-ensure-lost-cards-swimlane": "Ensure Lost Cards Swimlane", + "step-restore-lists": "Restore Lists", + "step-restore-cards": "Restore Cards", + "step-restore-swimlanes": "Restore Swimlanes", + "step-fix-missing-ids": "Fix Missing IDs", + "step-scan-users": "Checking board member avatars", + "step-scan-files": "Checking board file attachments", + "step-fix-file-urls": "Fixing file URLs", "cleanup": "Cleanup", "cleanup-old-jobs": "Cleanup Old Jobs", "completed": "Completed", diff --git a/imports/i18n/data/pl-PL.i18n.json b/imports/i18n/data/pl-PL.i18n.json index 641328e6d..313e3db40 100644 --- a/imports/i18n/data/pl-PL.i18n.json +++ b/imports/i18n/data/pl-PL.i18n.json @@ -1404,7 +1404,70 @@ "back-to-settings": "Back to Settings", "board-id": "Board ID", "board-migration": "Board Migration", + "board-migrations": "Board Migrations", "card-show-lists-on-minicard": "Show Lists on Minicard", + "comprehensive-board-migration": "Comprehensive Board Migration", + "comprehensive-board-migration-description": "Performs comprehensive checks and fixes for board data integrity, including list ordering, card positions, and swimlane structure.", + "delete-duplicate-empty-lists-migration": "Delete Duplicate Empty Lists", + "delete-duplicate-empty-lists-migration-description": "Safely deletes empty duplicate lists. Only removes lists that have no cards AND have another list with the same title that contains cards.", + "lost-cards": "Lost Cards", + "lost-cards-list": "Restored Items", + "restore-lost-cards-migration": "Restore Lost Cards", + "restore-lost-cards-migration-description": "Finds and restores cards and lists with missing swimlaneId or listId. Creates a 'Lost Cards' swimlane to make all lost items visible again.", + "restore-all-archived-migration": "Restore All Archived", + "restore-all-archived-migration-description": "Restores all archived swimlanes, lists, and cards. Automatically fixes any missing swimlaneId or listId to make items visible.", + "fix-missing-lists-migration": "Fix Missing Lists", + "fix-missing-lists-migration-description": "Detects and repairs missing or corrupted lists in the board structure.", + "fix-avatar-urls-migration": "Fix Avatar URLs", + "fix-avatar-urls-migration-description": "Updates avatar URLs for board members to use the correct storage backend and fixes broken avatar references.", + "fix-all-file-urls-migration": "Fix All File URLs", + "fix-all-file-urls-migration-description": "Updates all file attachment URLs on this board to use the correct storage backend and fixes broken file references.", + "migration-needed": "Migration Needed", + "migration-complete": "Complete", + "migration-running": "Running...", + "migration-successful": "Migration completed successfully", + "migration-failed": "Migration failed", + "migrations": "Migrations", + "migrations-admin-only": "Only board administrators can run migrations", + "migrations-description": "Run data integrity checks and repairs for this board. Each migration can be executed individually.", + "no-issues-found": "No issues found", + "run-migration": "Run Migration", + "run-comprehensive-migration-confirm": "This will perform a comprehensive migration to check and fix board data integrity. This may take a few moments. Continue?", + "run-delete-duplicate-empty-lists-migration-confirm": "This will first convert any shared lists to per-swimlane lists, then delete empty lists that have a duplicate list with the same title containing cards. Only truly redundant empty lists will be removed. Continue?", + "run-restore-lost-cards-migration-confirm": "This will create a 'Lost Cards' swimlane and restore all cards and lists with missing swimlaneId or listId. This only affects non-archived items. Continue?", + "run-restore-all-archived-migration-confirm": "This will restore ALL archived swimlanes, lists, and cards, making them visible again. Any items with missing IDs will be automatically fixed. This cannot be easily undone. Continue?", + "run-fix-missing-lists-migration-confirm": "This will detect and repair missing or corrupted lists in the board structure. Continue?", + "run-fix-avatar-urls-migration-confirm": "This will update avatar URLs for board members to use the correct storage backend. Continue?", + "run-fix-all-file-urls-migration-confirm": "This will update all file attachment URLs on this board to use the correct storage backend. Continue?", + "restore-lost-cards-nothing-to-restore": "No lost swimlanes, lists, or cards to restore", + + "migration-progress-title": "Board Migration in Progress", + "migration-progress-overall": "Overall Progress", + "migration-progress-current-step": "Current Step", + "migration-progress-status": "Status", + "migration-progress-details": "Szczegóły", + "migration-progress-note": "Please wait while we migrate your board to the latest structure...", + + "step-analyze-board-structure": "Analyze Board Structure", + "step-fix-orphaned-cards": "Fix Orphaned Cards", + "step-convert-shared-lists": "Convert Shared Lists", + "step-ensure-per-swimlane-lists": "Ensure Per-Swimlane Lists", + "step-validate-migration": "Validate Migration", + "step-fix-avatar-urls": "Fix Avatar URLs", + "step-fix-attachment-urls": "Fix Attachment URLs", + "step-analyze-lists": "Analyze Lists", + "step-create-missing-lists": "Create Missing Lists", + "step-update-cards": "Update Cards", + "step-finalize": "Finalize", + "step-delete-duplicate-empty-lists": "Delete Duplicate Empty Lists", + "step-ensure-lost-cards-swimlane": "Ensure Lost Cards Swimlane", + "step-restore-lists": "Restore Lists", + "step-restore-cards": "Restore Cards", + "step-restore-swimlanes": "Restore Swimlanes", + "step-fix-missing-ids": "Fix Missing IDs", + "step-scan-users": "Checking board member avatars", + "step-scan-files": "Checking board file attachments", + "step-fix-file-urls": "Fixing file URLs", "cleanup": "Cleanup", "cleanup-old-jobs": "Cleanup Old Jobs", "completed": "ukończona", diff --git a/imports/i18n/data/pl.i18n.json b/imports/i18n/data/pl.i18n.json index 7ae12a506..5d38e2647 100644 --- a/imports/i18n/data/pl.i18n.json +++ b/imports/i18n/data/pl.i18n.json @@ -1404,7 +1404,70 @@ "back-to-settings": "Back to Settings", "board-id": "Board ID", "board-migration": "Board Migration", + "board-migrations": "Board Migrations", "card-show-lists-on-minicard": "Show Lists on Minicard", + "comprehensive-board-migration": "Comprehensive Board Migration", + "comprehensive-board-migration-description": "Performs comprehensive checks and fixes for board data integrity, including list ordering, card positions, and swimlane structure.", + "delete-duplicate-empty-lists-migration": "Delete Duplicate Empty Lists", + "delete-duplicate-empty-lists-migration-description": "Safely deletes empty duplicate lists. Only removes lists that have no cards AND have another list with the same title that contains cards.", + "lost-cards": "Lost Cards", + "lost-cards-list": "Restored Items", + "restore-lost-cards-migration": "Restore Lost Cards", + "restore-lost-cards-migration-description": "Finds and restores cards and lists with missing swimlaneId or listId. Creates a 'Lost Cards' swimlane to make all lost items visible again.", + "restore-all-archived-migration": "Restore All Archived", + "restore-all-archived-migration-description": "Restores all archived swimlanes, lists, and cards. Automatically fixes any missing swimlaneId or listId to make items visible.", + "fix-missing-lists-migration": "Fix Missing Lists", + "fix-missing-lists-migration-description": "Detects and repairs missing or corrupted lists in the board structure.", + "fix-avatar-urls-migration": "Fix Avatar URLs", + "fix-avatar-urls-migration-description": "Updates avatar URLs for board members to use the correct storage backend and fixes broken avatar references.", + "fix-all-file-urls-migration": "Fix All File URLs", + "fix-all-file-urls-migration-description": "Updates all file attachment URLs on this board to use the correct storage backend and fixes broken file references.", + "migration-needed": "Migration Needed", + "migration-complete": "Complete", + "migration-running": "Running...", + "migration-successful": "Migration completed successfully", + "migration-failed": "Migration failed", + "migrations": "Migrations", + "migrations-admin-only": "Only board administrators can run migrations", + "migrations-description": "Run data integrity checks and repairs for this board. Each migration can be executed individually.", + "no-issues-found": "No issues found", + "run-migration": "Run Migration", + "run-comprehensive-migration-confirm": "This will perform a comprehensive migration to check and fix board data integrity. This may take a few moments. Continue?", + "run-delete-duplicate-empty-lists-migration-confirm": "This will first convert any shared lists to per-swimlane lists, then delete empty lists that have a duplicate list with the same title containing cards. Only truly redundant empty lists will be removed. Continue?", + "run-restore-lost-cards-migration-confirm": "This will create a 'Lost Cards' swimlane and restore all cards and lists with missing swimlaneId or listId. This only affects non-archived items. Continue?", + "run-restore-all-archived-migration-confirm": "This will restore ALL archived swimlanes, lists, and cards, making them visible again. Any items with missing IDs will be automatically fixed. This cannot be easily undone. Continue?", + "run-fix-missing-lists-migration-confirm": "This will detect and repair missing or corrupted lists in the board structure. Continue?", + "run-fix-avatar-urls-migration-confirm": "This will update avatar URLs for board members to use the correct storage backend. Continue?", + "run-fix-all-file-urls-migration-confirm": "This will update all file attachment URLs on this board to use the correct storage backend. Continue?", + "restore-lost-cards-nothing-to-restore": "No lost swimlanes, lists, or cards to restore", + + "migration-progress-title": "Board Migration in Progress", + "migration-progress-overall": "Overall Progress", + "migration-progress-current-step": "Current Step", + "migration-progress-status": "Status", + "migration-progress-details": "Szczegóły", + "migration-progress-note": "Please wait while we migrate your board to the latest structure...", + + "step-analyze-board-structure": "Analyze Board Structure", + "step-fix-orphaned-cards": "Fix Orphaned Cards", + "step-convert-shared-lists": "Convert Shared Lists", + "step-ensure-per-swimlane-lists": "Ensure Per-Swimlane Lists", + "step-validate-migration": "Validate Migration", + "step-fix-avatar-urls": "Fix Avatar URLs", + "step-fix-attachment-urls": "Fix Attachment URLs", + "step-analyze-lists": "Analyze Lists", + "step-create-missing-lists": "Create Missing Lists", + "step-update-cards": "Update Cards", + "step-finalize": "Finalize", + "step-delete-duplicate-empty-lists": "Delete Duplicate Empty Lists", + "step-ensure-lost-cards-swimlane": "Ensure Lost Cards Swimlane", + "step-restore-lists": "Restore Lists", + "step-restore-cards": "Restore Cards", + "step-restore-swimlanes": "Restore Swimlanes", + "step-fix-missing-ids": "Fix Missing IDs", + "step-scan-users": "Checking board member avatars", + "step-scan-files": "Checking board file attachments", + "step-fix-file-urls": "Fixing file URLs", "cleanup": "Cleanup", "cleanup-old-jobs": "Cleanup Old Jobs", "completed": "ukończona", diff --git a/imports/i18n/data/pt-BR.i18n.json b/imports/i18n/data/pt-BR.i18n.json index 5a135f543..89965f75b 100644 --- a/imports/i18n/data/pt-BR.i18n.json +++ b/imports/i18n/data/pt-BR.i18n.json @@ -1404,7 +1404,70 @@ "back-to-settings": "Voltar às Configurações", "board-id": "ID do Quadro", "board-migration": "Migração de Quadro", + "board-migrations": "Board Migrations", "card-show-lists-on-minicard": "Mostrar Listas no Mini cartão", + "comprehensive-board-migration": "Comprehensive Board Migration", + "comprehensive-board-migration-description": "Performs comprehensive checks and fixes for board data integrity, including list ordering, card positions, and swimlane structure.", + "delete-duplicate-empty-lists-migration": "Delete Duplicate Empty Lists", + "delete-duplicate-empty-lists-migration-description": "Safely deletes empty duplicate lists. Only removes lists that have no cards AND have another list with the same title that contains cards.", + "lost-cards": "Lost Cards", + "lost-cards-list": "Restored Items", + "restore-lost-cards-migration": "Restore Lost Cards", + "restore-lost-cards-migration-description": "Finds and restores cards and lists with missing swimlaneId or listId. Creates a 'Lost Cards' swimlane to make all lost items visible again.", + "restore-all-archived-migration": "Restore All Archived", + "restore-all-archived-migration-description": "Restores all archived swimlanes, lists, and cards. Automatically fixes any missing swimlaneId or listId to make items visible.", + "fix-missing-lists-migration": "Fix Missing Lists", + "fix-missing-lists-migration-description": "Detects and repairs missing or corrupted lists in the board structure.", + "fix-avatar-urls-migration": "Fix Avatar URLs", + "fix-avatar-urls-migration-description": "Updates avatar URLs for board members to use the correct storage backend and fixes broken avatar references.", + "fix-all-file-urls-migration": "Fix All File URLs", + "fix-all-file-urls-migration-description": "Updates all file attachment URLs on this board to use the correct storage backend and fixes broken file references.", + "migration-needed": "Migration Needed", + "migration-complete": "Concluído", + "migration-running": "Running...", + "migration-successful": "Migration completed successfully", + "migration-failed": "Migration failed", + "migrations": "Migrations", + "migrations-admin-only": "Only board administrators can run migrations", + "migrations-description": "Run data integrity checks and repairs for this board. Each migration can be executed individually.", + "no-issues-found": "No issues found", + "run-migration": "Run Migration", + "run-comprehensive-migration-confirm": "This will perform a comprehensive migration to check and fix board data integrity. This may take a few moments. Continue?", + "run-delete-duplicate-empty-lists-migration-confirm": "This will first convert any shared lists to per-swimlane lists, then delete empty lists that have a duplicate list with the same title containing cards. Only truly redundant empty lists will be removed. Continue?", + "run-restore-lost-cards-migration-confirm": "This will create a 'Lost Cards' swimlane and restore all cards and lists with missing swimlaneId or listId. This only affects non-archived items. Continue?", + "run-restore-all-archived-migration-confirm": "This will restore ALL archived swimlanes, lists, and cards, making them visible again. Any items with missing IDs will be automatically fixed. This cannot be easily undone. Continue?", + "run-fix-missing-lists-migration-confirm": "This will detect and repair missing or corrupted lists in the board structure. Continue?", + "run-fix-avatar-urls-migration-confirm": "This will update avatar URLs for board members to use the correct storage backend. Continue?", + "run-fix-all-file-urls-migration-confirm": "This will update all file attachment URLs on this board to use the correct storage backend. Continue?", + "restore-lost-cards-nothing-to-restore": "No lost swimlanes, lists, or cards to restore", + + "migration-progress-title": "Board Migration in Progress", + "migration-progress-overall": "Progresso Geral", + "migration-progress-current-step": "Current Step", + "migration-progress-status": "Status", + "migration-progress-details": "Detalhes", + "migration-progress-note": "Please wait while we migrate your board to the latest structure...", + + "step-analyze-board-structure": "Analyze Board Structure", + "step-fix-orphaned-cards": "Fix Orphaned Cards", + "step-convert-shared-lists": "Convert Shared Lists", + "step-ensure-per-swimlane-lists": "Ensure Per-Swimlane Lists", + "step-validate-migration": "Validate Migration", + "step-fix-avatar-urls": "Fix Avatar URLs", + "step-fix-attachment-urls": "Fix Attachment URLs", + "step-analyze-lists": "Analyze Lists", + "step-create-missing-lists": "Create Missing Lists", + "step-update-cards": "Update Cards", + "step-finalize": "Finalize", + "step-delete-duplicate-empty-lists": "Delete Duplicate Empty Lists", + "step-ensure-lost-cards-swimlane": "Ensure Lost Cards Swimlane", + "step-restore-lists": "Restore Lists", + "step-restore-cards": "Restore Cards", + "step-restore-swimlanes": "Restore Swimlanes", + "step-fix-missing-ids": "Fix Missing IDs", + "step-scan-users": "Checking board member avatars", + "step-scan-files": "Checking board file attachments", + "step-fix-file-urls": "Fixing file URLs", "cleanup": "Limpeza", "cleanup-old-jobs": "Limpar Trabalhos Antigos", "completed": "Completado", diff --git a/imports/i18n/data/pt.i18n.json b/imports/i18n/data/pt.i18n.json index 1f228c838..61052311f 100644 --- a/imports/i18n/data/pt.i18n.json +++ b/imports/i18n/data/pt.i18n.json @@ -1404,7 +1404,70 @@ "back-to-settings": "Back to Settings", "board-id": "Board ID", "board-migration": "Board Migration", + "board-migrations": "Board Migrations", "card-show-lists-on-minicard": "Show Lists on Minicard", + "comprehensive-board-migration": "Comprehensive Board Migration", + "comprehensive-board-migration-description": "Performs comprehensive checks and fixes for board data integrity, including list ordering, card positions, and swimlane structure.", + "delete-duplicate-empty-lists-migration": "Delete Duplicate Empty Lists", + "delete-duplicate-empty-lists-migration-description": "Safely deletes empty duplicate lists. Only removes lists that have no cards AND have another list with the same title that contains cards.", + "lost-cards": "Lost Cards", + "lost-cards-list": "Restored Items", + "restore-lost-cards-migration": "Restore Lost Cards", + "restore-lost-cards-migration-description": "Finds and restores cards and lists with missing swimlaneId or listId. Creates a 'Lost Cards' swimlane to make all lost items visible again.", + "restore-all-archived-migration": "Restore All Archived", + "restore-all-archived-migration-description": "Restores all archived swimlanes, lists, and cards. Automatically fixes any missing swimlaneId or listId to make items visible.", + "fix-missing-lists-migration": "Fix Missing Lists", + "fix-missing-lists-migration-description": "Detects and repairs missing or corrupted lists in the board structure.", + "fix-avatar-urls-migration": "Fix Avatar URLs", + "fix-avatar-urls-migration-description": "Updates avatar URLs for board members to use the correct storage backend and fixes broken avatar references.", + "fix-all-file-urls-migration": "Fix All File URLs", + "fix-all-file-urls-migration-description": "Updates all file attachment URLs on this board to use the correct storage backend and fixes broken file references.", + "migration-needed": "Migration Needed", + "migration-complete": "Concluído", + "migration-running": "Running...", + "migration-successful": "Migration completed successfully", + "migration-failed": "Migration failed", + "migrations": "Migrations", + "migrations-admin-only": "Only board administrators can run migrations", + "migrations-description": "Run data integrity checks and repairs for this board. Each migration can be executed individually.", + "no-issues-found": "No issues found", + "run-migration": "Run Migration", + "run-comprehensive-migration-confirm": "This will perform a comprehensive migration to check and fix board data integrity. This may take a few moments. Continue?", + "run-delete-duplicate-empty-lists-migration-confirm": "This will first convert any shared lists to per-swimlane lists, then delete empty lists that have a duplicate list with the same title containing cards. Only truly redundant empty lists will be removed. Continue?", + "run-restore-lost-cards-migration-confirm": "This will create a 'Lost Cards' swimlane and restore all cards and lists with missing swimlaneId or listId. This only affects non-archived items. Continue?", + "run-restore-all-archived-migration-confirm": "This will restore ALL archived swimlanes, lists, and cards, making them visible again. Any items with missing IDs will be automatically fixed. This cannot be easily undone. Continue?", + "run-fix-missing-lists-migration-confirm": "This will detect and repair missing or corrupted lists in the board structure. Continue?", + "run-fix-avatar-urls-migration-confirm": "This will update avatar URLs for board members to use the correct storage backend. Continue?", + "run-fix-all-file-urls-migration-confirm": "This will update all file attachment URLs on this board to use the correct storage backend. Continue?", + "restore-lost-cards-nothing-to-restore": "No lost swimlanes, lists, or cards to restore", + + "migration-progress-title": "Board Migration in Progress", + "migration-progress-overall": "Overall Progress", + "migration-progress-current-step": "Current Step", + "migration-progress-status": "Estado", + "migration-progress-details": "Detalhes", + "migration-progress-note": "Please wait while we migrate your board to the latest structure...", + + "step-analyze-board-structure": "Analyze Board Structure", + "step-fix-orphaned-cards": "Fix Orphaned Cards", + "step-convert-shared-lists": "Convert Shared Lists", + "step-ensure-per-swimlane-lists": "Ensure Per-Swimlane Lists", + "step-validate-migration": "Validate Migration", + "step-fix-avatar-urls": "Fix Avatar URLs", + "step-fix-attachment-urls": "Fix Attachment URLs", + "step-analyze-lists": "Analyze Lists", + "step-create-missing-lists": "Create Missing Lists", + "step-update-cards": "Update Cards", + "step-finalize": "Finalize", + "step-delete-duplicate-empty-lists": "Delete Duplicate Empty Lists", + "step-ensure-lost-cards-swimlane": "Ensure Lost Cards Swimlane", + "step-restore-lists": "Restore Lists", + "step-restore-cards": "Restore Cards", + "step-restore-swimlanes": "Restore Swimlanes", + "step-fix-missing-ids": "Fix Missing IDs", + "step-scan-users": "Checking board member avatars", + "step-scan-files": "Checking board file attachments", + "step-fix-file-urls": "Fixing file URLs", "cleanup": "Cleanup", "cleanup-old-jobs": "Cleanup Old Jobs", "completed": "Completada", diff --git a/imports/i18n/data/pt_PT.i18n.json b/imports/i18n/data/pt_PT.i18n.json index fa36c3204..260261018 100644 --- a/imports/i18n/data/pt_PT.i18n.json +++ b/imports/i18n/data/pt_PT.i18n.json @@ -1404,7 +1404,70 @@ "back-to-settings": "Back to Settings", "board-id": "Board ID", "board-migration": "Board Migration", + "board-migrations": "Board Migrations", "card-show-lists-on-minicard": "Show Lists on Minicard", + "comprehensive-board-migration": "Comprehensive Board Migration", + "comprehensive-board-migration-description": "Performs comprehensive checks and fixes for board data integrity, including list ordering, card positions, and swimlane structure.", + "delete-duplicate-empty-lists-migration": "Delete Duplicate Empty Lists", + "delete-duplicate-empty-lists-migration-description": "Safely deletes empty duplicate lists. Only removes lists that have no cards AND have another list with the same title that contains cards.", + "lost-cards": "Lost Cards", + "lost-cards-list": "Restored Items", + "restore-lost-cards-migration": "Restore Lost Cards", + "restore-lost-cards-migration-description": "Finds and restores cards and lists with missing swimlaneId or listId. Creates a 'Lost Cards' swimlane to make all lost items visible again.", + "restore-all-archived-migration": "Restore All Archived", + "restore-all-archived-migration-description": "Restores all archived swimlanes, lists, and cards. Automatically fixes any missing swimlaneId or listId to make items visible.", + "fix-missing-lists-migration": "Fix Missing Lists", + "fix-missing-lists-migration-description": "Detects and repairs missing or corrupted lists in the board structure.", + "fix-avatar-urls-migration": "Fix Avatar URLs", + "fix-avatar-urls-migration-description": "Updates avatar URLs for board members to use the correct storage backend and fixes broken avatar references.", + "fix-all-file-urls-migration": "Fix All File URLs", + "fix-all-file-urls-migration-description": "Updates all file attachment URLs on this board to use the correct storage backend and fixes broken file references.", + "migration-needed": "Migration Needed", + "migration-complete": "Complete", + "migration-running": "Running...", + "migration-successful": "Migration completed successfully", + "migration-failed": "Migration failed", + "migrations": "Migrations", + "migrations-admin-only": "Only board administrators can run migrations", + "migrations-description": "Run data integrity checks and repairs for this board. Each migration can be executed individually.", + "no-issues-found": "No issues found", + "run-migration": "Run Migration", + "run-comprehensive-migration-confirm": "This will perform a comprehensive migration to check and fix board data integrity. This may take a few moments. Continue?", + "run-delete-duplicate-empty-lists-migration-confirm": "This will first convert any shared lists to per-swimlane lists, then delete empty lists that have a duplicate list with the same title containing cards. Only truly redundant empty lists will be removed. Continue?", + "run-restore-lost-cards-migration-confirm": "This will create a 'Lost Cards' swimlane and restore all cards and lists with missing swimlaneId or listId. This only affects non-archived items. Continue?", + "run-restore-all-archived-migration-confirm": "This will restore ALL archived swimlanes, lists, and cards, making them visible again. Any items with missing IDs will be automatically fixed. This cannot be easily undone. Continue?", + "run-fix-missing-lists-migration-confirm": "This will detect and repair missing or corrupted lists in the board structure. Continue?", + "run-fix-avatar-urls-migration-confirm": "This will update avatar URLs for board members to use the correct storage backend. Continue?", + "run-fix-all-file-urls-migration-confirm": "This will update all file attachment URLs on this board to use the correct storage backend. Continue?", + "restore-lost-cards-nothing-to-restore": "No lost swimlanes, lists, or cards to restore", + + "migration-progress-title": "Board Migration in Progress", + "migration-progress-overall": "Overall Progress", + "migration-progress-current-step": "Current Step", + "migration-progress-status": "Estado", + "migration-progress-details": "Detalhes", + "migration-progress-note": "Please wait while we migrate your board to the latest structure...", + + "step-analyze-board-structure": "Analyze Board Structure", + "step-fix-orphaned-cards": "Fix Orphaned Cards", + "step-convert-shared-lists": "Convert Shared Lists", + "step-ensure-per-swimlane-lists": "Ensure Per-Swimlane Lists", + "step-validate-migration": "Validate Migration", + "step-fix-avatar-urls": "Fix Avatar URLs", + "step-fix-attachment-urls": "Fix Attachment URLs", + "step-analyze-lists": "Analyze Lists", + "step-create-missing-lists": "Create Missing Lists", + "step-update-cards": "Update Cards", + "step-finalize": "Finalize", + "step-delete-duplicate-empty-lists": "Delete Duplicate Empty Lists", + "step-ensure-lost-cards-swimlane": "Ensure Lost Cards Swimlane", + "step-restore-lists": "Restore Lists", + "step-restore-cards": "Restore Cards", + "step-restore-swimlanes": "Restore Swimlanes", + "step-fix-missing-ids": "Fix Missing IDs", + "step-scan-users": "Checking board member avatars", + "step-scan-files": "Checking board file attachments", + "step-fix-file-urls": "Fixing file URLs", "cleanup": "Cleanup", "cleanup-old-jobs": "Cleanup Old Jobs", "completed": "Completada", diff --git a/imports/i18n/data/ro-RO.i18n.json b/imports/i18n/data/ro-RO.i18n.json index 0adfeaf0e..11fa7fcf3 100644 --- a/imports/i18n/data/ro-RO.i18n.json +++ b/imports/i18n/data/ro-RO.i18n.json @@ -1404,7 +1404,70 @@ "back-to-settings": "Back to Settings", "board-id": "Board ID", "board-migration": "Board Migration", + "board-migrations": "Board Migrations", "card-show-lists-on-minicard": "Show Lists on Minicard", + "comprehensive-board-migration": "Comprehensive Board Migration", + "comprehensive-board-migration-description": "Performs comprehensive checks and fixes for board data integrity, including list ordering, card positions, and swimlane structure.", + "delete-duplicate-empty-lists-migration": "Delete Duplicate Empty Lists", + "delete-duplicate-empty-lists-migration-description": "Safely deletes empty duplicate lists. Only removes lists that have no cards AND have another list with the same title that contains cards.", + "lost-cards": "Lost Cards", + "lost-cards-list": "Restored Items", + "restore-lost-cards-migration": "Restore Lost Cards", + "restore-lost-cards-migration-description": "Finds and restores cards and lists with missing swimlaneId or listId. Creates a 'Lost Cards' swimlane to make all lost items visible again.", + "restore-all-archived-migration": "Restore All Archived", + "restore-all-archived-migration-description": "Restores all archived swimlanes, lists, and cards. Automatically fixes any missing swimlaneId or listId to make items visible.", + "fix-missing-lists-migration": "Fix Missing Lists", + "fix-missing-lists-migration-description": "Detects and repairs missing or corrupted lists in the board structure.", + "fix-avatar-urls-migration": "Fix Avatar URLs", + "fix-avatar-urls-migration-description": "Updates avatar URLs for board members to use the correct storage backend and fixes broken avatar references.", + "fix-all-file-urls-migration": "Fix All File URLs", + "fix-all-file-urls-migration-description": "Updates all file attachment URLs on this board to use the correct storage backend and fixes broken file references.", + "migration-needed": "Migration Needed", + "migration-complete": "Complete", + "migration-running": "Running...", + "migration-successful": "Migration completed successfully", + "migration-failed": "Migration failed", + "migrations": "Migrations", + "migrations-admin-only": "Only board administrators can run migrations", + "migrations-description": "Run data integrity checks and repairs for this board. Each migration can be executed individually.", + "no-issues-found": "No issues found", + "run-migration": "Run Migration", + "run-comprehensive-migration-confirm": "This will perform a comprehensive migration to check and fix board data integrity. This may take a few moments. Continue?", + "run-delete-duplicate-empty-lists-migration-confirm": "This will first convert any shared lists to per-swimlane lists, then delete empty lists that have a duplicate list with the same title containing cards. Only truly redundant empty lists will be removed. Continue?", + "run-restore-lost-cards-migration-confirm": "This will create a 'Lost Cards' swimlane and restore all cards and lists with missing swimlaneId or listId. This only affects non-archived items. Continue?", + "run-restore-all-archived-migration-confirm": "This will restore ALL archived swimlanes, lists, and cards, making them visible again. Any items with missing IDs will be automatically fixed. This cannot be easily undone. Continue?", + "run-fix-missing-lists-migration-confirm": "This will detect and repair missing or corrupted lists in the board structure. Continue?", + "run-fix-avatar-urls-migration-confirm": "This will update avatar URLs for board members to use the correct storage backend. Continue?", + "run-fix-all-file-urls-migration-confirm": "This will update all file attachment URLs on this board to use the correct storage backend. Continue?", + "restore-lost-cards-nothing-to-restore": "No lost swimlanes, lists, or cards to restore", + + "migration-progress-title": "Board Migration in Progress", + "migration-progress-overall": "Overall Progress", + "migration-progress-current-step": "Current Step", + "migration-progress-status": "Status", + "migration-progress-details": "Details", + "migration-progress-note": "Please wait while we migrate your board to the latest structure...", + + "step-analyze-board-structure": "Analyze Board Structure", + "step-fix-orphaned-cards": "Fix Orphaned Cards", + "step-convert-shared-lists": "Convert Shared Lists", + "step-ensure-per-swimlane-lists": "Ensure Per-Swimlane Lists", + "step-validate-migration": "Validate Migration", + "step-fix-avatar-urls": "Fix Avatar URLs", + "step-fix-attachment-urls": "Fix Attachment URLs", + "step-analyze-lists": "Analyze Lists", + "step-create-missing-lists": "Create Missing Lists", + "step-update-cards": "Update Cards", + "step-finalize": "Finalize", + "step-delete-duplicate-empty-lists": "Delete Duplicate Empty Lists", + "step-ensure-lost-cards-swimlane": "Ensure Lost Cards Swimlane", + "step-restore-lists": "Restore Lists", + "step-restore-cards": "Restore Cards", + "step-restore-swimlanes": "Restore Swimlanes", + "step-fix-missing-ids": "Fix Missing IDs", + "step-scan-users": "Checking board member avatars", + "step-scan-files": "Checking board file attachments", + "step-fix-file-urls": "Fixing file URLs", "cleanup": "Cleanup", "cleanup-old-jobs": "Cleanup Old Jobs", "completed": "Completed", diff --git a/imports/i18n/data/ro.i18n.json b/imports/i18n/data/ro.i18n.json index 5fbd58448..d41b0abaa 100644 --- a/imports/i18n/data/ro.i18n.json +++ b/imports/i18n/data/ro.i18n.json @@ -1404,7 +1404,70 @@ "back-to-settings": "Back to Settings", "board-id": "Board ID", "board-migration": "Board Migration", + "board-migrations": "Board Migrations", "card-show-lists-on-minicard": "Show Lists on Minicard", + "comprehensive-board-migration": "Comprehensive Board Migration", + "comprehensive-board-migration-description": "Performs comprehensive checks and fixes for board data integrity, including list ordering, card positions, and swimlane structure.", + "delete-duplicate-empty-lists-migration": "Delete Duplicate Empty Lists", + "delete-duplicate-empty-lists-migration-description": "Safely deletes empty duplicate lists. Only removes lists that have no cards AND have another list with the same title that contains cards.", + "lost-cards": "Lost Cards", + "lost-cards-list": "Restored Items", + "restore-lost-cards-migration": "Restore Lost Cards", + "restore-lost-cards-migration-description": "Finds and restores cards and lists with missing swimlaneId or listId. Creates a 'Lost Cards' swimlane to make all lost items visible again.", + "restore-all-archived-migration": "Restore All Archived", + "restore-all-archived-migration-description": "Restores all archived swimlanes, lists, and cards. Automatically fixes any missing swimlaneId or listId to make items visible.", + "fix-missing-lists-migration": "Fix Missing Lists", + "fix-missing-lists-migration-description": "Detects and repairs missing or corrupted lists in the board structure.", + "fix-avatar-urls-migration": "Fix Avatar URLs", + "fix-avatar-urls-migration-description": "Updates avatar URLs for board members to use the correct storage backend and fixes broken avatar references.", + "fix-all-file-urls-migration": "Fix All File URLs", + "fix-all-file-urls-migration-description": "Updates all file attachment URLs on this board to use the correct storage backend and fixes broken file references.", + "migration-needed": "Migration Needed", + "migration-complete": "Complete", + "migration-running": "Running...", + "migration-successful": "Migration completed successfully", + "migration-failed": "Migration failed", + "migrations": "Migrations", + "migrations-admin-only": "Only board administrators can run migrations", + "migrations-description": "Run data integrity checks and repairs for this board. Each migration can be executed individually.", + "no-issues-found": "No issues found", + "run-migration": "Run Migration", + "run-comprehensive-migration-confirm": "This will perform a comprehensive migration to check and fix board data integrity. This may take a few moments. Continue?", + "run-delete-duplicate-empty-lists-migration-confirm": "This will first convert any shared lists to per-swimlane lists, then delete empty lists that have a duplicate list with the same title containing cards. Only truly redundant empty lists will be removed. Continue?", + "run-restore-lost-cards-migration-confirm": "This will create a 'Lost Cards' swimlane and restore all cards and lists with missing swimlaneId or listId. This only affects non-archived items. Continue?", + "run-restore-all-archived-migration-confirm": "This will restore ALL archived swimlanes, lists, and cards, making them visible again. Any items with missing IDs will be automatically fixed. This cannot be easily undone. Continue?", + "run-fix-missing-lists-migration-confirm": "This will detect and repair missing or corrupted lists in the board structure. Continue?", + "run-fix-avatar-urls-migration-confirm": "This will update avatar URLs for board members to use the correct storage backend. Continue?", + "run-fix-all-file-urls-migration-confirm": "This will update all file attachment URLs on this board to use the correct storage backend. Continue?", + "restore-lost-cards-nothing-to-restore": "No lost swimlanes, lists, or cards to restore", + + "migration-progress-title": "Board Migration in Progress", + "migration-progress-overall": "Overall Progress", + "migration-progress-current-step": "Current Step", + "migration-progress-status": "Status", + "migration-progress-details": "Details", + "migration-progress-note": "Please wait while we migrate your board to the latest structure...", + + "step-analyze-board-structure": "Analyze Board Structure", + "step-fix-orphaned-cards": "Fix Orphaned Cards", + "step-convert-shared-lists": "Convert Shared Lists", + "step-ensure-per-swimlane-lists": "Ensure Per-Swimlane Lists", + "step-validate-migration": "Validate Migration", + "step-fix-avatar-urls": "Fix Avatar URLs", + "step-fix-attachment-urls": "Fix Attachment URLs", + "step-analyze-lists": "Analyze Lists", + "step-create-missing-lists": "Create Missing Lists", + "step-update-cards": "Update Cards", + "step-finalize": "Finalize", + "step-delete-duplicate-empty-lists": "Delete Duplicate Empty Lists", + "step-ensure-lost-cards-swimlane": "Ensure Lost Cards Swimlane", + "step-restore-lists": "Restore Lists", + "step-restore-cards": "Restore Cards", + "step-restore-swimlanes": "Restore Swimlanes", + "step-fix-missing-ids": "Fix Missing IDs", + "step-scan-users": "Checking board member avatars", + "step-scan-files": "Checking board file attachments", + "step-fix-file-urls": "Fixing file URLs", "cleanup": "Cleanup", "cleanup-old-jobs": "Cleanup Old Jobs", "completed": "Completed", diff --git a/imports/i18n/data/ru-UA.i18n.json b/imports/i18n/data/ru-UA.i18n.json index 055c85cdc..65a3eec81 100644 --- a/imports/i18n/data/ru-UA.i18n.json +++ b/imports/i18n/data/ru-UA.i18n.json @@ -1404,7 +1404,70 @@ "back-to-settings": "Back to Settings", "board-id": "Board ID", "board-migration": "Board Migration", + "board-migrations": "Board Migrations", "card-show-lists-on-minicard": "Show Lists on Minicard", + "comprehensive-board-migration": "Comprehensive Board Migration", + "comprehensive-board-migration-description": "Performs comprehensive checks and fixes for board data integrity, including list ordering, card positions, and swimlane structure.", + "delete-duplicate-empty-lists-migration": "Delete Duplicate Empty Lists", + "delete-duplicate-empty-lists-migration-description": "Safely deletes empty duplicate lists. Only removes lists that have no cards AND have another list with the same title that contains cards.", + "lost-cards": "Lost Cards", + "lost-cards-list": "Restored Items", + "restore-lost-cards-migration": "Restore Lost Cards", + "restore-lost-cards-migration-description": "Finds and restores cards and lists with missing swimlaneId or listId. Creates a 'Lost Cards' swimlane to make all lost items visible again.", + "restore-all-archived-migration": "Restore All Archived", + "restore-all-archived-migration-description": "Restores all archived swimlanes, lists, and cards. Automatically fixes any missing swimlaneId or listId to make items visible.", + "fix-missing-lists-migration": "Fix Missing Lists", + "fix-missing-lists-migration-description": "Detects and repairs missing or corrupted lists in the board structure.", + "fix-avatar-urls-migration": "Fix Avatar URLs", + "fix-avatar-urls-migration-description": "Updates avatar URLs for board members to use the correct storage backend and fixes broken avatar references.", + "fix-all-file-urls-migration": "Fix All File URLs", + "fix-all-file-urls-migration-description": "Updates all file attachment URLs on this board to use the correct storage backend and fixes broken file references.", + "migration-needed": "Migration Needed", + "migration-complete": "Complete", + "migration-running": "Running...", + "migration-successful": "Migration completed successfully", + "migration-failed": "Migration failed", + "migrations": "Migrations", + "migrations-admin-only": "Only board administrators can run migrations", + "migrations-description": "Run data integrity checks and repairs for this board. Each migration can be executed individually.", + "no-issues-found": "No issues found", + "run-migration": "Run Migration", + "run-comprehensive-migration-confirm": "This will perform a comprehensive migration to check and fix board data integrity. This may take a few moments. Continue?", + "run-delete-duplicate-empty-lists-migration-confirm": "This will first convert any shared lists to per-swimlane lists, then delete empty lists that have a duplicate list with the same title containing cards. Only truly redundant empty lists will be removed. Continue?", + "run-restore-lost-cards-migration-confirm": "This will create a 'Lost Cards' swimlane and restore all cards and lists with missing swimlaneId or listId. This only affects non-archived items. Continue?", + "run-restore-all-archived-migration-confirm": "This will restore ALL archived swimlanes, lists, and cards, making them visible again. Any items with missing IDs will be automatically fixed. This cannot be easily undone. Continue?", + "run-fix-missing-lists-migration-confirm": "This will detect and repair missing or corrupted lists in the board structure. Continue?", + "run-fix-avatar-urls-migration-confirm": "This will update avatar URLs for board members to use the correct storage backend. Continue?", + "run-fix-all-file-urls-migration-confirm": "This will update all file attachment URLs on this board to use the correct storage backend. Continue?", + "restore-lost-cards-nothing-to-restore": "No lost swimlanes, lists, or cards to restore", + + "migration-progress-title": "Board Migration in Progress", + "migration-progress-overall": "Overall Progress", + "migration-progress-current-step": "Current Step", + "migration-progress-status": "Статус", + "migration-progress-details": "Details", + "migration-progress-note": "Please wait while we migrate your board to the latest structure...", + + "step-analyze-board-structure": "Analyze Board Structure", + "step-fix-orphaned-cards": "Fix Orphaned Cards", + "step-convert-shared-lists": "Convert Shared Lists", + "step-ensure-per-swimlane-lists": "Ensure Per-Swimlane Lists", + "step-validate-migration": "Validate Migration", + "step-fix-avatar-urls": "Fix Avatar URLs", + "step-fix-attachment-urls": "Fix Attachment URLs", + "step-analyze-lists": "Analyze Lists", + "step-create-missing-lists": "Create Missing Lists", + "step-update-cards": "Update Cards", + "step-finalize": "Finalize", + "step-delete-duplicate-empty-lists": "Delete Duplicate Empty Lists", + "step-ensure-lost-cards-swimlane": "Ensure Lost Cards Swimlane", + "step-restore-lists": "Restore Lists", + "step-restore-cards": "Restore Cards", + "step-restore-swimlanes": "Restore Swimlanes", + "step-fix-missing-ids": "Fix Missing IDs", + "step-scan-users": "Checking board member avatars", + "step-scan-files": "Checking board file attachments", + "step-fix-file-urls": "Fixing file URLs", "cleanup": "Cleanup", "cleanup-old-jobs": "Cleanup Old Jobs", "completed": "Завершен", diff --git a/imports/i18n/data/ru.i18n.json b/imports/i18n/data/ru.i18n.json index 354f21a23..1b70576d7 100644 --- a/imports/i18n/data/ru.i18n.json +++ b/imports/i18n/data/ru.i18n.json @@ -1404,7 +1404,70 @@ "back-to-settings": "Back to Settings", "board-id": "Board ID", "board-migration": "Board Migration", + "board-migrations": "Board Migrations", "card-show-lists-on-minicard": "Show Lists on Minicard", + "comprehensive-board-migration": "Comprehensive Board Migration", + "comprehensive-board-migration-description": "Performs comprehensive checks and fixes for board data integrity, including list ordering, card positions, and swimlane structure.", + "delete-duplicate-empty-lists-migration": "Delete Duplicate Empty Lists", + "delete-duplicate-empty-lists-migration-description": "Safely deletes empty duplicate lists. Only removes lists that have no cards AND have another list with the same title that contains cards.", + "lost-cards": "Lost Cards", + "lost-cards-list": "Restored Items", + "restore-lost-cards-migration": "Restore Lost Cards", + "restore-lost-cards-migration-description": "Finds and restores cards and lists with missing swimlaneId or listId. Creates a 'Lost Cards' swimlane to make all lost items visible again.", + "restore-all-archived-migration": "Restore All Archived", + "restore-all-archived-migration-description": "Restores all archived swimlanes, lists, and cards. Automatically fixes any missing swimlaneId or listId to make items visible.", + "fix-missing-lists-migration": "Fix Missing Lists", + "fix-missing-lists-migration-description": "Detects and repairs missing or corrupted lists in the board structure.", + "fix-avatar-urls-migration": "Fix Avatar URLs", + "fix-avatar-urls-migration-description": "Updates avatar URLs for board members to use the correct storage backend and fixes broken avatar references.", + "fix-all-file-urls-migration": "Fix All File URLs", + "fix-all-file-urls-migration-description": "Updates all file attachment URLs on this board to use the correct storage backend and fixes broken file references.", + "migration-needed": "Migration Needed", + "migration-complete": "Завершено", + "migration-running": "Running...", + "migration-successful": "Migration completed successfully", + "migration-failed": "Migration failed", + "migrations": "Migrations", + "migrations-admin-only": "Only board administrators can run migrations", + "migrations-description": "Run data integrity checks and repairs for this board. Each migration can be executed individually.", + "no-issues-found": "No issues found", + "run-migration": "Run Migration", + "run-comprehensive-migration-confirm": "This will perform a comprehensive migration to check and fix board data integrity. This may take a few moments. Continue?", + "run-delete-duplicate-empty-lists-migration-confirm": "This will first convert any shared lists to per-swimlane lists, then delete empty lists that have a duplicate list with the same title containing cards. Only truly redundant empty lists will be removed. Continue?", + "run-restore-lost-cards-migration-confirm": "This will create a 'Lost Cards' swimlane and restore all cards and lists with missing swimlaneId or listId. This only affects non-archived items. Continue?", + "run-restore-all-archived-migration-confirm": "This will restore ALL archived swimlanes, lists, and cards, making them visible again. Any items with missing IDs will be automatically fixed. This cannot be easily undone. Continue?", + "run-fix-missing-lists-migration-confirm": "This will detect and repair missing or corrupted lists in the board structure. Continue?", + "run-fix-avatar-urls-migration-confirm": "This will update avatar URLs for board members to use the correct storage backend. Continue?", + "run-fix-all-file-urls-migration-confirm": "This will update all file attachment URLs on this board to use the correct storage backend. Continue?", + "restore-lost-cards-nothing-to-restore": "No lost swimlanes, lists, or cards to restore", + + "migration-progress-title": "Board Migration in Progress", + "migration-progress-overall": "Overall Progress", + "migration-progress-current-step": "Current Step", + "migration-progress-status": "Статус", + "migration-progress-details": "Детали", + "migration-progress-note": "Please wait while we migrate your board to the latest structure...", + + "step-analyze-board-structure": "Analyze Board Structure", + "step-fix-orphaned-cards": "Fix Orphaned Cards", + "step-convert-shared-lists": "Convert Shared Lists", + "step-ensure-per-swimlane-lists": "Ensure Per-Swimlane Lists", + "step-validate-migration": "Validate Migration", + "step-fix-avatar-urls": "Fix Avatar URLs", + "step-fix-attachment-urls": "Fix Attachment URLs", + "step-analyze-lists": "Analyze Lists", + "step-create-missing-lists": "Create Missing Lists", + "step-update-cards": "Update Cards", + "step-finalize": "Finalize", + "step-delete-duplicate-empty-lists": "Delete Duplicate Empty Lists", + "step-ensure-lost-cards-swimlane": "Ensure Lost Cards Swimlane", + "step-restore-lists": "Restore Lists", + "step-restore-cards": "Restore Cards", + "step-restore-swimlanes": "Restore Swimlanes", + "step-fix-missing-ids": "Fix Missing IDs", + "step-scan-users": "Checking board member avatars", + "step-scan-files": "Checking board file attachments", + "step-fix-file-urls": "Fixing file URLs", "cleanup": "Cleanup", "cleanup-old-jobs": "Cleanup Old Jobs", "completed": "Завершен", diff --git a/imports/i18n/data/sk.i18n.json b/imports/i18n/data/sk.i18n.json index 4eff2776c..6285db533 100644 --- a/imports/i18n/data/sk.i18n.json +++ b/imports/i18n/data/sk.i18n.json @@ -1404,7 +1404,70 @@ "back-to-settings": "Back to Settings", "board-id": "Board ID", "board-migration": "Board Migration", + "board-migrations": "Board Migrations", "card-show-lists-on-minicard": "Show Lists on Minicard", + "comprehensive-board-migration": "Comprehensive Board Migration", + "comprehensive-board-migration-description": "Performs comprehensive checks and fixes for board data integrity, including list ordering, card positions, and swimlane structure.", + "delete-duplicate-empty-lists-migration": "Delete Duplicate Empty Lists", + "delete-duplicate-empty-lists-migration-description": "Safely deletes empty duplicate lists. Only removes lists that have no cards AND have another list with the same title that contains cards.", + "lost-cards": "Lost Cards", + "lost-cards-list": "Restored Items", + "restore-lost-cards-migration": "Restore Lost Cards", + "restore-lost-cards-migration-description": "Finds and restores cards and lists with missing swimlaneId or listId. Creates a 'Lost Cards' swimlane to make all lost items visible again.", + "restore-all-archived-migration": "Restore All Archived", + "restore-all-archived-migration-description": "Restores all archived swimlanes, lists, and cards. Automatically fixes any missing swimlaneId or listId to make items visible.", + "fix-missing-lists-migration": "Fix Missing Lists", + "fix-missing-lists-migration-description": "Detects and repairs missing or corrupted lists in the board structure.", + "fix-avatar-urls-migration": "Fix Avatar URLs", + "fix-avatar-urls-migration-description": "Updates avatar URLs for board members to use the correct storage backend and fixes broken avatar references.", + "fix-all-file-urls-migration": "Fix All File URLs", + "fix-all-file-urls-migration-description": "Updates all file attachment URLs on this board to use the correct storage backend and fixes broken file references.", + "migration-needed": "Migration Needed", + "migration-complete": "Complete", + "migration-running": "Running...", + "migration-successful": "Migration completed successfully", + "migration-failed": "Migration failed", + "migrations": "Migrations", + "migrations-admin-only": "Only board administrators can run migrations", + "migrations-description": "Run data integrity checks and repairs for this board. Each migration can be executed individually.", + "no-issues-found": "No issues found", + "run-migration": "Run Migration", + "run-comprehensive-migration-confirm": "This will perform a comprehensive migration to check and fix board data integrity. This may take a few moments. Continue?", + "run-delete-duplicate-empty-lists-migration-confirm": "This will first convert any shared lists to per-swimlane lists, then delete empty lists that have a duplicate list with the same title containing cards. Only truly redundant empty lists will be removed. Continue?", + "run-restore-lost-cards-migration-confirm": "This will create a 'Lost Cards' swimlane and restore all cards and lists with missing swimlaneId or listId. This only affects non-archived items. Continue?", + "run-restore-all-archived-migration-confirm": "This will restore ALL archived swimlanes, lists, and cards, making them visible again. Any items with missing IDs will be automatically fixed. This cannot be easily undone. Continue?", + "run-fix-missing-lists-migration-confirm": "This will detect and repair missing or corrupted lists in the board structure. Continue?", + "run-fix-avatar-urls-migration-confirm": "This will update avatar URLs for board members to use the correct storage backend. Continue?", + "run-fix-all-file-urls-migration-confirm": "This will update all file attachment URLs on this board to use the correct storage backend. Continue?", + "restore-lost-cards-nothing-to-restore": "No lost swimlanes, lists, or cards to restore", + + "migration-progress-title": "Board Migration in Progress", + "migration-progress-overall": "Overall Progress", + "migration-progress-current-step": "Current Step", + "migration-progress-status": "Status", + "migration-progress-details": "Details", + "migration-progress-note": "Please wait while we migrate your board to the latest structure...", + + "step-analyze-board-structure": "Analyze Board Structure", + "step-fix-orphaned-cards": "Fix Orphaned Cards", + "step-convert-shared-lists": "Convert Shared Lists", + "step-ensure-per-swimlane-lists": "Ensure Per-Swimlane Lists", + "step-validate-migration": "Validate Migration", + "step-fix-avatar-urls": "Fix Avatar URLs", + "step-fix-attachment-urls": "Fix Attachment URLs", + "step-analyze-lists": "Analyze Lists", + "step-create-missing-lists": "Create Missing Lists", + "step-update-cards": "Update Cards", + "step-finalize": "Finalize", + "step-delete-duplicate-empty-lists": "Delete Duplicate Empty Lists", + "step-ensure-lost-cards-swimlane": "Ensure Lost Cards Swimlane", + "step-restore-lists": "Restore Lists", + "step-restore-cards": "Restore Cards", + "step-restore-swimlanes": "Restore Swimlanes", + "step-fix-missing-ids": "Fix Missing IDs", + "step-scan-users": "Checking board member avatars", + "step-scan-files": "Checking board file attachments", + "step-fix-file-urls": "Fixing file URLs", "cleanup": "Cleanup", "cleanup-old-jobs": "Cleanup Old Jobs", "completed": "Completed", diff --git a/imports/i18n/data/sl.i18n.json b/imports/i18n/data/sl.i18n.json index 7c7f0d684..b9a1b7522 100644 --- a/imports/i18n/data/sl.i18n.json +++ b/imports/i18n/data/sl.i18n.json @@ -1404,7 +1404,70 @@ "back-to-settings": "Back to Settings", "board-id": "Board ID", "board-migration": "Board Migration", + "board-migrations": "Board Migrations", "card-show-lists-on-minicard": "Show Lists on Minicard", + "comprehensive-board-migration": "Comprehensive Board Migration", + "comprehensive-board-migration-description": "Performs comprehensive checks and fixes for board data integrity, including list ordering, card positions, and swimlane structure.", + "delete-duplicate-empty-lists-migration": "Delete Duplicate Empty Lists", + "delete-duplicate-empty-lists-migration-description": "Safely deletes empty duplicate lists. Only removes lists that have no cards AND have another list with the same title that contains cards.", + "lost-cards": "Lost Cards", + "lost-cards-list": "Restored Items", + "restore-lost-cards-migration": "Restore Lost Cards", + "restore-lost-cards-migration-description": "Finds and restores cards and lists with missing swimlaneId or listId. Creates a 'Lost Cards' swimlane to make all lost items visible again.", + "restore-all-archived-migration": "Restore All Archived", + "restore-all-archived-migration-description": "Restores all archived swimlanes, lists, and cards. Automatically fixes any missing swimlaneId or listId to make items visible.", + "fix-missing-lists-migration": "Fix Missing Lists", + "fix-missing-lists-migration-description": "Detects and repairs missing or corrupted lists in the board structure.", + "fix-avatar-urls-migration": "Fix Avatar URLs", + "fix-avatar-urls-migration-description": "Updates avatar URLs for board members to use the correct storage backend and fixes broken avatar references.", + "fix-all-file-urls-migration": "Fix All File URLs", + "fix-all-file-urls-migration-description": "Updates all file attachment URLs on this board to use the correct storage backend and fixes broken file references.", + "migration-needed": "Migration Needed", + "migration-complete": "Complete", + "migration-running": "Running...", + "migration-successful": "Migration completed successfully", + "migration-failed": "Migration failed", + "migrations": "Migrations", + "migrations-admin-only": "Only board administrators can run migrations", + "migrations-description": "Run data integrity checks and repairs for this board. Each migration can be executed individually.", + "no-issues-found": "No issues found", + "run-migration": "Run Migration", + "run-comprehensive-migration-confirm": "This will perform a comprehensive migration to check and fix board data integrity. This may take a few moments. Continue?", + "run-delete-duplicate-empty-lists-migration-confirm": "This will first convert any shared lists to per-swimlane lists, then delete empty lists that have a duplicate list with the same title containing cards. Only truly redundant empty lists will be removed. Continue?", + "run-restore-lost-cards-migration-confirm": "This will create a 'Lost Cards' swimlane and restore all cards and lists with missing swimlaneId or listId. This only affects non-archived items. Continue?", + "run-restore-all-archived-migration-confirm": "This will restore ALL archived swimlanes, lists, and cards, making them visible again. Any items with missing IDs will be automatically fixed. This cannot be easily undone. Continue?", + "run-fix-missing-lists-migration-confirm": "This will detect and repair missing or corrupted lists in the board structure. Continue?", + "run-fix-avatar-urls-migration-confirm": "This will update avatar URLs for board members to use the correct storage backend. Continue?", + "run-fix-all-file-urls-migration-confirm": "This will update all file attachment URLs on this board to use the correct storage backend. Continue?", + "restore-lost-cards-nothing-to-restore": "No lost swimlanes, lists, or cards to restore", + + "migration-progress-title": "Board Migration in Progress", + "migration-progress-overall": "Overall Progress", + "migration-progress-current-step": "Current Step", + "migration-progress-status": "Status", + "migration-progress-details": "Details", + "migration-progress-note": "Please wait while we migrate your board to the latest structure...", + + "step-analyze-board-structure": "Analyze Board Structure", + "step-fix-orphaned-cards": "Fix Orphaned Cards", + "step-convert-shared-lists": "Convert Shared Lists", + "step-ensure-per-swimlane-lists": "Ensure Per-Swimlane Lists", + "step-validate-migration": "Validate Migration", + "step-fix-avatar-urls": "Fix Avatar URLs", + "step-fix-attachment-urls": "Fix Attachment URLs", + "step-analyze-lists": "Analyze Lists", + "step-create-missing-lists": "Create Missing Lists", + "step-update-cards": "Update Cards", + "step-finalize": "Finalize", + "step-delete-duplicate-empty-lists": "Delete Duplicate Empty Lists", + "step-ensure-lost-cards-swimlane": "Ensure Lost Cards Swimlane", + "step-restore-lists": "Restore Lists", + "step-restore-cards": "Restore Cards", + "step-restore-swimlanes": "Restore Swimlanes", + "step-fix-missing-ids": "Fix Missing IDs", + "step-scan-users": "Checking board member avatars", + "step-scan-files": "Checking board file attachments", + "step-fix-file-urls": "Fixing file URLs", "cleanup": "Cleanup", "cleanup-old-jobs": "Cleanup Old Jobs", "completed": "zaključen", diff --git a/imports/i18n/data/sr.i18n.json b/imports/i18n/data/sr.i18n.json index 2022488c1..302397003 100644 --- a/imports/i18n/data/sr.i18n.json +++ b/imports/i18n/data/sr.i18n.json @@ -1404,7 +1404,70 @@ "back-to-settings": "Back to Settings", "board-id": "Board ID", "board-migration": "Board Migration", + "board-migrations": "Board Migrations", "card-show-lists-on-minicard": "Show Lists on Minicard", + "comprehensive-board-migration": "Comprehensive Board Migration", + "comprehensive-board-migration-description": "Performs comprehensive checks and fixes for board data integrity, including list ordering, card positions, and swimlane structure.", + "delete-duplicate-empty-lists-migration": "Delete Duplicate Empty Lists", + "delete-duplicate-empty-lists-migration-description": "Safely deletes empty duplicate lists. Only removes lists that have no cards AND have another list with the same title that contains cards.", + "lost-cards": "Lost Cards", + "lost-cards-list": "Restored Items", + "restore-lost-cards-migration": "Restore Lost Cards", + "restore-lost-cards-migration-description": "Finds and restores cards and lists with missing swimlaneId or listId. Creates a 'Lost Cards' swimlane to make all lost items visible again.", + "restore-all-archived-migration": "Restore All Archived", + "restore-all-archived-migration-description": "Restores all archived swimlanes, lists, and cards. Automatically fixes any missing swimlaneId or listId to make items visible.", + "fix-missing-lists-migration": "Fix Missing Lists", + "fix-missing-lists-migration-description": "Detects and repairs missing or corrupted lists in the board structure.", + "fix-avatar-urls-migration": "Fix Avatar URLs", + "fix-avatar-urls-migration-description": "Updates avatar URLs for board members to use the correct storage backend and fixes broken avatar references.", + "fix-all-file-urls-migration": "Fix All File URLs", + "fix-all-file-urls-migration-description": "Updates all file attachment URLs on this board to use the correct storage backend and fixes broken file references.", + "migration-needed": "Migration Needed", + "migration-complete": "Complete", + "migration-running": "Running...", + "migration-successful": "Migration completed successfully", + "migration-failed": "Migration failed", + "migrations": "Migrations", + "migrations-admin-only": "Only board administrators can run migrations", + "migrations-description": "Run data integrity checks and repairs for this board. Each migration can be executed individually.", + "no-issues-found": "No issues found", + "run-migration": "Run Migration", + "run-comprehensive-migration-confirm": "This will perform a comprehensive migration to check and fix board data integrity. This may take a few moments. Continue?", + "run-delete-duplicate-empty-lists-migration-confirm": "This will first convert any shared lists to per-swimlane lists, then delete empty lists that have a duplicate list with the same title containing cards. Only truly redundant empty lists will be removed. Continue?", + "run-restore-lost-cards-migration-confirm": "This will create a 'Lost Cards' swimlane and restore all cards and lists with missing swimlaneId or listId. This only affects non-archived items. Continue?", + "run-restore-all-archived-migration-confirm": "This will restore ALL archived swimlanes, lists, and cards, making them visible again. Any items with missing IDs will be automatically fixed. This cannot be easily undone. Continue?", + "run-fix-missing-lists-migration-confirm": "This will detect and repair missing or corrupted lists in the board structure. Continue?", + "run-fix-avatar-urls-migration-confirm": "This will update avatar URLs for board members to use the correct storage backend. Continue?", + "run-fix-all-file-urls-migration-confirm": "This will update all file attachment URLs on this board to use the correct storage backend. Continue?", + "restore-lost-cards-nothing-to-restore": "No lost swimlanes, lists, or cards to restore", + + "migration-progress-title": "Board Migration in Progress", + "migration-progress-overall": "Overall Progress", + "migration-progress-current-step": "Current Step", + "migration-progress-status": "Стање", + "migration-progress-details": "Детаљи", + "migration-progress-note": "Please wait while we migrate your board to the latest structure...", + + "step-analyze-board-structure": "Analyze Board Structure", + "step-fix-orphaned-cards": "Fix Orphaned Cards", + "step-convert-shared-lists": "Convert Shared Lists", + "step-ensure-per-swimlane-lists": "Ensure Per-Swimlane Lists", + "step-validate-migration": "Validate Migration", + "step-fix-avatar-urls": "Fix Avatar URLs", + "step-fix-attachment-urls": "Fix Attachment URLs", + "step-analyze-lists": "Analyze Lists", + "step-create-missing-lists": "Create Missing Lists", + "step-update-cards": "Update Cards", + "step-finalize": "Finalize", + "step-delete-duplicate-empty-lists": "Delete Duplicate Empty Lists", + "step-ensure-lost-cards-swimlane": "Ensure Lost Cards Swimlane", + "step-restore-lists": "Restore Lists", + "step-restore-cards": "Restore Cards", + "step-restore-swimlanes": "Restore Swimlanes", + "step-fix-missing-ids": "Fix Missing IDs", + "step-scan-users": "Checking board member avatars", + "step-scan-files": "Checking board file attachments", + "step-fix-file-urls": "Fixing file URLs", "cleanup": "Cleanup", "cleanup-old-jobs": "Cleanup Old Jobs", "completed": "Обављен", diff --git a/imports/i18n/data/sv.i18n.json b/imports/i18n/data/sv.i18n.json index 2297e547b..177a6c147 100644 --- a/imports/i18n/data/sv.i18n.json +++ b/imports/i18n/data/sv.i18n.json @@ -1404,7 +1404,70 @@ "back-to-settings": "Back to Settings", "board-id": "Board ID", "board-migration": "Board Migration", + "board-migrations": "Board Migrations", "card-show-lists-on-minicard": "Show Lists on Minicard", + "comprehensive-board-migration": "Comprehensive Board Migration", + "comprehensive-board-migration-description": "Performs comprehensive checks and fixes for board data integrity, including list ordering, card positions, and swimlane structure.", + "delete-duplicate-empty-lists-migration": "Delete Duplicate Empty Lists", + "delete-duplicate-empty-lists-migration-description": "Safely deletes empty duplicate lists. Only removes lists that have no cards AND have another list with the same title that contains cards.", + "lost-cards": "Lost Cards", + "lost-cards-list": "Restored Items", + "restore-lost-cards-migration": "Restore Lost Cards", + "restore-lost-cards-migration-description": "Finds and restores cards and lists with missing swimlaneId or listId. Creates a 'Lost Cards' swimlane to make all lost items visible again.", + "restore-all-archived-migration": "Restore All Archived", + "restore-all-archived-migration-description": "Restores all archived swimlanes, lists, and cards. Automatically fixes any missing swimlaneId or listId to make items visible.", + "fix-missing-lists-migration": "Fix Missing Lists", + "fix-missing-lists-migration-description": "Detects and repairs missing or corrupted lists in the board structure.", + "fix-avatar-urls-migration": "Fix Avatar URLs", + "fix-avatar-urls-migration-description": "Updates avatar URLs for board members to use the correct storage backend and fixes broken avatar references.", + "fix-all-file-urls-migration": "Fix All File URLs", + "fix-all-file-urls-migration-description": "Updates all file attachment URLs on this board to use the correct storage backend and fixes broken file references.", + "migration-needed": "Migration Needed", + "migration-complete": "Complete", + "migration-running": "Running...", + "migration-successful": "Migration completed successfully", + "migration-failed": "Migration failed", + "migrations": "Migrations", + "migrations-admin-only": "Only board administrators can run migrations", + "migrations-description": "Run data integrity checks and repairs for this board. Each migration can be executed individually.", + "no-issues-found": "No issues found", + "run-migration": "Run Migration", + "run-comprehensive-migration-confirm": "This will perform a comprehensive migration to check and fix board data integrity. This may take a few moments. Continue?", + "run-delete-duplicate-empty-lists-migration-confirm": "This will first convert any shared lists to per-swimlane lists, then delete empty lists that have a duplicate list with the same title containing cards. Only truly redundant empty lists will be removed. Continue?", + "run-restore-lost-cards-migration-confirm": "This will create a 'Lost Cards' swimlane and restore all cards and lists with missing swimlaneId or listId. This only affects non-archived items. Continue?", + "run-restore-all-archived-migration-confirm": "This will restore ALL archived swimlanes, lists, and cards, making them visible again. Any items with missing IDs will be automatically fixed. This cannot be easily undone. Continue?", + "run-fix-missing-lists-migration-confirm": "This will detect and repair missing or corrupted lists in the board structure. Continue?", + "run-fix-avatar-urls-migration-confirm": "This will update avatar URLs for board members to use the correct storage backend. Continue?", + "run-fix-all-file-urls-migration-confirm": "This will update all file attachment URLs on this board to use the correct storage backend. Continue?", + "restore-lost-cards-nothing-to-restore": "No lost swimlanes, lists, or cards to restore", + + "migration-progress-title": "Board Migration in Progress", + "migration-progress-overall": "Overall Progress", + "migration-progress-current-step": "Current Step", + "migration-progress-status": "Status", + "migration-progress-details": "Detaljer", + "migration-progress-note": "Please wait while we migrate your board to the latest structure...", + + "step-analyze-board-structure": "Analyze Board Structure", + "step-fix-orphaned-cards": "Fix Orphaned Cards", + "step-convert-shared-lists": "Convert Shared Lists", + "step-ensure-per-swimlane-lists": "Ensure Per-Swimlane Lists", + "step-validate-migration": "Validate Migration", + "step-fix-avatar-urls": "Fix Avatar URLs", + "step-fix-attachment-urls": "Fix Attachment URLs", + "step-analyze-lists": "Analyze Lists", + "step-create-missing-lists": "Create Missing Lists", + "step-update-cards": "Update Cards", + "step-finalize": "Finalize", + "step-delete-duplicate-empty-lists": "Delete Duplicate Empty Lists", + "step-ensure-lost-cards-swimlane": "Ensure Lost Cards Swimlane", + "step-restore-lists": "Restore Lists", + "step-restore-cards": "Restore Cards", + "step-restore-swimlanes": "Restore Swimlanes", + "step-fix-missing-ids": "Fix Missing IDs", + "step-scan-users": "Checking board member avatars", + "step-scan-files": "Checking board file attachments", + "step-fix-file-urls": "Fixing file URLs", "cleanup": "Cleanup", "cleanup-old-jobs": "Cleanup Old Jobs", "completed": "Avslutad", diff --git a/imports/i18n/data/sw.i18n.json b/imports/i18n/data/sw.i18n.json index ae5c41a0d..c5ea981db 100644 --- a/imports/i18n/data/sw.i18n.json +++ b/imports/i18n/data/sw.i18n.json @@ -1404,7 +1404,70 @@ "back-to-settings": "Back to Settings", "board-id": "Board ID", "board-migration": "Board Migration", + "board-migrations": "Board Migrations", "card-show-lists-on-minicard": "Show Lists on Minicard", + "comprehensive-board-migration": "Comprehensive Board Migration", + "comprehensive-board-migration-description": "Performs comprehensive checks and fixes for board data integrity, including list ordering, card positions, and swimlane structure.", + "delete-duplicate-empty-lists-migration": "Delete Duplicate Empty Lists", + "delete-duplicate-empty-lists-migration-description": "Safely deletes empty duplicate lists. Only removes lists that have no cards AND have another list with the same title that contains cards.", + "lost-cards": "Lost Cards", + "lost-cards-list": "Restored Items", + "restore-lost-cards-migration": "Restore Lost Cards", + "restore-lost-cards-migration-description": "Finds and restores cards and lists with missing swimlaneId or listId. Creates a 'Lost Cards' swimlane to make all lost items visible again.", + "restore-all-archived-migration": "Restore All Archived", + "restore-all-archived-migration-description": "Restores all archived swimlanes, lists, and cards. Automatically fixes any missing swimlaneId or listId to make items visible.", + "fix-missing-lists-migration": "Fix Missing Lists", + "fix-missing-lists-migration-description": "Detects and repairs missing or corrupted lists in the board structure.", + "fix-avatar-urls-migration": "Fix Avatar URLs", + "fix-avatar-urls-migration-description": "Updates avatar URLs for board members to use the correct storage backend and fixes broken avatar references.", + "fix-all-file-urls-migration": "Fix All File URLs", + "fix-all-file-urls-migration-description": "Updates all file attachment URLs on this board to use the correct storage backend and fixes broken file references.", + "migration-needed": "Migration Needed", + "migration-complete": "Complete", + "migration-running": "Running...", + "migration-successful": "Migration completed successfully", + "migration-failed": "Migration failed", + "migrations": "Migrations", + "migrations-admin-only": "Only board administrators can run migrations", + "migrations-description": "Run data integrity checks and repairs for this board. Each migration can be executed individually.", + "no-issues-found": "No issues found", + "run-migration": "Run Migration", + "run-comprehensive-migration-confirm": "This will perform a comprehensive migration to check and fix board data integrity. This may take a few moments. Continue?", + "run-delete-duplicate-empty-lists-migration-confirm": "This will first convert any shared lists to per-swimlane lists, then delete empty lists that have a duplicate list with the same title containing cards. Only truly redundant empty lists will be removed. Continue?", + "run-restore-lost-cards-migration-confirm": "This will create a 'Lost Cards' swimlane and restore all cards and lists with missing swimlaneId or listId. This only affects non-archived items. Continue?", + "run-restore-all-archived-migration-confirm": "This will restore ALL archived swimlanes, lists, and cards, making them visible again. Any items with missing IDs will be automatically fixed. This cannot be easily undone. Continue?", + "run-fix-missing-lists-migration-confirm": "This will detect and repair missing or corrupted lists in the board structure. Continue?", + "run-fix-avatar-urls-migration-confirm": "This will update avatar URLs for board members to use the correct storage backend. Continue?", + "run-fix-all-file-urls-migration-confirm": "This will update all file attachment URLs on this board to use the correct storage backend. Continue?", + "restore-lost-cards-nothing-to-restore": "No lost swimlanes, lists, or cards to restore", + + "migration-progress-title": "Board Migration in Progress", + "migration-progress-overall": "Overall Progress", + "migration-progress-current-step": "Current Step", + "migration-progress-status": "Status", + "migration-progress-details": "Details", + "migration-progress-note": "Please wait while we migrate your board to the latest structure...", + + "step-analyze-board-structure": "Analyze Board Structure", + "step-fix-orphaned-cards": "Fix Orphaned Cards", + "step-convert-shared-lists": "Convert Shared Lists", + "step-ensure-per-swimlane-lists": "Ensure Per-Swimlane Lists", + "step-validate-migration": "Validate Migration", + "step-fix-avatar-urls": "Fix Avatar URLs", + "step-fix-attachment-urls": "Fix Attachment URLs", + "step-analyze-lists": "Analyze Lists", + "step-create-missing-lists": "Create Missing Lists", + "step-update-cards": "Update Cards", + "step-finalize": "Finalize", + "step-delete-duplicate-empty-lists": "Delete Duplicate Empty Lists", + "step-ensure-lost-cards-swimlane": "Ensure Lost Cards Swimlane", + "step-restore-lists": "Restore Lists", + "step-restore-cards": "Restore Cards", + "step-restore-swimlanes": "Restore Swimlanes", + "step-fix-missing-ids": "Fix Missing IDs", + "step-scan-users": "Checking board member avatars", + "step-scan-files": "Checking board file attachments", + "step-fix-file-urls": "Fixing file URLs", "cleanup": "Cleanup", "cleanup-old-jobs": "Cleanup Old Jobs", "completed": "Completed", diff --git a/imports/i18n/data/ta.i18n.json b/imports/i18n/data/ta.i18n.json index c2e38c9eb..2c7332730 100644 --- a/imports/i18n/data/ta.i18n.json +++ b/imports/i18n/data/ta.i18n.json @@ -1404,7 +1404,70 @@ "back-to-settings": "Back to Settings", "board-id": "Board ID", "board-migration": "Board Migration", + "board-migrations": "Board Migrations", "card-show-lists-on-minicard": "Show Lists on Minicard", + "comprehensive-board-migration": "Comprehensive Board Migration", + "comprehensive-board-migration-description": "Performs comprehensive checks and fixes for board data integrity, including list ordering, card positions, and swimlane structure.", + "delete-duplicate-empty-lists-migration": "Delete Duplicate Empty Lists", + "delete-duplicate-empty-lists-migration-description": "Safely deletes empty duplicate lists. Only removes lists that have no cards AND have another list with the same title that contains cards.", + "lost-cards": "Lost Cards", + "lost-cards-list": "Restored Items", + "restore-lost-cards-migration": "Restore Lost Cards", + "restore-lost-cards-migration-description": "Finds and restores cards and lists with missing swimlaneId or listId. Creates a 'Lost Cards' swimlane to make all lost items visible again.", + "restore-all-archived-migration": "Restore All Archived", + "restore-all-archived-migration-description": "Restores all archived swimlanes, lists, and cards. Automatically fixes any missing swimlaneId or listId to make items visible.", + "fix-missing-lists-migration": "Fix Missing Lists", + "fix-missing-lists-migration-description": "Detects and repairs missing or corrupted lists in the board structure.", + "fix-avatar-urls-migration": "Fix Avatar URLs", + "fix-avatar-urls-migration-description": "Updates avatar URLs for board members to use the correct storage backend and fixes broken avatar references.", + "fix-all-file-urls-migration": "Fix All File URLs", + "fix-all-file-urls-migration-description": "Updates all file attachment URLs on this board to use the correct storage backend and fixes broken file references.", + "migration-needed": "Migration Needed", + "migration-complete": "Complete", + "migration-running": "Running...", + "migration-successful": "Migration completed successfully", + "migration-failed": "Migration failed", + "migrations": "Migrations", + "migrations-admin-only": "Only board administrators can run migrations", + "migrations-description": "Run data integrity checks and repairs for this board. Each migration can be executed individually.", + "no-issues-found": "No issues found", + "run-migration": "Run Migration", + "run-comprehensive-migration-confirm": "This will perform a comprehensive migration to check and fix board data integrity. This may take a few moments. Continue?", + "run-delete-duplicate-empty-lists-migration-confirm": "This will first convert any shared lists to per-swimlane lists, then delete empty lists that have a duplicate list with the same title containing cards. Only truly redundant empty lists will be removed. Continue?", + "run-restore-lost-cards-migration-confirm": "This will create a 'Lost Cards' swimlane and restore all cards and lists with missing swimlaneId or listId. This only affects non-archived items. Continue?", + "run-restore-all-archived-migration-confirm": "This will restore ALL archived swimlanes, lists, and cards, making them visible again. Any items with missing IDs will be automatically fixed. This cannot be easily undone. Continue?", + "run-fix-missing-lists-migration-confirm": "This will detect and repair missing or corrupted lists in the board structure. Continue?", + "run-fix-avatar-urls-migration-confirm": "This will update avatar URLs for board members to use the correct storage backend. Continue?", + "run-fix-all-file-urls-migration-confirm": "This will update all file attachment URLs on this board to use the correct storage backend. Continue?", + "restore-lost-cards-nothing-to-restore": "No lost swimlanes, lists, or cards to restore", + + "migration-progress-title": "Board Migration in Progress", + "migration-progress-overall": "Overall Progress", + "migration-progress-current-step": "Current Step", + "migration-progress-status": "Status", + "migration-progress-details": "Details", + "migration-progress-note": "Please wait while we migrate your board to the latest structure...", + + "step-analyze-board-structure": "Analyze Board Structure", + "step-fix-orphaned-cards": "Fix Orphaned Cards", + "step-convert-shared-lists": "Convert Shared Lists", + "step-ensure-per-swimlane-lists": "Ensure Per-Swimlane Lists", + "step-validate-migration": "Validate Migration", + "step-fix-avatar-urls": "Fix Avatar URLs", + "step-fix-attachment-urls": "Fix Attachment URLs", + "step-analyze-lists": "Analyze Lists", + "step-create-missing-lists": "Create Missing Lists", + "step-update-cards": "Update Cards", + "step-finalize": "Finalize", + "step-delete-duplicate-empty-lists": "Delete Duplicate Empty Lists", + "step-ensure-lost-cards-swimlane": "Ensure Lost Cards Swimlane", + "step-restore-lists": "Restore Lists", + "step-restore-cards": "Restore Cards", + "step-restore-swimlanes": "Restore Swimlanes", + "step-fix-missing-ids": "Fix Missing IDs", + "step-scan-users": "Checking board member avatars", + "step-scan-files": "Checking board file attachments", + "step-fix-file-urls": "Fixing file URLs", "cleanup": "Cleanup", "cleanup-old-jobs": "Cleanup Old Jobs", "completed": "Completed", diff --git a/imports/i18n/data/te-IN.i18n.json b/imports/i18n/data/te-IN.i18n.json index eef27e1fe..958d8313a 100644 --- a/imports/i18n/data/te-IN.i18n.json +++ b/imports/i18n/data/te-IN.i18n.json @@ -1404,7 +1404,70 @@ "back-to-settings": "Back to Settings", "board-id": "Board ID", "board-migration": "Board Migration", + "board-migrations": "Board Migrations", "card-show-lists-on-minicard": "Show Lists on Minicard", + "comprehensive-board-migration": "Comprehensive Board Migration", + "comprehensive-board-migration-description": "Performs comprehensive checks and fixes for board data integrity, including list ordering, card positions, and swimlane structure.", + "delete-duplicate-empty-lists-migration": "Delete Duplicate Empty Lists", + "delete-duplicate-empty-lists-migration-description": "Safely deletes empty duplicate lists. Only removes lists that have no cards AND have another list with the same title that contains cards.", + "lost-cards": "Lost Cards", + "lost-cards-list": "Restored Items", + "restore-lost-cards-migration": "Restore Lost Cards", + "restore-lost-cards-migration-description": "Finds and restores cards and lists with missing swimlaneId or listId. Creates a 'Lost Cards' swimlane to make all lost items visible again.", + "restore-all-archived-migration": "Restore All Archived", + "restore-all-archived-migration-description": "Restores all archived swimlanes, lists, and cards. Automatically fixes any missing swimlaneId or listId to make items visible.", + "fix-missing-lists-migration": "Fix Missing Lists", + "fix-missing-lists-migration-description": "Detects and repairs missing or corrupted lists in the board structure.", + "fix-avatar-urls-migration": "Fix Avatar URLs", + "fix-avatar-urls-migration-description": "Updates avatar URLs for board members to use the correct storage backend and fixes broken avatar references.", + "fix-all-file-urls-migration": "Fix All File URLs", + "fix-all-file-urls-migration-description": "Updates all file attachment URLs on this board to use the correct storage backend and fixes broken file references.", + "migration-needed": "Migration Needed", + "migration-complete": "Complete", + "migration-running": "Running...", + "migration-successful": "Migration completed successfully", + "migration-failed": "Migration failed", + "migrations": "Migrations", + "migrations-admin-only": "Only board administrators can run migrations", + "migrations-description": "Run data integrity checks and repairs for this board. Each migration can be executed individually.", + "no-issues-found": "No issues found", + "run-migration": "Run Migration", + "run-comprehensive-migration-confirm": "This will perform a comprehensive migration to check and fix board data integrity. This may take a few moments. Continue?", + "run-delete-duplicate-empty-lists-migration-confirm": "This will first convert any shared lists to per-swimlane lists, then delete empty lists that have a duplicate list with the same title containing cards. Only truly redundant empty lists will be removed. Continue?", + "run-restore-lost-cards-migration-confirm": "This will create a 'Lost Cards' swimlane and restore all cards and lists with missing swimlaneId or listId. This only affects non-archived items. Continue?", + "run-restore-all-archived-migration-confirm": "This will restore ALL archived swimlanes, lists, and cards, making them visible again. Any items with missing IDs will be automatically fixed. This cannot be easily undone. Continue?", + "run-fix-missing-lists-migration-confirm": "This will detect and repair missing or corrupted lists in the board structure. Continue?", + "run-fix-avatar-urls-migration-confirm": "This will update avatar URLs for board members to use the correct storage backend. Continue?", + "run-fix-all-file-urls-migration-confirm": "This will update all file attachment URLs on this board to use the correct storage backend. Continue?", + "restore-lost-cards-nothing-to-restore": "No lost swimlanes, lists, or cards to restore", + + "migration-progress-title": "Board Migration in Progress", + "migration-progress-overall": "Overall Progress", + "migration-progress-current-step": "Current Step", + "migration-progress-status": "Status", + "migration-progress-details": "Details", + "migration-progress-note": "Please wait while we migrate your board to the latest structure...", + + "step-analyze-board-structure": "Analyze Board Structure", + "step-fix-orphaned-cards": "Fix Orphaned Cards", + "step-convert-shared-lists": "Convert Shared Lists", + "step-ensure-per-swimlane-lists": "Ensure Per-Swimlane Lists", + "step-validate-migration": "Validate Migration", + "step-fix-avatar-urls": "Fix Avatar URLs", + "step-fix-attachment-urls": "Fix Attachment URLs", + "step-analyze-lists": "Analyze Lists", + "step-create-missing-lists": "Create Missing Lists", + "step-update-cards": "Update Cards", + "step-finalize": "Finalize", + "step-delete-duplicate-empty-lists": "Delete Duplicate Empty Lists", + "step-ensure-lost-cards-swimlane": "Ensure Lost Cards Swimlane", + "step-restore-lists": "Restore Lists", + "step-restore-cards": "Restore Cards", + "step-restore-swimlanes": "Restore Swimlanes", + "step-fix-missing-ids": "Fix Missing IDs", + "step-scan-users": "Checking board member avatars", + "step-scan-files": "Checking board file attachments", + "step-fix-file-urls": "Fixing file URLs", "cleanup": "Cleanup", "cleanup-old-jobs": "Cleanup Old Jobs", "completed": "Completed", diff --git a/imports/i18n/data/th.i18n.json b/imports/i18n/data/th.i18n.json index fc992cb5d..2b9c09b1c 100644 --- a/imports/i18n/data/th.i18n.json +++ b/imports/i18n/data/th.i18n.json @@ -1404,7 +1404,70 @@ "back-to-settings": "Back to Settings", "board-id": "Board ID", "board-migration": "Board Migration", + "board-migrations": "Board Migrations", "card-show-lists-on-minicard": "Show Lists on Minicard", + "comprehensive-board-migration": "Comprehensive Board Migration", + "comprehensive-board-migration-description": "Performs comprehensive checks and fixes for board data integrity, including list ordering, card positions, and swimlane structure.", + "delete-duplicate-empty-lists-migration": "Delete Duplicate Empty Lists", + "delete-duplicate-empty-lists-migration-description": "Safely deletes empty duplicate lists. Only removes lists that have no cards AND have another list with the same title that contains cards.", + "lost-cards": "Lost Cards", + "lost-cards-list": "Restored Items", + "restore-lost-cards-migration": "Restore Lost Cards", + "restore-lost-cards-migration-description": "Finds and restores cards and lists with missing swimlaneId or listId. Creates a 'Lost Cards' swimlane to make all lost items visible again.", + "restore-all-archived-migration": "Restore All Archived", + "restore-all-archived-migration-description": "Restores all archived swimlanes, lists, and cards. Automatically fixes any missing swimlaneId or listId to make items visible.", + "fix-missing-lists-migration": "Fix Missing Lists", + "fix-missing-lists-migration-description": "Detects and repairs missing or corrupted lists in the board structure.", + "fix-avatar-urls-migration": "Fix Avatar URLs", + "fix-avatar-urls-migration-description": "Updates avatar URLs for board members to use the correct storage backend and fixes broken avatar references.", + "fix-all-file-urls-migration": "Fix All File URLs", + "fix-all-file-urls-migration-description": "Updates all file attachment URLs on this board to use the correct storage backend and fixes broken file references.", + "migration-needed": "Migration Needed", + "migration-complete": "Complete", + "migration-running": "Running...", + "migration-successful": "Migration completed successfully", + "migration-failed": "Migration failed", + "migrations": "Migrations", + "migrations-admin-only": "Only board administrators can run migrations", + "migrations-description": "Run data integrity checks and repairs for this board. Each migration can be executed individually.", + "no-issues-found": "No issues found", + "run-migration": "Run Migration", + "run-comprehensive-migration-confirm": "This will perform a comprehensive migration to check and fix board data integrity. This may take a few moments. Continue?", + "run-delete-duplicate-empty-lists-migration-confirm": "This will first convert any shared lists to per-swimlane lists, then delete empty lists that have a duplicate list with the same title containing cards. Only truly redundant empty lists will be removed. Continue?", + "run-restore-lost-cards-migration-confirm": "This will create a 'Lost Cards' swimlane and restore all cards and lists with missing swimlaneId or listId. This only affects non-archived items. Continue?", + "run-restore-all-archived-migration-confirm": "This will restore ALL archived swimlanes, lists, and cards, making them visible again. Any items with missing IDs will be automatically fixed. This cannot be easily undone. Continue?", + "run-fix-missing-lists-migration-confirm": "This will detect and repair missing or corrupted lists in the board structure. Continue?", + "run-fix-avatar-urls-migration-confirm": "This will update avatar URLs for board members to use the correct storage backend. Continue?", + "run-fix-all-file-urls-migration-confirm": "This will update all file attachment URLs on this board to use the correct storage backend. Continue?", + "restore-lost-cards-nothing-to-restore": "No lost swimlanes, lists, or cards to restore", + + "migration-progress-title": "Board Migration in Progress", + "migration-progress-overall": "Overall Progress", + "migration-progress-current-step": "Current Step", + "migration-progress-status": "Status", + "migration-progress-details": "Details", + "migration-progress-note": "Please wait while we migrate your board to the latest structure...", + + "step-analyze-board-structure": "Analyze Board Structure", + "step-fix-orphaned-cards": "Fix Orphaned Cards", + "step-convert-shared-lists": "Convert Shared Lists", + "step-ensure-per-swimlane-lists": "Ensure Per-Swimlane Lists", + "step-validate-migration": "Validate Migration", + "step-fix-avatar-urls": "Fix Avatar URLs", + "step-fix-attachment-urls": "Fix Attachment URLs", + "step-analyze-lists": "Analyze Lists", + "step-create-missing-lists": "Create Missing Lists", + "step-update-cards": "Update Cards", + "step-finalize": "Finalize", + "step-delete-duplicate-empty-lists": "Delete Duplicate Empty Lists", + "step-ensure-lost-cards-swimlane": "Ensure Lost Cards Swimlane", + "step-restore-lists": "Restore Lists", + "step-restore-cards": "Restore Cards", + "step-restore-swimlanes": "Restore Swimlanes", + "step-fix-missing-ids": "Fix Missing IDs", + "step-scan-users": "Checking board member avatars", + "step-scan-files": "Checking board file attachments", + "step-fix-file-urls": "Fixing file URLs", "cleanup": "Cleanup", "cleanup-old-jobs": "Cleanup Old Jobs", "completed": "Completed", diff --git a/imports/i18n/data/tk_TM.i18n.json b/imports/i18n/data/tk_TM.i18n.json index eef27e1fe..958d8313a 100644 --- a/imports/i18n/data/tk_TM.i18n.json +++ b/imports/i18n/data/tk_TM.i18n.json @@ -1404,7 +1404,70 @@ "back-to-settings": "Back to Settings", "board-id": "Board ID", "board-migration": "Board Migration", + "board-migrations": "Board Migrations", "card-show-lists-on-minicard": "Show Lists on Minicard", + "comprehensive-board-migration": "Comprehensive Board Migration", + "comprehensive-board-migration-description": "Performs comprehensive checks and fixes for board data integrity, including list ordering, card positions, and swimlane structure.", + "delete-duplicate-empty-lists-migration": "Delete Duplicate Empty Lists", + "delete-duplicate-empty-lists-migration-description": "Safely deletes empty duplicate lists. Only removes lists that have no cards AND have another list with the same title that contains cards.", + "lost-cards": "Lost Cards", + "lost-cards-list": "Restored Items", + "restore-lost-cards-migration": "Restore Lost Cards", + "restore-lost-cards-migration-description": "Finds and restores cards and lists with missing swimlaneId or listId. Creates a 'Lost Cards' swimlane to make all lost items visible again.", + "restore-all-archived-migration": "Restore All Archived", + "restore-all-archived-migration-description": "Restores all archived swimlanes, lists, and cards. Automatically fixes any missing swimlaneId or listId to make items visible.", + "fix-missing-lists-migration": "Fix Missing Lists", + "fix-missing-lists-migration-description": "Detects and repairs missing or corrupted lists in the board structure.", + "fix-avatar-urls-migration": "Fix Avatar URLs", + "fix-avatar-urls-migration-description": "Updates avatar URLs for board members to use the correct storage backend and fixes broken avatar references.", + "fix-all-file-urls-migration": "Fix All File URLs", + "fix-all-file-urls-migration-description": "Updates all file attachment URLs on this board to use the correct storage backend and fixes broken file references.", + "migration-needed": "Migration Needed", + "migration-complete": "Complete", + "migration-running": "Running...", + "migration-successful": "Migration completed successfully", + "migration-failed": "Migration failed", + "migrations": "Migrations", + "migrations-admin-only": "Only board administrators can run migrations", + "migrations-description": "Run data integrity checks and repairs for this board. Each migration can be executed individually.", + "no-issues-found": "No issues found", + "run-migration": "Run Migration", + "run-comprehensive-migration-confirm": "This will perform a comprehensive migration to check and fix board data integrity. This may take a few moments. Continue?", + "run-delete-duplicate-empty-lists-migration-confirm": "This will first convert any shared lists to per-swimlane lists, then delete empty lists that have a duplicate list with the same title containing cards. Only truly redundant empty lists will be removed. Continue?", + "run-restore-lost-cards-migration-confirm": "This will create a 'Lost Cards' swimlane and restore all cards and lists with missing swimlaneId or listId. This only affects non-archived items. Continue?", + "run-restore-all-archived-migration-confirm": "This will restore ALL archived swimlanes, lists, and cards, making them visible again. Any items with missing IDs will be automatically fixed. This cannot be easily undone. Continue?", + "run-fix-missing-lists-migration-confirm": "This will detect and repair missing or corrupted lists in the board structure. Continue?", + "run-fix-avatar-urls-migration-confirm": "This will update avatar URLs for board members to use the correct storage backend. Continue?", + "run-fix-all-file-urls-migration-confirm": "This will update all file attachment URLs on this board to use the correct storage backend. Continue?", + "restore-lost-cards-nothing-to-restore": "No lost swimlanes, lists, or cards to restore", + + "migration-progress-title": "Board Migration in Progress", + "migration-progress-overall": "Overall Progress", + "migration-progress-current-step": "Current Step", + "migration-progress-status": "Status", + "migration-progress-details": "Details", + "migration-progress-note": "Please wait while we migrate your board to the latest structure...", + + "step-analyze-board-structure": "Analyze Board Structure", + "step-fix-orphaned-cards": "Fix Orphaned Cards", + "step-convert-shared-lists": "Convert Shared Lists", + "step-ensure-per-swimlane-lists": "Ensure Per-Swimlane Lists", + "step-validate-migration": "Validate Migration", + "step-fix-avatar-urls": "Fix Avatar URLs", + "step-fix-attachment-urls": "Fix Attachment URLs", + "step-analyze-lists": "Analyze Lists", + "step-create-missing-lists": "Create Missing Lists", + "step-update-cards": "Update Cards", + "step-finalize": "Finalize", + "step-delete-duplicate-empty-lists": "Delete Duplicate Empty Lists", + "step-ensure-lost-cards-swimlane": "Ensure Lost Cards Swimlane", + "step-restore-lists": "Restore Lists", + "step-restore-cards": "Restore Cards", + "step-restore-swimlanes": "Restore Swimlanes", + "step-fix-missing-ids": "Fix Missing IDs", + "step-scan-users": "Checking board member avatars", + "step-scan-files": "Checking board file attachments", + "step-fix-file-urls": "Fixing file URLs", "cleanup": "Cleanup", "cleanup-old-jobs": "Cleanup Old Jobs", "completed": "Completed", diff --git a/imports/i18n/data/tlh.i18n.json b/imports/i18n/data/tlh.i18n.json index eef27e1fe..958d8313a 100644 --- a/imports/i18n/data/tlh.i18n.json +++ b/imports/i18n/data/tlh.i18n.json @@ -1404,7 +1404,70 @@ "back-to-settings": "Back to Settings", "board-id": "Board ID", "board-migration": "Board Migration", + "board-migrations": "Board Migrations", "card-show-lists-on-minicard": "Show Lists on Minicard", + "comprehensive-board-migration": "Comprehensive Board Migration", + "comprehensive-board-migration-description": "Performs comprehensive checks and fixes for board data integrity, including list ordering, card positions, and swimlane structure.", + "delete-duplicate-empty-lists-migration": "Delete Duplicate Empty Lists", + "delete-duplicate-empty-lists-migration-description": "Safely deletes empty duplicate lists. Only removes lists that have no cards AND have another list with the same title that contains cards.", + "lost-cards": "Lost Cards", + "lost-cards-list": "Restored Items", + "restore-lost-cards-migration": "Restore Lost Cards", + "restore-lost-cards-migration-description": "Finds and restores cards and lists with missing swimlaneId or listId. Creates a 'Lost Cards' swimlane to make all lost items visible again.", + "restore-all-archived-migration": "Restore All Archived", + "restore-all-archived-migration-description": "Restores all archived swimlanes, lists, and cards. Automatically fixes any missing swimlaneId or listId to make items visible.", + "fix-missing-lists-migration": "Fix Missing Lists", + "fix-missing-lists-migration-description": "Detects and repairs missing or corrupted lists in the board structure.", + "fix-avatar-urls-migration": "Fix Avatar URLs", + "fix-avatar-urls-migration-description": "Updates avatar URLs for board members to use the correct storage backend and fixes broken avatar references.", + "fix-all-file-urls-migration": "Fix All File URLs", + "fix-all-file-urls-migration-description": "Updates all file attachment URLs on this board to use the correct storage backend and fixes broken file references.", + "migration-needed": "Migration Needed", + "migration-complete": "Complete", + "migration-running": "Running...", + "migration-successful": "Migration completed successfully", + "migration-failed": "Migration failed", + "migrations": "Migrations", + "migrations-admin-only": "Only board administrators can run migrations", + "migrations-description": "Run data integrity checks and repairs for this board. Each migration can be executed individually.", + "no-issues-found": "No issues found", + "run-migration": "Run Migration", + "run-comprehensive-migration-confirm": "This will perform a comprehensive migration to check and fix board data integrity. This may take a few moments. Continue?", + "run-delete-duplicate-empty-lists-migration-confirm": "This will first convert any shared lists to per-swimlane lists, then delete empty lists that have a duplicate list with the same title containing cards. Only truly redundant empty lists will be removed. Continue?", + "run-restore-lost-cards-migration-confirm": "This will create a 'Lost Cards' swimlane and restore all cards and lists with missing swimlaneId or listId. This only affects non-archived items. Continue?", + "run-restore-all-archived-migration-confirm": "This will restore ALL archived swimlanes, lists, and cards, making them visible again. Any items with missing IDs will be automatically fixed. This cannot be easily undone. Continue?", + "run-fix-missing-lists-migration-confirm": "This will detect and repair missing or corrupted lists in the board structure. Continue?", + "run-fix-avatar-urls-migration-confirm": "This will update avatar URLs for board members to use the correct storage backend. Continue?", + "run-fix-all-file-urls-migration-confirm": "This will update all file attachment URLs on this board to use the correct storage backend. Continue?", + "restore-lost-cards-nothing-to-restore": "No lost swimlanes, lists, or cards to restore", + + "migration-progress-title": "Board Migration in Progress", + "migration-progress-overall": "Overall Progress", + "migration-progress-current-step": "Current Step", + "migration-progress-status": "Status", + "migration-progress-details": "Details", + "migration-progress-note": "Please wait while we migrate your board to the latest structure...", + + "step-analyze-board-structure": "Analyze Board Structure", + "step-fix-orphaned-cards": "Fix Orphaned Cards", + "step-convert-shared-lists": "Convert Shared Lists", + "step-ensure-per-swimlane-lists": "Ensure Per-Swimlane Lists", + "step-validate-migration": "Validate Migration", + "step-fix-avatar-urls": "Fix Avatar URLs", + "step-fix-attachment-urls": "Fix Attachment URLs", + "step-analyze-lists": "Analyze Lists", + "step-create-missing-lists": "Create Missing Lists", + "step-update-cards": "Update Cards", + "step-finalize": "Finalize", + "step-delete-duplicate-empty-lists": "Delete Duplicate Empty Lists", + "step-ensure-lost-cards-swimlane": "Ensure Lost Cards Swimlane", + "step-restore-lists": "Restore Lists", + "step-restore-cards": "Restore Cards", + "step-restore-swimlanes": "Restore Swimlanes", + "step-fix-missing-ids": "Fix Missing IDs", + "step-scan-users": "Checking board member avatars", + "step-scan-files": "Checking board file attachments", + "step-fix-file-urls": "Fixing file URLs", "cleanup": "Cleanup", "cleanup-old-jobs": "Cleanup Old Jobs", "completed": "Completed", diff --git a/imports/i18n/data/tr.i18n.json b/imports/i18n/data/tr.i18n.json index a4d61f39c..b4690871e 100644 --- a/imports/i18n/data/tr.i18n.json +++ b/imports/i18n/data/tr.i18n.json @@ -1404,7 +1404,70 @@ "back-to-settings": "Back to Settings", "board-id": "Board ID", "board-migration": "Board Migration", + "board-migrations": "Board Migrations", "card-show-lists-on-minicard": "Show Lists on Minicard", + "comprehensive-board-migration": "Comprehensive Board Migration", + "comprehensive-board-migration-description": "Performs comprehensive checks and fixes for board data integrity, including list ordering, card positions, and swimlane structure.", + "delete-duplicate-empty-lists-migration": "Delete Duplicate Empty Lists", + "delete-duplicate-empty-lists-migration-description": "Safely deletes empty duplicate lists. Only removes lists that have no cards AND have another list with the same title that contains cards.", + "lost-cards": "Lost Cards", + "lost-cards-list": "Restored Items", + "restore-lost-cards-migration": "Restore Lost Cards", + "restore-lost-cards-migration-description": "Finds and restores cards and lists with missing swimlaneId or listId. Creates a 'Lost Cards' swimlane to make all lost items visible again.", + "restore-all-archived-migration": "Restore All Archived", + "restore-all-archived-migration-description": "Restores all archived swimlanes, lists, and cards. Automatically fixes any missing swimlaneId or listId to make items visible.", + "fix-missing-lists-migration": "Fix Missing Lists", + "fix-missing-lists-migration-description": "Detects and repairs missing or corrupted lists in the board structure.", + "fix-avatar-urls-migration": "Fix Avatar URLs", + "fix-avatar-urls-migration-description": "Updates avatar URLs for board members to use the correct storage backend and fixes broken avatar references.", + "fix-all-file-urls-migration": "Fix All File URLs", + "fix-all-file-urls-migration-description": "Updates all file attachment URLs on this board to use the correct storage backend and fixes broken file references.", + "migration-needed": "Migration Needed", + "migration-complete": "Complete", + "migration-running": "Running...", + "migration-successful": "Migration completed successfully", + "migration-failed": "Migration failed", + "migrations": "Migrations", + "migrations-admin-only": "Only board administrators can run migrations", + "migrations-description": "Run data integrity checks and repairs for this board. Each migration can be executed individually.", + "no-issues-found": "No issues found", + "run-migration": "Run Migration", + "run-comprehensive-migration-confirm": "This will perform a comprehensive migration to check and fix board data integrity. This may take a few moments. Continue?", + "run-delete-duplicate-empty-lists-migration-confirm": "This will first convert any shared lists to per-swimlane lists, then delete empty lists that have a duplicate list with the same title containing cards. Only truly redundant empty lists will be removed. Continue?", + "run-restore-lost-cards-migration-confirm": "This will create a 'Lost Cards' swimlane and restore all cards and lists with missing swimlaneId or listId. This only affects non-archived items. Continue?", + "run-restore-all-archived-migration-confirm": "This will restore ALL archived swimlanes, lists, and cards, making them visible again. Any items with missing IDs will be automatically fixed. This cannot be easily undone. Continue?", + "run-fix-missing-lists-migration-confirm": "This will detect and repair missing or corrupted lists in the board structure. Continue?", + "run-fix-avatar-urls-migration-confirm": "This will update avatar URLs for board members to use the correct storage backend. Continue?", + "run-fix-all-file-urls-migration-confirm": "This will update all file attachment URLs on this board to use the correct storage backend. Continue?", + "restore-lost-cards-nothing-to-restore": "No lost swimlanes, lists, or cards to restore", + + "migration-progress-title": "Board Migration in Progress", + "migration-progress-overall": "Overall Progress", + "migration-progress-current-step": "Current Step", + "migration-progress-status": "Durum", + "migration-progress-details": "Detaylar", + "migration-progress-note": "Please wait while we migrate your board to the latest structure...", + + "step-analyze-board-structure": "Analyze Board Structure", + "step-fix-orphaned-cards": "Fix Orphaned Cards", + "step-convert-shared-lists": "Convert Shared Lists", + "step-ensure-per-swimlane-lists": "Ensure Per-Swimlane Lists", + "step-validate-migration": "Validate Migration", + "step-fix-avatar-urls": "Fix Avatar URLs", + "step-fix-attachment-urls": "Fix Attachment URLs", + "step-analyze-lists": "Analyze Lists", + "step-create-missing-lists": "Create Missing Lists", + "step-update-cards": "Update Cards", + "step-finalize": "Finalize", + "step-delete-duplicate-empty-lists": "Delete Duplicate Empty Lists", + "step-ensure-lost-cards-swimlane": "Ensure Lost Cards Swimlane", + "step-restore-lists": "Restore Lists", + "step-restore-cards": "Restore Cards", + "step-restore-swimlanes": "Restore Swimlanes", + "step-fix-missing-ids": "Fix Missing IDs", + "step-scan-users": "Checking board member avatars", + "step-scan-files": "Checking board file attachments", + "step-fix-file-urls": "Fixing file URLs", "cleanup": "Cleanup", "cleanup-old-jobs": "Cleanup Old Jobs", "completed": "Tamamlandı", diff --git a/imports/i18n/data/ug.i18n.json b/imports/i18n/data/ug.i18n.json index eef27e1fe..958d8313a 100644 --- a/imports/i18n/data/ug.i18n.json +++ b/imports/i18n/data/ug.i18n.json @@ -1404,7 +1404,70 @@ "back-to-settings": "Back to Settings", "board-id": "Board ID", "board-migration": "Board Migration", + "board-migrations": "Board Migrations", "card-show-lists-on-minicard": "Show Lists on Minicard", + "comprehensive-board-migration": "Comprehensive Board Migration", + "comprehensive-board-migration-description": "Performs comprehensive checks and fixes for board data integrity, including list ordering, card positions, and swimlane structure.", + "delete-duplicate-empty-lists-migration": "Delete Duplicate Empty Lists", + "delete-duplicate-empty-lists-migration-description": "Safely deletes empty duplicate lists. Only removes lists that have no cards AND have another list with the same title that contains cards.", + "lost-cards": "Lost Cards", + "lost-cards-list": "Restored Items", + "restore-lost-cards-migration": "Restore Lost Cards", + "restore-lost-cards-migration-description": "Finds and restores cards and lists with missing swimlaneId or listId. Creates a 'Lost Cards' swimlane to make all lost items visible again.", + "restore-all-archived-migration": "Restore All Archived", + "restore-all-archived-migration-description": "Restores all archived swimlanes, lists, and cards. Automatically fixes any missing swimlaneId or listId to make items visible.", + "fix-missing-lists-migration": "Fix Missing Lists", + "fix-missing-lists-migration-description": "Detects and repairs missing or corrupted lists in the board structure.", + "fix-avatar-urls-migration": "Fix Avatar URLs", + "fix-avatar-urls-migration-description": "Updates avatar URLs for board members to use the correct storage backend and fixes broken avatar references.", + "fix-all-file-urls-migration": "Fix All File URLs", + "fix-all-file-urls-migration-description": "Updates all file attachment URLs on this board to use the correct storage backend and fixes broken file references.", + "migration-needed": "Migration Needed", + "migration-complete": "Complete", + "migration-running": "Running...", + "migration-successful": "Migration completed successfully", + "migration-failed": "Migration failed", + "migrations": "Migrations", + "migrations-admin-only": "Only board administrators can run migrations", + "migrations-description": "Run data integrity checks and repairs for this board. Each migration can be executed individually.", + "no-issues-found": "No issues found", + "run-migration": "Run Migration", + "run-comprehensive-migration-confirm": "This will perform a comprehensive migration to check and fix board data integrity. This may take a few moments. Continue?", + "run-delete-duplicate-empty-lists-migration-confirm": "This will first convert any shared lists to per-swimlane lists, then delete empty lists that have a duplicate list with the same title containing cards. Only truly redundant empty lists will be removed. Continue?", + "run-restore-lost-cards-migration-confirm": "This will create a 'Lost Cards' swimlane and restore all cards and lists with missing swimlaneId or listId. This only affects non-archived items. Continue?", + "run-restore-all-archived-migration-confirm": "This will restore ALL archived swimlanes, lists, and cards, making them visible again. Any items with missing IDs will be automatically fixed. This cannot be easily undone. Continue?", + "run-fix-missing-lists-migration-confirm": "This will detect and repair missing or corrupted lists in the board structure. Continue?", + "run-fix-avatar-urls-migration-confirm": "This will update avatar URLs for board members to use the correct storage backend. Continue?", + "run-fix-all-file-urls-migration-confirm": "This will update all file attachment URLs on this board to use the correct storage backend. Continue?", + "restore-lost-cards-nothing-to-restore": "No lost swimlanes, lists, or cards to restore", + + "migration-progress-title": "Board Migration in Progress", + "migration-progress-overall": "Overall Progress", + "migration-progress-current-step": "Current Step", + "migration-progress-status": "Status", + "migration-progress-details": "Details", + "migration-progress-note": "Please wait while we migrate your board to the latest structure...", + + "step-analyze-board-structure": "Analyze Board Structure", + "step-fix-orphaned-cards": "Fix Orphaned Cards", + "step-convert-shared-lists": "Convert Shared Lists", + "step-ensure-per-swimlane-lists": "Ensure Per-Swimlane Lists", + "step-validate-migration": "Validate Migration", + "step-fix-avatar-urls": "Fix Avatar URLs", + "step-fix-attachment-urls": "Fix Attachment URLs", + "step-analyze-lists": "Analyze Lists", + "step-create-missing-lists": "Create Missing Lists", + "step-update-cards": "Update Cards", + "step-finalize": "Finalize", + "step-delete-duplicate-empty-lists": "Delete Duplicate Empty Lists", + "step-ensure-lost-cards-swimlane": "Ensure Lost Cards Swimlane", + "step-restore-lists": "Restore Lists", + "step-restore-cards": "Restore Cards", + "step-restore-swimlanes": "Restore Swimlanes", + "step-fix-missing-ids": "Fix Missing IDs", + "step-scan-users": "Checking board member avatars", + "step-scan-files": "Checking board file attachments", + "step-fix-file-urls": "Fixing file URLs", "cleanup": "Cleanup", "cleanup-old-jobs": "Cleanup Old Jobs", "completed": "Completed", diff --git a/imports/i18n/data/uk-UA.i18n.json b/imports/i18n/data/uk-UA.i18n.json index 037a7a3e5..e4a704602 100644 --- a/imports/i18n/data/uk-UA.i18n.json +++ b/imports/i18n/data/uk-UA.i18n.json @@ -1404,7 +1404,70 @@ "back-to-settings": "Back to Settings", "board-id": "Board ID", "board-migration": "Board Migration", + "board-migrations": "Board Migrations", "card-show-lists-on-minicard": "Show Lists on Minicard", + "comprehensive-board-migration": "Comprehensive Board Migration", + "comprehensive-board-migration-description": "Performs comprehensive checks and fixes for board data integrity, including list ordering, card positions, and swimlane structure.", + "delete-duplicate-empty-lists-migration": "Delete Duplicate Empty Lists", + "delete-duplicate-empty-lists-migration-description": "Safely deletes empty duplicate lists. Only removes lists that have no cards AND have another list with the same title that contains cards.", + "lost-cards": "Lost Cards", + "lost-cards-list": "Restored Items", + "restore-lost-cards-migration": "Restore Lost Cards", + "restore-lost-cards-migration-description": "Finds and restores cards and lists with missing swimlaneId or listId. Creates a 'Lost Cards' swimlane to make all lost items visible again.", + "restore-all-archived-migration": "Restore All Archived", + "restore-all-archived-migration-description": "Restores all archived swimlanes, lists, and cards. Automatically fixes any missing swimlaneId or listId to make items visible.", + "fix-missing-lists-migration": "Fix Missing Lists", + "fix-missing-lists-migration-description": "Detects and repairs missing or corrupted lists in the board structure.", + "fix-avatar-urls-migration": "Fix Avatar URLs", + "fix-avatar-urls-migration-description": "Updates avatar URLs for board members to use the correct storage backend and fixes broken avatar references.", + "fix-all-file-urls-migration": "Fix All File URLs", + "fix-all-file-urls-migration-description": "Updates all file attachment URLs on this board to use the correct storage backend and fixes broken file references.", + "migration-needed": "Migration Needed", + "migration-complete": "Complete", + "migration-running": "Running...", + "migration-successful": "Migration completed successfully", + "migration-failed": "Migration failed", + "migrations": "Migrations", + "migrations-admin-only": "Only board administrators can run migrations", + "migrations-description": "Run data integrity checks and repairs for this board. Each migration can be executed individually.", + "no-issues-found": "No issues found", + "run-migration": "Run Migration", + "run-comprehensive-migration-confirm": "This will perform a comprehensive migration to check and fix board data integrity. This may take a few moments. Continue?", + "run-delete-duplicate-empty-lists-migration-confirm": "This will first convert any shared lists to per-swimlane lists, then delete empty lists that have a duplicate list with the same title containing cards. Only truly redundant empty lists will be removed. Continue?", + "run-restore-lost-cards-migration-confirm": "This will create a 'Lost Cards' swimlane and restore all cards and lists with missing swimlaneId or listId. This only affects non-archived items. Continue?", + "run-restore-all-archived-migration-confirm": "This will restore ALL archived swimlanes, lists, and cards, making them visible again. Any items with missing IDs will be automatically fixed. This cannot be easily undone. Continue?", + "run-fix-missing-lists-migration-confirm": "This will detect and repair missing or corrupted lists in the board structure. Continue?", + "run-fix-avatar-urls-migration-confirm": "This will update avatar URLs for board members to use the correct storage backend. Continue?", + "run-fix-all-file-urls-migration-confirm": "This will update all file attachment URLs on this board to use the correct storage backend. Continue?", + "restore-lost-cards-nothing-to-restore": "No lost swimlanes, lists, or cards to restore", + + "migration-progress-title": "Board Migration in Progress", + "migration-progress-overall": "Overall Progress", + "migration-progress-current-step": "Current Step", + "migration-progress-status": "Статус", + "migration-progress-details": "Деталі", + "migration-progress-note": "Please wait while we migrate your board to the latest structure...", + + "step-analyze-board-structure": "Analyze Board Structure", + "step-fix-orphaned-cards": "Fix Orphaned Cards", + "step-convert-shared-lists": "Convert Shared Lists", + "step-ensure-per-swimlane-lists": "Ensure Per-Swimlane Lists", + "step-validate-migration": "Validate Migration", + "step-fix-avatar-urls": "Fix Avatar URLs", + "step-fix-attachment-urls": "Fix Attachment URLs", + "step-analyze-lists": "Analyze Lists", + "step-create-missing-lists": "Create Missing Lists", + "step-update-cards": "Update Cards", + "step-finalize": "Finalize", + "step-delete-duplicate-empty-lists": "Delete Duplicate Empty Lists", + "step-ensure-lost-cards-swimlane": "Ensure Lost Cards Swimlane", + "step-restore-lists": "Restore Lists", + "step-restore-cards": "Restore Cards", + "step-restore-swimlanes": "Restore Swimlanes", + "step-fix-missing-ids": "Fix Missing IDs", + "step-scan-users": "Checking board member avatars", + "step-scan-files": "Checking board file attachments", + "step-fix-file-urls": "Fixing file URLs", "cleanup": "Cleanup", "cleanup-old-jobs": "Cleanup Old Jobs", "completed": "завершено", diff --git a/imports/i18n/data/uk.i18n.json b/imports/i18n/data/uk.i18n.json index e1d01b105..1efcb3f10 100644 --- a/imports/i18n/data/uk.i18n.json +++ b/imports/i18n/data/uk.i18n.json @@ -1404,7 +1404,70 @@ "back-to-settings": "Back to Settings", "board-id": "Board ID", "board-migration": "Board Migration", + "board-migrations": "Board Migrations", "card-show-lists-on-minicard": "Show Lists on Minicard", + "comprehensive-board-migration": "Comprehensive Board Migration", + "comprehensive-board-migration-description": "Performs comprehensive checks and fixes for board data integrity, including list ordering, card positions, and swimlane structure.", + "delete-duplicate-empty-lists-migration": "Delete Duplicate Empty Lists", + "delete-duplicate-empty-lists-migration-description": "Safely deletes empty duplicate lists. Only removes lists that have no cards AND have another list with the same title that contains cards.", + "lost-cards": "Lost Cards", + "lost-cards-list": "Restored Items", + "restore-lost-cards-migration": "Restore Lost Cards", + "restore-lost-cards-migration-description": "Finds and restores cards and lists with missing swimlaneId or listId. Creates a 'Lost Cards' swimlane to make all lost items visible again.", + "restore-all-archived-migration": "Restore All Archived", + "restore-all-archived-migration-description": "Restores all archived swimlanes, lists, and cards. Automatically fixes any missing swimlaneId or listId to make items visible.", + "fix-missing-lists-migration": "Fix Missing Lists", + "fix-missing-lists-migration-description": "Detects and repairs missing or corrupted lists in the board structure.", + "fix-avatar-urls-migration": "Fix Avatar URLs", + "fix-avatar-urls-migration-description": "Updates avatar URLs for board members to use the correct storage backend and fixes broken avatar references.", + "fix-all-file-urls-migration": "Fix All File URLs", + "fix-all-file-urls-migration-description": "Updates all file attachment URLs on this board to use the correct storage backend and fixes broken file references.", + "migration-needed": "Migration Needed", + "migration-complete": "Complete", + "migration-running": "Running...", + "migration-successful": "Migration completed successfully", + "migration-failed": "Migration failed", + "migrations": "Migrations", + "migrations-admin-only": "Only board administrators can run migrations", + "migrations-description": "Run data integrity checks and repairs for this board. Each migration can be executed individually.", + "no-issues-found": "No issues found", + "run-migration": "Run Migration", + "run-comprehensive-migration-confirm": "This will perform a comprehensive migration to check and fix board data integrity. This may take a few moments. Continue?", + "run-delete-duplicate-empty-lists-migration-confirm": "This will first convert any shared lists to per-swimlane lists, then delete empty lists that have a duplicate list with the same title containing cards. Only truly redundant empty lists will be removed. Continue?", + "run-restore-lost-cards-migration-confirm": "This will create a 'Lost Cards' swimlane and restore all cards and lists with missing swimlaneId or listId. This only affects non-archived items. Continue?", + "run-restore-all-archived-migration-confirm": "This will restore ALL archived swimlanes, lists, and cards, making them visible again. Any items with missing IDs will be automatically fixed. This cannot be easily undone. Continue?", + "run-fix-missing-lists-migration-confirm": "This will detect and repair missing or corrupted lists in the board structure. Continue?", + "run-fix-avatar-urls-migration-confirm": "This will update avatar URLs for board members to use the correct storage backend. Continue?", + "run-fix-all-file-urls-migration-confirm": "This will update all file attachment URLs on this board to use the correct storage backend. Continue?", + "restore-lost-cards-nothing-to-restore": "No lost swimlanes, lists, or cards to restore", + + "migration-progress-title": "Board Migration in Progress", + "migration-progress-overall": "Overall Progress", + "migration-progress-current-step": "Current Step", + "migration-progress-status": "Статус", + "migration-progress-details": "Деталі", + "migration-progress-note": "Please wait while we migrate your board to the latest structure...", + + "step-analyze-board-structure": "Analyze Board Structure", + "step-fix-orphaned-cards": "Fix Orphaned Cards", + "step-convert-shared-lists": "Convert Shared Lists", + "step-ensure-per-swimlane-lists": "Ensure Per-Swimlane Lists", + "step-validate-migration": "Validate Migration", + "step-fix-avatar-urls": "Fix Avatar URLs", + "step-fix-attachment-urls": "Fix Attachment URLs", + "step-analyze-lists": "Analyze Lists", + "step-create-missing-lists": "Create Missing Lists", + "step-update-cards": "Update Cards", + "step-finalize": "Finalize", + "step-delete-duplicate-empty-lists": "Delete Duplicate Empty Lists", + "step-ensure-lost-cards-swimlane": "Ensure Lost Cards Swimlane", + "step-restore-lists": "Restore Lists", + "step-restore-cards": "Restore Cards", + "step-restore-swimlanes": "Restore Swimlanes", + "step-fix-missing-ids": "Fix Missing IDs", + "step-scan-users": "Checking board member avatars", + "step-scan-files": "Checking board file attachments", + "step-fix-file-urls": "Fixing file URLs", "cleanup": "Cleanup", "cleanup-old-jobs": "Cleanup Old Jobs", "completed": "завершено", diff --git a/imports/i18n/data/uz-AR.i18n.json b/imports/i18n/data/uz-AR.i18n.json index eef27e1fe..958d8313a 100644 --- a/imports/i18n/data/uz-AR.i18n.json +++ b/imports/i18n/data/uz-AR.i18n.json @@ -1404,7 +1404,70 @@ "back-to-settings": "Back to Settings", "board-id": "Board ID", "board-migration": "Board Migration", + "board-migrations": "Board Migrations", "card-show-lists-on-minicard": "Show Lists on Minicard", + "comprehensive-board-migration": "Comprehensive Board Migration", + "comprehensive-board-migration-description": "Performs comprehensive checks and fixes for board data integrity, including list ordering, card positions, and swimlane structure.", + "delete-duplicate-empty-lists-migration": "Delete Duplicate Empty Lists", + "delete-duplicate-empty-lists-migration-description": "Safely deletes empty duplicate lists. Only removes lists that have no cards AND have another list with the same title that contains cards.", + "lost-cards": "Lost Cards", + "lost-cards-list": "Restored Items", + "restore-lost-cards-migration": "Restore Lost Cards", + "restore-lost-cards-migration-description": "Finds and restores cards and lists with missing swimlaneId or listId. Creates a 'Lost Cards' swimlane to make all lost items visible again.", + "restore-all-archived-migration": "Restore All Archived", + "restore-all-archived-migration-description": "Restores all archived swimlanes, lists, and cards. Automatically fixes any missing swimlaneId or listId to make items visible.", + "fix-missing-lists-migration": "Fix Missing Lists", + "fix-missing-lists-migration-description": "Detects and repairs missing or corrupted lists in the board structure.", + "fix-avatar-urls-migration": "Fix Avatar URLs", + "fix-avatar-urls-migration-description": "Updates avatar URLs for board members to use the correct storage backend and fixes broken avatar references.", + "fix-all-file-urls-migration": "Fix All File URLs", + "fix-all-file-urls-migration-description": "Updates all file attachment URLs on this board to use the correct storage backend and fixes broken file references.", + "migration-needed": "Migration Needed", + "migration-complete": "Complete", + "migration-running": "Running...", + "migration-successful": "Migration completed successfully", + "migration-failed": "Migration failed", + "migrations": "Migrations", + "migrations-admin-only": "Only board administrators can run migrations", + "migrations-description": "Run data integrity checks and repairs for this board. Each migration can be executed individually.", + "no-issues-found": "No issues found", + "run-migration": "Run Migration", + "run-comprehensive-migration-confirm": "This will perform a comprehensive migration to check and fix board data integrity. This may take a few moments. Continue?", + "run-delete-duplicate-empty-lists-migration-confirm": "This will first convert any shared lists to per-swimlane lists, then delete empty lists that have a duplicate list with the same title containing cards. Only truly redundant empty lists will be removed. Continue?", + "run-restore-lost-cards-migration-confirm": "This will create a 'Lost Cards' swimlane and restore all cards and lists with missing swimlaneId or listId. This only affects non-archived items. Continue?", + "run-restore-all-archived-migration-confirm": "This will restore ALL archived swimlanes, lists, and cards, making them visible again. Any items with missing IDs will be automatically fixed. This cannot be easily undone. Continue?", + "run-fix-missing-lists-migration-confirm": "This will detect and repair missing or corrupted lists in the board structure. Continue?", + "run-fix-avatar-urls-migration-confirm": "This will update avatar URLs for board members to use the correct storage backend. Continue?", + "run-fix-all-file-urls-migration-confirm": "This will update all file attachment URLs on this board to use the correct storage backend. Continue?", + "restore-lost-cards-nothing-to-restore": "No lost swimlanes, lists, or cards to restore", + + "migration-progress-title": "Board Migration in Progress", + "migration-progress-overall": "Overall Progress", + "migration-progress-current-step": "Current Step", + "migration-progress-status": "Status", + "migration-progress-details": "Details", + "migration-progress-note": "Please wait while we migrate your board to the latest structure...", + + "step-analyze-board-structure": "Analyze Board Structure", + "step-fix-orphaned-cards": "Fix Orphaned Cards", + "step-convert-shared-lists": "Convert Shared Lists", + "step-ensure-per-swimlane-lists": "Ensure Per-Swimlane Lists", + "step-validate-migration": "Validate Migration", + "step-fix-avatar-urls": "Fix Avatar URLs", + "step-fix-attachment-urls": "Fix Attachment URLs", + "step-analyze-lists": "Analyze Lists", + "step-create-missing-lists": "Create Missing Lists", + "step-update-cards": "Update Cards", + "step-finalize": "Finalize", + "step-delete-duplicate-empty-lists": "Delete Duplicate Empty Lists", + "step-ensure-lost-cards-swimlane": "Ensure Lost Cards Swimlane", + "step-restore-lists": "Restore Lists", + "step-restore-cards": "Restore Cards", + "step-restore-swimlanes": "Restore Swimlanes", + "step-fix-missing-ids": "Fix Missing IDs", + "step-scan-users": "Checking board member avatars", + "step-scan-files": "Checking board file attachments", + "step-fix-file-urls": "Fixing file URLs", "cleanup": "Cleanup", "cleanup-old-jobs": "Cleanup Old Jobs", "completed": "Completed", diff --git a/imports/i18n/data/uz-LA.i18n.json b/imports/i18n/data/uz-LA.i18n.json index eef27e1fe..958d8313a 100644 --- a/imports/i18n/data/uz-LA.i18n.json +++ b/imports/i18n/data/uz-LA.i18n.json @@ -1404,7 +1404,70 @@ "back-to-settings": "Back to Settings", "board-id": "Board ID", "board-migration": "Board Migration", + "board-migrations": "Board Migrations", "card-show-lists-on-minicard": "Show Lists on Minicard", + "comprehensive-board-migration": "Comprehensive Board Migration", + "comprehensive-board-migration-description": "Performs comprehensive checks and fixes for board data integrity, including list ordering, card positions, and swimlane structure.", + "delete-duplicate-empty-lists-migration": "Delete Duplicate Empty Lists", + "delete-duplicate-empty-lists-migration-description": "Safely deletes empty duplicate lists. Only removes lists that have no cards AND have another list with the same title that contains cards.", + "lost-cards": "Lost Cards", + "lost-cards-list": "Restored Items", + "restore-lost-cards-migration": "Restore Lost Cards", + "restore-lost-cards-migration-description": "Finds and restores cards and lists with missing swimlaneId or listId. Creates a 'Lost Cards' swimlane to make all lost items visible again.", + "restore-all-archived-migration": "Restore All Archived", + "restore-all-archived-migration-description": "Restores all archived swimlanes, lists, and cards. Automatically fixes any missing swimlaneId or listId to make items visible.", + "fix-missing-lists-migration": "Fix Missing Lists", + "fix-missing-lists-migration-description": "Detects and repairs missing or corrupted lists in the board structure.", + "fix-avatar-urls-migration": "Fix Avatar URLs", + "fix-avatar-urls-migration-description": "Updates avatar URLs for board members to use the correct storage backend and fixes broken avatar references.", + "fix-all-file-urls-migration": "Fix All File URLs", + "fix-all-file-urls-migration-description": "Updates all file attachment URLs on this board to use the correct storage backend and fixes broken file references.", + "migration-needed": "Migration Needed", + "migration-complete": "Complete", + "migration-running": "Running...", + "migration-successful": "Migration completed successfully", + "migration-failed": "Migration failed", + "migrations": "Migrations", + "migrations-admin-only": "Only board administrators can run migrations", + "migrations-description": "Run data integrity checks and repairs for this board. Each migration can be executed individually.", + "no-issues-found": "No issues found", + "run-migration": "Run Migration", + "run-comprehensive-migration-confirm": "This will perform a comprehensive migration to check and fix board data integrity. This may take a few moments. Continue?", + "run-delete-duplicate-empty-lists-migration-confirm": "This will first convert any shared lists to per-swimlane lists, then delete empty lists that have a duplicate list with the same title containing cards. Only truly redundant empty lists will be removed. Continue?", + "run-restore-lost-cards-migration-confirm": "This will create a 'Lost Cards' swimlane and restore all cards and lists with missing swimlaneId or listId. This only affects non-archived items. Continue?", + "run-restore-all-archived-migration-confirm": "This will restore ALL archived swimlanes, lists, and cards, making them visible again. Any items with missing IDs will be automatically fixed. This cannot be easily undone. Continue?", + "run-fix-missing-lists-migration-confirm": "This will detect and repair missing or corrupted lists in the board structure. Continue?", + "run-fix-avatar-urls-migration-confirm": "This will update avatar URLs for board members to use the correct storage backend. Continue?", + "run-fix-all-file-urls-migration-confirm": "This will update all file attachment URLs on this board to use the correct storage backend. Continue?", + "restore-lost-cards-nothing-to-restore": "No lost swimlanes, lists, or cards to restore", + + "migration-progress-title": "Board Migration in Progress", + "migration-progress-overall": "Overall Progress", + "migration-progress-current-step": "Current Step", + "migration-progress-status": "Status", + "migration-progress-details": "Details", + "migration-progress-note": "Please wait while we migrate your board to the latest structure...", + + "step-analyze-board-structure": "Analyze Board Structure", + "step-fix-orphaned-cards": "Fix Orphaned Cards", + "step-convert-shared-lists": "Convert Shared Lists", + "step-ensure-per-swimlane-lists": "Ensure Per-Swimlane Lists", + "step-validate-migration": "Validate Migration", + "step-fix-avatar-urls": "Fix Avatar URLs", + "step-fix-attachment-urls": "Fix Attachment URLs", + "step-analyze-lists": "Analyze Lists", + "step-create-missing-lists": "Create Missing Lists", + "step-update-cards": "Update Cards", + "step-finalize": "Finalize", + "step-delete-duplicate-empty-lists": "Delete Duplicate Empty Lists", + "step-ensure-lost-cards-swimlane": "Ensure Lost Cards Swimlane", + "step-restore-lists": "Restore Lists", + "step-restore-cards": "Restore Cards", + "step-restore-swimlanes": "Restore Swimlanes", + "step-fix-missing-ids": "Fix Missing IDs", + "step-scan-users": "Checking board member avatars", + "step-scan-files": "Checking board file attachments", + "step-fix-file-urls": "Fixing file URLs", "cleanup": "Cleanup", "cleanup-old-jobs": "Cleanup Old Jobs", "completed": "Completed", diff --git a/imports/i18n/data/uz-UZ.i18n.json b/imports/i18n/data/uz-UZ.i18n.json index eef27e1fe..958d8313a 100644 --- a/imports/i18n/data/uz-UZ.i18n.json +++ b/imports/i18n/data/uz-UZ.i18n.json @@ -1404,7 +1404,70 @@ "back-to-settings": "Back to Settings", "board-id": "Board ID", "board-migration": "Board Migration", + "board-migrations": "Board Migrations", "card-show-lists-on-minicard": "Show Lists on Minicard", + "comprehensive-board-migration": "Comprehensive Board Migration", + "comprehensive-board-migration-description": "Performs comprehensive checks and fixes for board data integrity, including list ordering, card positions, and swimlane structure.", + "delete-duplicate-empty-lists-migration": "Delete Duplicate Empty Lists", + "delete-duplicate-empty-lists-migration-description": "Safely deletes empty duplicate lists. Only removes lists that have no cards AND have another list with the same title that contains cards.", + "lost-cards": "Lost Cards", + "lost-cards-list": "Restored Items", + "restore-lost-cards-migration": "Restore Lost Cards", + "restore-lost-cards-migration-description": "Finds and restores cards and lists with missing swimlaneId or listId. Creates a 'Lost Cards' swimlane to make all lost items visible again.", + "restore-all-archived-migration": "Restore All Archived", + "restore-all-archived-migration-description": "Restores all archived swimlanes, lists, and cards. Automatically fixes any missing swimlaneId or listId to make items visible.", + "fix-missing-lists-migration": "Fix Missing Lists", + "fix-missing-lists-migration-description": "Detects and repairs missing or corrupted lists in the board structure.", + "fix-avatar-urls-migration": "Fix Avatar URLs", + "fix-avatar-urls-migration-description": "Updates avatar URLs for board members to use the correct storage backend and fixes broken avatar references.", + "fix-all-file-urls-migration": "Fix All File URLs", + "fix-all-file-urls-migration-description": "Updates all file attachment URLs on this board to use the correct storage backend and fixes broken file references.", + "migration-needed": "Migration Needed", + "migration-complete": "Complete", + "migration-running": "Running...", + "migration-successful": "Migration completed successfully", + "migration-failed": "Migration failed", + "migrations": "Migrations", + "migrations-admin-only": "Only board administrators can run migrations", + "migrations-description": "Run data integrity checks and repairs for this board. Each migration can be executed individually.", + "no-issues-found": "No issues found", + "run-migration": "Run Migration", + "run-comprehensive-migration-confirm": "This will perform a comprehensive migration to check and fix board data integrity. This may take a few moments. Continue?", + "run-delete-duplicate-empty-lists-migration-confirm": "This will first convert any shared lists to per-swimlane lists, then delete empty lists that have a duplicate list with the same title containing cards. Only truly redundant empty lists will be removed. Continue?", + "run-restore-lost-cards-migration-confirm": "This will create a 'Lost Cards' swimlane and restore all cards and lists with missing swimlaneId or listId. This only affects non-archived items. Continue?", + "run-restore-all-archived-migration-confirm": "This will restore ALL archived swimlanes, lists, and cards, making them visible again. Any items with missing IDs will be automatically fixed. This cannot be easily undone. Continue?", + "run-fix-missing-lists-migration-confirm": "This will detect and repair missing or corrupted lists in the board structure. Continue?", + "run-fix-avatar-urls-migration-confirm": "This will update avatar URLs for board members to use the correct storage backend. Continue?", + "run-fix-all-file-urls-migration-confirm": "This will update all file attachment URLs on this board to use the correct storage backend. Continue?", + "restore-lost-cards-nothing-to-restore": "No lost swimlanes, lists, or cards to restore", + + "migration-progress-title": "Board Migration in Progress", + "migration-progress-overall": "Overall Progress", + "migration-progress-current-step": "Current Step", + "migration-progress-status": "Status", + "migration-progress-details": "Details", + "migration-progress-note": "Please wait while we migrate your board to the latest structure...", + + "step-analyze-board-structure": "Analyze Board Structure", + "step-fix-orphaned-cards": "Fix Orphaned Cards", + "step-convert-shared-lists": "Convert Shared Lists", + "step-ensure-per-swimlane-lists": "Ensure Per-Swimlane Lists", + "step-validate-migration": "Validate Migration", + "step-fix-avatar-urls": "Fix Avatar URLs", + "step-fix-attachment-urls": "Fix Attachment URLs", + "step-analyze-lists": "Analyze Lists", + "step-create-missing-lists": "Create Missing Lists", + "step-update-cards": "Update Cards", + "step-finalize": "Finalize", + "step-delete-duplicate-empty-lists": "Delete Duplicate Empty Lists", + "step-ensure-lost-cards-swimlane": "Ensure Lost Cards Swimlane", + "step-restore-lists": "Restore Lists", + "step-restore-cards": "Restore Cards", + "step-restore-swimlanes": "Restore Swimlanes", + "step-fix-missing-ids": "Fix Missing IDs", + "step-scan-users": "Checking board member avatars", + "step-scan-files": "Checking board file attachments", + "step-fix-file-urls": "Fixing file URLs", "cleanup": "Cleanup", "cleanup-old-jobs": "Cleanup Old Jobs", "completed": "Completed", diff --git a/imports/i18n/data/uz.i18n.json b/imports/i18n/data/uz.i18n.json index eef27e1fe..958d8313a 100644 --- a/imports/i18n/data/uz.i18n.json +++ b/imports/i18n/data/uz.i18n.json @@ -1404,7 +1404,70 @@ "back-to-settings": "Back to Settings", "board-id": "Board ID", "board-migration": "Board Migration", + "board-migrations": "Board Migrations", "card-show-lists-on-minicard": "Show Lists on Minicard", + "comprehensive-board-migration": "Comprehensive Board Migration", + "comprehensive-board-migration-description": "Performs comprehensive checks and fixes for board data integrity, including list ordering, card positions, and swimlane structure.", + "delete-duplicate-empty-lists-migration": "Delete Duplicate Empty Lists", + "delete-duplicate-empty-lists-migration-description": "Safely deletes empty duplicate lists. Only removes lists that have no cards AND have another list with the same title that contains cards.", + "lost-cards": "Lost Cards", + "lost-cards-list": "Restored Items", + "restore-lost-cards-migration": "Restore Lost Cards", + "restore-lost-cards-migration-description": "Finds and restores cards and lists with missing swimlaneId or listId. Creates a 'Lost Cards' swimlane to make all lost items visible again.", + "restore-all-archived-migration": "Restore All Archived", + "restore-all-archived-migration-description": "Restores all archived swimlanes, lists, and cards. Automatically fixes any missing swimlaneId or listId to make items visible.", + "fix-missing-lists-migration": "Fix Missing Lists", + "fix-missing-lists-migration-description": "Detects and repairs missing or corrupted lists in the board structure.", + "fix-avatar-urls-migration": "Fix Avatar URLs", + "fix-avatar-urls-migration-description": "Updates avatar URLs for board members to use the correct storage backend and fixes broken avatar references.", + "fix-all-file-urls-migration": "Fix All File URLs", + "fix-all-file-urls-migration-description": "Updates all file attachment URLs on this board to use the correct storage backend and fixes broken file references.", + "migration-needed": "Migration Needed", + "migration-complete": "Complete", + "migration-running": "Running...", + "migration-successful": "Migration completed successfully", + "migration-failed": "Migration failed", + "migrations": "Migrations", + "migrations-admin-only": "Only board administrators can run migrations", + "migrations-description": "Run data integrity checks and repairs for this board. Each migration can be executed individually.", + "no-issues-found": "No issues found", + "run-migration": "Run Migration", + "run-comprehensive-migration-confirm": "This will perform a comprehensive migration to check and fix board data integrity. This may take a few moments. Continue?", + "run-delete-duplicate-empty-lists-migration-confirm": "This will first convert any shared lists to per-swimlane lists, then delete empty lists that have a duplicate list with the same title containing cards. Only truly redundant empty lists will be removed. Continue?", + "run-restore-lost-cards-migration-confirm": "This will create a 'Lost Cards' swimlane and restore all cards and lists with missing swimlaneId or listId. This only affects non-archived items. Continue?", + "run-restore-all-archived-migration-confirm": "This will restore ALL archived swimlanes, lists, and cards, making them visible again. Any items with missing IDs will be automatically fixed. This cannot be easily undone. Continue?", + "run-fix-missing-lists-migration-confirm": "This will detect and repair missing or corrupted lists in the board structure. Continue?", + "run-fix-avatar-urls-migration-confirm": "This will update avatar URLs for board members to use the correct storage backend. Continue?", + "run-fix-all-file-urls-migration-confirm": "This will update all file attachment URLs on this board to use the correct storage backend. Continue?", + "restore-lost-cards-nothing-to-restore": "No lost swimlanes, lists, or cards to restore", + + "migration-progress-title": "Board Migration in Progress", + "migration-progress-overall": "Overall Progress", + "migration-progress-current-step": "Current Step", + "migration-progress-status": "Status", + "migration-progress-details": "Details", + "migration-progress-note": "Please wait while we migrate your board to the latest structure...", + + "step-analyze-board-structure": "Analyze Board Structure", + "step-fix-orphaned-cards": "Fix Orphaned Cards", + "step-convert-shared-lists": "Convert Shared Lists", + "step-ensure-per-swimlane-lists": "Ensure Per-Swimlane Lists", + "step-validate-migration": "Validate Migration", + "step-fix-avatar-urls": "Fix Avatar URLs", + "step-fix-attachment-urls": "Fix Attachment URLs", + "step-analyze-lists": "Analyze Lists", + "step-create-missing-lists": "Create Missing Lists", + "step-update-cards": "Update Cards", + "step-finalize": "Finalize", + "step-delete-duplicate-empty-lists": "Delete Duplicate Empty Lists", + "step-ensure-lost-cards-swimlane": "Ensure Lost Cards Swimlane", + "step-restore-lists": "Restore Lists", + "step-restore-cards": "Restore Cards", + "step-restore-swimlanes": "Restore Swimlanes", + "step-fix-missing-ids": "Fix Missing IDs", + "step-scan-users": "Checking board member avatars", + "step-scan-files": "Checking board file attachments", + "step-fix-file-urls": "Fixing file URLs", "cleanup": "Cleanup", "cleanup-old-jobs": "Cleanup Old Jobs", "completed": "Completed", diff --git a/imports/i18n/data/ve-CC.i18n.json b/imports/i18n/data/ve-CC.i18n.json index eef27e1fe..958d8313a 100644 --- a/imports/i18n/data/ve-CC.i18n.json +++ b/imports/i18n/data/ve-CC.i18n.json @@ -1404,7 +1404,70 @@ "back-to-settings": "Back to Settings", "board-id": "Board ID", "board-migration": "Board Migration", + "board-migrations": "Board Migrations", "card-show-lists-on-minicard": "Show Lists on Minicard", + "comprehensive-board-migration": "Comprehensive Board Migration", + "comprehensive-board-migration-description": "Performs comprehensive checks and fixes for board data integrity, including list ordering, card positions, and swimlane structure.", + "delete-duplicate-empty-lists-migration": "Delete Duplicate Empty Lists", + "delete-duplicate-empty-lists-migration-description": "Safely deletes empty duplicate lists. Only removes lists that have no cards AND have another list with the same title that contains cards.", + "lost-cards": "Lost Cards", + "lost-cards-list": "Restored Items", + "restore-lost-cards-migration": "Restore Lost Cards", + "restore-lost-cards-migration-description": "Finds and restores cards and lists with missing swimlaneId or listId. Creates a 'Lost Cards' swimlane to make all lost items visible again.", + "restore-all-archived-migration": "Restore All Archived", + "restore-all-archived-migration-description": "Restores all archived swimlanes, lists, and cards. Automatically fixes any missing swimlaneId or listId to make items visible.", + "fix-missing-lists-migration": "Fix Missing Lists", + "fix-missing-lists-migration-description": "Detects and repairs missing or corrupted lists in the board structure.", + "fix-avatar-urls-migration": "Fix Avatar URLs", + "fix-avatar-urls-migration-description": "Updates avatar URLs for board members to use the correct storage backend and fixes broken avatar references.", + "fix-all-file-urls-migration": "Fix All File URLs", + "fix-all-file-urls-migration-description": "Updates all file attachment URLs on this board to use the correct storage backend and fixes broken file references.", + "migration-needed": "Migration Needed", + "migration-complete": "Complete", + "migration-running": "Running...", + "migration-successful": "Migration completed successfully", + "migration-failed": "Migration failed", + "migrations": "Migrations", + "migrations-admin-only": "Only board administrators can run migrations", + "migrations-description": "Run data integrity checks and repairs for this board. Each migration can be executed individually.", + "no-issues-found": "No issues found", + "run-migration": "Run Migration", + "run-comprehensive-migration-confirm": "This will perform a comprehensive migration to check and fix board data integrity. This may take a few moments. Continue?", + "run-delete-duplicate-empty-lists-migration-confirm": "This will first convert any shared lists to per-swimlane lists, then delete empty lists that have a duplicate list with the same title containing cards. Only truly redundant empty lists will be removed. Continue?", + "run-restore-lost-cards-migration-confirm": "This will create a 'Lost Cards' swimlane and restore all cards and lists with missing swimlaneId or listId. This only affects non-archived items. Continue?", + "run-restore-all-archived-migration-confirm": "This will restore ALL archived swimlanes, lists, and cards, making them visible again. Any items with missing IDs will be automatically fixed. This cannot be easily undone. Continue?", + "run-fix-missing-lists-migration-confirm": "This will detect and repair missing or corrupted lists in the board structure. Continue?", + "run-fix-avatar-urls-migration-confirm": "This will update avatar URLs for board members to use the correct storage backend. Continue?", + "run-fix-all-file-urls-migration-confirm": "This will update all file attachment URLs on this board to use the correct storage backend. Continue?", + "restore-lost-cards-nothing-to-restore": "No lost swimlanes, lists, or cards to restore", + + "migration-progress-title": "Board Migration in Progress", + "migration-progress-overall": "Overall Progress", + "migration-progress-current-step": "Current Step", + "migration-progress-status": "Status", + "migration-progress-details": "Details", + "migration-progress-note": "Please wait while we migrate your board to the latest structure...", + + "step-analyze-board-structure": "Analyze Board Structure", + "step-fix-orphaned-cards": "Fix Orphaned Cards", + "step-convert-shared-lists": "Convert Shared Lists", + "step-ensure-per-swimlane-lists": "Ensure Per-Swimlane Lists", + "step-validate-migration": "Validate Migration", + "step-fix-avatar-urls": "Fix Avatar URLs", + "step-fix-attachment-urls": "Fix Attachment URLs", + "step-analyze-lists": "Analyze Lists", + "step-create-missing-lists": "Create Missing Lists", + "step-update-cards": "Update Cards", + "step-finalize": "Finalize", + "step-delete-duplicate-empty-lists": "Delete Duplicate Empty Lists", + "step-ensure-lost-cards-swimlane": "Ensure Lost Cards Swimlane", + "step-restore-lists": "Restore Lists", + "step-restore-cards": "Restore Cards", + "step-restore-swimlanes": "Restore Swimlanes", + "step-fix-missing-ids": "Fix Missing IDs", + "step-scan-users": "Checking board member avatars", + "step-scan-files": "Checking board file attachments", + "step-fix-file-urls": "Fixing file URLs", "cleanup": "Cleanup", "cleanup-old-jobs": "Cleanup Old Jobs", "completed": "Completed", diff --git a/imports/i18n/data/ve-PP.i18n.json b/imports/i18n/data/ve-PP.i18n.json index eef27e1fe..958d8313a 100644 --- a/imports/i18n/data/ve-PP.i18n.json +++ b/imports/i18n/data/ve-PP.i18n.json @@ -1404,7 +1404,70 @@ "back-to-settings": "Back to Settings", "board-id": "Board ID", "board-migration": "Board Migration", + "board-migrations": "Board Migrations", "card-show-lists-on-minicard": "Show Lists on Minicard", + "comprehensive-board-migration": "Comprehensive Board Migration", + "comprehensive-board-migration-description": "Performs comprehensive checks and fixes for board data integrity, including list ordering, card positions, and swimlane structure.", + "delete-duplicate-empty-lists-migration": "Delete Duplicate Empty Lists", + "delete-duplicate-empty-lists-migration-description": "Safely deletes empty duplicate lists. Only removes lists that have no cards AND have another list with the same title that contains cards.", + "lost-cards": "Lost Cards", + "lost-cards-list": "Restored Items", + "restore-lost-cards-migration": "Restore Lost Cards", + "restore-lost-cards-migration-description": "Finds and restores cards and lists with missing swimlaneId or listId. Creates a 'Lost Cards' swimlane to make all lost items visible again.", + "restore-all-archived-migration": "Restore All Archived", + "restore-all-archived-migration-description": "Restores all archived swimlanes, lists, and cards. Automatically fixes any missing swimlaneId or listId to make items visible.", + "fix-missing-lists-migration": "Fix Missing Lists", + "fix-missing-lists-migration-description": "Detects and repairs missing or corrupted lists in the board structure.", + "fix-avatar-urls-migration": "Fix Avatar URLs", + "fix-avatar-urls-migration-description": "Updates avatar URLs for board members to use the correct storage backend and fixes broken avatar references.", + "fix-all-file-urls-migration": "Fix All File URLs", + "fix-all-file-urls-migration-description": "Updates all file attachment URLs on this board to use the correct storage backend and fixes broken file references.", + "migration-needed": "Migration Needed", + "migration-complete": "Complete", + "migration-running": "Running...", + "migration-successful": "Migration completed successfully", + "migration-failed": "Migration failed", + "migrations": "Migrations", + "migrations-admin-only": "Only board administrators can run migrations", + "migrations-description": "Run data integrity checks and repairs for this board. Each migration can be executed individually.", + "no-issues-found": "No issues found", + "run-migration": "Run Migration", + "run-comprehensive-migration-confirm": "This will perform a comprehensive migration to check and fix board data integrity. This may take a few moments. Continue?", + "run-delete-duplicate-empty-lists-migration-confirm": "This will first convert any shared lists to per-swimlane lists, then delete empty lists that have a duplicate list with the same title containing cards. Only truly redundant empty lists will be removed. Continue?", + "run-restore-lost-cards-migration-confirm": "This will create a 'Lost Cards' swimlane and restore all cards and lists with missing swimlaneId or listId. This only affects non-archived items. Continue?", + "run-restore-all-archived-migration-confirm": "This will restore ALL archived swimlanes, lists, and cards, making them visible again. Any items with missing IDs will be automatically fixed. This cannot be easily undone. Continue?", + "run-fix-missing-lists-migration-confirm": "This will detect and repair missing or corrupted lists in the board structure. Continue?", + "run-fix-avatar-urls-migration-confirm": "This will update avatar URLs for board members to use the correct storage backend. Continue?", + "run-fix-all-file-urls-migration-confirm": "This will update all file attachment URLs on this board to use the correct storage backend. Continue?", + "restore-lost-cards-nothing-to-restore": "No lost swimlanes, lists, or cards to restore", + + "migration-progress-title": "Board Migration in Progress", + "migration-progress-overall": "Overall Progress", + "migration-progress-current-step": "Current Step", + "migration-progress-status": "Status", + "migration-progress-details": "Details", + "migration-progress-note": "Please wait while we migrate your board to the latest structure...", + + "step-analyze-board-structure": "Analyze Board Structure", + "step-fix-orphaned-cards": "Fix Orphaned Cards", + "step-convert-shared-lists": "Convert Shared Lists", + "step-ensure-per-swimlane-lists": "Ensure Per-Swimlane Lists", + "step-validate-migration": "Validate Migration", + "step-fix-avatar-urls": "Fix Avatar URLs", + "step-fix-attachment-urls": "Fix Attachment URLs", + "step-analyze-lists": "Analyze Lists", + "step-create-missing-lists": "Create Missing Lists", + "step-update-cards": "Update Cards", + "step-finalize": "Finalize", + "step-delete-duplicate-empty-lists": "Delete Duplicate Empty Lists", + "step-ensure-lost-cards-swimlane": "Ensure Lost Cards Swimlane", + "step-restore-lists": "Restore Lists", + "step-restore-cards": "Restore Cards", + "step-restore-swimlanes": "Restore Swimlanes", + "step-fix-missing-ids": "Fix Missing IDs", + "step-scan-users": "Checking board member avatars", + "step-scan-files": "Checking board file attachments", + "step-fix-file-urls": "Fixing file URLs", "cleanup": "Cleanup", "cleanup-old-jobs": "Cleanup Old Jobs", "completed": "Completed", diff --git a/imports/i18n/data/ve.i18n.json b/imports/i18n/data/ve.i18n.json index eef27e1fe..958d8313a 100644 --- a/imports/i18n/data/ve.i18n.json +++ b/imports/i18n/data/ve.i18n.json @@ -1404,7 +1404,70 @@ "back-to-settings": "Back to Settings", "board-id": "Board ID", "board-migration": "Board Migration", + "board-migrations": "Board Migrations", "card-show-lists-on-minicard": "Show Lists on Minicard", + "comprehensive-board-migration": "Comprehensive Board Migration", + "comprehensive-board-migration-description": "Performs comprehensive checks and fixes for board data integrity, including list ordering, card positions, and swimlane structure.", + "delete-duplicate-empty-lists-migration": "Delete Duplicate Empty Lists", + "delete-duplicate-empty-lists-migration-description": "Safely deletes empty duplicate lists. Only removes lists that have no cards AND have another list with the same title that contains cards.", + "lost-cards": "Lost Cards", + "lost-cards-list": "Restored Items", + "restore-lost-cards-migration": "Restore Lost Cards", + "restore-lost-cards-migration-description": "Finds and restores cards and lists with missing swimlaneId or listId. Creates a 'Lost Cards' swimlane to make all lost items visible again.", + "restore-all-archived-migration": "Restore All Archived", + "restore-all-archived-migration-description": "Restores all archived swimlanes, lists, and cards. Automatically fixes any missing swimlaneId or listId to make items visible.", + "fix-missing-lists-migration": "Fix Missing Lists", + "fix-missing-lists-migration-description": "Detects and repairs missing or corrupted lists in the board structure.", + "fix-avatar-urls-migration": "Fix Avatar URLs", + "fix-avatar-urls-migration-description": "Updates avatar URLs for board members to use the correct storage backend and fixes broken avatar references.", + "fix-all-file-urls-migration": "Fix All File URLs", + "fix-all-file-urls-migration-description": "Updates all file attachment URLs on this board to use the correct storage backend and fixes broken file references.", + "migration-needed": "Migration Needed", + "migration-complete": "Complete", + "migration-running": "Running...", + "migration-successful": "Migration completed successfully", + "migration-failed": "Migration failed", + "migrations": "Migrations", + "migrations-admin-only": "Only board administrators can run migrations", + "migrations-description": "Run data integrity checks and repairs for this board. Each migration can be executed individually.", + "no-issues-found": "No issues found", + "run-migration": "Run Migration", + "run-comprehensive-migration-confirm": "This will perform a comprehensive migration to check and fix board data integrity. This may take a few moments. Continue?", + "run-delete-duplicate-empty-lists-migration-confirm": "This will first convert any shared lists to per-swimlane lists, then delete empty lists that have a duplicate list with the same title containing cards. Only truly redundant empty lists will be removed. Continue?", + "run-restore-lost-cards-migration-confirm": "This will create a 'Lost Cards' swimlane and restore all cards and lists with missing swimlaneId or listId. This only affects non-archived items. Continue?", + "run-restore-all-archived-migration-confirm": "This will restore ALL archived swimlanes, lists, and cards, making them visible again. Any items with missing IDs will be automatically fixed. This cannot be easily undone. Continue?", + "run-fix-missing-lists-migration-confirm": "This will detect and repair missing or corrupted lists in the board structure. Continue?", + "run-fix-avatar-urls-migration-confirm": "This will update avatar URLs for board members to use the correct storage backend. Continue?", + "run-fix-all-file-urls-migration-confirm": "This will update all file attachment URLs on this board to use the correct storage backend. Continue?", + "restore-lost-cards-nothing-to-restore": "No lost swimlanes, lists, or cards to restore", + + "migration-progress-title": "Board Migration in Progress", + "migration-progress-overall": "Overall Progress", + "migration-progress-current-step": "Current Step", + "migration-progress-status": "Status", + "migration-progress-details": "Details", + "migration-progress-note": "Please wait while we migrate your board to the latest structure...", + + "step-analyze-board-structure": "Analyze Board Structure", + "step-fix-orphaned-cards": "Fix Orphaned Cards", + "step-convert-shared-lists": "Convert Shared Lists", + "step-ensure-per-swimlane-lists": "Ensure Per-Swimlane Lists", + "step-validate-migration": "Validate Migration", + "step-fix-avatar-urls": "Fix Avatar URLs", + "step-fix-attachment-urls": "Fix Attachment URLs", + "step-analyze-lists": "Analyze Lists", + "step-create-missing-lists": "Create Missing Lists", + "step-update-cards": "Update Cards", + "step-finalize": "Finalize", + "step-delete-duplicate-empty-lists": "Delete Duplicate Empty Lists", + "step-ensure-lost-cards-swimlane": "Ensure Lost Cards Swimlane", + "step-restore-lists": "Restore Lists", + "step-restore-cards": "Restore Cards", + "step-restore-swimlanes": "Restore Swimlanes", + "step-fix-missing-ids": "Fix Missing IDs", + "step-scan-users": "Checking board member avatars", + "step-scan-files": "Checking board file attachments", + "step-fix-file-urls": "Fixing file URLs", "cleanup": "Cleanup", "cleanup-old-jobs": "Cleanup Old Jobs", "completed": "Completed", diff --git a/imports/i18n/data/vi-VN.i18n.json b/imports/i18n/data/vi-VN.i18n.json index 9dbb74fa4..14e4ab2c2 100644 --- a/imports/i18n/data/vi-VN.i18n.json +++ b/imports/i18n/data/vi-VN.i18n.json @@ -1404,7 +1404,70 @@ "back-to-settings": "Back to Settings", "board-id": "Board ID", "board-migration": "Board Migration", + "board-migrations": "Board Migrations", "card-show-lists-on-minicard": "Show Lists on Minicard", + "comprehensive-board-migration": "Comprehensive Board Migration", + "comprehensive-board-migration-description": "Performs comprehensive checks and fixes for board data integrity, including list ordering, card positions, and swimlane structure.", + "delete-duplicate-empty-lists-migration": "Delete Duplicate Empty Lists", + "delete-duplicate-empty-lists-migration-description": "Safely deletes empty duplicate lists. Only removes lists that have no cards AND have another list with the same title that contains cards.", + "lost-cards": "Lost Cards", + "lost-cards-list": "Restored Items", + "restore-lost-cards-migration": "Restore Lost Cards", + "restore-lost-cards-migration-description": "Finds and restores cards and lists with missing swimlaneId or listId. Creates a 'Lost Cards' swimlane to make all lost items visible again.", + "restore-all-archived-migration": "Restore All Archived", + "restore-all-archived-migration-description": "Restores all archived swimlanes, lists, and cards. Automatically fixes any missing swimlaneId or listId to make items visible.", + "fix-missing-lists-migration": "Fix Missing Lists", + "fix-missing-lists-migration-description": "Detects and repairs missing or corrupted lists in the board structure.", + "fix-avatar-urls-migration": "Fix Avatar URLs", + "fix-avatar-urls-migration-description": "Updates avatar URLs for board members to use the correct storage backend and fixes broken avatar references.", + "fix-all-file-urls-migration": "Fix All File URLs", + "fix-all-file-urls-migration-description": "Updates all file attachment URLs on this board to use the correct storage backend and fixes broken file references.", + "migration-needed": "Migration Needed", + "migration-complete": "Complete", + "migration-running": "Running...", + "migration-successful": "Migration completed successfully", + "migration-failed": "Migration failed", + "migrations": "Migrations", + "migrations-admin-only": "Only board administrators can run migrations", + "migrations-description": "Run data integrity checks and repairs for this board. Each migration can be executed individually.", + "no-issues-found": "No issues found", + "run-migration": "Run Migration", + "run-comprehensive-migration-confirm": "This will perform a comprehensive migration to check and fix board data integrity. This may take a few moments. Continue?", + "run-delete-duplicate-empty-lists-migration-confirm": "This will first convert any shared lists to per-swimlane lists, then delete empty lists that have a duplicate list with the same title containing cards. Only truly redundant empty lists will be removed. Continue?", + "run-restore-lost-cards-migration-confirm": "This will create a 'Lost Cards' swimlane and restore all cards and lists with missing swimlaneId or listId. This only affects non-archived items. Continue?", + "run-restore-all-archived-migration-confirm": "This will restore ALL archived swimlanes, lists, and cards, making them visible again. Any items with missing IDs will be automatically fixed. This cannot be easily undone. Continue?", + "run-fix-missing-lists-migration-confirm": "This will detect and repair missing or corrupted lists in the board structure. Continue?", + "run-fix-avatar-urls-migration-confirm": "This will update avatar URLs for board members to use the correct storage backend. Continue?", + "run-fix-all-file-urls-migration-confirm": "This will update all file attachment URLs on this board to use the correct storage backend. Continue?", + "restore-lost-cards-nothing-to-restore": "No lost swimlanes, lists, or cards to restore", + + "migration-progress-title": "Board Migration in Progress", + "migration-progress-overall": "Overall Progress", + "migration-progress-current-step": "Current Step", + "migration-progress-status": "Status", + "migration-progress-details": "Details", + "migration-progress-note": "Please wait while we migrate your board to the latest structure...", + + "step-analyze-board-structure": "Analyze Board Structure", + "step-fix-orphaned-cards": "Fix Orphaned Cards", + "step-convert-shared-lists": "Convert Shared Lists", + "step-ensure-per-swimlane-lists": "Ensure Per-Swimlane Lists", + "step-validate-migration": "Validate Migration", + "step-fix-avatar-urls": "Fix Avatar URLs", + "step-fix-attachment-urls": "Fix Attachment URLs", + "step-analyze-lists": "Analyze Lists", + "step-create-missing-lists": "Create Missing Lists", + "step-update-cards": "Update Cards", + "step-finalize": "Finalize", + "step-delete-duplicate-empty-lists": "Delete Duplicate Empty Lists", + "step-ensure-lost-cards-swimlane": "Ensure Lost Cards Swimlane", + "step-restore-lists": "Restore Lists", + "step-restore-cards": "Restore Cards", + "step-restore-swimlanes": "Restore Swimlanes", + "step-fix-missing-ids": "Fix Missing IDs", + "step-scan-users": "Checking board member avatars", + "step-scan-files": "Checking board file attachments", + "step-fix-file-urls": "Fixing file URLs", "cleanup": "Cleanup", "cleanup-old-jobs": "Cleanup Old Jobs", "completed": "Completed", diff --git a/imports/i18n/data/vi.i18n.json b/imports/i18n/data/vi.i18n.json index 8843c5833..d7b031a95 100644 --- a/imports/i18n/data/vi.i18n.json +++ b/imports/i18n/data/vi.i18n.json @@ -1404,7 +1404,70 @@ "back-to-settings": "Back to Settings", "board-id": "Board ID", "board-migration": "Board Migration", + "board-migrations": "Board Migrations", "card-show-lists-on-minicard": "Show Lists on Minicard", + "comprehensive-board-migration": "Comprehensive Board Migration", + "comprehensive-board-migration-description": "Performs comprehensive checks and fixes for board data integrity, including list ordering, card positions, and swimlane structure.", + "delete-duplicate-empty-lists-migration": "Delete Duplicate Empty Lists", + "delete-duplicate-empty-lists-migration-description": "Safely deletes empty duplicate lists. Only removes lists that have no cards AND have another list with the same title that contains cards.", + "lost-cards": "Lost Cards", + "lost-cards-list": "Restored Items", + "restore-lost-cards-migration": "Restore Lost Cards", + "restore-lost-cards-migration-description": "Finds and restores cards and lists with missing swimlaneId or listId. Creates a 'Lost Cards' swimlane to make all lost items visible again.", + "restore-all-archived-migration": "Restore All Archived", + "restore-all-archived-migration-description": "Restores all archived swimlanes, lists, and cards. Automatically fixes any missing swimlaneId or listId to make items visible.", + "fix-missing-lists-migration": "Fix Missing Lists", + "fix-missing-lists-migration-description": "Detects and repairs missing or corrupted lists in the board structure.", + "fix-avatar-urls-migration": "Fix Avatar URLs", + "fix-avatar-urls-migration-description": "Updates avatar URLs for board members to use the correct storage backend and fixes broken avatar references.", + "fix-all-file-urls-migration": "Fix All File URLs", + "fix-all-file-urls-migration-description": "Updates all file attachment URLs on this board to use the correct storage backend and fixes broken file references.", + "migration-needed": "Migration Needed", + "migration-complete": "Complete", + "migration-running": "Running...", + "migration-successful": "Migration completed successfully", + "migration-failed": "Migration failed", + "migrations": "Migrations", + "migrations-admin-only": "Only board administrators can run migrations", + "migrations-description": "Run data integrity checks and repairs for this board. Each migration can be executed individually.", + "no-issues-found": "No issues found", + "run-migration": "Run Migration", + "run-comprehensive-migration-confirm": "This will perform a comprehensive migration to check and fix board data integrity. This may take a few moments. Continue?", + "run-delete-duplicate-empty-lists-migration-confirm": "This will first convert any shared lists to per-swimlane lists, then delete empty lists that have a duplicate list with the same title containing cards. Only truly redundant empty lists will be removed. Continue?", + "run-restore-lost-cards-migration-confirm": "This will create a 'Lost Cards' swimlane and restore all cards and lists with missing swimlaneId or listId. This only affects non-archived items. Continue?", + "run-restore-all-archived-migration-confirm": "This will restore ALL archived swimlanes, lists, and cards, making them visible again. Any items with missing IDs will be automatically fixed. This cannot be easily undone. Continue?", + "run-fix-missing-lists-migration-confirm": "This will detect and repair missing or corrupted lists in the board structure. Continue?", + "run-fix-avatar-urls-migration-confirm": "This will update avatar URLs for board members to use the correct storage backend. Continue?", + "run-fix-all-file-urls-migration-confirm": "This will update all file attachment URLs on this board to use the correct storage backend. Continue?", + "restore-lost-cards-nothing-to-restore": "No lost swimlanes, lists, or cards to restore", + + "migration-progress-title": "Board Migration in Progress", + "migration-progress-overall": "Overall Progress", + "migration-progress-current-step": "Current Step", + "migration-progress-status": "Trạng thái", + "migration-progress-details": "Details", + "migration-progress-note": "Please wait while we migrate your board to the latest structure...", + + "step-analyze-board-structure": "Analyze Board Structure", + "step-fix-orphaned-cards": "Fix Orphaned Cards", + "step-convert-shared-lists": "Convert Shared Lists", + "step-ensure-per-swimlane-lists": "Ensure Per-Swimlane Lists", + "step-validate-migration": "Validate Migration", + "step-fix-avatar-urls": "Fix Avatar URLs", + "step-fix-attachment-urls": "Fix Attachment URLs", + "step-analyze-lists": "Analyze Lists", + "step-create-missing-lists": "Create Missing Lists", + "step-update-cards": "Update Cards", + "step-finalize": "Finalize", + "step-delete-duplicate-empty-lists": "Delete Duplicate Empty Lists", + "step-ensure-lost-cards-swimlane": "Ensure Lost Cards Swimlane", + "step-restore-lists": "Restore Lists", + "step-restore-cards": "Restore Cards", + "step-restore-swimlanes": "Restore Swimlanes", + "step-fix-missing-ids": "Fix Missing IDs", + "step-scan-users": "Checking board member avatars", + "step-scan-files": "Checking board file attachments", + "step-fix-file-urls": "Fixing file URLs", "cleanup": "Cleanup", "cleanup-old-jobs": "Cleanup Old Jobs", "completed": "Đã hoàn thành", diff --git a/imports/i18n/data/vl-SS.i18n.json b/imports/i18n/data/vl-SS.i18n.json index eef27e1fe..958d8313a 100644 --- a/imports/i18n/data/vl-SS.i18n.json +++ b/imports/i18n/data/vl-SS.i18n.json @@ -1404,7 +1404,70 @@ "back-to-settings": "Back to Settings", "board-id": "Board ID", "board-migration": "Board Migration", + "board-migrations": "Board Migrations", "card-show-lists-on-minicard": "Show Lists on Minicard", + "comprehensive-board-migration": "Comprehensive Board Migration", + "comprehensive-board-migration-description": "Performs comprehensive checks and fixes for board data integrity, including list ordering, card positions, and swimlane structure.", + "delete-duplicate-empty-lists-migration": "Delete Duplicate Empty Lists", + "delete-duplicate-empty-lists-migration-description": "Safely deletes empty duplicate lists. Only removes lists that have no cards AND have another list with the same title that contains cards.", + "lost-cards": "Lost Cards", + "lost-cards-list": "Restored Items", + "restore-lost-cards-migration": "Restore Lost Cards", + "restore-lost-cards-migration-description": "Finds and restores cards and lists with missing swimlaneId or listId. Creates a 'Lost Cards' swimlane to make all lost items visible again.", + "restore-all-archived-migration": "Restore All Archived", + "restore-all-archived-migration-description": "Restores all archived swimlanes, lists, and cards. Automatically fixes any missing swimlaneId or listId to make items visible.", + "fix-missing-lists-migration": "Fix Missing Lists", + "fix-missing-lists-migration-description": "Detects and repairs missing or corrupted lists in the board structure.", + "fix-avatar-urls-migration": "Fix Avatar URLs", + "fix-avatar-urls-migration-description": "Updates avatar URLs for board members to use the correct storage backend and fixes broken avatar references.", + "fix-all-file-urls-migration": "Fix All File URLs", + "fix-all-file-urls-migration-description": "Updates all file attachment URLs on this board to use the correct storage backend and fixes broken file references.", + "migration-needed": "Migration Needed", + "migration-complete": "Complete", + "migration-running": "Running...", + "migration-successful": "Migration completed successfully", + "migration-failed": "Migration failed", + "migrations": "Migrations", + "migrations-admin-only": "Only board administrators can run migrations", + "migrations-description": "Run data integrity checks and repairs for this board. Each migration can be executed individually.", + "no-issues-found": "No issues found", + "run-migration": "Run Migration", + "run-comprehensive-migration-confirm": "This will perform a comprehensive migration to check and fix board data integrity. This may take a few moments. Continue?", + "run-delete-duplicate-empty-lists-migration-confirm": "This will first convert any shared lists to per-swimlane lists, then delete empty lists that have a duplicate list with the same title containing cards. Only truly redundant empty lists will be removed. Continue?", + "run-restore-lost-cards-migration-confirm": "This will create a 'Lost Cards' swimlane and restore all cards and lists with missing swimlaneId or listId. This only affects non-archived items. Continue?", + "run-restore-all-archived-migration-confirm": "This will restore ALL archived swimlanes, lists, and cards, making them visible again. Any items with missing IDs will be automatically fixed. This cannot be easily undone. Continue?", + "run-fix-missing-lists-migration-confirm": "This will detect and repair missing or corrupted lists in the board structure. Continue?", + "run-fix-avatar-urls-migration-confirm": "This will update avatar URLs for board members to use the correct storage backend. Continue?", + "run-fix-all-file-urls-migration-confirm": "This will update all file attachment URLs on this board to use the correct storage backend. Continue?", + "restore-lost-cards-nothing-to-restore": "No lost swimlanes, lists, or cards to restore", + + "migration-progress-title": "Board Migration in Progress", + "migration-progress-overall": "Overall Progress", + "migration-progress-current-step": "Current Step", + "migration-progress-status": "Status", + "migration-progress-details": "Details", + "migration-progress-note": "Please wait while we migrate your board to the latest structure...", + + "step-analyze-board-structure": "Analyze Board Structure", + "step-fix-orphaned-cards": "Fix Orphaned Cards", + "step-convert-shared-lists": "Convert Shared Lists", + "step-ensure-per-swimlane-lists": "Ensure Per-Swimlane Lists", + "step-validate-migration": "Validate Migration", + "step-fix-avatar-urls": "Fix Avatar URLs", + "step-fix-attachment-urls": "Fix Attachment URLs", + "step-analyze-lists": "Analyze Lists", + "step-create-missing-lists": "Create Missing Lists", + "step-update-cards": "Update Cards", + "step-finalize": "Finalize", + "step-delete-duplicate-empty-lists": "Delete Duplicate Empty Lists", + "step-ensure-lost-cards-swimlane": "Ensure Lost Cards Swimlane", + "step-restore-lists": "Restore Lists", + "step-restore-cards": "Restore Cards", + "step-restore-swimlanes": "Restore Swimlanes", + "step-fix-missing-ids": "Fix Missing IDs", + "step-scan-users": "Checking board member avatars", + "step-scan-files": "Checking board file attachments", + "step-fix-file-urls": "Fixing file URLs", "cleanup": "Cleanup", "cleanup-old-jobs": "Cleanup Old Jobs", "completed": "Completed", diff --git a/imports/i18n/data/vo.i18n.json b/imports/i18n/data/vo.i18n.json index eef27e1fe..958d8313a 100644 --- a/imports/i18n/data/vo.i18n.json +++ b/imports/i18n/data/vo.i18n.json @@ -1404,7 +1404,70 @@ "back-to-settings": "Back to Settings", "board-id": "Board ID", "board-migration": "Board Migration", + "board-migrations": "Board Migrations", "card-show-lists-on-minicard": "Show Lists on Minicard", + "comprehensive-board-migration": "Comprehensive Board Migration", + "comprehensive-board-migration-description": "Performs comprehensive checks and fixes for board data integrity, including list ordering, card positions, and swimlane structure.", + "delete-duplicate-empty-lists-migration": "Delete Duplicate Empty Lists", + "delete-duplicate-empty-lists-migration-description": "Safely deletes empty duplicate lists. Only removes lists that have no cards AND have another list with the same title that contains cards.", + "lost-cards": "Lost Cards", + "lost-cards-list": "Restored Items", + "restore-lost-cards-migration": "Restore Lost Cards", + "restore-lost-cards-migration-description": "Finds and restores cards and lists with missing swimlaneId or listId. Creates a 'Lost Cards' swimlane to make all lost items visible again.", + "restore-all-archived-migration": "Restore All Archived", + "restore-all-archived-migration-description": "Restores all archived swimlanes, lists, and cards. Automatically fixes any missing swimlaneId or listId to make items visible.", + "fix-missing-lists-migration": "Fix Missing Lists", + "fix-missing-lists-migration-description": "Detects and repairs missing or corrupted lists in the board structure.", + "fix-avatar-urls-migration": "Fix Avatar URLs", + "fix-avatar-urls-migration-description": "Updates avatar URLs for board members to use the correct storage backend and fixes broken avatar references.", + "fix-all-file-urls-migration": "Fix All File URLs", + "fix-all-file-urls-migration-description": "Updates all file attachment URLs on this board to use the correct storage backend and fixes broken file references.", + "migration-needed": "Migration Needed", + "migration-complete": "Complete", + "migration-running": "Running...", + "migration-successful": "Migration completed successfully", + "migration-failed": "Migration failed", + "migrations": "Migrations", + "migrations-admin-only": "Only board administrators can run migrations", + "migrations-description": "Run data integrity checks and repairs for this board. Each migration can be executed individually.", + "no-issues-found": "No issues found", + "run-migration": "Run Migration", + "run-comprehensive-migration-confirm": "This will perform a comprehensive migration to check and fix board data integrity. This may take a few moments. Continue?", + "run-delete-duplicate-empty-lists-migration-confirm": "This will first convert any shared lists to per-swimlane lists, then delete empty lists that have a duplicate list with the same title containing cards. Only truly redundant empty lists will be removed. Continue?", + "run-restore-lost-cards-migration-confirm": "This will create a 'Lost Cards' swimlane and restore all cards and lists with missing swimlaneId or listId. This only affects non-archived items. Continue?", + "run-restore-all-archived-migration-confirm": "This will restore ALL archived swimlanes, lists, and cards, making them visible again. Any items with missing IDs will be automatically fixed. This cannot be easily undone. Continue?", + "run-fix-missing-lists-migration-confirm": "This will detect and repair missing or corrupted lists in the board structure. Continue?", + "run-fix-avatar-urls-migration-confirm": "This will update avatar URLs for board members to use the correct storage backend. Continue?", + "run-fix-all-file-urls-migration-confirm": "This will update all file attachment URLs on this board to use the correct storage backend. Continue?", + "restore-lost-cards-nothing-to-restore": "No lost swimlanes, lists, or cards to restore", + + "migration-progress-title": "Board Migration in Progress", + "migration-progress-overall": "Overall Progress", + "migration-progress-current-step": "Current Step", + "migration-progress-status": "Status", + "migration-progress-details": "Details", + "migration-progress-note": "Please wait while we migrate your board to the latest structure...", + + "step-analyze-board-structure": "Analyze Board Structure", + "step-fix-orphaned-cards": "Fix Orphaned Cards", + "step-convert-shared-lists": "Convert Shared Lists", + "step-ensure-per-swimlane-lists": "Ensure Per-Swimlane Lists", + "step-validate-migration": "Validate Migration", + "step-fix-avatar-urls": "Fix Avatar URLs", + "step-fix-attachment-urls": "Fix Attachment URLs", + "step-analyze-lists": "Analyze Lists", + "step-create-missing-lists": "Create Missing Lists", + "step-update-cards": "Update Cards", + "step-finalize": "Finalize", + "step-delete-duplicate-empty-lists": "Delete Duplicate Empty Lists", + "step-ensure-lost-cards-swimlane": "Ensure Lost Cards Swimlane", + "step-restore-lists": "Restore Lists", + "step-restore-cards": "Restore Cards", + "step-restore-swimlanes": "Restore Swimlanes", + "step-fix-missing-ids": "Fix Missing IDs", + "step-scan-users": "Checking board member avatars", + "step-scan-files": "Checking board file attachments", + "step-fix-file-urls": "Fixing file URLs", "cleanup": "Cleanup", "cleanup-old-jobs": "Cleanup Old Jobs", "completed": "Completed", diff --git a/imports/i18n/data/wa-RR.i18n.json b/imports/i18n/data/wa-RR.i18n.json index eef27e1fe..958d8313a 100644 --- a/imports/i18n/data/wa-RR.i18n.json +++ b/imports/i18n/data/wa-RR.i18n.json @@ -1404,7 +1404,70 @@ "back-to-settings": "Back to Settings", "board-id": "Board ID", "board-migration": "Board Migration", + "board-migrations": "Board Migrations", "card-show-lists-on-minicard": "Show Lists on Minicard", + "comprehensive-board-migration": "Comprehensive Board Migration", + "comprehensive-board-migration-description": "Performs comprehensive checks and fixes for board data integrity, including list ordering, card positions, and swimlane structure.", + "delete-duplicate-empty-lists-migration": "Delete Duplicate Empty Lists", + "delete-duplicate-empty-lists-migration-description": "Safely deletes empty duplicate lists. Only removes lists that have no cards AND have another list with the same title that contains cards.", + "lost-cards": "Lost Cards", + "lost-cards-list": "Restored Items", + "restore-lost-cards-migration": "Restore Lost Cards", + "restore-lost-cards-migration-description": "Finds and restores cards and lists with missing swimlaneId or listId. Creates a 'Lost Cards' swimlane to make all lost items visible again.", + "restore-all-archived-migration": "Restore All Archived", + "restore-all-archived-migration-description": "Restores all archived swimlanes, lists, and cards. Automatically fixes any missing swimlaneId or listId to make items visible.", + "fix-missing-lists-migration": "Fix Missing Lists", + "fix-missing-lists-migration-description": "Detects and repairs missing or corrupted lists in the board structure.", + "fix-avatar-urls-migration": "Fix Avatar URLs", + "fix-avatar-urls-migration-description": "Updates avatar URLs for board members to use the correct storage backend and fixes broken avatar references.", + "fix-all-file-urls-migration": "Fix All File URLs", + "fix-all-file-urls-migration-description": "Updates all file attachment URLs on this board to use the correct storage backend and fixes broken file references.", + "migration-needed": "Migration Needed", + "migration-complete": "Complete", + "migration-running": "Running...", + "migration-successful": "Migration completed successfully", + "migration-failed": "Migration failed", + "migrations": "Migrations", + "migrations-admin-only": "Only board administrators can run migrations", + "migrations-description": "Run data integrity checks and repairs for this board. Each migration can be executed individually.", + "no-issues-found": "No issues found", + "run-migration": "Run Migration", + "run-comprehensive-migration-confirm": "This will perform a comprehensive migration to check and fix board data integrity. This may take a few moments. Continue?", + "run-delete-duplicate-empty-lists-migration-confirm": "This will first convert any shared lists to per-swimlane lists, then delete empty lists that have a duplicate list with the same title containing cards. Only truly redundant empty lists will be removed. Continue?", + "run-restore-lost-cards-migration-confirm": "This will create a 'Lost Cards' swimlane and restore all cards and lists with missing swimlaneId or listId. This only affects non-archived items. Continue?", + "run-restore-all-archived-migration-confirm": "This will restore ALL archived swimlanes, lists, and cards, making them visible again. Any items with missing IDs will be automatically fixed. This cannot be easily undone. Continue?", + "run-fix-missing-lists-migration-confirm": "This will detect and repair missing or corrupted lists in the board structure. Continue?", + "run-fix-avatar-urls-migration-confirm": "This will update avatar URLs for board members to use the correct storage backend. Continue?", + "run-fix-all-file-urls-migration-confirm": "This will update all file attachment URLs on this board to use the correct storage backend. Continue?", + "restore-lost-cards-nothing-to-restore": "No lost swimlanes, lists, or cards to restore", + + "migration-progress-title": "Board Migration in Progress", + "migration-progress-overall": "Overall Progress", + "migration-progress-current-step": "Current Step", + "migration-progress-status": "Status", + "migration-progress-details": "Details", + "migration-progress-note": "Please wait while we migrate your board to the latest structure...", + + "step-analyze-board-structure": "Analyze Board Structure", + "step-fix-orphaned-cards": "Fix Orphaned Cards", + "step-convert-shared-lists": "Convert Shared Lists", + "step-ensure-per-swimlane-lists": "Ensure Per-Swimlane Lists", + "step-validate-migration": "Validate Migration", + "step-fix-avatar-urls": "Fix Avatar URLs", + "step-fix-attachment-urls": "Fix Attachment URLs", + "step-analyze-lists": "Analyze Lists", + "step-create-missing-lists": "Create Missing Lists", + "step-update-cards": "Update Cards", + "step-finalize": "Finalize", + "step-delete-duplicate-empty-lists": "Delete Duplicate Empty Lists", + "step-ensure-lost-cards-swimlane": "Ensure Lost Cards Swimlane", + "step-restore-lists": "Restore Lists", + "step-restore-cards": "Restore Cards", + "step-restore-swimlanes": "Restore Swimlanes", + "step-fix-missing-ids": "Fix Missing IDs", + "step-scan-users": "Checking board member avatars", + "step-scan-files": "Checking board file attachments", + "step-fix-file-urls": "Fixing file URLs", "cleanup": "Cleanup", "cleanup-old-jobs": "Cleanup Old Jobs", "completed": "Completed", diff --git a/imports/i18n/data/wa.i18n.json b/imports/i18n/data/wa.i18n.json index eef27e1fe..958d8313a 100644 --- a/imports/i18n/data/wa.i18n.json +++ b/imports/i18n/data/wa.i18n.json @@ -1404,7 +1404,70 @@ "back-to-settings": "Back to Settings", "board-id": "Board ID", "board-migration": "Board Migration", + "board-migrations": "Board Migrations", "card-show-lists-on-minicard": "Show Lists on Minicard", + "comprehensive-board-migration": "Comprehensive Board Migration", + "comprehensive-board-migration-description": "Performs comprehensive checks and fixes for board data integrity, including list ordering, card positions, and swimlane structure.", + "delete-duplicate-empty-lists-migration": "Delete Duplicate Empty Lists", + "delete-duplicate-empty-lists-migration-description": "Safely deletes empty duplicate lists. Only removes lists that have no cards AND have another list with the same title that contains cards.", + "lost-cards": "Lost Cards", + "lost-cards-list": "Restored Items", + "restore-lost-cards-migration": "Restore Lost Cards", + "restore-lost-cards-migration-description": "Finds and restores cards and lists with missing swimlaneId or listId. Creates a 'Lost Cards' swimlane to make all lost items visible again.", + "restore-all-archived-migration": "Restore All Archived", + "restore-all-archived-migration-description": "Restores all archived swimlanes, lists, and cards. Automatically fixes any missing swimlaneId or listId to make items visible.", + "fix-missing-lists-migration": "Fix Missing Lists", + "fix-missing-lists-migration-description": "Detects and repairs missing or corrupted lists in the board structure.", + "fix-avatar-urls-migration": "Fix Avatar URLs", + "fix-avatar-urls-migration-description": "Updates avatar URLs for board members to use the correct storage backend and fixes broken avatar references.", + "fix-all-file-urls-migration": "Fix All File URLs", + "fix-all-file-urls-migration-description": "Updates all file attachment URLs on this board to use the correct storage backend and fixes broken file references.", + "migration-needed": "Migration Needed", + "migration-complete": "Complete", + "migration-running": "Running...", + "migration-successful": "Migration completed successfully", + "migration-failed": "Migration failed", + "migrations": "Migrations", + "migrations-admin-only": "Only board administrators can run migrations", + "migrations-description": "Run data integrity checks and repairs for this board. Each migration can be executed individually.", + "no-issues-found": "No issues found", + "run-migration": "Run Migration", + "run-comprehensive-migration-confirm": "This will perform a comprehensive migration to check and fix board data integrity. This may take a few moments. Continue?", + "run-delete-duplicate-empty-lists-migration-confirm": "This will first convert any shared lists to per-swimlane lists, then delete empty lists that have a duplicate list with the same title containing cards. Only truly redundant empty lists will be removed. Continue?", + "run-restore-lost-cards-migration-confirm": "This will create a 'Lost Cards' swimlane and restore all cards and lists with missing swimlaneId or listId. This only affects non-archived items. Continue?", + "run-restore-all-archived-migration-confirm": "This will restore ALL archived swimlanes, lists, and cards, making them visible again. Any items with missing IDs will be automatically fixed. This cannot be easily undone. Continue?", + "run-fix-missing-lists-migration-confirm": "This will detect and repair missing or corrupted lists in the board structure. Continue?", + "run-fix-avatar-urls-migration-confirm": "This will update avatar URLs for board members to use the correct storage backend. Continue?", + "run-fix-all-file-urls-migration-confirm": "This will update all file attachment URLs on this board to use the correct storage backend. Continue?", + "restore-lost-cards-nothing-to-restore": "No lost swimlanes, lists, or cards to restore", + + "migration-progress-title": "Board Migration in Progress", + "migration-progress-overall": "Overall Progress", + "migration-progress-current-step": "Current Step", + "migration-progress-status": "Status", + "migration-progress-details": "Details", + "migration-progress-note": "Please wait while we migrate your board to the latest structure...", + + "step-analyze-board-structure": "Analyze Board Structure", + "step-fix-orphaned-cards": "Fix Orphaned Cards", + "step-convert-shared-lists": "Convert Shared Lists", + "step-ensure-per-swimlane-lists": "Ensure Per-Swimlane Lists", + "step-validate-migration": "Validate Migration", + "step-fix-avatar-urls": "Fix Avatar URLs", + "step-fix-attachment-urls": "Fix Attachment URLs", + "step-analyze-lists": "Analyze Lists", + "step-create-missing-lists": "Create Missing Lists", + "step-update-cards": "Update Cards", + "step-finalize": "Finalize", + "step-delete-duplicate-empty-lists": "Delete Duplicate Empty Lists", + "step-ensure-lost-cards-swimlane": "Ensure Lost Cards Swimlane", + "step-restore-lists": "Restore Lists", + "step-restore-cards": "Restore Cards", + "step-restore-swimlanes": "Restore Swimlanes", + "step-fix-missing-ids": "Fix Missing IDs", + "step-scan-users": "Checking board member avatars", + "step-scan-files": "Checking board file attachments", + "step-fix-file-urls": "Fixing file URLs", "cleanup": "Cleanup", "cleanup-old-jobs": "Cleanup Old Jobs", "completed": "Completed", diff --git a/imports/i18n/data/wo.i18n.json b/imports/i18n/data/wo.i18n.json index eef27e1fe..958d8313a 100644 --- a/imports/i18n/data/wo.i18n.json +++ b/imports/i18n/data/wo.i18n.json @@ -1404,7 +1404,70 @@ "back-to-settings": "Back to Settings", "board-id": "Board ID", "board-migration": "Board Migration", + "board-migrations": "Board Migrations", "card-show-lists-on-minicard": "Show Lists on Minicard", + "comprehensive-board-migration": "Comprehensive Board Migration", + "comprehensive-board-migration-description": "Performs comprehensive checks and fixes for board data integrity, including list ordering, card positions, and swimlane structure.", + "delete-duplicate-empty-lists-migration": "Delete Duplicate Empty Lists", + "delete-duplicate-empty-lists-migration-description": "Safely deletes empty duplicate lists. Only removes lists that have no cards AND have another list with the same title that contains cards.", + "lost-cards": "Lost Cards", + "lost-cards-list": "Restored Items", + "restore-lost-cards-migration": "Restore Lost Cards", + "restore-lost-cards-migration-description": "Finds and restores cards and lists with missing swimlaneId or listId. Creates a 'Lost Cards' swimlane to make all lost items visible again.", + "restore-all-archived-migration": "Restore All Archived", + "restore-all-archived-migration-description": "Restores all archived swimlanes, lists, and cards. Automatically fixes any missing swimlaneId or listId to make items visible.", + "fix-missing-lists-migration": "Fix Missing Lists", + "fix-missing-lists-migration-description": "Detects and repairs missing or corrupted lists in the board structure.", + "fix-avatar-urls-migration": "Fix Avatar URLs", + "fix-avatar-urls-migration-description": "Updates avatar URLs for board members to use the correct storage backend and fixes broken avatar references.", + "fix-all-file-urls-migration": "Fix All File URLs", + "fix-all-file-urls-migration-description": "Updates all file attachment URLs on this board to use the correct storage backend and fixes broken file references.", + "migration-needed": "Migration Needed", + "migration-complete": "Complete", + "migration-running": "Running...", + "migration-successful": "Migration completed successfully", + "migration-failed": "Migration failed", + "migrations": "Migrations", + "migrations-admin-only": "Only board administrators can run migrations", + "migrations-description": "Run data integrity checks and repairs for this board. Each migration can be executed individually.", + "no-issues-found": "No issues found", + "run-migration": "Run Migration", + "run-comprehensive-migration-confirm": "This will perform a comprehensive migration to check and fix board data integrity. This may take a few moments. Continue?", + "run-delete-duplicate-empty-lists-migration-confirm": "This will first convert any shared lists to per-swimlane lists, then delete empty lists that have a duplicate list with the same title containing cards. Only truly redundant empty lists will be removed. Continue?", + "run-restore-lost-cards-migration-confirm": "This will create a 'Lost Cards' swimlane and restore all cards and lists with missing swimlaneId or listId. This only affects non-archived items. Continue?", + "run-restore-all-archived-migration-confirm": "This will restore ALL archived swimlanes, lists, and cards, making them visible again. Any items with missing IDs will be automatically fixed. This cannot be easily undone. Continue?", + "run-fix-missing-lists-migration-confirm": "This will detect and repair missing or corrupted lists in the board structure. Continue?", + "run-fix-avatar-urls-migration-confirm": "This will update avatar URLs for board members to use the correct storage backend. Continue?", + "run-fix-all-file-urls-migration-confirm": "This will update all file attachment URLs on this board to use the correct storage backend. Continue?", + "restore-lost-cards-nothing-to-restore": "No lost swimlanes, lists, or cards to restore", + + "migration-progress-title": "Board Migration in Progress", + "migration-progress-overall": "Overall Progress", + "migration-progress-current-step": "Current Step", + "migration-progress-status": "Status", + "migration-progress-details": "Details", + "migration-progress-note": "Please wait while we migrate your board to the latest structure...", + + "step-analyze-board-structure": "Analyze Board Structure", + "step-fix-orphaned-cards": "Fix Orphaned Cards", + "step-convert-shared-lists": "Convert Shared Lists", + "step-ensure-per-swimlane-lists": "Ensure Per-Swimlane Lists", + "step-validate-migration": "Validate Migration", + "step-fix-avatar-urls": "Fix Avatar URLs", + "step-fix-attachment-urls": "Fix Attachment URLs", + "step-analyze-lists": "Analyze Lists", + "step-create-missing-lists": "Create Missing Lists", + "step-update-cards": "Update Cards", + "step-finalize": "Finalize", + "step-delete-duplicate-empty-lists": "Delete Duplicate Empty Lists", + "step-ensure-lost-cards-swimlane": "Ensure Lost Cards Swimlane", + "step-restore-lists": "Restore Lists", + "step-restore-cards": "Restore Cards", + "step-restore-swimlanes": "Restore Swimlanes", + "step-fix-missing-ids": "Fix Missing IDs", + "step-scan-users": "Checking board member avatars", + "step-scan-files": "Checking board file attachments", + "step-fix-file-urls": "Fixing file URLs", "cleanup": "Cleanup", "cleanup-old-jobs": "Cleanup Old Jobs", "completed": "Completed", diff --git a/imports/i18n/data/wuu-Hans.i18n.json b/imports/i18n/data/wuu-Hans.i18n.json index eef27e1fe..958d8313a 100644 --- a/imports/i18n/data/wuu-Hans.i18n.json +++ b/imports/i18n/data/wuu-Hans.i18n.json @@ -1404,7 +1404,70 @@ "back-to-settings": "Back to Settings", "board-id": "Board ID", "board-migration": "Board Migration", + "board-migrations": "Board Migrations", "card-show-lists-on-minicard": "Show Lists on Minicard", + "comprehensive-board-migration": "Comprehensive Board Migration", + "comprehensive-board-migration-description": "Performs comprehensive checks and fixes for board data integrity, including list ordering, card positions, and swimlane structure.", + "delete-duplicate-empty-lists-migration": "Delete Duplicate Empty Lists", + "delete-duplicate-empty-lists-migration-description": "Safely deletes empty duplicate lists. Only removes lists that have no cards AND have another list with the same title that contains cards.", + "lost-cards": "Lost Cards", + "lost-cards-list": "Restored Items", + "restore-lost-cards-migration": "Restore Lost Cards", + "restore-lost-cards-migration-description": "Finds and restores cards and lists with missing swimlaneId or listId. Creates a 'Lost Cards' swimlane to make all lost items visible again.", + "restore-all-archived-migration": "Restore All Archived", + "restore-all-archived-migration-description": "Restores all archived swimlanes, lists, and cards. Automatically fixes any missing swimlaneId or listId to make items visible.", + "fix-missing-lists-migration": "Fix Missing Lists", + "fix-missing-lists-migration-description": "Detects and repairs missing or corrupted lists in the board structure.", + "fix-avatar-urls-migration": "Fix Avatar URLs", + "fix-avatar-urls-migration-description": "Updates avatar URLs for board members to use the correct storage backend and fixes broken avatar references.", + "fix-all-file-urls-migration": "Fix All File URLs", + "fix-all-file-urls-migration-description": "Updates all file attachment URLs on this board to use the correct storage backend and fixes broken file references.", + "migration-needed": "Migration Needed", + "migration-complete": "Complete", + "migration-running": "Running...", + "migration-successful": "Migration completed successfully", + "migration-failed": "Migration failed", + "migrations": "Migrations", + "migrations-admin-only": "Only board administrators can run migrations", + "migrations-description": "Run data integrity checks and repairs for this board. Each migration can be executed individually.", + "no-issues-found": "No issues found", + "run-migration": "Run Migration", + "run-comprehensive-migration-confirm": "This will perform a comprehensive migration to check and fix board data integrity. This may take a few moments. Continue?", + "run-delete-duplicate-empty-lists-migration-confirm": "This will first convert any shared lists to per-swimlane lists, then delete empty lists that have a duplicate list with the same title containing cards. Only truly redundant empty lists will be removed. Continue?", + "run-restore-lost-cards-migration-confirm": "This will create a 'Lost Cards' swimlane and restore all cards and lists with missing swimlaneId or listId. This only affects non-archived items. Continue?", + "run-restore-all-archived-migration-confirm": "This will restore ALL archived swimlanes, lists, and cards, making them visible again. Any items with missing IDs will be automatically fixed. This cannot be easily undone. Continue?", + "run-fix-missing-lists-migration-confirm": "This will detect and repair missing or corrupted lists in the board structure. Continue?", + "run-fix-avatar-urls-migration-confirm": "This will update avatar URLs for board members to use the correct storage backend. Continue?", + "run-fix-all-file-urls-migration-confirm": "This will update all file attachment URLs on this board to use the correct storage backend. Continue?", + "restore-lost-cards-nothing-to-restore": "No lost swimlanes, lists, or cards to restore", + + "migration-progress-title": "Board Migration in Progress", + "migration-progress-overall": "Overall Progress", + "migration-progress-current-step": "Current Step", + "migration-progress-status": "Status", + "migration-progress-details": "Details", + "migration-progress-note": "Please wait while we migrate your board to the latest structure...", + + "step-analyze-board-structure": "Analyze Board Structure", + "step-fix-orphaned-cards": "Fix Orphaned Cards", + "step-convert-shared-lists": "Convert Shared Lists", + "step-ensure-per-swimlane-lists": "Ensure Per-Swimlane Lists", + "step-validate-migration": "Validate Migration", + "step-fix-avatar-urls": "Fix Avatar URLs", + "step-fix-attachment-urls": "Fix Attachment URLs", + "step-analyze-lists": "Analyze Lists", + "step-create-missing-lists": "Create Missing Lists", + "step-update-cards": "Update Cards", + "step-finalize": "Finalize", + "step-delete-duplicate-empty-lists": "Delete Duplicate Empty Lists", + "step-ensure-lost-cards-swimlane": "Ensure Lost Cards Swimlane", + "step-restore-lists": "Restore Lists", + "step-restore-cards": "Restore Cards", + "step-restore-swimlanes": "Restore Swimlanes", + "step-fix-missing-ids": "Fix Missing IDs", + "step-scan-users": "Checking board member avatars", + "step-scan-files": "Checking board file attachments", + "step-fix-file-urls": "Fixing file URLs", "cleanup": "Cleanup", "cleanup-old-jobs": "Cleanup Old Jobs", "completed": "Completed", diff --git a/imports/i18n/data/xh.i18n.json b/imports/i18n/data/xh.i18n.json index eef27e1fe..958d8313a 100644 --- a/imports/i18n/data/xh.i18n.json +++ b/imports/i18n/data/xh.i18n.json @@ -1404,7 +1404,70 @@ "back-to-settings": "Back to Settings", "board-id": "Board ID", "board-migration": "Board Migration", + "board-migrations": "Board Migrations", "card-show-lists-on-minicard": "Show Lists on Minicard", + "comprehensive-board-migration": "Comprehensive Board Migration", + "comprehensive-board-migration-description": "Performs comprehensive checks and fixes for board data integrity, including list ordering, card positions, and swimlane structure.", + "delete-duplicate-empty-lists-migration": "Delete Duplicate Empty Lists", + "delete-duplicate-empty-lists-migration-description": "Safely deletes empty duplicate lists. Only removes lists that have no cards AND have another list with the same title that contains cards.", + "lost-cards": "Lost Cards", + "lost-cards-list": "Restored Items", + "restore-lost-cards-migration": "Restore Lost Cards", + "restore-lost-cards-migration-description": "Finds and restores cards and lists with missing swimlaneId or listId. Creates a 'Lost Cards' swimlane to make all lost items visible again.", + "restore-all-archived-migration": "Restore All Archived", + "restore-all-archived-migration-description": "Restores all archived swimlanes, lists, and cards. Automatically fixes any missing swimlaneId or listId to make items visible.", + "fix-missing-lists-migration": "Fix Missing Lists", + "fix-missing-lists-migration-description": "Detects and repairs missing or corrupted lists in the board structure.", + "fix-avatar-urls-migration": "Fix Avatar URLs", + "fix-avatar-urls-migration-description": "Updates avatar URLs for board members to use the correct storage backend and fixes broken avatar references.", + "fix-all-file-urls-migration": "Fix All File URLs", + "fix-all-file-urls-migration-description": "Updates all file attachment URLs on this board to use the correct storage backend and fixes broken file references.", + "migration-needed": "Migration Needed", + "migration-complete": "Complete", + "migration-running": "Running...", + "migration-successful": "Migration completed successfully", + "migration-failed": "Migration failed", + "migrations": "Migrations", + "migrations-admin-only": "Only board administrators can run migrations", + "migrations-description": "Run data integrity checks and repairs for this board. Each migration can be executed individually.", + "no-issues-found": "No issues found", + "run-migration": "Run Migration", + "run-comprehensive-migration-confirm": "This will perform a comprehensive migration to check and fix board data integrity. This may take a few moments. Continue?", + "run-delete-duplicate-empty-lists-migration-confirm": "This will first convert any shared lists to per-swimlane lists, then delete empty lists that have a duplicate list with the same title containing cards. Only truly redundant empty lists will be removed. Continue?", + "run-restore-lost-cards-migration-confirm": "This will create a 'Lost Cards' swimlane and restore all cards and lists with missing swimlaneId or listId. This only affects non-archived items. Continue?", + "run-restore-all-archived-migration-confirm": "This will restore ALL archived swimlanes, lists, and cards, making them visible again. Any items with missing IDs will be automatically fixed. This cannot be easily undone. Continue?", + "run-fix-missing-lists-migration-confirm": "This will detect and repair missing or corrupted lists in the board structure. Continue?", + "run-fix-avatar-urls-migration-confirm": "This will update avatar URLs for board members to use the correct storage backend. Continue?", + "run-fix-all-file-urls-migration-confirm": "This will update all file attachment URLs on this board to use the correct storage backend. Continue?", + "restore-lost-cards-nothing-to-restore": "No lost swimlanes, lists, or cards to restore", + + "migration-progress-title": "Board Migration in Progress", + "migration-progress-overall": "Overall Progress", + "migration-progress-current-step": "Current Step", + "migration-progress-status": "Status", + "migration-progress-details": "Details", + "migration-progress-note": "Please wait while we migrate your board to the latest structure...", + + "step-analyze-board-structure": "Analyze Board Structure", + "step-fix-orphaned-cards": "Fix Orphaned Cards", + "step-convert-shared-lists": "Convert Shared Lists", + "step-ensure-per-swimlane-lists": "Ensure Per-Swimlane Lists", + "step-validate-migration": "Validate Migration", + "step-fix-avatar-urls": "Fix Avatar URLs", + "step-fix-attachment-urls": "Fix Attachment URLs", + "step-analyze-lists": "Analyze Lists", + "step-create-missing-lists": "Create Missing Lists", + "step-update-cards": "Update Cards", + "step-finalize": "Finalize", + "step-delete-duplicate-empty-lists": "Delete Duplicate Empty Lists", + "step-ensure-lost-cards-swimlane": "Ensure Lost Cards Swimlane", + "step-restore-lists": "Restore Lists", + "step-restore-cards": "Restore Cards", + "step-restore-swimlanes": "Restore Swimlanes", + "step-fix-missing-ids": "Fix Missing IDs", + "step-scan-users": "Checking board member avatars", + "step-scan-files": "Checking board file attachments", + "step-fix-file-urls": "Fixing file URLs", "cleanup": "Cleanup", "cleanup-old-jobs": "Cleanup Old Jobs", "completed": "Completed", diff --git a/imports/i18n/data/yi.i18n.json b/imports/i18n/data/yi.i18n.json index eef27e1fe..958d8313a 100644 --- a/imports/i18n/data/yi.i18n.json +++ b/imports/i18n/data/yi.i18n.json @@ -1404,7 +1404,70 @@ "back-to-settings": "Back to Settings", "board-id": "Board ID", "board-migration": "Board Migration", + "board-migrations": "Board Migrations", "card-show-lists-on-minicard": "Show Lists on Minicard", + "comprehensive-board-migration": "Comprehensive Board Migration", + "comprehensive-board-migration-description": "Performs comprehensive checks and fixes for board data integrity, including list ordering, card positions, and swimlane structure.", + "delete-duplicate-empty-lists-migration": "Delete Duplicate Empty Lists", + "delete-duplicate-empty-lists-migration-description": "Safely deletes empty duplicate lists. Only removes lists that have no cards AND have another list with the same title that contains cards.", + "lost-cards": "Lost Cards", + "lost-cards-list": "Restored Items", + "restore-lost-cards-migration": "Restore Lost Cards", + "restore-lost-cards-migration-description": "Finds and restores cards and lists with missing swimlaneId or listId. Creates a 'Lost Cards' swimlane to make all lost items visible again.", + "restore-all-archived-migration": "Restore All Archived", + "restore-all-archived-migration-description": "Restores all archived swimlanes, lists, and cards. Automatically fixes any missing swimlaneId or listId to make items visible.", + "fix-missing-lists-migration": "Fix Missing Lists", + "fix-missing-lists-migration-description": "Detects and repairs missing or corrupted lists in the board structure.", + "fix-avatar-urls-migration": "Fix Avatar URLs", + "fix-avatar-urls-migration-description": "Updates avatar URLs for board members to use the correct storage backend and fixes broken avatar references.", + "fix-all-file-urls-migration": "Fix All File URLs", + "fix-all-file-urls-migration-description": "Updates all file attachment URLs on this board to use the correct storage backend and fixes broken file references.", + "migration-needed": "Migration Needed", + "migration-complete": "Complete", + "migration-running": "Running...", + "migration-successful": "Migration completed successfully", + "migration-failed": "Migration failed", + "migrations": "Migrations", + "migrations-admin-only": "Only board administrators can run migrations", + "migrations-description": "Run data integrity checks and repairs for this board. Each migration can be executed individually.", + "no-issues-found": "No issues found", + "run-migration": "Run Migration", + "run-comprehensive-migration-confirm": "This will perform a comprehensive migration to check and fix board data integrity. This may take a few moments. Continue?", + "run-delete-duplicate-empty-lists-migration-confirm": "This will first convert any shared lists to per-swimlane lists, then delete empty lists that have a duplicate list with the same title containing cards. Only truly redundant empty lists will be removed. Continue?", + "run-restore-lost-cards-migration-confirm": "This will create a 'Lost Cards' swimlane and restore all cards and lists with missing swimlaneId or listId. This only affects non-archived items. Continue?", + "run-restore-all-archived-migration-confirm": "This will restore ALL archived swimlanes, lists, and cards, making them visible again. Any items with missing IDs will be automatically fixed. This cannot be easily undone. Continue?", + "run-fix-missing-lists-migration-confirm": "This will detect and repair missing or corrupted lists in the board structure. Continue?", + "run-fix-avatar-urls-migration-confirm": "This will update avatar URLs for board members to use the correct storage backend. Continue?", + "run-fix-all-file-urls-migration-confirm": "This will update all file attachment URLs on this board to use the correct storage backend. Continue?", + "restore-lost-cards-nothing-to-restore": "No lost swimlanes, lists, or cards to restore", + + "migration-progress-title": "Board Migration in Progress", + "migration-progress-overall": "Overall Progress", + "migration-progress-current-step": "Current Step", + "migration-progress-status": "Status", + "migration-progress-details": "Details", + "migration-progress-note": "Please wait while we migrate your board to the latest structure...", + + "step-analyze-board-structure": "Analyze Board Structure", + "step-fix-orphaned-cards": "Fix Orphaned Cards", + "step-convert-shared-lists": "Convert Shared Lists", + "step-ensure-per-swimlane-lists": "Ensure Per-Swimlane Lists", + "step-validate-migration": "Validate Migration", + "step-fix-avatar-urls": "Fix Avatar URLs", + "step-fix-attachment-urls": "Fix Attachment URLs", + "step-analyze-lists": "Analyze Lists", + "step-create-missing-lists": "Create Missing Lists", + "step-update-cards": "Update Cards", + "step-finalize": "Finalize", + "step-delete-duplicate-empty-lists": "Delete Duplicate Empty Lists", + "step-ensure-lost-cards-swimlane": "Ensure Lost Cards Swimlane", + "step-restore-lists": "Restore Lists", + "step-restore-cards": "Restore Cards", + "step-restore-swimlanes": "Restore Swimlanes", + "step-fix-missing-ids": "Fix Missing IDs", + "step-scan-users": "Checking board member avatars", + "step-scan-files": "Checking board file attachments", + "step-fix-file-urls": "Fixing file URLs", "cleanup": "Cleanup", "cleanup-old-jobs": "Cleanup Old Jobs", "completed": "Completed", diff --git a/imports/i18n/data/yo.i18n.json b/imports/i18n/data/yo.i18n.json index eef27e1fe..958d8313a 100644 --- a/imports/i18n/data/yo.i18n.json +++ b/imports/i18n/data/yo.i18n.json @@ -1404,7 +1404,70 @@ "back-to-settings": "Back to Settings", "board-id": "Board ID", "board-migration": "Board Migration", + "board-migrations": "Board Migrations", "card-show-lists-on-minicard": "Show Lists on Minicard", + "comprehensive-board-migration": "Comprehensive Board Migration", + "comprehensive-board-migration-description": "Performs comprehensive checks and fixes for board data integrity, including list ordering, card positions, and swimlane structure.", + "delete-duplicate-empty-lists-migration": "Delete Duplicate Empty Lists", + "delete-duplicate-empty-lists-migration-description": "Safely deletes empty duplicate lists. Only removes lists that have no cards AND have another list with the same title that contains cards.", + "lost-cards": "Lost Cards", + "lost-cards-list": "Restored Items", + "restore-lost-cards-migration": "Restore Lost Cards", + "restore-lost-cards-migration-description": "Finds and restores cards and lists with missing swimlaneId or listId. Creates a 'Lost Cards' swimlane to make all lost items visible again.", + "restore-all-archived-migration": "Restore All Archived", + "restore-all-archived-migration-description": "Restores all archived swimlanes, lists, and cards. Automatically fixes any missing swimlaneId or listId to make items visible.", + "fix-missing-lists-migration": "Fix Missing Lists", + "fix-missing-lists-migration-description": "Detects and repairs missing or corrupted lists in the board structure.", + "fix-avatar-urls-migration": "Fix Avatar URLs", + "fix-avatar-urls-migration-description": "Updates avatar URLs for board members to use the correct storage backend and fixes broken avatar references.", + "fix-all-file-urls-migration": "Fix All File URLs", + "fix-all-file-urls-migration-description": "Updates all file attachment URLs on this board to use the correct storage backend and fixes broken file references.", + "migration-needed": "Migration Needed", + "migration-complete": "Complete", + "migration-running": "Running...", + "migration-successful": "Migration completed successfully", + "migration-failed": "Migration failed", + "migrations": "Migrations", + "migrations-admin-only": "Only board administrators can run migrations", + "migrations-description": "Run data integrity checks and repairs for this board. Each migration can be executed individually.", + "no-issues-found": "No issues found", + "run-migration": "Run Migration", + "run-comprehensive-migration-confirm": "This will perform a comprehensive migration to check and fix board data integrity. This may take a few moments. Continue?", + "run-delete-duplicate-empty-lists-migration-confirm": "This will first convert any shared lists to per-swimlane lists, then delete empty lists that have a duplicate list with the same title containing cards. Only truly redundant empty lists will be removed. Continue?", + "run-restore-lost-cards-migration-confirm": "This will create a 'Lost Cards' swimlane and restore all cards and lists with missing swimlaneId or listId. This only affects non-archived items. Continue?", + "run-restore-all-archived-migration-confirm": "This will restore ALL archived swimlanes, lists, and cards, making them visible again. Any items with missing IDs will be automatically fixed. This cannot be easily undone. Continue?", + "run-fix-missing-lists-migration-confirm": "This will detect and repair missing or corrupted lists in the board structure. Continue?", + "run-fix-avatar-urls-migration-confirm": "This will update avatar URLs for board members to use the correct storage backend. Continue?", + "run-fix-all-file-urls-migration-confirm": "This will update all file attachment URLs on this board to use the correct storage backend. Continue?", + "restore-lost-cards-nothing-to-restore": "No lost swimlanes, lists, or cards to restore", + + "migration-progress-title": "Board Migration in Progress", + "migration-progress-overall": "Overall Progress", + "migration-progress-current-step": "Current Step", + "migration-progress-status": "Status", + "migration-progress-details": "Details", + "migration-progress-note": "Please wait while we migrate your board to the latest structure...", + + "step-analyze-board-structure": "Analyze Board Structure", + "step-fix-orphaned-cards": "Fix Orphaned Cards", + "step-convert-shared-lists": "Convert Shared Lists", + "step-ensure-per-swimlane-lists": "Ensure Per-Swimlane Lists", + "step-validate-migration": "Validate Migration", + "step-fix-avatar-urls": "Fix Avatar URLs", + "step-fix-attachment-urls": "Fix Attachment URLs", + "step-analyze-lists": "Analyze Lists", + "step-create-missing-lists": "Create Missing Lists", + "step-update-cards": "Update Cards", + "step-finalize": "Finalize", + "step-delete-duplicate-empty-lists": "Delete Duplicate Empty Lists", + "step-ensure-lost-cards-swimlane": "Ensure Lost Cards Swimlane", + "step-restore-lists": "Restore Lists", + "step-restore-cards": "Restore Cards", + "step-restore-swimlanes": "Restore Swimlanes", + "step-fix-missing-ids": "Fix Missing IDs", + "step-scan-users": "Checking board member avatars", + "step-scan-files": "Checking board file attachments", + "step-fix-file-urls": "Fixing file URLs", "cleanup": "Cleanup", "cleanup-old-jobs": "Cleanup Old Jobs", "completed": "Completed", diff --git a/imports/i18n/data/yue_CN.i18n.json b/imports/i18n/data/yue_CN.i18n.json index eef27e1fe..958d8313a 100644 --- a/imports/i18n/data/yue_CN.i18n.json +++ b/imports/i18n/data/yue_CN.i18n.json @@ -1404,7 +1404,70 @@ "back-to-settings": "Back to Settings", "board-id": "Board ID", "board-migration": "Board Migration", + "board-migrations": "Board Migrations", "card-show-lists-on-minicard": "Show Lists on Minicard", + "comprehensive-board-migration": "Comprehensive Board Migration", + "comprehensive-board-migration-description": "Performs comprehensive checks and fixes for board data integrity, including list ordering, card positions, and swimlane structure.", + "delete-duplicate-empty-lists-migration": "Delete Duplicate Empty Lists", + "delete-duplicate-empty-lists-migration-description": "Safely deletes empty duplicate lists. Only removes lists that have no cards AND have another list with the same title that contains cards.", + "lost-cards": "Lost Cards", + "lost-cards-list": "Restored Items", + "restore-lost-cards-migration": "Restore Lost Cards", + "restore-lost-cards-migration-description": "Finds and restores cards and lists with missing swimlaneId or listId. Creates a 'Lost Cards' swimlane to make all lost items visible again.", + "restore-all-archived-migration": "Restore All Archived", + "restore-all-archived-migration-description": "Restores all archived swimlanes, lists, and cards. Automatically fixes any missing swimlaneId or listId to make items visible.", + "fix-missing-lists-migration": "Fix Missing Lists", + "fix-missing-lists-migration-description": "Detects and repairs missing or corrupted lists in the board structure.", + "fix-avatar-urls-migration": "Fix Avatar URLs", + "fix-avatar-urls-migration-description": "Updates avatar URLs for board members to use the correct storage backend and fixes broken avatar references.", + "fix-all-file-urls-migration": "Fix All File URLs", + "fix-all-file-urls-migration-description": "Updates all file attachment URLs on this board to use the correct storage backend and fixes broken file references.", + "migration-needed": "Migration Needed", + "migration-complete": "Complete", + "migration-running": "Running...", + "migration-successful": "Migration completed successfully", + "migration-failed": "Migration failed", + "migrations": "Migrations", + "migrations-admin-only": "Only board administrators can run migrations", + "migrations-description": "Run data integrity checks and repairs for this board. Each migration can be executed individually.", + "no-issues-found": "No issues found", + "run-migration": "Run Migration", + "run-comprehensive-migration-confirm": "This will perform a comprehensive migration to check and fix board data integrity. This may take a few moments. Continue?", + "run-delete-duplicate-empty-lists-migration-confirm": "This will first convert any shared lists to per-swimlane lists, then delete empty lists that have a duplicate list with the same title containing cards. Only truly redundant empty lists will be removed. Continue?", + "run-restore-lost-cards-migration-confirm": "This will create a 'Lost Cards' swimlane and restore all cards and lists with missing swimlaneId or listId. This only affects non-archived items. Continue?", + "run-restore-all-archived-migration-confirm": "This will restore ALL archived swimlanes, lists, and cards, making them visible again. Any items with missing IDs will be automatically fixed. This cannot be easily undone. Continue?", + "run-fix-missing-lists-migration-confirm": "This will detect and repair missing or corrupted lists in the board structure. Continue?", + "run-fix-avatar-urls-migration-confirm": "This will update avatar URLs for board members to use the correct storage backend. Continue?", + "run-fix-all-file-urls-migration-confirm": "This will update all file attachment URLs on this board to use the correct storage backend. Continue?", + "restore-lost-cards-nothing-to-restore": "No lost swimlanes, lists, or cards to restore", + + "migration-progress-title": "Board Migration in Progress", + "migration-progress-overall": "Overall Progress", + "migration-progress-current-step": "Current Step", + "migration-progress-status": "Status", + "migration-progress-details": "Details", + "migration-progress-note": "Please wait while we migrate your board to the latest structure...", + + "step-analyze-board-structure": "Analyze Board Structure", + "step-fix-orphaned-cards": "Fix Orphaned Cards", + "step-convert-shared-lists": "Convert Shared Lists", + "step-ensure-per-swimlane-lists": "Ensure Per-Swimlane Lists", + "step-validate-migration": "Validate Migration", + "step-fix-avatar-urls": "Fix Avatar URLs", + "step-fix-attachment-urls": "Fix Attachment URLs", + "step-analyze-lists": "Analyze Lists", + "step-create-missing-lists": "Create Missing Lists", + "step-update-cards": "Update Cards", + "step-finalize": "Finalize", + "step-delete-duplicate-empty-lists": "Delete Duplicate Empty Lists", + "step-ensure-lost-cards-swimlane": "Ensure Lost Cards Swimlane", + "step-restore-lists": "Restore Lists", + "step-restore-cards": "Restore Cards", + "step-restore-swimlanes": "Restore Swimlanes", + "step-fix-missing-ids": "Fix Missing IDs", + "step-scan-users": "Checking board member avatars", + "step-scan-files": "Checking board file attachments", + "step-fix-file-urls": "Fixing file URLs", "cleanup": "Cleanup", "cleanup-old-jobs": "Cleanup Old Jobs", "completed": "Completed", diff --git a/imports/i18n/data/zgh.i18n.json b/imports/i18n/data/zgh.i18n.json index eef27e1fe..958d8313a 100644 --- a/imports/i18n/data/zgh.i18n.json +++ b/imports/i18n/data/zgh.i18n.json @@ -1404,7 +1404,70 @@ "back-to-settings": "Back to Settings", "board-id": "Board ID", "board-migration": "Board Migration", + "board-migrations": "Board Migrations", "card-show-lists-on-minicard": "Show Lists on Minicard", + "comprehensive-board-migration": "Comprehensive Board Migration", + "comprehensive-board-migration-description": "Performs comprehensive checks and fixes for board data integrity, including list ordering, card positions, and swimlane structure.", + "delete-duplicate-empty-lists-migration": "Delete Duplicate Empty Lists", + "delete-duplicate-empty-lists-migration-description": "Safely deletes empty duplicate lists. Only removes lists that have no cards AND have another list with the same title that contains cards.", + "lost-cards": "Lost Cards", + "lost-cards-list": "Restored Items", + "restore-lost-cards-migration": "Restore Lost Cards", + "restore-lost-cards-migration-description": "Finds and restores cards and lists with missing swimlaneId or listId. Creates a 'Lost Cards' swimlane to make all lost items visible again.", + "restore-all-archived-migration": "Restore All Archived", + "restore-all-archived-migration-description": "Restores all archived swimlanes, lists, and cards. Automatically fixes any missing swimlaneId or listId to make items visible.", + "fix-missing-lists-migration": "Fix Missing Lists", + "fix-missing-lists-migration-description": "Detects and repairs missing or corrupted lists in the board structure.", + "fix-avatar-urls-migration": "Fix Avatar URLs", + "fix-avatar-urls-migration-description": "Updates avatar URLs for board members to use the correct storage backend and fixes broken avatar references.", + "fix-all-file-urls-migration": "Fix All File URLs", + "fix-all-file-urls-migration-description": "Updates all file attachment URLs on this board to use the correct storage backend and fixes broken file references.", + "migration-needed": "Migration Needed", + "migration-complete": "Complete", + "migration-running": "Running...", + "migration-successful": "Migration completed successfully", + "migration-failed": "Migration failed", + "migrations": "Migrations", + "migrations-admin-only": "Only board administrators can run migrations", + "migrations-description": "Run data integrity checks and repairs for this board. Each migration can be executed individually.", + "no-issues-found": "No issues found", + "run-migration": "Run Migration", + "run-comprehensive-migration-confirm": "This will perform a comprehensive migration to check and fix board data integrity. This may take a few moments. Continue?", + "run-delete-duplicate-empty-lists-migration-confirm": "This will first convert any shared lists to per-swimlane lists, then delete empty lists that have a duplicate list with the same title containing cards. Only truly redundant empty lists will be removed. Continue?", + "run-restore-lost-cards-migration-confirm": "This will create a 'Lost Cards' swimlane and restore all cards and lists with missing swimlaneId or listId. This only affects non-archived items. Continue?", + "run-restore-all-archived-migration-confirm": "This will restore ALL archived swimlanes, lists, and cards, making them visible again. Any items with missing IDs will be automatically fixed. This cannot be easily undone. Continue?", + "run-fix-missing-lists-migration-confirm": "This will detect and repair missing or corrupted lists in the board structure. Continue?", + "run-fix-avatar-urls-migration-confirm": "This will update avatar URLs for board members to use the correct storage backend. Continue?", + "run-fix-all-file-urls-migration-confirm": "This will update all file attachment URLs on this board to use the correct storage backend. Continue?", + "restore-lost-cards-nothing-to-restore": "No lost swimlanes, lists, or cards to restore", + + "migration-progress-title": "Board Migration in Progress", + "migration-progress-overall": "Overall Progress", + "migration-progress-current-step": "Current Step", + "migration-progress-status": "Status", + "migration-progress-details": "Details", + "migration-progress-note": "Please wait while we migrate your board to the latest structure...", + + "step-analyze-board-structure": "Analyze Board Structure", + "step-fix-orphaned-cards": "Fix Orphaned Cards", + "step-convert-shared-lists": "Convert Shared Lists", + "step-ensure-per-swimlane-lists": "Ensure Per-Swimlane Lists", + "step-validate-migration": "Validate Migration", + "step-fix-avatar-urls": "Fix Avatar URLs", + "step-fix-attachment-urls": "Fix Attachment URLs", + "step-analyze-lists": "Analyze Lists", + "step-create-missing-lists": "Create Missing Lists", + "step-update-cards": "Update Cards", + "step-finalize": "Finalize", + "step-delete-duplicate-empty-lists": "Delete Duplicate Empty Lists", + "step-ensure-lost-cards-swimlane": "Ensure Lost Cards Swimlane", + "step-restore-lists": "Restore Lists", + "step-restore-cards": "Restore Cards", + "step-restore-swimlanes": "Restore Swimlanes", + "step-fix-missing-ids": "Fix Missing IDs", + "step-scan-users": "Checking board member avatars", + "step-scan-files": "Checking board file attachments", + "step-fix-file-urls": "Fixing file URLs", "cleanup": "Cleanup", "cleanup-old-jobs": "Cleanup Old Jobs", "completed": "Completed", diff --git a/imports/i18n/data/zh-CN.i18n.json b/imports/i18n/data/zh-CN.i18n.json index 2f3dba633..76590fc80 100644 --- a/imports/i18n/data/zh-CN.i18n.json +++ b/imports/i18n/data/zh-CN.i18n.json @@ -1404,7 +1404,70 @@ "back-to-settings": "Back to Settings", "board-id": "Board ID", "board-migration": "Board Migration", + "board-migrations": "Board Migrations", "card-show-lists-on-minicard": "Show Lists on Minicard", + "comprehensive-board-migration": "Comprehensive Board Migration", + "comprehensive-board-migration-description": "Performs comprehensive checks and fixes for board data integrity, including list ordering, card positions, and swimlane structure.", + "delete-duplicate-empty-lists-migration": "Delete Duplicate Empty Lists", + "delete-duplicate-empty-lists-migration-description": "Safely deletes empty duplicate lists. Only removes lists that have no cards AND have another list with the same title that contains cards.", + "lost-cards": "Lost Cards", + "lost-cards-list": "Restored Items", + "restore-lost-cards-migration": "Restore Lost Cards", + "restore-lost-cards-migration-description": "Finds and restores cards and lists with missing swimlaneId or listId. Creates a 'Lost Cards' swimlane to make all lost items visible again.", + "restore-all-archived-migration": "Restore All Archived", + "restore-all-archived-migration-description": "Restores all archived swimlanes, lists, and cards. Automatically fixes any missing swimlaneId or listId to make items visible.", + "fix-missing-lists-migration": "Fix Missing Lists", + "fix-missing-lists-migration-description": "Detects and repairs missing or corrupted lists in the board structure.", + "fix-avatar-urls-migration": "Fix Avatar URLs", + "fix-avatar-urls-migration-description": "Updates avatar URLs for board members to use the correct storage backend and fixes broken avatar references.", + "fix-all-file-urls-migration": "Fix All File URLs", + "fix-all-file-urls-migration-description": "Updates all file attachment URLs on this board to use the correct storage backend and fixes broken file references.", + "migration-needed": "Migration Needed", + "migration-complete": "完成", + "migration-running": "Running...", + "migration-successful": "Migration completed successfully", + "migration-failed": "Migration failed", + "migrations": "Migrations", + "migrations-admin-only": "Only board administrators can run migrations", + "migrations-description": "Run data integrity checks and repairs for this board. Each migration can be executed individually.", + "no-issues-found": "No issues found", + "run-migration": "Run Migration", + "run-comprehensive-migration-confirm": "This will perform a comprehensive migration to check and fix board data integrity. This may take a few moments. Continue?", + "run-delete-duplicate-empty-lists-migration-confirm": "This will first convert any shared lists to per-swimlane lists, then delete empty lists that have a duplicate list with the same title containing cards. Only truly redundant empty lists will be removed. Continue?", + "run-restore-lost-cards-migration-confirm": "This will create a 'Lost Cards' swimlane and restore all cards and lists with missing swimlaneId or listId. This only affects non-archived items. Continue?", + "run-restore-all-archived-migration-confirm": "This will restore ALL archived swimlanes, lists, and cards, making them visible again. Any items with missing IDs will be automatically fixed. This cannot be easily undone. Continue?", + "run-fix-missing-lists-migration-confirm": "This will detect and repair missing or corrupted lists in the board structure. Continue?", + "run-fix-avatar-urls-migration-confirm": "This will update avatar URLs for board members to use the correct storage backend. Continue?", + "run-fix-all-file-urls-migration-confirm": "This will update all file attachment URLs on this board to use the correct storage backend. Continue?", + "restore-lost-cards-nothing-to-restore": "No lost swimlanes, lists, or cards to restore", + + "migration-progress-title": "Board Migration in Progress", + "migration-progress-overall": "Overall Progress", + "migration-progress-current-step": "Current Step", + "migration-progress-status": "状态", + "migration-progress-details": "详情", + "migration-progress-note": "Please wait while we migrate your board to the latest structure...", + + "step-analyze-board-structure": "Analyze Board Structure", + "step-fix-orphaned-cards": "Fix Orphaned Cards", + "step-convert-shared-lists": "Convert Shared Lists", + "step-ensure-per-swimlane-lists": "Ensure Per-Swimlane Lists", + "step-validate-migration": "Validate Migration", + "step-fix-avatar-urls": "Fix Avatar URLs", + "step-fix-attachment-urls": "Fix Attachment URLs", + "step-analyze-lists": "Analyze Lists", + "step-create-missing-lists": "Create Missing Lists", + "step-update-cards": "Update Cards", + "step-finalize": "Finalize", + "step-delete-duplicate-empty-lists": "Delete Duplicate Empty Lists", + "step-ensure-lost-cards-swimlane": "Ensure Lost Cards Swimlane", + "step-restore-lists": "Restore Lists", + "step-restore-cards": "Restore Cards", + "step-restore-swimlanes": "Restore Swimlanes", + "step-fix-missing-ids": "Fix Missing IDs", + "step-scan-users": "Checking board member avatars", + "step-scan-files": "Checking board file attachments", + "step-fix-file-urls": "Fixing file URLs", "cleanup": "Cleanup", "cleanup-old-jobs": "Cleanup Old Jobs", "completed": "已完成", diff --git a/imports/i18n/data/zh-GB.i18n.json b/imports/i18n/data/zh-GB.i18n.json index 151bcf1fd..3f96ee90b 100644 --- a/imports/i18n/data/zh-GB.i18n.json +++ b/imports/i18n/data/zh-GB.i18n.json @@ -1404,7 +1404,70 @@ "back-to-settings": "Back to Settings", "board-id": "Board ID", "board-migration": "Board Migration", + "board-migrations": "Board Migrations", "card-show-lists-on-minicard": "Show Lists on Minicard", + "comprehensive-board-migration": "Comprehensive Board Migration", + "comprehensive-board-migration-description": "Performs comprehensive checks and fixes for board data integrity, including list ordering, card positions, and swimlane structure.", + "delete-duplicate-empty-lists-migration": "Delete Duplicate Empty Lists", + "delete-duplicate-empty-lists-migration-description": "Safely deletes empty duplicate lists. Only removes lists that have no cards AND have another list with the same title that contains cards.", + "lost-cards": "Lost Cards", + "lost-cards-list": "Restored Items", + "restore-lost-cards-migration": "Restore Lost Cards", + "restore-lost-cards-migration-description": "Finds and restores cards and lists with missing swimlaneId or listId. Creates a 'Lost Cards' swimlane to make all lost items visible again.", + "restore-all-archived-migration": "Restore All Archived", + "restore-all-archived-migration-description": "Restores all archived swimlanes, lists, and cards. Automatically fixes any missing swimlaneId or listId to make items visible.", + "fix-missing-lists-migration": "Fix Missing Lists", + "fix-missing-lists-migration-description": "Detects and repairs missing or corrupted lists in the board structure.", + "fix-avatar-urls-migration": "Fix Avatar URLs", + "fix-avatar-urls-migration-description": "Updates avatar URLs for board members to use the correct storage backend and fixes broken avatar references.", + "fix-all-file-urls-migration": "Fix All File URLs", + "fix-all-file-urls-migration-description": "Updates all file attachment URLs on this board to use the correct storage backend and fixes broken file references.", + "migration-needed": "Migration Needed", + "migration-complete": "Complete", + "migration-running": "Running...", + "migration-successful": "Migration completed successfully", + "migration-failed": "Migration failed", + "migrations": "Migrations", + "migrations-admin-only": "Only board administrators can run migrations", + "migrations-description": "Run data integrity checks and repairs for this board. Each migration can be executed individually.", + "no-issues-found": "No issues found", + "run-migration": "Run Migration", + "run-comprehensive-migration-confirm": "This will perform a comprehensive migration to check and fix board data integrity. This may take a few moments. Continue?", + "run-delete-duplicate-empty-lists-migration-confirm": "This will first convert any shared lists to per-swimlane lists, then delete empty lists that have a duplicate list with the same title containing cards. Only truly redundant empty lists will be removed. Continue?", + "run-restore-lost-cards-migration-confirm": "This will create a 'Lost Cards' swimlane and restore all cards and lists with missing swimlaneId or listId. This only affects non-archived items. Continue?", + "run-restore-all-archived-migration-confirm": "This will restore ALL archived swimlanes, lists, and cards, making them visible again. Any items with missing IDs will be automatically fixed. This cannot be easily undone. Continue?", + "run-fix-missing-lists-migration-confirm": "This will detect and repair missing or corrupted lists in the board structure. Continue?", + "run-fix-avatar-urls-migration-confirm": "This will update avatar URLs for board members to use the correct storage backend. Continue?", + "run-fix-all-file-urls-migration-confirm": "This will update all file attachment URLs on this board to use the correct storage backend. Continue?", + "restore-lost-cards-nothing-to-restore": "No lost swimlanes, lists, or cards to restore", + + "migration-progress-title": "Board Migration in Progress", + "migration-progress-overall": "Overall Progress", + "migration-progress-current-step": "Current Step", + "migration-progress-status": "Status", + "migration-progress-details": "Details", + "migration-progress-note": "Please wait while we migrate your board to the latest structure...", + + "step-analyze-board-structure": "Analyze Board Structure", + "step-fix-orphaned-cards": "Fix Orphaned Cards", + "step-convert-shared-lists": "Convert Shared Lists", + "step-ensure-per-swimlane-lists": "Ensure Per-Swimlane Lists", + "step-validate-migration": "Validate Migration", + "step-fix-avatar-urls": "Fix Avatar URLs", + "step-fix-attachment-urls": "Fix Attachment URLs", + "step-analyze-lists": "Analyze Lists", + "step-create-missing-lists": "Create Missing Lists", + "step-update-cards": "Update Cards", + "step-finalize": "Finalize", + "step-delete-duplicate-empty-lists": "Delete Duplicate Empty Lists", + "step-ensure-lost-cards-swimlane": "Ensure Lost Cards Swimlane", + "step-restore-lists": "Restore Lists", + "step-restore-cards": "Restore Cards", + "step-restore-swimlanes": "Restore Swimlanes", + "step-fix-missing-ids": "Fix Missing IDs", + "step-scan-users": "Checking board member avatars", + "step-scan-files": "Checking board file attachments", + "step-fix-file-urls": "Fixing file URLs", "cleanup": "Cleanup", "cleanup-old-jobs": "Cleanup Old Jobs", "completed": "Completed", diff --git a/imports/i18n/data/zh-HK.i18n.json b/imports/i18n/data/zh-HK.i18n.json index 5c71d764d..577b25b58 100644 --- a/imports/i18n/data/zh-HK.i18n.json +++ b/imports/i18n/data/zh-HK.i18n.json @@ -1404,7 +1404,70 @@ "back-to-settings": "Back to Settings", "board-id": "Board ID", "board-migration": "Board Migration", + "board-migrations": "Board Migrations", "card-show-lists-on-minicard": "Show Lists on Minicard", + "comprehensive-board-migration": "Comprehensive Board Migration", + "comprehensive-board-migration-description": "Performs comprehensive checks and fixes for board data integrity, including list ordering, card positions, and swimlane structure.", + "delete-duplicate-empty-lists-migration": "Delete Duplicate Empty Lists", + "delete-duplicate-empty-lists-migration-description": "Safely deletes empty duplicate lists. Only removes lists that have no cards AND have another list with the same title that contains cards.", + "lost-cards": "Lost Cards", + "lost-cards-list": "Restored Items", + "restore-lost-cards-migration": "Restore Lost Cards", + "restore-lost-cards-migration-description": "Finds and restores cards and lists with missing swimlaneId or listId. Creates a 'Lost Cards' swimlane to make all lost items visible again.", + "restore-all-archived-migration": "Restore All Archived", + "restore-all-archived-migration-description": "Restores all archived swimlanes, lists, and cards. Automatically fixes any missing swimlaneId or listId to make items visible.", + "fix-missing-lists-migration": "Fix Missing Lists", + "fix-missing-lists-migration-description": "Detects and repairs missing or corrupted lists in the board structure.", + "fix-avatar-urls-migration": "Fix Avatar URLs", + "fix-avatar-urls-migration-description": "Updates avatar URLs for board members to use the correct storage backend and fixes broken avatar references.", + "fix-all-file-urls-migration": "Fix All File URLs", + "fix-all-file-urls-migration-description": "Updates all file attachment URLs on this board to use the correct storage backend and fixes broken file references.", + "migration-needed": "Migration Needed", + "migration-complete": "Complete", + "migration-running": "Running...", + "migration-successful": "Migration completed successfully", + "migration-failed": "Migration failed", + "migrations": "Migrations", + "migrations-admin-only": "Only board administrators can run migrations", + "migrations-description": "Run data integrity checks and repairs for this board. Each migration can be executed individually.", + "no-issues-found": "No issues found", + "run-migration": "Run Migration", + "run-comprehensive-migration-confirm": "This will perform a comprehensive migration to check and fix board data integrity. This may take a few moments. Continue?", + "run-delete-duplicate-empty-lists-migration-confirm": "This will first convert any shared lists to per-swimlane lists, then delete empty lists that have a duplicate list with the same title containing cards. Only truly redundant empty lists will be removed. Continue?", + "run-restore-lost-cards-migration-confirm": "This will create a 'Lost Cards' swimlane and restore all cards and lists with missing swimlaneId or listId. This only affects non-archived items. Continue?", + "run-restore-all-archived-migration-confirm": "This will restore ALL archived swimlanes, lists, and cards, making them visible again. Any items with missing IDs will be automatically fixed. This cannot be easily undone. Continue?", + "run-fix-missing-lists-migration-confirm": "This will detect and repair missing or corrupted lists in the board structure. Continue?", + "run-fix-avatar-urls-migration-confirm": "This will update avatar URLs for board members to use the correct storage backend. Continue?", + "run-fix-all-file-urls-migration-confirm": "This will update all file attachment URLs on this board to use the correct storage backend. Continue?", + "restore-lost-cards-nothing-to-restore": "No lost swimlanes, lists, or cards to restore", + + "migration-progress-title": "Board Migration in Progress", + "migration-progress-overall": "Overall Progress", + "migration-progress-current-step": "Current Step", + "migration-progress-status": "Status", + "migration-progress-details": "Details", + "migration-progress-note": "Please wait while we migrate your board to the latest structure...", + + "step-analyze-board-structure": "Analyze Board Structure", + "step-fix-orphaned-cards": "Fix Orphaned Cards", + "step-convert-shared-lists": "Convert Shared Lists", + "step-ensure-per-swimlane-lists": "Ensure Per-Swimlane Lists", + "step-validate-migration": "Validate Migration", + "step-fix-avatar-urls": "Fix Avatar URLs", + "step-fix-attachment-urls": "Fix Attachment URLs", + "step-analyze-lists": "Analyze Lists", + "step-create-missing-lists": "Create Missing Lists", + "step-update-cards": "Update Cards", + "step-finalize": "Finalize", + "step-delete-duplicate-empty-lists": "Delete Duplicate Empty Lists", + "step-ensure-lost-cards-swimlane": "Ensure Lost Cards Swimlane", + "step-restore-lists": "Restore Lists", + "step-restore-cards": "Restore Cards", + "step-restore-swimlanes": "Restore Swimlanes", + "step-fix-missing-ids": "Fix Missing IDs", + "step-scan-users": "Checking board member avatars", + "step-scan-files": "Checking board file attachments", + "step-fix-file-urls": "Fixing file URLs", "cleanup": "Cleanup", "cleanup-old-jobs": "Cleanup Old Jobs", "completed": "Completed", diff --git a/imports/i18n/data/zh-Hans.i18n.json b/imports/i18n/data/zh-Hans.i18n.json index 1938e0590..3a9ff35d7 100644 --- a/imports/i18n/data/zh-Hans.i18n.json +++ b/imports/i18n/data/zh-Hans.i18n.json @@ -1404,7 +1404,70 @@ "back-to-settings": "Back to Settings", "board-id": "Board ID", "board-migration": "Board Migration", + "board-migrations": "Board Migrations", "card-show-lists-on-minicard": "Show Lists on Minicard", + "comprehensive-board-migration": "Comprehensive Board Migration", + "comprehensive-board-migration-description": "Performs comprehensive checks and fixes for board data integrity, including list ordering, card positions, and swimlane structure.", + "delete-duplicate-empty-lists-migration": "Delete Duplicate Empty Lists", + "delete-duplicate-empty-lists-migration-description": "Safely deletes empty duplicate lists. Only removes lists that have no cards AND have another list with the same title that contains cards.", + "lost-cards": "Lost Cards", + "lost-cards-list": "Restored Items", + "restore-lost-cards-migration": "Restore Lost Cards", + "restore-lost-cards-migration-description": "Finds and restores cards and lists with missing swimlaneId or listId. Creates a 'Lost Cards' swimlane to make all lost items visible again.", + "restore-all-archived-migration": "Restore All Archived", + "restore-all-archived-migration-description": "Restores all archived swimlanes, lists, and cards. Automatically fixes any missing swimlaneId or listId to make items visible.", + "fix-missing-lists-migration": "Fix Missing Lists", + "fix-missing-lists-migration-description": "Detects and repairs missing or corrupted lists in the board structure.", + "fix-avatar-urls-migration": "Fix Avatar URLs", + "fix-avatar-urls-migration-description": "Updates avatar URLs for board members to use the correct storage backend and fixes broken avatar references.", + "fix-all-file-urls-migration": "Fix All File URLs", + "fix-all-file-urls-migration-description": "Updates all file attachment URLs on this board to use the correct storage backend and fixes broken file references.", + "migration-needed": "Migration Needed", + "migration-complete": "Complete", + "migration-running": "Running...", + "migration-successful": "Migration completed successfully", + "migration-failed": "Migration failed", + "migrations": "Migrations", + "migrations-admin-only": "Only board administrators can run migrations", + "migrations-description": "Run data integrity checks and repairs for this board. Each migration can be executed individually.", + "no-issues-found": "No issues found", + "run-migration": "Run Migration", + "run-comprehensive-migration-confirm": "This will perform a comprehensive migration to check and fix board data integrity. This may take a few moments. Continue?", + "run-delete-duplicate-empty-lists-migration-confirm": "This will first convert any shared lists to per-swimlane lists, then delete empty lists that have a duplicate list with the same title containing cards. Only truly redundant empty lists will be removed. Continue?", + "run-restore-lost-cards-migration-confirm": "This will create a 'Lost Cards' swimlane and restore all cards and lists with missing swimlaneId or listId. This only affects non-archived items. Continue?", + "run-restore-all-archived-migration-confirm": "This will restore ALL archived swimlanes, lists, and cards, making them visible again. Any items with missing IDs will be automatically fixed. This cannot be easily undone. Continue?", + "run-fix-missing-lists-migration-confirm": "This will detect and repair missing or corrupted lists in the board structure. Continue?", + "run-fix-avatar-urls-migration-confirm": "This will update avatar URLs for board members to use the correct storage backend. Continue?", + "run-fix-all-file-urls-migration-confirm": "This will update all file attachment URLs on this board to use the correct storage backend. Continue?", + "restore-lost-cards-nothing-to-restore": "No lost swimlanes, lists, or cards to restore", + + "migration-progress-title": "Board Migration in Progress", + "migration-progress-overall": "Overall Progress", + "migration-progress-current-step": "Current Step", + "migration-progress-status": "Status", + "migration-progress-details": "Details", + "migration-progress-note": "Please wait while we migrate your board to the latest structure...", + + "step-analyze-board-structure": "Analyze Board Structure", + "step-fix-orphaned-cards": "Fix Orphaned Cards", + "step-convert-shared-lists": "Convert Shared Lists", + "step-ensure-per-swimlane-lists": "Ensure Per-Swimlane Lists", + "step-validate-migration": "Validate Migration", + "step-fix-avatar-urls": "Fix Avatar URLs", + "step-fix-attachment-urls": "Fix Attachment URLs", + "step-analyze-lists": "Analyze Lists", + "step-create-missing-lists": "Create Missing Lists", + "step-update-cards": "Update Cards", + "step-finalize": "Finalize", + "step-delete-duplicate-empty-lists": "Delete Duplicate Empty Lists", + "step-ensure-lost-cards-swimlane": "Ensure Lost Cards Swimlane", + "step-restore-lists": "Restore Lists", + "step-restore-cards": "Restore Cards", + "step-restore-swimlanes": "Restore Swimlanes", + "step-fix-missing-ids": "Fix Missing IDs", + "step-scan-users": "Checking board member avatars", + "step-scan-files": "Checking board file attachments", + "step-fix-file-urls": "Fixing file URLs", "cleanup": "Cleanup", "cleanup-old-jobs": "Cleanup Old Jobs", "completed": "Completed", diff --git a/imports/i18n/data/zh-Hant.i18n.json b/imports/i18n/data/zh-Hant.i18n.json index c1d5ed8b7..4f00f5998 100644 --- a/imports/i18n/data/zh-Hant.i18n.json +++ b/imports/i18n/data/zh-Hant.i18n.json @@ -1404,7 +1404,70 @@ "back-to-settings": "Back to Settings", "board-id": "Board ID", "board-migration": "Board Migration", + "board-migrations": "Board Migrations", "card-show-lists-on-minicard": "Show Lists on Minicard", + "comprehensive-board-migration": "Comprehensive Board Migration", + "comprehensive-board-migration-description": "Performs comprehensive checks and fixes for board data integrity, including list ordering, card positions, and swimlane structure.", + "delete-duplicate-empty-lists-migration": "Delete Duplicate Empty Lists", + "delete-duplicate-empty-lists-migration-description": "Safely deletes empty duplicate lists. Only removes lists that have no cards AND have another list with the same title that contains cards.", + "lost-cards": "Lost Cards", + "lost-cards-list": "Restored Items", + "restore-lost-cards-migration": "Restore Lost Cards", + "restore-lost-cards-migration-description": "Finds and restores cards and lists with missing swimlaneId or listId. Creates a 'Lost Cards' swimlane to make all lost items visible again.", + "restore-all-archived-migration": "Restore All Archived", + "restore-all-archived-migration-description": "Restores all archived swimlanes, lists, and cards. Automatically fixes any missing swimlaneId or listId to make items visible.", + "fix-missing-lists-migration": "Fix Missing Lists", + "fix-missing-lists-migration-description": "Detects and repairs missing or corrupted lists in the board structure.", + "fix-avatar-urls-migration": "Fix Avatar URLs", + "fix-avatar-urls-migration-description": "Updates avatar URLs for board members to use the correct storage backend and fixes broken avatar references.", + "fix-all-file-urls-migration": "Fix All File URLs", + "fix-all-file-urls-migration-description": "Updates all file attachment URLs on this board to use the correct storage backend and fixes broken file references.", + "migration-needed": "Migration Needed", + "migration-complete": "Complete", + "migration-running": "Running...", + "migration-successful": "Migration completed successfully", + "migration-failed": "Migration failed", + "migrations": "Migrations", + "migrations-admin-only": "Only board administrators can run migrations", + "migrations-description": "Run data integrity checks and repairs for this board. Each migration can be executed individually.", + "no-issues-found": "No issues found", + "run-migration": "Run Migration", + "run-comprehensive-migration-confirm": "This will perform a comprehensive migration to check and fix board data integrity. This may take a few moments. Continue?", + "run-delete-duplicate-empty-lists-migration-confirm": "This will first convert any shared lists to per-swimlane lists, then delete empty lists that have a duplicate list with the same title containing cards. Only truly redundant empty lists will be removed. Continue?", + "run-restore-lost-cards-migration-confirm": "This will create a 'Lost Cards' swimlane and restore all cards and lists with missing swimlaneId or listId. This only affects non-archived items. Continue?", + "run-restore-all-archived-migration-confirm": "This will restore ALL archived swimlanes, lists, and cards, making them visible again. Any items with missing IDs will be automatically fixed. This cannot be easily undone. Continue?", + "run-fix-missing-lists-migration-confirm": "This will detect and repair missing or corrupted lists in the board structure. Continue?", + "run-fix-avatar-urls-migration-confirm": "This will update avatar URLs for board members to use the correct storage backend. Continue?", + "run-fix-all-file-urls-migration-confirm": "This will update all file attachment URLs on this board to use the correct storage backend. Continue?", + "restore-lost-cards-nothing-to-restore": "No lost swimlanes, lists, or cards to restore", + + "migration-progress-title": "Board Migration in Progress", + "migration-progress-overall": "Overall Progress", + "migration-progress-current-step": "Current Step", + "migration-progress-status": "Status", + "migration-progress-details": "Details", + "migration-progress-note": "Please wait while we migrate your board to the latest structure...", + + "step-analyze-board-structure": "Analyze Board Structure", + "step-fix-orphaned-cards": "Fix Orphaned Cards", + "step-convert-shared-lists": "Convert Shared Lists", + "step-ensure-per-swimlane-lists": "Ensure Per-Swimlane Lists", + "step-validate-migration": "Validate Migration", + "step-fix-avatar-urls": "Fix Avatar URLs", + "step-fix-attachment-urls": "Fix Attachment URLs", + "step-analyze-lists": "Analyze Lists", + "step-create-missing-lists": "Create Missing Lists", + "step-update-cards": "Update Cards", + "step-finalize": "Finalize", + "step-delete-duplicate-empty-lists": "Delete Duplicate Empty Lists", + "step-ensure-lost-cards-swimlane": "Ensure Lost Cards Swimlane", + "step-restore-lists": "Restore Lists", + "step-restore-cards": "Restore Cards", + "step-restore-swimlanes": "Restore Swimlanes", + "step-fix-missing-ids": "Fix Missing IDs", + "step-scan-users": "Checking board member avatars", + "step-scan-files": "Checking board file attachments", + "step-fix-file-urls": "Fixing file URLs", "cleanup": "Cleanup", "cleanup-old-jobs": "Cleanup Old Jobs", "completed": "Completed", diff --git a/imports/i18n/data/zh-TW.i18n.json b/imports/i18n/data/zh-TW.i18n.json index a6f66caed..a8a292b44 100644 --- a/imports/i18n/data/zh-TW.i18n.json +++ b/imports/i18n/data/zh-TW.i18n.json @@ -1404,7 +1404,70 @@ "back-to-settings": "回到設定", "board-id": "看板 ID", "board-migration": "看板遷移", + "board-migrations": "Board Migrations", "card-show-lists-on-minicard": "在迷你卡片上顯示清單", + "comprehensive-board-migration": "Comprehensive Board Migration", + "comprehensive-board-migration-description": "Performs comprehensive checks and fixes for board data integrity, including list ordering, card positions, and swimlane structure.", + "delete-duplicate-empty-lists-migration": "Delete Duplicate Empty Lists", + "delete-duplicate-empty-lists-migration-description": "Safely deletes empty duplicate lists. Only removes lists that have no cards AND have another list with the same title that contains cards.", + "lost-cards": "Lost Cards", + "lost-cards-list": "Restored Items", + "restore-lost-cards-migration": "Restore Lost Cards", + "restore-lost-cards-migration-description": "Finds and restores cards and lists with missing swimlaneId or listId. Creates a 'Lost Cards' swimlane to make all lost items visible again.", + "restore-all-archived-migration": "Restore All Archived", + "restore-all-archived-migration-description": "Restores all archived swimlanes, lists, and cards. Automatically fixes any missing swimlaneId or listId to make items visible.", + "fix-missing-lists-migration": "Fix Missing Lists", + "fix-missing-lists-migration-description": "Detects and repairs missing or corrupted lists in the board structure.", + "fix-avatar-urls-migration": "Fix Avatar URLs", + "fix-avatar-urls-migration-description": "Updates avatar URLs for board members to use the correct storage backend and fixes broken avatar references.", + "fix-all-file-urls-migration": "Fix All File URLs", + "fix-all-file-urls-migration-description": "Updates all file attachment URLs on this board to use the correct storage backend and fixes broken file references.", + "migration-needed": "Migration Needed", + "migration-complete": "完成", + "migration-running": "Running...", + "migration-successful": "Migration completed successfully", + "migration-failed": "Migration failed", + "migrations": "Migrations", + "migrations-admin-only": "Only board administrators can run migrations", + "migrations-description": "Run data integrity checks and repairs for this board. Each migration can be executed individually.", + "no-issues-found": "No issues found", + "run-migration": "Run Migration", + "run-comprehensive-migration-confirm": "This will perform a comprehensive migration to check and fix board data integrity. This may take a few moments. Continue?", + "run-delete-duplicate-empty-lists-migration-confirm": "This will first convert any shared lists to per-swimlane lists, then delete empty lists that have a duplicate list with the same title containing cards. Only truly redundant empty lists will be removed. Continue?", + "run-restore-lost-cards-migration-confirm": "This will create a 'Lost Cards' swimlane and restore all cards and lists with missing swimlaneId or listId. This only affects non-archived items. Continue?", + "run-restore-all-archived-migration-confirm": "This will restore ALL archived swimlanes, lists, and cards, making them visible again. Any items with missing IDs will be automatically fixed. This cannot be easily undone. Continue?", + "run-fix-missing-lists-migration-confirm": "This will detect and repair missing or corrupted lists in the board structure. Continue?", + "run-fix-avatar-urls-migration-confirm": "This will update avatar URLs for board members to use the correct storage backend. Continue?", + "run-fix-all-file-urls-migration-confirm": "This will update all file attachment URLs on this board to use the correct storage backend. Continue?", + "restore-lost-cards-nothing-to-restore": "No lost swimlanes, lists, or cards to restore", + + "migration-progress-title": "Board Migration in Progress", + "migration-progress-overall": "整體進度", + "migration-progress-current-step": "Current Step", + "migration-progress-status": "狀態", + "migration-progress-details": "內容", + "migration-progress-note": "Please wait while we migrate your board to the latest structure...", + + "step-analyze-board-structure": "Analyze Board Structure", + "step-fix-orphaned-cards": "Fix Orphaned Cards", + "step-convert-shared-lists": "Convert Shared Lists", + "step-ensure-per-swimlane-lists": "Ensure Per-Swimlane Lists", + "step-validate-migration": "Validate Migration", + "step-fix-avatar-urls": "Fix Avatar URLs", + "step-fix-attachment-urls": "Fix Attachment URLs", + "step-analyze-lists": "Analyze Lists", + "step-create-missing-lists": "Create Missing Lists", + "step-update-cards": "Update Cards", + "step-finalize": "Finalize", + "step-delete-duplicate-empty-lists": "Delete Duplicate Empty Lists", + "step-ensure-lost-cards-swimlane": "Ensure Lost Cards Swimlane", + "step-restore-lists": "Restore Lists", + "step-restore-cards": "Restore Cards", + "step-restore-swimlanes": "Restore Swimlanes", + "step-fix-missing-ids": "Fix Missing IDs", + "step-scan-users": "Checking board member avatars", + "step-scan-files": "Checking board file attachments", + "step-fix-file-urls": "Fixing file URLs", "cleanup": "清理", "cleanup-old-jobs": "清理舊工作", "completed": "已完成", diff --git a/imports/i18n/data/zh.i18n.json b/imports/i18n/data/zh.i18n.json index 78fd22be4..237c5e595 100644 --- a/imports/i18n/data/zh.i18n.json +++ b/imports/i18n/data/zh.i18n.json @@ -1404,7 +1404,70 @@ "back-to-settings": "Back to Settings", "board-id": "Board ID", "board-migration": "Board Migration", + "board-migrations": "Board Migrations", "card-show-lists-on-minicard": "Show Lists on Minicard", + "comprehensive-board-migration": "Comprehensive Board Migration", + "comprehensive-board-migration-description": "Performs comprehensive checks and fixes for board data integrity, including list ordering, card positions, and swimlane structure.", + "delete-duplicate-empty-lists-migration": "Delete Duplicate Empty Lists", + "delete-duplicate-empty-lists-migration-description": "Safely deletes empty duplicate lists. Only removes lists that have no cards AND have another list with the same title that contains cards.", + "lost-cards": "Lost Cards", + "lost-cards-list": "Restored Items", + "restore-lost-cards-migration": "Restore Lost Cards", + "restore-lost-cards-migration-description": "Finds and restores cards and lists with missing swimlaneId or listId. Creates a 'Lost Cards' swimlane to make all lost items visible again.", + "restore-all-archived-migration": "Restore All Archived", + "restore-all-archived-migration-description": "Restores all archived swimlanes, lists, and cards. Automatically fixes any missing swimlaneId or listId to make items visible.", + "fix-missing-lists-migration": "Fix Missing Lists", + "fix-missing-lists-migration-description": "Detects and repairs missing or corrupted lists in the board structure.", + "fix-avatar-urls-migration": "Fix Avatar URLs", + "fix-avatar-urls-migration-description": "Updates avatar URLs for board members to use the correct storage backend and fixes broken avatar references.", + "fix-all-file-urls-migration": "Fix All File URLs", + "fix-all-file-urls-migration-description": "Updates all file attachment URLs on this board to use the correct storage backend and fixes broken file references.", + "migration-needed": "Migration Needed", + "migration-complete": "Complete", + "migration-running": "Running...", + "migration-successful": "Migration completed successfully", + "migration-failed": "Migration failed", + "migrations": "Migrations", + "migrations-admin-only": "Only board administrators can run migrations", + "migrations-description": "Run data integrity checks and repairs for this board. Each migration can be executed individually.", + "no-issues-found": "No issues found", + "run-migration": "Run Migration", + "run-comprehensive-migration-confirm": "This will perform a comprehensive migration to check and fix board data integrity. This may take a few moments. Continue?", + "run-delete-duplicate-empty-lists-migration-confirm": "This will first convert any shared lists to per-swimlane lists, then delete empty lists that have a duplicate list with the same title containing cards. Only truly redundant empty lists will be removed. Continue?", + "run-restore-lost-cards-migration-confirm": "This will create a 'Lost Cards' swimlane and restore all cards and lists with missing swimlaneId or listId. This only affects non-archived items. Continue?", + "run-restore-all-archived-migration-confirm": "This will restore ALL archived swimlanes, lists, and cards, making them visible again. Any items with missing IDs will be automatically fixed. This cannot be easily undone. Continue?", + "run-fix-missing-lists-migration-confirm": "This will detect and repair missing or corrupted lists in the board structure. Continue?", + "run-fix-avatar-urls-migration-confirm": "This will update avatar URLs for board members to use the correct storage backend. Continue?", + "run-fix-all-file-urls-migration-confirm": "This will update all file attachment URLs on this board to use the correct storage backend. Continue?", + "restore-lost-cards-nothing-to-restore": "No lost swimlanes, lists, or cards to restore", + + "migration-progress-title": "Board Migration in Progress", + "migration-progress-overall": "Overall Progress", + "migration-progress-current-step": "Current Step", + "migration-progress-status": "Status", + "migration-progress-details": "Details", + "migration-progress-note": "Please wait while we migrate your board to the latest structure...", + + "step-analyze-board-structure": "Analyze Board Structure", + "step-fix-orphaned-cards": "Fix Orphaned Cards", + "step-convert-shared-lists": "Convert Shared Lists", + "step-ensure-per-swimlane-lists": "Ensure Per-Swimlane Lists", + "step-validate-migration": "Validate Migration", + "step-fix-avatar-urls": "Fix Avatar URLs", + "step-fix-attachment-urls": "Fix Attachment URLs", + "step-analyze-lists": "Analyze Lists", + "step-create-missing-lists": "Create Missing Lists", + "step-update-cards": "Update Cards", + "step-finalize": "Finalize", + "step-delete-duplicate-empty-lists": "Delete Duplicate Empty Lists", + "step-ensure-lost-cards-swimlane": "Ensure Lost Cards Swimlane", + "step-restore-lists": "Restore Lists", + "step-restore-cards": "Restore Cards", + "step-restore-swimlanes": "Restore Swimlanes", + "step-fix-missing-ids": "Fix Missing IDs", + "step-scan-users": "Checking board member avatars", + "step-scan-files": "Checking board file attachments", + "step-fix-file-urls": "Fixing file URLs", "cleanup": "Cleanup", "cleanup-old-jobs": "Cleanup Old Jobs", "completed": "Completed", diff --git a/imports/i18n/data/zu-ZA.i18n.json b/imports/i18n/data/zu-ZA.i18n.json index eef27e1fe..958d8313a 100644 --- a/imports/i18n/data/zu-ZA.i18n.json +++ b/imports/i18n/data/zu-ZA.i18n.json @@ -1404,7 +1404,70 @@ "back-to-settings": "Back to Settings", "board-id": "Board ID", "board-migration": "Board Migration", + "board-migrations": "Board Migrations", "card-show-lists-on-minicard": "Show Lists on Minicard", + "comprehensive-board-migration": "Comprehensive Board Migration", + "comprehensive-board-migration-description": "Performs comprehensive checks and fixes for board data integrity, including list ordering, card positions, and swimlane structure.", + "delete-duplicate-empty-lists-migration": "Delete Duplicate Empty Lists", + "delete-duplicate-empty-lists-migration-description": "Safely deletes empty duplicate lists. Only removes lists that have no cards AND have another list with the same title that contains cards.", + "lost-cards": "Lost Cards", + "lost-cards-list": "Restored Items", + "restore-lost-cards-migration": "Restore Lost Cards", + "restore-lost-cards-migration-description": "Finds and restores cards and lists with missing swimlaneId or listId. Creates a 'Lost Cards' swimlane to make all lost items visible again.", + "restore-all-archived-migration": "Restore All Archived", + "restore-all-archived-migration-description": "Restores all archived swimlanes, lists, and cards. Automatically fixes any missing swimlaneId or listId to make items visible.", + "fix-missing-lists-migration": "Fix Missing Lists", + "fix-missing-lists-migration-description": "Detects and repairs missing or corrupted lists in the board structure.", + "fix-avatar-urls-migration": "Fix Avatar URLs", + "fix-avatar-urls-migration-description": "Updates avatar URLs for board members to use the correct storage backend and fixes broken avatar references.", + "fix-all-file-urls-migration": "Fix All File URLs", + "fix-all-file-urls-migration-description": "Updates all file attachment URLs on this board to use the correct storage backend and fixes broken file references.", + "migration-needed": "Migration Needed", + "migration-complete": "Complete", + "migration-running": "Running...", + "migration-successful": "Migration completed successfully", + "migration-failed": "Migration failed", + "migrations": "Migrations", + "migrations-admin-only": "Only board administrators can run migrations", + "migrations-description": "Run data integrity checks and repairs for this board. Each migration can be executed individually.", + "no-issues-found": "No issues found", + "run-migration": "Run Migration", + "run-comprehensive-migration-confirm": "This will perform a comprehensive migration to check and fix board data integrity. This may take a few moments. Continue?", + "run-delete-duplicate-empty-lists-migration-confirm": "This will first convert any shared lists to per-swimlane lists, then delete empty lists that have a duplicate list with the same title containing cards. Only truly redundant empty lists will be removed. Continue?", + "run-restore-lost-cards-migration-confirm": "This will create a 'Lost Cards' swimlane and restore all cards and lists with missing swimlaneId or listId. This only affects non-archived items. Continue?", + "run-restore-all-archived-migration-confirm": "This will restore ALL archived swimlanes, lists, and cards, making them visible again. Any items with missing IDs will be automatically fixed. This cannot be easily undone. Continue?", + "run-fix-missing-lists-migration-confirm": "This will detect and repair missing or corrupted lists in the board structure. Continue?", + "run-fix-avatar-urls-migration-confirm": "This will update avatar URLs for board members to use the correct storage backend. Continue?", + "run-fix-all-file-urls-migration-confirm": "This will update all file attachment URLs on this board to use the correct storage backend. Continue?", + "restore-lost-cards-nothing-to-restore": "No lost swimlanes, lists, or cards to restore", + + "migration-progress-title": "Board Migration in Progress", + "migration-progress-overall": "Overall Progress", + "migration-progress-current-step": "Current Step", + "migration-progress-status": "Status", + "migration-progress-details": "Details", + "migration-progress-note": "Please wait while we migrate your board to the latest structure...", + + "step-analyze-board-structure": "Analyze Board Structure", + "step-fix-orphaned-cards": "Fix Orphaned Cards", + "step-convert-shared-lists": "Convert Shared Lists", + "step-ensure-per-swimlane-lists": "Ensure Per-Swimlane Lists", + "step-validate-migration": "Validate Migration", + "step-fix-avatar-urls": "Fix Avatar URLs", + "step-fix-attachment-urls": "Fix Attachment URLs", + "step-analyze-lists": "Analyze Lists", + "step-create-missing-lists": "Create Missing Lists", + "step-update-cards": "Update Cards", + "step-finalize": "Finalize", + "step-delete-duplicate-empty-lists": "Delete Duplicate Empty Lists", + "step-ensure-lost-cards-swimlane": "Ensure Lost Cards Swimlane", + "step-restore-lists": "Restore Lists", + "step-restore-cards": "Restore Cards", + "step-restore-swimlanes": "Restore Swimlanes", + "step-fix-missing-ids": "Fix Missing IDs", + "step-scan-users": "Checking board member avatars", + "step-scan-files": "Checking board file attachments", + "step-fix-file-urls": "Fixing file URLs", "cleanup": "Cleanup", "cleanup-old-jobs": "Cleanup Old Jobs", "completed": "Completed", diff --git a/imports/i18n/data/zu.i18n.json b/imports/i18n/data/zu.i18n.json index eef27e1fe..958d8313a 100644 --- a/imports/i18n/data/zu.i18n.json +++ b/imports/i18n/data/zu.i18n.json @@ -1404,7 +1404,70 @@ "back-to-settings": "Back to Settings", "board-id": "Board ID", "board-migration": "Board Migration", + "board-migrations": "Board Migrations", "card-show-lists-on-minicard": "Show Lists on Minicard", + "comprehensive-board-migration": "Comprehensive Board Migration", + "comprehensive-board-migration-description": "Performs comprehensive checks and fixes for board data integrity, including list ordering, card positions, and swimlane structure.", + "delete-duplicate-empty-lists-migration": "Delete Duplicate Empty Lists", + "delete-duplicate-empty-lists-migration-description": "Safely deletes empty duplicate lists. Only removes lists that have no cards AND have another list with the same title that contains cards.", + "lost-cards": "Lost Cards", + "lost-cards-list": "Restored Items", + "restore-lost-cards-migration": "Restore Lost Cards", + "restore-lost-cards-migration-description": "Finds and restores cards and lists with missing swimlaneId or listId. Creates a 'Lost Cards' swimlane to make all lost items visible again.", + "restore-all-archived-migration": "Restore All Archived", + "restore-all-archived-migration-description": "Restores all archived swimlanes, lists, and cards. Automatically fixes any missing swimlaneId or listId to make items visible.", + "fix-missing-lists-migration": "Fix Missing Lists", + "fix-missing-lists-migration-description": "Detects and repairs missing or corrupted lists in the board structure.", + "fix-avatar-urls-migration": "Fix Avatar URLs", + "fix-avatar-urls-migration-description": "Updates avatar URLs for board members to use the correct storage backend and fixes broken avatar references.", + "fix-all-file-urls-migration": "Fix All File URLs", + "fix-all-file-urls-migration-description": "Updates all file attachment URLs on this board to use the correct storage backend and fixes broken file references.", + "migration-needed": "Migration Needed", + "migration-complete": "Complete", + "migration-running": "Running...", + "migration-successful": "Migration completed successfully", + "migration-failed": "Migration failed", + "migrations": "Migrations", + "migrations-admin-only": "Only board administrators can run migrations", + "migrations-description": "Run data integrity checks and repairs for this board. Each migration can be executed individually.", + "no-issues-found": "No issues found", + "run-migration": "Run Migration", + "run-comprehensive-migration-confirm": "This will perform a comprehensive migration to check and fix board data integrity. This may take a few moments. Continue?", + "run-delete-duplicate-empty-lists-migration-confirm": "This will first convert any shared lists to per-swimlane lists, then delete empty lists that have a duplicate list with the same title containing cards. Only truly redundant empty lists will be removed. Continue?", + "run-restore-lost-cards-migration-confirm": "This will create a 'Lost Cards' swimlane and restore all cards and lists with missing swimlaneId or listId. This only affects non-archived items. Continue?", + "run-restore-all-archived-migration-confirm": "This will restore ALL archived swimlanes, lists, and cards, making them visible again. Any items with missing IDs will be automatically fixed. This cannot be easily undone. Continue?", + "run-fix-missing-lists-migration-confirm": "This will detect and repair missing or corrupted lists in the board structure. Continue?", + "run-fix-avatar-urls-migration-confirm": "This will update avatar URLs for board members to use the correct storage backend. Continue?", + "run-fix-all-file-urls-migration-confirm": "This will update all file attachment URLs on this board to use the correct storage backend. Continue?", + "restore-lost-cards-nothing-to-restore": "No lost swimlanes, lists, or cards to restore", + + "migration-progress-title": "Board Migration in Progress", + "migration-progress-overall": "Overall Progress", + "migration-progress-current-step": "Current Step", + "migration-progress-status": "Status", + "migration-progress-details": "Details", + "migration-progress-note": "Please wait while we migrate your board to the latest structure...", + + "step-analyze-board-structure": "Analyze Board Structure", + "step-fix-orphaned-cards": "Fix Orphaned Cards", + "step-convert-shared-lists": "Convert Shared Lists", + "step-ensure-per-swimlane-lists": "Ensure Per-Swimlane Lists", + "step-validate-migration": "Validate Migration", + "step-fix-avatar-urls": "Fix Avatar URLs", + "step-fix-attachment-urls": "Fix Attachment URLs", + "step-analyze-lists": "Analyze Lists", + "step-create-missing-lists": "Create Missing Lists", + "step-update-cards": "Update Cards", + "step-finalize": "Finalize", + "step-delete-duplicate-empty-lists": "Delete Duplicate Empty Lists", + "step-ensure-lost-cards-swimlane": "Ensure Lost Cards Swimlane", + "step-restore-lists": "Restore Lists", + "step-restore-cards": "Restore Cards", + "step-restore-swimlanes": "Restore Swimlanes", + "step-fix-missing-ids": "Fix Missing IDs", + "step-scan-users": "Checking board member avatars", + "step-scan-files": "Checking board file attachments", + "step-fix-file-urls": "Fixing file URLs", "cleanup": "Cleanup", "cleanup-old-jobs": "Cleanup Old Jobs", "completed": "Completed", From 8711b476be30496b96b845529b5717bb6e685c27 Mon Sep 17 00:00:00 2001 From: Lauri Ojansivu Date: Wed, 5 Nov 2025 20:50:28 +0200 Subject: [PATCH 46/84] Fix star board. Thanks to xet7 ! --- client/components/boards/boardHeader.jade | 23 ++++++++++++----------- client/components/boards/boardHeader.js | 5 ++++- client/components/boards/boardsList.js | 4 +++- client/components/users/userHeader.jade | 10 +++++----- models/users.js | 21 +++++++++++++++++++++ 5 files changed, 45 insertions(+), 18 deletions(-) diff --git a/client/components/boards/boardHeader.jade b/client/components/boards/boardHeader.jade index 013cb3619..fa395dc70 100644 --- a/client/components/boards/boardHeader.jade +++ b/client/components/boards/boardHeader.jade @@ -16,13 +16,6 @@ template(name="boardHeaderBar") a.board-header-btn(class="{{#if currentUser.isBoardAdmin}}js-edit-board-title{{else}}is-disabled{{/if}}" title="{{_ 'edit'}}" value=title) | ✏️ - a.board-header-btn.js-star-board(class="{{#if isStarred}}is-active{{/if}}" - title="{{#if isStarred}}{{_ 'star-board-short-unstar'}}{{else}}{{_ 'star-board-short-star'}}{{/if}}" aria-label="{{#if isStarred}}{{_ 'star-board-short-unstar'}}{{else}}{{_ 'star-board-short-star'}}{{/if}}") - | {{#if isStarred}}⭐{{else}}☆{{/if}} - if showStarCounter - span - = currentBoard.stars - a.board-header-btn( class="{{#if currentUser.isBoardAdmin}}js-change-visibility{{else}}is-disabled{{/if}}" title="{{_ currentBoard.permission}}") @@ -38,6 +31,13 @@ template(name="boardHeaderBar") if $eq watchLevel "muted" | 🔕 span {{_ watchLevel}} + a.board-header-btn.js-star-board(title="{{_ 'star-board'}}") + if isStarred + | ⭐ + else + | ☆ + if showStarCounter + span.board-star-counter {{currentBoard.stars}} a.board-header-btn(title="{{_ 'sort-cards'}}" class="{{#if isSortActive }}emphasis{{else}} js-sort-cards {{/if}}") | {{sortCardsIcon}} span {{#if isSortActive }}{{_ 'sort-is-on'}}{{else}}{{_ 'sort-cards'}}{{/if}} @@ -61,10 +61,6 @@ template(name="boardHeaderBar") a.board-header-btn(class="{{#if currentUser.isBoardAdmin}}js-edit-board-title{{else}}is-disabled{{/if}}" title="{{_ 'edit'}}" value=title) | ✏️ - a.board-header-btn.js-star-board(class="{{#if isStarred}}is-active{{/if}}" - title="{{#if isStarred}}{{_ 'click-to-unstar'}}{{else}}{{_ 'click-to-star'}}{{/if}} {{_ 'starred-boards-description'}}") - | {{#if isStarred}}⭐{{else}}☆{{/if}} - a.board-header-btn( class="{{#if currentUser.isBoardAdmin}}js-change-visibility{{else}}is-disabled{{/if}}" title="{{_ currentBoard.permission}}") @@ -78,6 +74,11 @@ template(name="boardHeaderBar") | 🔔 if $eq watchLevel "muted" | 🔕 + a.board-header-btn.js-star-board(title="{{_ 'star-board'}}") + if isStarred + | ⭐ + else + | ☆ a.board-header-btn(title="{{_ 'sort-cards'}}" class="{{#if isSortActive }}emphasis{{else}} js-sort-cards {{/if}}") | {{sortCardsIcon}} if isSortActive diff --git a/client/components/boards/boardHeader.js b/client/components/boards/boardHeader.js index b95a45395..b79b49ba4 100644 --- a/client/components/boards/boardHeader.js +++ b/client/components/boards/boardHeader.js @@ -72,7 +72,10 @@ BlazeComponent.extendComponent({ { 'click .js-edit-board-title': Popup.open('boardChangeTitle'), 'click .js-star-board'() { - ReactiveCache.getCurrentUser().toggleBoardStar(Session.get('currentBoard')); + const boardId = Session.get('currentBoard'); + if (boardId) { + Meteor.call('toggleBoardStar', boardId); + } }, 'click .js-open-board-menu': Popup.open('boardMenu'), 'click .js-change-visibility': Popup.open('boardChangeVisibility'), diff --git a/client/components/boards/boardsList.js b/client/components/boards/boardsList.js index db2ed2446..3f7600174 100644 --- a/client/components/boards/boardsList.js +++ b/client/components/boards/boardsList.js @@ -243,7 +243,9 @@ BlazeComponent.extendComponent({ 'click .js-add-board': Popup.open('createBoard'), 'click .js-star-board'(evt) { const boardId = this.currentData()._id; - ReactiveCache.getCurrentUser().toggleBoardStar(boardId); + if (boardId) { + Meteor.call('toggleBoardStar', boardId); + } evt.preventDefault(); }, 'click .js-clone-board'(evt) { diff --git a/client/components/users/userHeader.jade b/client/components/users/userHeader.jade index 8934ddbc4..668777dbb 100644 --- a/client/components/users/userHeader.jade +++ b/client/components/users/userHeader.jade @@ -12,11 +12,6 @@ template(name="headerUserBar") template(name="memberMenuPopup") ul.pop-over-list - // Bookmarks at the very top - li - a.js-open-bookmarks - | 🔖 - | {{_ 'bookmarks'}} with currentUser li a.js-my-cards(href="{{pathFor 'my-cards'}}") @@ -32,6 +27,11 @@ template(name="memberMenuPopup") | {{_ 'globalSearch-title'}} li a(href="{{pathFor 'home'}}") + | ⭐ + | {{_ 'my-bookmarks'}} + li + a(href="{{pathFor 'home'}}") + | 🏠 | 🏠 | {{_ 'all-boards'}} li diff --git a/models/users.js b/models/users.js index 417528272..bebe9b633 100644 --- a/models/users.js +++ b/models/users.js @@ -1643,6 +1643,27 @@ Meteor.methods({ check(value, String); ReactiveCache.getCurrentUser().setListSortBy(value); }, + toggleBoardStar(boardId) { + check(boardId, String); + if (!this.userId) { + throw new Meteor.Error('not-logged-in', 'User must be logged in'); + } + const user = Users.findOne(this.userId); + if (!user) { + throw new Meteor.Error('user-not-found', 'User not found'); + } + + // Check if board is already starred + const starredBoards = (user.profile && user.profile.starredBoards) || []; + const isStarred = starredBoards.includes(boardId); + + // Build update object + const updateObject = isStarred + ? { $pull: { 'profile.starredBoards': boardId } } + : { $addToSet: { 'profile.starredBoards': boardId } }; + + Users.update(this.userId, updateObject); + }, toggleDesktopDragHandles() { const user = ReactiveCache.getCurrentUser(); user.toggleDesktopHandles(user.hasShowDesktopDragHandles()); From 16a74bb748ea01040eab1d3130d6012a10300a22 Mon Sep 17 00:00:00 2001 From: Lauri Ojansivu Date: Wed, 5 Nov 2025 20:51:44 +0200 Subject: [PATCH 47/84] Updated ChangeLog. --- CHANGELOG.md | 2 ++ 1 file changed, 2 insertions(+) diff --git a/CHANGELOG.md b/CHANGELOG.md index 538b59d8b..e10e7c894 100644 --- a/CHANGELOG.md +++ b/CHANGELOG.md @@ -36,6 +36,8 @@ This release fixes the following bugs: Thanks to xet7. - [Fixed sidebar migrations to be per-board, not global. Clarified translations](https://github.com/wekan/wekan/commit/e4638d5fbcbe004ac393462331805cac3ba25097). Thanks to xet7. +- [Fix star board](https://github.com/wekan/wekan/commit/8711b476be30496b96b845529b5717bb6e685c27). + Thanks to xet7. Thanks to above GitHub users for their contributions and translators for their translations. From 0afbdc95b49537e06b4f9cf98f51a669ef249384 Mon Sep 17 00:00:00 2001 From: Lauri Ojansivu Date: Thu, 6 Nov 2025 00:26:35 +0200 Subject: [PATCH 48/84] Feature: Workspaces, at All Boards page. Thanks to xet7 ! --- client/components/boards/boardHeader.jade | 30 ++ client/components/boards/boardHeader.js | 25 ++ client/components/boards/boardsList.css | 434 ++++++++++++++++++- client/components/boards/boardsList.jade | 311 ++++++++------ client/components/boards/boardsList.js | 500 +++++++++++++++++++++- client/lib/boardMultiSelection.js | 73 ++++ imports/i18n/data/af.i18n.json | 13 + imports/i18n/data/af_ZA.i18n.json | 13 + imports/i18n/data/ar-DZ.i18n.json | 13 + imports/i18n/data/ar-EG.i18n.json | 13 + imports/i18n/data/ar.i18n.json | 13 + imports/i18n/data/ary.i18n.json | 13 + imports/i18n/data/ast-ES.i18n.json | 13 + imports/i18n/data/az-AZ.i18n.json | 13 + imports/i18n/data/az-LA.i18n.json | 13 + imports/i18n/data/az.i18n.json | 13 + imports/i18n/data/bg.i18n.json | 13 + imports/i18n/data/br.i18n.json | 13 + imports/i18n/data/ca.i18n.json | 13 + imports/i18n/data/ca@valencia.i18n.json | 13 + imports/i18n/data/ca_ES.i18n.json | 13 + imports/i18n/data/cmn.i18n.json | 13 + imports/i18n/data/cs-CZ.i18n.json | 13 + imports/i18n/data/cs.i18n.json | 13 + imports/i18n/data/cy-GB.i18n.json | 13 + imports/i18n/data/cy.i18n.json | 13 + imports/i18n/data/da.i18n.json | 13 + imports/i18n/data/de-AT.i18n.json | 13 + imports/i18n/data/de-CH.i18n.json | 13 + imports/i18n/data/de.i18n.json | 13 + imports/i18n/data/de_DE.i18n.json | 13 + imports/i18n/data/el-GR.i18n.json | 13 + imports/i18n/data/el.i18n.json | 13 + imports/i18n/data/en-BR.i18n.json | 13 + imports/i18n/data/en-DE.i18n.json | 13 + imports/i18n/data/en-GB.i18n.json | 13 + imports/i18n/data/en-IT.i18n.json | 13 + imports/i18n/data/en-MY.i18n.json | 13 + imports/i18n/data/en-YS.i18n.json | 13 + imports/i18n/data/en.i18n.json | 13 + imports/i18n/data/en_AU.i18n.json | 13 + imports/i18n/data/en_ID.i18n.json | 13 + imports/i18n/data/en_SG.i18n.json | 13 + imports/i18n/data/en_TR.i18n.json | 13 + imports/i18n/data/en_ZA.i18n.json | 13 + imports/i18n/data/eo.i18n.json | 13 + imports/i18n/data/es-AR.i18n.json | 13 + imports/i18n/data/es-CL.i18n.json | 13 + imports/i18n/data/es-LA.i18n.json | 13 + imports/i18n/data/es-MX.i18n.json | 13 + imports/i18n/data/es-PE.i18n.json | 13 + imports/i18n/data/es-PY.i18n.json | 13 + imports/i18n/data/es.i18n.json | 13 + imports/i18n/data/es_CO.i18n.json | 13 + imports/i18n/data/et-EE.i18n.json | 13 + imports/i18n/data/eu.i18n.json | 13 + imports/i18n/data/fa-IR.i18n.json | 13 + imports/i18n/data/fa.i18n.json | 13 + imports/i18n/data/fi.i18n.json | 13 + imports/i18n/data/fr-CH.i18n.json | 13 + imports/i18n/data/fr-FR.i18n.json | 13 + imports/i18n/data/fr.i18n.json | 13 + imports/i18n/data/fy-NL.i18n.json | 13 + imports/i18n/data/fy.i18n.json | 13 + imports/i18n/data/gl-ES.i18n.json | 13 + imports/i18n/data/gl.i18n.json | 13 + imports/i18n/data/gu-IN.i18n.json | 13 + imports/i18n/data/he-IL.i18n.json | 13 + imports/i18n/data/he.i18n.json | 13 + imports/i18n/data/hi-IN.i18n.json | 13 + imports/i18n/data/hi.i18n.json | 13 + imports/i18n/data/hr.i18n.json | 13 + imports/i18n/data/hu.i18n.json | 13 + imports/i18n/data/hy.i18n.json | 13 + imports/i18n/data/id.i18n.json | 13 + imports/i18n/data/ig.i18n.json | 13 + imports/i18n/data/it.i18n.json | 13 + imports/i18n/data/ja-HI.i18n.json | 13 + imports/i18n/data/ja.i18n.json | 13 + imports/i18n/data/ka.i18n.json | 13 + imports/i18n/data/km.i18n.json | 13 + imports/i18n/data/ko-KR.i18n.json | 13 + imports/i18n/data/ko.i18n.json | 13 + imports/i18n/data/lt.i18n.json | 13 + imports/i18n/data/lv.i18n.json | 13 + imports/i18n/data/mk.i18n.json | 13 + imports/i18n/data/mn.i18n.json | 13 + imports/i18n/data/ms-MY.i18n.json | 13 + imports/i18n/data/ms.i18n.json | 13 + imports/i18n/data/nb.i18n.json | 13 + imports/i18n/data/nl-NL.i18n.json | 13 + imports/i18n/data/nl.i18n.json | 13 + imports/i18n/data/oc.i18n.json | 13 + imports/i18n/data/or_IN.i18n.json | 13 + imports/i18n/data/pa.i18n.json | 13 + imports/i18n/data/pl-PL.i18n.json | 13 + imports/i18n/data/pl.i18n.json | 13 + imports/i18n/data/pt-BR.i18n.json | 15 +- imports/i18n/data/pt.i18n.json | 13 + imports/i18n/data/pt_PT.i18n.json | 13 + imports/i18n/data/ro-RO.i18n.json | 13 + imports/i18n/data/ro.i18n.json | 13 + imports/i18n/data/ru-UA.i18n.json | 13 + imports/i18n/data/ru.i18n.json | 13 + imports/i18n/data/sk.i18n.json | 13 + imports/i18n/data/sl.i18n.json | 13 + imports/i18n/data/sr.i18n.json | 13 + imports/i18n/data/sv.i18n.json | 13 + imports/i18n/data/sw.i18n.json | 13 + imports/i18n/data/ta.i18n.json | 13 + imports/i18n/data/te-IN.i18n.json | 13 + imports/i18n/data/th.i18n.json | 13 + imports/i18n/data/tk_TM.i18n.json | 13 + imports/i18n/data/tlh.i18n.json | 13 + imports/i18n/data/tr.i18n.json | 13 + imports/i18n/data/ug.i18n.json | 13 + imports/i18n/data/uk-UA.i18n.json | 13 + imports/i18n/data/uk.i18n.json | 13 + imports/i18n/data/uz-AR.i18n.json | 13 + imports/i18n/data/uz-LA.i18n.json | 13 + imports/i18n/data/uz-UZ.i18n.json | 13 + imports/i18n/data/uz.i18n.json | 13 + imports/i18n/data/ve-CC.i18n.json | 13 + imports/i18n/data/ve-PP.i18n.json | 13 + imports/i18n/data/ve.i18n.json | 13 + imports/i18n/data/vi-VN.i18n.json | 13 + imports/i18n/data/vi.i18n.json | 13 + imports/i18n/data/vl-SS.i18n.json | 13 + imports/i18n/data/vo.i18n.json | 13 + imports/i18n/data/wa-RR.i18n.json | 13 + imports/i18n/data/wa.i18n.json | 13 + imports/i18n/data/wo.i18n.json | 13 + imports/i18n/data/wuu-Hans.i18n.json | 13 + imports/i18n/data/xh.i18n.json | 13 + imports/i18n/data/yi.i18n.json | 13 + imports/i18n/data/yo.i18n.json | 13 + imports/i18n/data/yue_CN.i18n.json | 13 + imports/i18n/data/zgh.i18n.json | 13 + imports/i18n/data/zh-CN.i18n.json | 13 + imports/i18n/data/zh-GB.i18n.json | 13 + imports/i18n/data/zh-HK.i18n.json | 13 + imports/i18n/data/zh-Hans.i18n.json | 13 + imports/i18n/data/zh-Hant.i18n.json | 13 + imports/i18n/data/zh-TW.i18n.json | 13 + imports/i18n/data/zh.i18n.json | 13 + imports/i18n/data/zu-ZA.i18n.json | 13 + imports/i18n/data/zu.i18n.json | 13 + models/users.js | 91 ++++ 148 files changed, 3137 insertions(+), 162 deletions(-) create mode 100644 client/lib/boardMultiSelection.js diff --git a/client/components/boards/boardHeader.jade b/client/components/boards/boardHeader.jade index fa395dc70..bac4216ed 100644 --- a/client/components/boards/boardHeader.jade +++ b/client/components/boards/boardHeader.jade @@ -267,6 +267,36 @@ template(name="createBoardPopup") | / a.js-board-template {{_ 'template'}} +// New popup for Template Container creation; shares the same form content +template(name="createTemplateContainerPopup") + form + label + | {{_ 'title'}} + input.js-new-board-title(type="text" placeholder="{{_ 'bucket-example'}}" autofocus required) + if visibilityMenuIsOpen.get + +boardVisibilityList + else + p.quiet + if $eq visibility.get 'public' + span 🌐 + = " " + | {{{_ 'board-public-info'}}} + else + span 🔒 + = " " + | {{{_ 'board-private-info'}}} + a.js-change-visibility {{_ 'change'}}. + a.flex.js-toggle-add-template-container + .materialCheckBox#add-template-container + span {{_ 'add-template-container'}} + input.primary.wide(type="submit" value="{{_ 'create'}}") + span.quiet + | {{_ 'or'}} + a.js-import-board {{_ 'import'}} + span.quiet + | / + a.js-board-template {{_ 'template'}} + //template(name="listsortPopup") // h2 // | {{_ 'list-sort-by'}} diff --git a/client/components/boards/boardHeader.js b/client/components/boards/boardHeader.js index b79b49ba4..c84b593c6 100644 --- a/client/components/boards/boardHeader.js +++ b/client/components/boards/boardHeader.js @@ -294,6 +294,15 @@ const CreateBoard = BlazeComponent.extendComponent({ }, ); + // Assign to space if one was selected + const spaceId = Session.get('createBoardInWorkspace'); + if (spaceId) { + Meteor.call('assignBoardToWorkspace', this.boardId.get(), spaceId, (err) => { + if (err) console.error('Error assigning board to space:', err); + }); + Session.set('createBoardInWorkspace', null); // Clear after use + } + Utils.goBoardId(this.boardId.get()); } else { @@ -312,6 +321,15 @@ const CreateBoard = BlazeComponent.extendComponent({ boardId: this.boardId.get(), }); + // Assign to space if one was selected + const spaceId = Session.get('createBoardInWorkspace'); + if (spaceId) { + Meteor.call('assignBoardToWorkspace', this.boardId.get(), spaceId, (err) => { + if (err) console.error('Error assigning board to space:', err); + }); + Session.set('createBoardInWorkspace', null); // Clear after use + } + Utils.goBoardId(this.boardId.get()); } }, @@ -333,6 +351,13 @@ const CreateBoard = BlazeComponent.extendComponent({ }, }).register('createBoardPopup'); +(class CreateTemplateContainerPopup extends CreateBoard { + onRendered() { + // Always pre-check the template container checkbox for this popup + $('#add-template-container').addClass('is-checked'); + } +}).register('createTemplateContainerPopup'); + (class HeaderBarCreateBoard extends CreateBoard { onSubmit(event) { super.onSubmit(event); diff --git a/client/components/boards/boardsList.css b/client/components/boards/boardsList.css index f834af830..e17b77b12 100644 --- a/client/components/boards/boardsList.css +++ b/client/components/boards/boardsList.css @@ -8,6 +8,273 @@ padding: 1vh 0; } +/* Two-column layout for All Boards */ +.boards-layout { + display: grid; + grid-template-columns: 260px 1fr; + gap: 16px; +} + +.boards-left-menu { + border-right: 1px solid #e0e0e0; + padding-right: 12px; +} + +.boards-left-menu ul.menu { + list-style: none; + padding: 0; + margin: 0 0 12px 0; +} + +.boards-left-menu .menu-item { + margin: 4px 0; +} +.boards-left-menu .menu-item a { + display: flex; + justify-content: space-between; + align-items: center; + padding: 8px 10px; + border-radius: 4px; + cursor: pointer; +} +.boards-left-menu .menu-item .menu-label { + flex: 1; +} +.boards-left-menu .menu-item .menu-count { + background: #ddd; + padding: 2px 8px; + border-radius: 12px; + font-size: 12px; + font-weight: bold; + margin-left: 8px; +} +.boards-left-menu .menu-item.active a, +.boards-left-menu .menu-item a:hover { + background: #f0f0f0; +} +.boards-left-menu .menu-item.active .menu-count { + background: #bbb; +} + +/* Drag-over state for menu items (for dropping boards on Remaining) */ +.boards-left-menu .menu-item a.drag-over { + background: #d0e8ff; + border: 2px dashed #2196F3; +} + +.workspaces-header { + display: flex; + align-items: center; + justify-content: space-between; + font-weight: bold; + margin-top: 12px; +} +.workspaces-header .js-add-space { + text-decoration: none; + font-weight: bold; + border: 1px solid #ccc; + padding: 2px 8px; + border-radius: 4px; +} + +.workspace-tree { + list-style: none; + padding-left: 10px; +} + +.workspace-node { + margin: 2px 0; + position: relative; +} + +.workspace-node-content { + display: flex; + align-items: center; + gap: 4px; + padding: 4px; + border-radius: 4px; + transition: background-color 0.2s; +} + +.workspace-node.dragging > .workspace-node-content { + opacity: 0.5; + background: #e0e0e0; +} + +.workspace-node.drag-over > .workspace-node-content { + background: #d0e8ff; + border: 2px dashed #2196F3; +} + +.workspace-drag-handle { + cursor: grab; + color: #999; + font-size: 14px; + padding: 0 4px; + user-select: none; +} + +.workspace-drag-handle:active { + cursor: grabbing; +} + +.workspace-node .js-select-space { + display: flex; + align-items: center; + gap: 6px; + padding: 4px 8px; + border-radius: 4px; + cursor: pointer; + flex: 1; + text-decoration: none; +} + +.workspace-node .workspace-icon { + font-size: 16px; + line-height: 1; +} + +.workspace-node .workspace-name { + flex: 1; +} + +.workspace-node .workspace-count { + background: #ddd; + padding: 2px 6px; + border-radius: 10px; + font-size: 11px; + font-weight: bold; + min-width: 20px; + text-align: center; +} + +.workspace-node .js-edit-space, +.workspace-node .js-add-subspace { + padding: 2px 6px; + border-radius: 3px; + cursor: pointer; + text-decoration: none; + font-size: 14px; + opacity: 0.6; + transition: opacity 0.2s; +} + +.workspace-node .js-edit-space:hover, +.workspace-node .js-add-subspace:hover { + opacity: 1; + background: #e0e0e0; +} + +.workspace-node.active > .workspace-node-content .js-select-space, +.workspace-node > .workspace-node-content:hover .js-select-space { + background: #f0f0f0; +} + +.workspace-node.active .workspace-count { + background: #bbb; +} + +.boards-right-grid { + min-height: 200px; +} + +.boards-path-header { + display: flex; + align-items: center; + justify-content: space-between; + gap: 8px; + padding: 12px 16px; + margin-bottom: 16px; + background: #f5f5f5; + border-radius: 6px; + font-size: 16px; + font-weight: 500; +} + +.boards-path-header .path-left { + display: flex; + align-items: center; + gap: 8px; + flex: 1; +} + +.boards-path-header .multiselection-hint { + background: #FFF3CD; + color: #856404; + padding: 4px 12px; + border-radius: 4px; + font-size: 13px; + font-weight: normal; + border: 1px solid #FFE69C; + animation: pulse 2s ease-in-out infinite; +} + +@keyframes pulse { + 0%, 100% { opacity: 1; } + 50% { opacity: 0.7; } +} + +.boards-path-header .path-right { + display: flex; + align-items: center; + gap: 8px; +} + +.boards-path-header .path-icon { + font-size: 18px; +} + +.boards-path-header .path-text { + color: #333; +} + +.boards-path-header .board-header-btn { + padding: 6px 12px; + background: #fff; + border: 1px solid #ddd; + border-radius: 4px; + cursor: pointer; + display: flex; + align-items: center; + gap: 6px; + font-size: 14px; + transition: all 0.2s; +} + +.boards-path-header .board-header-btn:hover { + background: #f0f0f0; + border-color: #bbb; +} + +.boards-path-header .board-header-btn.emphasis { + background: #2196F3; + color: #fff; + border-color: #2196F3; + font-weight: bold; + box-shadow: 0 2px 8px rgba(33, 150, 243, 0.5); + transform: scale(1.05); +} + +.boards-path-header .board-header-btn.emphasis:hover { + background: #1976D2; + box-shadow: 0 3px 12px rgba(33, 150, 243, 0.7); +} + +.boards-path-header .board-header-btn-close { + padding: 4px 10px; + background: #f44336; + color: #000; + border: none; + border-radius: 4px; + cursor: pointer; + font-size: 16px; + margin-left: 10px; /* Extra space between MultiSelection toggle and Remove Filter */ +} + +.boards-path-header .board-header-btn-close:hover { + background: #d32f2f; +} + .zoom-controls { display: flex; align-items: center; @@ -109,20 +376,26 @@ } .board-list .board-list-item { overflow: hidden; - background-color: #999; + background-color: inherit; /* Inherit board color from parent li.js-board */ color: #f6f6f6; min-height: 100px; font-size: 16px; line-height: 22px; - border-radius: 3px; + border-radius: 0; /* No border-radius - parent .js-board has it */ display: block; font-weight: 700; - padding: 8px; - margin: 8px; + padding: 36px 8px 32px 8px; /* Top padding for drag handle, bottom for checkbox */ + margin: 0; /* No margin - moved to parent .js-board */ position: relative; text-decoration: none; word-wrap: break-word; } + +.board-list .board-list-item > .js-open-board { + text-decoration: none; + color: inherit; + display: block; +} .board-list .board-list-item.template-container { border: 4px solid #fff; } @@ -150,9 +423,16 @@ .board-list .js-add-board .label { font-weight: normal; line-height: 56px; + min-height: 100px; + display: flex; + align-items: center; + justify-content: center; + background-color: #999; /* Darker background for better text contrast */ + border-radius: 3px; + padding: 36px 8px 32px 8px; } -.board-list .js-add-board :hover { - background-color: #939393; +.board-list .js-add-board .label:hover { + background-color: #808080; /* Even darker on hover */ } .board-list .is-star-active, .board-list .is-not-star-active { @@ -238,6 +518,95 @@ .board-list li:hover a .is-not-star-active { opacity: 1; } + +/* Board drag handle - always visible and positioned at top */ +.board-list .board-handle { + position: absolute; + padding: 4px 6px; + top: 4px; + left: 50%; + transform: translateX(-50%); + font-size: 14px; + color: #fff; + background: rgba(0,0,0,0.4); + border-radius: 4px; + display: flex; + align-items: center; + justify-content: center; + z-index: 10; + transition: background-color 0.2s ease; + cursor: grab; + opacity: 1; + user-select: none; +} + +.board-list .board-handle:active { + cursor: grabbing; +} + +.board-list .board-handle:hover { + background: rgba(255, 255, 0, 0.8) !important; + color: #000; +} + +/* Multiselection checkbox on board items */ +.board-list .board-list-item .multi-selection-checkbox { + position: absolute !important; + bottom: 4px !important; + left: 4px !important; + top: auto !important; + width: 24px; + height: 24px; + border: 3px solid #fff; + background: rgba(0,0,0,0.5); + border-radius: 4px; + cursor: pointer; + z-index: 11; + display: flex; + align-items: center; + justify-content: center; + transition: all 0.2s; + box-shadow: 0 2px 4px rgba(0,0,0,0.3); + transform: none !important; + margin: 0 !important; +} + +.board-list .board-list-item .multi-selection-checkbox:hover { + background: rgba(0,0,0,0.7); + transform: scale(1.15) !important; + box-shadow: 0 3px 6px rgba(0,0,0,0.5); +} + +.board-list .board-list-item .multi-selection-checkbox.is-checked { + background: #2196F3; + border-color: #2196F3; + box-shadow: 0 2px 8px rgba(33, 150, 243, 0.6); + width: 24px !important; + height: 24px !important; + top: auto !important; + left: 4px !important; + transform: none !important; + border-radius: 4px !important; +} + +.board-list .board-list-item .multi-selection-checkbox.is-checked::after { + content: '✓'; + color: #fff; + font-size: 16px; + font-weight: bold; +} + +.board-list.is-multiselection-active .js-board.is-checked { + outline: 4px solid #2196F3; + outline-offset: -4px; + box-shadow: 0 4px 12px rgba(33, 150, 243, 0.4); +} + +/* Visual hint when multiselection is active */ +.board-list.is-multiselection-active .board-list-item { + border: 2px dashed rgba(33, 150, 243, 0.3); +} + .board-backgrounds-list .board-background-select { box-sizing: border-box; display: block; @@ -739,9 +1108,62 @@ #resetBtn { display: inline; } + +#resetBtn.filter-reset-btn { + background: #f44336; + color: #000; + border: none; + border-radius: 4px; + padding: 6px 12px; + cursor: pointer; + font-size: 14px; + display: inline-flex; + align-items: center; + gap: 6px; + transition: background 0.2s; +} + +#resetBtn.filter-reset-btn:hover { + background: #d32f2f; +} + +#resetBtn.filter-reset-btn .reset-icon { + font-size: 14px; +} + .js-board { display: block; + background-color: #999; /* Default gray background if no color class is applied */ + border-radius: 3px; /* Rounded corners for board items */ + overflow: hidden; /* Ensure children respect rounded corners */ + margin: 8px; /* Space between board items */ } + +/* Reset background for add-board button */ +.js-add-board { + background-color: transparent !important; + margin: 8px !important; /* Keep margin for add-board */ +} + +/* Apply board colors to li.js-board parent instead of just the link */ +.board-list .board-color-nephritis { background-color: #27ae60; } +.board-list .board-color-pomegranate { background-color: #c0392b; } +.board-list .board-color-belize { background-color: #2980b9; } +.board-list .board-color-wisteria { background-color: #8e44ad; } +.board-list .board-color-midnight { background-color: #2c3e50; } +.board-list .board-color-pumpkin { background-color: #e67e22; } +.board-list .board-color-moderatepink { background-color: #cd5a91; } +.board-list .board-color-strongcyan { background-color: #00aecc; } +.board-list .board-color-limegreen { background-color: #4bbf6b; } +.board-list .board-color-dark { background-color: #2c3e51; } +.board-list .board-color-relax { background-color: #27ae61; } +.board-list .board-color-corteza { background-color: #568ba2; } +.board-list .board-color-clearblue { background-color: #3498db; } +.board-list .board-color-natural { background-color: #596557; } +.board-list .board-color-modern { background-color: #2a80b8; } +.board-list .board-color-moderndark { background-color: #2a2a2a; } +.board-list .board-color-exodark { background-color: #222; } + .minicard-members { padding: 6px 0 6px 8px; width: 100%; diff --git a/client/components/boards/boardsList.jade b/client/components/boards/boardsList.jade index 3d01c19c9..5cf488c55 100644 --- a/client/components/boards/boardsList.jade +++ b/client/components/boards/boardsList.jade @@ -2,151 +2,160 @@ template(name="boardList") .wrapper .board-list-header - ul.AllBoardTeamsOrgs - li.AllBoardTeams - if userHasTeams - select.js-AllBoardTeams#jsAllBoardTeams("multiple") - option(value="-1") {{_ 'teams'}} : - each teamsDatas - option(value="{{teamId}}") {{_ teamDisplayName}} + .boards-layout + // Left menu + .boards-left-menu + ul.menu + li(class="menu-item {{#if isSelectedMenu 'starred'}}active{{/if}}") + a.js-select-menu(data-type="starred") + span.menu-label ⭐ {{_ 'allboards.starred'}} + span.menu-count {{menuItemCount 'starred'}} + li(class="menu-item {{#if isSelectedMenu 'templates'}}active{{/if}}") + a.js-select-menu(data-type="templates") + span.menu-label 📋 {{_ 'allboards.templates'}} + span.menu-count {{menuItemCount 'templates'}} + li(class="menu-item {{#if isSelectedMenu 'remaining'}}active{{/if}}") + a.js-select-menu(data-type="remaining") + span.menu-label 📂 {{_ 'allboards.remaining'}} + span.menu-count {{menuItemCount 'remaining'}} + .workspaces-header + span 🗂️ {{_ 'allboards.workspaces'}} + a.js-add-workspace(title="{{_ 'allboards.add-workspace'}}") + + // Workspaces tree + +workspaceTree(nodes=workspacesTree selectedWorkspaceId=selectedWorkspaceId) - li.AllBoardOrgs - if userHasOrgs - select.js-AllBoardOrgs#jsAllBoardOrgs("multiple") - option(value="-1") {{_ 'organizations'}} : - each orgsDatas - option(value="{{orgId}}") {{orgDisplayName}} + // Existing filter by orgs/teams (kept) + ul.AllBoardTeamsOrgs + li.AllBoardTeams + if userHasTeams + select.js-AllBoardTeams#jsAllBoardTeams("multiple") + option(value="-1") {{_ 'teams'}} : + each teamsDatas + option(value="{{teamId}}") {{_ teamDisplayName}} - //li.AllBoardTemplates - // if userHasTemplates - // select.js-AllBoardTemplates#jsAllBoardTemplates("multiple") - // option(value="-1") {{_ 'templates'}} : - // each templatesDatas - // option(value="{{templateId}}") {{_ templateDisplayName}} + li.AllBoardOrgs + if userHasOrgs + select.js-AllBoardOrgs#jsAllBoardOrgs("multiple") + option(value="-1") {{_ 'organizations'}} : + each orgsDatas + option(value="{{orgId}}") {{orgDisplayName}} - li.AllBoardBtns - div.AllBoardButtonsContainer - if userHasOrgsOrTeams - i.fa.fa-filter - input#filterBtn(type="button" value="{{_ 'filter'}}") - input#resetBtn(type="button" value="{{_ 'filter-clear'}}") + li.AllBoardBtns + div.AllBoardButtonsContainer + if userHasOrgsOrTeams + span 🔍 + input#filterBtn(type="button" value="{{_ 'filter'}}") + button#resetBtn.filter-reset-btn + span.reset-icon ❌ + span {{_ 'filter-clear'}} - ul.board-list.clearfix.js-boards(class="{{#if isMiniScreen}}mobile-view{{/if}}") - li.js-add-board - a.board-list-item.label(title="{{_ 'add-board'}}") - | {{_ 'add-board'}} - each boards - li(class="{{_id}}" class="{{#if isStarred}}starred{{/if}}" class=colorClass).js-board - if isInvited - .board-list-item - span.details - span.board-list-item-name= title - i.fa.js-star-board( - class="fa-star{{#if isStarred}} is-star-active{{else}}-o{{/if}}" - title="{{_ 'star-board-title'}}") - p.board-list-item-desc {{_ 'just-invited'}} - button.js-accept-invite.primary {{_ 'accept'}} - button.js-decline-invite {{_ 'decline'}} - else - if $eq type "template-container" - a.js-open-board.template-container.board-list-item(href="{{pathFor 'board' id=_id slug=slug}}") - span.details - span.board-list-item-name(title="{{_ 'template-container'}}") - +viewer - = title - i.fa.js-star-board( - class="fa-star{{#if isStarred}} is-star-active{{else}}-o{{/if}}" - title="{{_ 'star-board-title'}}") - p.board-list-item-desc - +viewer - = description - if hasSpentTimeCards - i.fa.js-has-spenttime-cards( - class="fa-circle{{#if hasOvertimeCards}} has-overtime-card-active{{else}} no-overtime-card-active{{/if}}" - title="{{#if hasOvertimeCards}}{{_ 'has-overtime-cards'}}{{else}}{{_ 'has-spenttime-cards'}}{{/if}}") - i.fa.board-handle( - class="fa-arrows" - title="{{_ 'drag-board'}}") - if isSandstorm - i.fa.js-clone-board( - class="fa-clone" - title="{{_ 'duplicate-board'}}") - i.fa.js-archive-board( - class="fa-archive" - title="{{_ 'archive-board'}}") - else if isAdministrable - i.fa.js-clone-board( - class="fa-clone" - title="{{_ 'duplicate-board'}}") - i.fa.js-archive-board( - class="fa-archive" - title="{{_ 'archive-board'}}") - else if currentUser.isAdmin - i.fa.js-clone-board( - class="fa-clone" - title="{{_ 'duplicate-board'}}") - i.fa.js-archive-board( - class="fa-archive" - title="{{_ 'archive-board'}}") + // Right boards grid + .boards-right-grid + .boards-path-header + .path-left + span.path-icon {{currentMenuPath.icon}} + span.path-text {{currentMenuPath.text}} + if BoardMultiSelection.isActive + span.multiselection-hint 📌 {{_ 'multi-selection-active'}} + .path-right + if canModifyBoards + if hasBoardsSelected + button.js-archive-selected-boards.board-header-btn + span 📦 + span {{_ 'archive-board'}} + button.js-duplicate-selected-boards.board-header-btn + span 📋 + span {{_ 'duplicate-board'}} + a.board-header-btn.js-multiselection-activate( + title="{{#if BoardMultiSelection.isActive}}{{_ 'multi-selection-on'}}{{else}}{{_ 'multi-selection'}}{{/if}}" + class="{{#if BoardMultiSelection.isActive}}emphasis{{/if}}") + | ☑️ + if BoardMultiSelection.isActive + a.board-header-btn-close.js-multiselection-reset(title="{{_ 'filter-clear'}}") + | ✖ + ul.board-list.clearfix.js-boards(class="{{#if isMiniScreen}}mobile-view{{/if}} {{#if BoardMultiSelection.isActive}}is-multiselection-active{{/if}}") + li.js-add-board + if isSelectedMenu 'templates' + a.board-list-item.label(title="{{_ 'add-template-container'}}") + | ➕ {{_ 'add-template-container'}} else - a.js-open-board.board-list-item(href="{{pathFor 'board' id=_id slug=slug}}") - span.details - span.board-list-item-name(title="{{_ 'board-drag-drop-reorder-or-click-open'}}") - +viewer - = title - unless currentSetting.hideBoardMemberList - if allowsBoardMemberList - .minicard-members - each member in boardMembers _id - a.name - +userAvatar(userId=member noRemove=true) - unless currentSetting.hideCardCounterList - if allowsCardCounterList - .minicard-lists.flex.flex-wrap - each list in boardLists _id - .item - | {{ list }} - a.js-star-board( - class="{{#if isStarred}}is-star-active{{else}}is-not-star-active{{/if}}" - title="{{_ 'star-board-title'}}") - | {{#if isStarred}}⭐{{else}}☆{{/if}} - p.board-list-item-desc - +viewer - = description - if hasSpentTimeCards - i.fa.js-has-spenttime-cards( - class="fa-circle{{#if hasOvertimeCards}} has-overtime-card-active{{else}} no-overtime-card-active{{/if}}" - title="{{#if hasOvertimeCards}}{{_ 'has-overtime-cards'}}{{else}}{{_ 'has-spenttime-cards'}}{{/if}}") - i.fa.board-handle( - class="fa-arrows" - title="{{_ 'drag-board'}}") - if isSandstorm - a.js-clone-board( - class="fa-clone" - title="{{_ 'duplicate-board'}}") - | 📋 - a.js-archive-board( - class="fa-archive" - title="{{_ 'archive-board'}}") - | 📦 - else if isAdministrable - a.js-clone-board( - class="fa-clone" - title="{{_ 'duplicate-board'}}") - | 📋 - a.js-archive-board( - class="fa-archive" - title="{{_ 'archive-board'}}") - | 📦 - else if currentUser.isAdmin - a.js-clone-board( - class="fa-clone" - title="{{_ 'duplicate-board'}}") - | 📋 - a.js-archive-board( - class="fa-archive" - title="{{_ 'archive-board'}}") - | 📦 + a.board-list-item.label(title="{{_ 'add-board'}}") + | ➕ {{_ 'add-board'}} + each boards + li.js-board(class="{{_id}} {{#if isStarred}}starred{{/if}} {{colorClass}} {{#if BoardMultiSelection.isSelected _id}}is-checked{{/if}}", draggable="true") + if isInvited + .board-list-item + if BoardMultiSelection.isActive + .materialCheckBox.multi-selection-checkbox.js-toggle-board-multi-selection( + class="{{#if BoardMultiSelection.isSelected _id}}is-checked{{/if}}") + span.details + span.board-list-item-name= title + span.js-star-board( + class="{{#if isStarred}}is-star-active{{else}}is-not-star-active{{/if}}" + title="{{_ 'star-board-title'}}") + | {{#if isStarred}}⭐{{else}}☆{{/if}} + p.board-list-item-desc {{_ 'just-invited'}} + button.js-accept-invite.primary {{_ 'accept'}} + button.js-decline-invite {{_ 'decline'}} + else + if $eq type "template-container" + .template-container.board-list-item + if BoardMultiSelection.isActive + .materialCheckBox.multi-selection-checkbox.js-toggle-board-multi-selection( + class="{{#if BoardMultiSelection.isSelected _id}}is-checked{{/if}}") + span.board-handle(title="{{_ 'drag-board'}}") ↕️ + a.js-open-board(href="{{pathFor 'board' id=_id slug=slug}}") + span.details + span.board-list-item-name(title="{{_ 'template-container'}}") + +viewer + = title + p.board-list-item-desc + +viewer + = description + if hasSpentTimeCards + span.js-has-spenttime-cards( + class="{{#if hasOvertimeCards}}has-overtime-card-active{{else}}no-overtime-card-active{{/if}}" + title="{{#if hasOvertimeCards}}{{_ 'has-overtime-cards'}}{{else}}{{_ 'has-spenttime-cards'}}{{/if}}") + | ⏱️ + span.js-star-board( + class="{{#if isStarred}}is-star-active{{else}}is-not-star-active{{/if}}" + title="{{_ 'star-board-title'}}") + | {{#if isStarred}}⭐{{else}}☆{{/if}} + else + .board-list-item + if BoardMultiSelection.isActive + .materialCheckBox.multi-selection-checkbox.js-toggle-board-multi-selection( + class="{{#if BoardMultiSelection.isSelected _id}}is-checked{{/if}}") + span.board-handle(title="{{_ 'drag-board'}}") ↕️ + a.js-open-board(href="{{pathFor 'board' id=_id slug=slug}}") + span.details + span.board-list-item-name(title="{{_ 'board-drag-drop-reorder-or-click-open'}}") + +viewer + = title + unless currentSetting.hideBoardMemberList + if allowsBoardMemberList + .minicard-members + each member in boardMembers _id + a.name + +userAvatar(userId=member noRemove=true) + unless currentSetting.hideCardCounterList + if allowsCardCounterList + .minicard-lists.flex.flex-wrap + each list in boardLists _id + .item + | {{ list }} + p.board-list-item-desc + +viewer + = description + if hasSpentTimeCards + span.js-has-spenttime-cards( + class="{{#if hasOvertimeCards}}has-overtime-card-active{{else}}no-overtime-card-active{{/if}}" + title="{{#if hasOvertimeCards}}{{_ 'has-overtime-cards'}}{{else}}{{_ 'has-spenttime-cards'}}{{/if}}") + | ⏱️ + a.js-star-board( + class="{{#if isStarred}}is-star-active{{else}}is-not-star-active{{/if}}" + title="{{_ 'star-board-title'}}") + | {{#if isStarred}}⭐{{else}}☆{{/if}} template(name="boardListHeaderBar") h1 {{_ title }} @@ -157,3 +166,25 @@ template(name="boardListHeaderBar") // a.board-header-btn(href="{{pathFor 'board' id=templatesBoardId slug=templatesBoardSlug}}") // i.fa.fa-clone // span {{_ 'templates'}} + +// Recursive template for workspaces tree +template(name="workspaceTree") + if nodes + ul.workspace-tree.js-workspace-tree + each nodes + li.workspace-node(class="{{#if $eq id selectedWorkspaceId}}active{{/if}}" data-workspace-id="{{id}}" draggable="true") + .workspace-node-content + span.workspace-drag-handle ↕️ + a.js-select-workspace(data-id="{{id}}") + span.workspace-icon + if icon + +viewer + = icon + else + | 📁 + span.workspace-name= name + a.js-edit-workspace(data-id="{{id}}" title="{{_ 'allboards.edit-workspace'}}") ✏️ + span.workspace-count {{workspaceCount id}} + a.js-add-subworkspace(data-id="{{id}}" title="{{_ 'allboards.add-subworkspace'}}") + + if children + +workspaceTree(nodes=children selectedWorkspaceId=selectedWorkspaceId) diff --git a/client/components/boards/boardsList.js b/client/components/boards/boardsList.js index 3f7600174..171c9196b 100644 --- a/client/components/boards/boardsList.js +++ b/client/components/boards/boardsList.js @@ -14,6 +14,9 @@ Template.boardList.helpers({ return Utils.isMiniScreen() && Session.get('currentBoard'); */ return true; }, + BoardMultiSelection() { + return BoardMultiSelection; + }, }) Template.boardListHeaderBar.events({ @@ -45,6 +48,9 @@ BlazeComponent.extendComponent({ onCreated() { Meteor.subscribe('setting'); Meteor.subscribe('tableVisibilityModeSettings'); + this.selectedMenu = new ReactiveVar('starred'); + this.selectedWorkspaceIdVar = new ReactiveVar(null); + this.workspacesTreeVar = new ReactiveVar([]); let currUser = ReactiveCache.getCurrentUser(); let userLanguage; if (currUser && currUser.profile) { @@ -53,9 +59,72 @@ BlazeComponent.extendComponent({ if (userLanguage) { TAPi18n.setLanguage(userLanguage); } + // Load workspaces tree reactively + this.autorun(() => { + const u = ReactiveCache.getCurrentUser(); + const tree = (u && u.profile && u.profile.boardWorkspacesTree) || []; + this.workspacesTreeVar.set(tree); + }); + }, + + reorderWorkspaces(draggedSpaceId, targetSpaceId) { + const tree = this.workspacesTreeVar.get(); + + // Helper to remove a space from tree + const removeSpace = (nodes, id) => { + for (let i = 0; i < nodes.length; i++) { + if (nodes[i].id === id) { + const removed = nodes.splice(i, 1)[0]; + return { tree: nodes, removed }; + } + if (nodes[i].children) { + const result = removeSpace(nodes[i].children, id); + if (result.removed) { + return { tree: nodes, removed: result.removed }; + } + } + } + return { tree: nodes, removed: null }; + }; + + // Helper to insert a space after target + const insertAfter = (nodes, targetId, spaceToInsert) => { + for (let i = 0; i < nodes.length; i++) { + if (nodes[i].id === targetId) { + nodes.splice(i + 1, 0, spaceToInsert); + return true; + } + if (nodes[i].children) { + if (insertAfter(nodes[i].children, targetId, spaceToInsert)) { + return true; + } + } + } + return false; + }; + + // Clone the tree + const newTree = EJSON.clone(tree); + + // Remove the dragged space + const { tree: treeAfterRemoval, removed } = removeSpace(newTree, draggedSpaceId); + + if (removed) { + // Insert after target + insertAfter(treeAfterRemoval, targetSpaceId, removed); + + // Save the new tree + Meteor.call('setWorkspacesTree', treeAfterRemoval, (err) => { + if (err) console.error(err); + }); + } }, onRendered() { + // jQuery sortable is disabled in favor of HTML5 drag-and-drop for space management + // The old sortable code has been removed to prevent conflicts + + /* OLD SORTABLE CODE - DISABLED const itemsSelector = '.js-board:not(.placeholder)'; const $boards = this.$('.js-boards'); @@ -73,20 +142,12 @@ BlazeComponent.extendComponent({ EscapeActions.executeUpTo('popup-close'); }, stop(evt, ui) { - // To attribute the new index number, we need to get the DOM element const prevBoardDom = ui.item.prev('.js-board').get(0); const nextBoardDom = ui.item.next('.js-board').get(0); const sortIndex = Utils.calculateIndex(prevBoardDom, nextBoardDom, 1); const boardDomElement = ui.item.get(0); const board = Blaze.getData(boardDomElement); - // Normally the jquery-ui sortable library moves the dragged DOM element - // to its new position, which disrupts Blaze reactive updates mechanism - // (especially when we move the last card of a list, or when multiple - // users move some cards at the same time). To prevent these UX glitches - // we ask sortable to gracefully cancel the move, and to put back the - // DOM in its initial state. The card move is then handled reactively by - // Blaze with the below query. $boards.sortable('cancel'); const currentUser = ReactiveCache.getCurrentUser(); if (currentUser && typeof currentUser.setBoardSortIndex === 'function') { @@ -95,7 +156,6 @@ BlazeComponent.extendComponent({ }, }); - // Disable drag-dropping if the current user is not a board member or is comment only this.autorun(() => { if (Utils.isTouchScreenOrShowDesktopDragHandles()) { $boards.sortable({ @@ -103,6 +163,7 @@ BlazeComponent.extendComponent({ }); } }); + */ }, userHasTeams() { if (ReactiveCache.getCurrentUser()?.teams?.length > 0) @@ -134,6 +195,41 @@ BlazeComponent.extendComponent({ const ret = this.userHasOrgs() || this.userHasTeams(); return ret; }, + currentMenuPath() { + const sel = this.selectedMenu.get(); + const currentUser = ReactiveCache.getCurrentUser(); + + // Helper to find space by id in tree + const findSpaceById = (nodes, targetId, path = []) => { + for (const node of nodes) { + if (node.id === targetId) { + return [...path, node]; + } + if (node.children && node.children.length > 0) { + const result = findSpaceById(node.children, targetId, [...path, node]); + if (result) return result; + } + } + return null; + }; + + if (sel === 'starred') { + return { icon: '⭐', text: TAPi18n.__('allboards.starred') }; + } else if (sel === 'templates') { + return { icon: '📋', text: TAPi18n.__('allboards.templates') }; + } else if (sel === 'remaining') { + return { icon: '📂', text: TAPi18n.__('allboards.remaining') }; + } else { + // sel is a workspaceId, build path + const tree = this.workspacesTreeVar.get(); + const spacePath = findSpaceById(tree, sel); + if (spacePath && spacePath.length > 0) { + const pathText = spacePath.map(s => s.name).join(' / '); + return { icon: '🗂️', text: `${TAPi18n.__('allboards.workspaces')} / ${pathText}` }; + } + return { icon: '🗂️', text: TAPi18n.__('allboards.workspaces') }; + } + }, boards() { let query = { // { type: 'board' }, @@ -188,11 +284,29 @@ BlazeComponent.extendComponent({ const boards = ReactiveCache.getBoards(query, {}); const currentUser = ReactiveCache.getCurrentUser(); - if (currentUser && typeof currentUser.sortBoardsForUser === 'function') { - return currentUser.sortBoardsForUser(boards); + let list = boards; + // Apply left menu filtering + const sel = this.selectedMenu.get(); + const assignments = (currentUser && currentUser.profile && currentUser.profile.boardWorkspaceAssignments) || {}; + if (sel === 'starred') { + list = list.filter(b => currentUser && currentUser.hasStarred(b._id)); + } else if (sel === 'templates') { + list = list.filter(b => b.type === 'template-container'); + } else if (sel === 'remaining') { + list = list.filter(b => + !assignments[b._id] && + b.type !== 'template-container' && + !(currentUser && currentUser.hasStarred(b._id)) + ); + } else { + // assume sel is a workspaceId + list = list.filter(b => assignments[b._id] === sel); } - // Fallback: deterministic title sort when no user mapping is available (e.g., public page) - return boards.slice().sort((a, b) => (a.title || '').localeCompare(b.title || '')); + + if (currentUser && typeof currentUser.sortBoardsForUser === 'function') { + return currentUser.sortBoardsForUser(list); + } + return list.slice().sort((a, b) => (a.title || '').localeCompare(b.title || '')); }, boardLists(boardId) { /* Bug Board icons random dance https://github.com/wekan/wekan/issues/4214 @@ -240,13 +354,65 @@ BlazeComponent.extendComponent({ events() { return [ { - 'click .js-add-board': Popup.open('createBoard'), + 'click .js-select-menu'(evt) { + const type = evt.currentTarget.getAttribute('data-type'); + this.selectedWorkspaceIdVar.set(null); + this.selectedMenu.set(type); + }, + 'click .js-select-workspace'(evt) { + const id = evt.currentTarget.getAttribute('data-id'); + this.selectedWorkspaceIdVar.set(id); + this.selectedMenu.set(id); + }, + 'click .js-add-workspace'(evt) { + evt.preventDefault(); + const name = prompt(TAPi18n.__('allboards.add-workspace-prompt') || 'New Space name'); + if (name && name.trim()) { + Meteor.call('createWorkspace', { parentId: null, name: name.trim() }, (err, res) => { + if (err) console.error(err); + }); + } + }, + 'click .js-add-board'(evt) { + // Store the currently selected workspace/menu for board creation + const selectedWorkspaceId = this.selectedWorkspaceIdVar.get(); + const selectedMenu = this.selectedMenu.get(); + + if (selectedWorkspaceId) { + Session.set('createBoardInWorkspace', selectedWorkspaceId); + } else { + Session.set('createBoardInWorkspace', null); + } + + // Open different popup based on context + if (selectedMenu === 'templates') { + Popup.open('createTemplateContainer')(evt); + } else { + Popup.open('createBoard')(evt); + } + }, 'click .js-star-board'(evt) { + evt.preventDefault(); + evt.stopPropagation(); const boardId = this.currentData()._id; if (boardId) { Meteor.call('toggleBoardStar', boardId); } - evt.preventDefault(); + }, + // HTML5 DnD from boards to spaces + 'dragstart .js-board'(evt) { + const boardId = this.currentData()._id; + + // Support multi-drag + if (BoardMultiSelection.isActive() && BoardMultiSelection.isSelected(boardId)) { + const selectedIds = BoardMultiSelection.getSelectedBoardIds(); + try { + evt.originalEvent.dataTransfer.setData('text/plain', JSON.stringify(selectedIds)); + evt.originalEvent.dataTransfer.setData('application/x-board-multi', 'true'); + } catch (e) {} + } else { + try { evt.originalEvent.dataTransfer.setData('text/plain', boardId); } catch (e) {} + } }, 'click .js-clone-board'(evt) { if (confirm(TAPi18n.__('duplicate-board-confirm'))) { @@ -297,6 +463,58 @@ BlazeComponent.extendComponent({ } }); }, + 'click .js-multiselection-activate'(evt) { + evt.preventDefault(); + if (BoardMultiSelection.isActive()) { + BoardMultiSelection.disable(); + } else { + BoardMultiSelection.activate(); + } + }, + 'click .js-multiselection-reset'(evt) { + evt.preventDefault(); + BoardMultiSelection.disable(); + }, + 'click .js-toggle-board-multi-selection'(evt) { + evt.preventDefault(); + evt.stopPropagation(); + const boardId = this.currentData()._id; + BoardMultiSelection.toogle(boardId); + }, + 'click .js-archive-selected-boards'(evt) { + evt.preventDefault(); + const selectedBoards = BoardMultiSelection.getSelectedBoardIds(); + if (selectedBoards.length > 0 && confirm(TAPi18n.__('archive-board-confirm'))) { + selectedBoards.forEach(boardId => { + Meteor.call('archiveBoard', boardId); + }); + BoardMultiSelection.reset(); + } + }, + 'click .js-duplicate-selected-boards'(evt) { + evt.preventDefault(); + const selectedBoards = BoardMultiSelection.getSelectedBoardIds(); + if (selectedBoards.length > 0 && confirm(TAPi18n.__('duplicate-board-confirm'))) { + selectedBoards.forEach(boardId => { + const board = ReactiveCache.getBoard(boardId); + if (board) { + Meteor.call( + 'copyBoard', + boardId, + { + sort: ReactiveCache.getBoards({ archived: false }).length, + type: 'board', + title: board.title, + }, + (err, res) => { + if (err) console.error(err); + } + ); + } + }); + BoardMultiSelection.reset(); + } + }, 'click #resetBtn'(event) { let allBoards = document.getElementsByClassName("js-board"); let currBoard; @@ -363,7 +581,259 @@ BlazeComponent.extendComponent({ } } }, + 'click .js-edit-workspace'(evt) { + evt.preventDefault(); + evt.stopPropagation(); + const workspaceId = evt.currentTarget.getAttribute('data-id'); + + // Find the space in the tree + const findSpace = (nodes, id) => { + for (const node of nodes) { + if (node.id === id) return node; + if (node.children) { + const found = findSpace(node.children, id); + if (found) return found; + } + } + return null; + }; + + const tree = this.workspacesTreeVar.get(); + const space = findSpace(tree, workspaceId); + + if (space) { + const newName = prompt(TAPi18n.__('allboards.edit-workspace-name') || 'Space name:', space.name); + const newIcon = prompt(TAPi18n.__('allboards.edit-workspace-icon') || 'Space icon (markdown):', space.icon || '📁'); + + if (newName !== null && newName.trim()) { + // Update space in tree + const updateSpaceInTree = (nodes, id, updates) => { + return nodes.map(node => { + if (node.id === id) { + return { ...node, ...updates }; + } + if (node.children) { + return { ...node, children: updateSpaceInTree(node.children, id, updates) }; + } + return node; + }); + }; + + const updatedTree = updateSpaceInTree(tree, workspaceId, { + name: newName.trim(), + icon: newIcon || '📁' + }); + + Meteor.call('setWorkspacesTree', updatedTree, (err) => { + if (err) console.error(err); + }); + } + } + }, + 'click .js-add-subworkspace'(evt) { + evt.preventDefault(); + evt.stopPropagation(); + const parentId = evt.currentTarget.getAttribute('data-id'); + const name = prompt(TAPi18n.__('allboards.add-subworkspace-prompt') || 'Subspace name:'); + + if (name && name.trim()) { + Meteor.call('createWorkspace', { parentId, name: name.trim() }, (err) => { + if (err) console.error(err); + }); + } + }, + 'dragstart .workspace-node'(evt) { + const workspaceId = evt.currentTarget.getAttribute('data-workspace-id'); + evt.originalEvent.dataTransfer.effectAllowed = 'move'; + evt.originalEvent.dataTransfer.setData('application/x-workspace-id', workspaceId); + + // Create a better drag image + const dragImage = evt.currentTarget.cloneNode(true); + dragImage.style.position = 'absolute'; + dragImage.style.top = '-9999px'; + dragImage.style.opacity = '0.8'; + document.body.appendChild(dragImage); + evt.originalEvent.dataTransfer.setDragImage(dragImage, 0, 0); + setTimeout(() => document.body.removeChild(dragImage), 0); + + evt.currentTarget.classList.add('dragging'); + }, + 'dragend .workspace-node'(evt) { + evt.currentTarget.classList.remove('dragging'); + document.querySelectorAll('.workspace-node').forEach(el => { + el.classList.remove('drag-over'); + }); + }, + 'dragover .workspace-node'(evt) { + evt.preventDefault(); + evt.stopPropagation(); + + const draggingEl = document.querySelector('.workspace-node.dragging'); + const targetEl = evt.currentTarget; + + // Allow dropping boards on any space + // Or allow dropping spaces on other spaces (but not on itself or descendants) + if (!draggingEl || (targetEl !== draggingEl && !draggingEl.contains(targetEl))) { + evt.originalEvent.dataTransfer.dropEffect = 'move'; + targetEl.classList.add('drag-over'); + } + }, + 'dragleave .workspace-node'(evt) { + evt.currentTarget.classList.remove('drag-over'); + }, + 'drop .workspace-node'(evt) { + evt.preventDefault(); + evt.stopPropagation(); + + const targetEl = evt.currentTarget; + targetEl.classList.remove('drag-over'); + + // Check what's being dropped - board or workspace + const draggedWorkspaceId = evt.originalEvent.dataTransfer.getData('application/x-workspace-id'); + const isMultiBoard = evt.originalEvent.dataTransfer.getData('application/x-board-multi'); + const boardData = evt.originalEvent.dataTransfer.getData('text/plain'); + + if (draggedWorkspaceId && !boardData) { + // This is a workspace reorder operation + const targetWorkspaceId = targetEl.getAttribute('data-workspace-id'); + + if (draggedWorkspaceId !== targetWorkspaceId) { + this.reorderWorkspaces(draggedWorkspaceId, targetWorkspaceId); + } + } else if (boardData) { + // This is a board assignment operation + // Get the workspace ID directly from the dropped workspace-node's data-workspace-id attribute + const workspaceId = targetEl.getAttribute('data-workspace-id'); + + if (workspaceId) { + if (isMultiBoard) { + // Multi-board drag + try { + const boardIds = JSON.parse(boardData); + boardIds.forEach(boardId => { + Meteor.call('assignBoardToWorkspace', boardId, workspaceId); + }); + } catch (e) { + // Error parsing multi-board data + } + } else { + // Single board drag + Meteor.call('assignBoardToWorkspace', boardData, workspaceId); + } + } + } + }, + 'dragover .js-select-menu'(evt) { + evt.preventDefault(); + evt.stopPropagation(); + + const menuType = evt.currentTarget.getAttribute('data-type'); + // Only allow drop on "remaining" menu to unassign boards from spaces + if (menuType === 'remaining') { + evt.originalEvent.dataTransfer.dropEffect = 'move'; + evt.currentTarget.classList.add('drag-over'); + } + }, + 'dragleave .js-select-menu'(evt) { + evt.currentTarget.classList.remove('drag-over'); + }, + 'drop .js-select-menu'(evt) { + evt.preventDefault(); + evt.stopPropagation(); + + const menuType = evt.currentTarget.getAttribute('data-type'); + evt.currentTarget.classList.remove('drag-over'); + + // Only handle drops on "remaining" menu + if (menuType !== 'remaining') return; + + const isMultiBoard = evt.originalEvent.dataTransfer.getData('application/x-board-multi'); + const boardData = evt.originalEvent.dataTransfer.getData('text/plain'); + + if (boardData) { + if (isMultiBoard) { + // Multi-board drag - unassign all from workspaces + try { + const boardIds = JSON.parse(boardData); + boardIds.forEach(boardId => { + Meteor.call('unassignBoardFromWorkspace', boardId); + }); + } catch (e) { + // Error parsing multi-board data + } + } else { + // Single board drag - unassign from workspace + Meteor.call('unassignBoardFromWorkspace', boardData); + } + } + }, }, ]; }, + // Helpers for templates + workspacesTree() { + return this.workspacesTreeVar.get(); + }, + selectedWorkspaceId() { + return this.selectedWorkspaceIdVar.get(); + }, + isSelectedMenu(type) { + return this.selectedMenu.get() === type; + }, + isSpaceSelected(id) { + return this.selectedWorkspaceIdVar.get() === id; + }, + menuItemCount(type) { + const currentUser = ReactiveCache.getCurrentUser(); + const assignments = (currentUser && currentUser.profile && currentUser.profile.boardWorkspaceAssignments) || {}; + + // Get all boards for counting + let query = { + $and: [ + { archived: false }, + { type: { $in: ['board', 'template-container'] } }, + { $or: [{ 'members.userId': Meteor.userId() }] }, + { title: { $not: { $regex: /^\^.*\^$/ } } } + ] + }; + const allBoards = ReactiveCache.getBoards(query, {}); + + if (type === 'starred') { + return allBoards.filter(b => currentUser && currentUser.hasStarred(b._id)).length; + } else if (type === 'templates') { + return allBoards.filter(b => b.type === 'template-container').length; + } else if (type === 'remaining') { + return allBoards.filter(b => + !assignments[b._id] && + b.type !== 'template-container' && + !(currentUser && currentUser.hasStarred(b._id)) + ).length; + } + return 0; + }, + workspaceCount(workspaceId) { + const currentUser = ReactiveCache.getCurrentUser(); + const assignments = (currentUser && currentUser.profile && currentUser.profile.boardWorkspaceAssignments) || {}; + + // Get all boards for counting + let query = { + $and: [ + { archived: false }, + { type: { $in: ['board', 'template-container'] } }, + { $or: [{ 'members.userId': Meteor.userId() }] }, + { title: { $not: { $regex: /^\^.*\^$/ } } } + ] + }; + const allBoards = ReactiveCache.getBoards(query, {}); + + // Count boards directly assigned to this space (not including children) + return allBoards.filter(b => assignments[b._id] === workspaceId).length; + }, + canModifyBoards() { + const currentUser = ReactiveCache.getCurrentUser(); + return currentUser && !currentUser.isCommentOnly(); + }, + hasBoardsSelected() { + return BoardMultiSelection.count() > 0; + }, }).register('boardList'); diff --git a/client/lib/boardMultiSelection.js b/client/lib/boardMultiSelection.js new file mode 100644 index 000000000..036c312b6 --- /dev/null +++ b/client/lib/boardMultiSelection.js @@ -0,0 +1,73 @@ +import { ReactiveCache } from '/imports/reactiveCache'; + +BoardMultiSelection = { + _selectedBoards: new ReactiveVar([]), + + _isActive: new ReactiveVar(false), + + reset() { + this._selectedBoards.set([]); + }, + + isActive() { + return this._isActive.get(); + }, + + count() { + return this._selectedBoards.get().length; + }, + + isEmpty() { + return this.count() === 0; + }, + + getSelectedBoardIds() { + return this._selectedBoards.get(); + }, + + activate() { + if (!this.isActive()) { + this._isActive.set(true); + Tracker.flush(); + } + }, + + disable() { + if (this.isActive()) { + this._isActive.set(false); + this.reset(); + } + }, + + add(boardIds) { + return this.toggle(boardIds, { add: true, remove: false }); + }, + + remove(boardIds) { + return this.toggle(boardIds, { add: false, remove: true }); + }, + + toogle(boardIds) { + return this.toggle(boardIds, { add: true, remove: true }); + }, + + toggle(boardIds, { add, remove } = {}) { + boardIds = _.isString(boardIds) ? [boardIds] : boardIds; + let selectedBoards = this._selectedBoards.get(); + + boardIds.forEach(boardId => { + const index = selectedBoards.indexOf(boardId); + if (index > -1 && remove) { + selectedBoards = selectedBoards.filter(id => id !== boardId); + } else if (index === -1 && add) { + selectedBoards.push(boardId); + } + }); + + this._selectedBoards.set(selectedBoards); + }, + + isSelected(boardId) { + return this._selectedBoards.get().includes(boardId); + }, +}; diff --git a/imports/i18n/data/af.i18n.json b/imports/i18n/data/af.i18n.json index 958d8313a..acf3c6934 100644 --- a/imports/i18n/data/af.i18n.json +++ b/imports/i18n/data/af.i18n.json @@ -78,6 +78,18 @@ "activity-deleteComment": "deleted comment %s", "activity-receivedDate": "edited received date to %s of %s", "activity-startDate": "edited start date to %s of %s", + "allboards.starred": "Starred", + "allboards.templates": "Templates", + "allboards.remaining": "Remaining", + "allboards.workspaces": "Workspaces", + "allboards.add-workspace": "Add Workspace", + "allboards.add-workspace-prompt": "Workspace name", + "allboards.add-subworkspace": "Add Subworkspace", + "allboards.add-subworkspace-prompt": "Subworkspace name", + "allboards.edit-workspace": "Edit workspace", + "allboards.edit-workspace-name": "Workspace name", + "allboards.edit-workspace-icon": "Workspace icon (markdown)", + "multi-selection-active": "Click checkboxes to select boards", "activity-dueDate": "edited due date to %s of %s", "activity-endDate": "edited end date to %s of %s", "add-attachment": "Add Attachment", @@ -337,6 +349,7 @@ "copyManyCardsPopup-format": "[ {\"title\": \"First card title\", \"description\":\"First card description\"}, {\"title\":\"Second card title\",\"description\":\"Second card description\"},{\"title\":\"Last card title\",\"description\":\"Last card description\"} ]", "create": "Create", "createBoardPopup-title": "Create Board", + "createTemplateContainerPopup-title": "Add Template Container", "chooseBoardSourcePopup-title": "Import board", "createLabelPopup-title": "Create Label", "createCustomField": "Create Field", diff --git a/imports/i18n/data/af_ZA.i18n.json b/imports/i18n/data/af_ZA.i18n.json index ff43f83d1..d397f5944 100644 --- a/imports/i18n/data/af_ZA.i18n.json +++ b/imports/i18n/data/af_ZA.i18n.json @@ -78,6 +78,18 @@ "activity-deleteComment": "deleted comment %s", "activity-receivedDate": "edited received date to %s of %s", "activity-startDate": "edited start date to %s of %s", + "allboards.starred": "Starred", + "allboards.templates": "Templates", + "allboards.remaining": "Remaining", + "allboards.workspaces": "Workspaces", + "allboards.add-workspace": "Add Workspace", + "allboards.add-workspace-prompt": "Workspace name", + "allboards.add-subworkspace": "Add Subworkspace", + "allboards.add-subworkspace-prompt": "Subworkspace name", + "allboards.edit-workspace": "Edit workspace", + "allboards.edit-workspace-name": "Workspace name", + "allboards.edit-workspace-icon": "Workspace icon (markdown)", + "multi-selection-active": "Click checkboxes to select boards", "activity-dueDate": "edited due date to %s of %s", "activity-endDate": "edited end date to %s of %s", "add-attachment": "Add Attachment", @@ -337,6 +349,7 @@ "copyManyCardsPopup-format": "[ {\"title\": \"First card title\", \"description\":\"First card description\"}, {\"title\":\"Second card title\",\"description\":\"Second card description\"},{\"title\":\"Last card title\",\"description\":\"Last card description\"} ]", "create": "Create", "createBoardPopup-title": "Create Board", + "createTemplateContainerPopup-title": "Add Template Container", "chooseBoardSourcePopup-title": "Import board", "createLabelPopup-title": "Create Label", "createCustomField": "Create Field", diff --git a/imports/i18n/data/ar-DZ.i18n.json b/imports/i18n/data/ar-DZ.i18n.json index 4804e6dff..b6aac02e5 100644 --- a/imports/i18n/data/ar-DZ.i18n.json +++ b/imports/i18n/data/ar-DZ.i18n.json @@ -78,6 +78,18 @@ "activity-deleteComment": "deleted comment %s", "activity-receivedDate": "edited received date to %s of %s", "activity-startDate": "edited start date to %s of %s", + "allboards.starred": "Starred", + "allboards.templates": "Templates", + "allboards.remaining": "Remaining", + "allboards.workspaces": "Workspaces", + "allboards.add-workspace": "Add Workspace", + "allboards.add-workspace-prompt": "Workspace name", + "allboards.add-subworkspace": "Add Subworkspace", + "allboards.add-subworkspace-prompt": "Subworkspace name", + "allboards.edit-workspace": "Edit workspace", + "allboards.edit-workspace-name": "Workspace name", + "allboards.edit-workspace-icon": "Workspace icon (markdown)", + "multi-selection-active": "Click checkboxes to select boards", "activity-dueDate": "edited due date to %s of %s", "activity-endDate": "edited end date to %s of %s", "add-attachment": "Add Attachment", @@ -337,6 +349,7 @@ "copyManyCardsPopup-format": "[ {\"title\": \"First card title\", \"description\":\"First card description\"}, {\"title\":\"Second card title\",\"description\":\"Second card description\"},{\"title\":\"Last card title\",\"description\":\"Last card description\"} ]", "create": "Create", "createBoardPopup-title": "Create Board", + "createTemplateContainerPopup-title": "Add Template Container", "chooseBoardSourcePopup-title": "Import board", "createLabelPopup-title": "Create Label", "createCustomField": "Create Field", diff --git a/imports/i18n/data/ar-EG.i18n.json b/imports/i18n/data/ar-EG.i18n.json index 4804e6dff..b6aac02e5 100644 --- a/imports/i18n/data/ar-EG.i18n.json +++ b/imports/i18n/data/ar-EG.i18n.json @@ -78,6 +78,18 @@ "activity-deleteComment": "deleted comment %s", "activity-receivedDate": "edited received date to %s of %s", "activity-startDate": "edited start date to %s of %s", + "allboards.starred": "Starred", + "allboards.templates": "Templates", + "allboards.remaining": "Remaining", + "allboards.workspaces": "Workspaces", + "allboards.add-workspace": "Add Workspace", + "allboards.add-workspace-prompt": "Workspace name", + "allboards.add-subworkspace": "Add Subworkspace", + "allboards.add-subworkspace-prompt": "Subworkspace name", + "allboards.edit-workspace": "Edit workspace", + "allboards.edit-workspace-name": "Workspace name", + "allboards.edit-workspace-icon": "Workspace icon (markdown)", + "multi-selection-active": "Click checkboxes to select boards", "activity-dueDate": "edited due date to %s of %s", "activity-endDate": "edited end date to %s of %s", "add-attachment": "Add Attachment", @@ -337,6 +349,7 @@ "copyManyCardsPopup-format": "[ {\"title\": \"First card title\", \"description\":\"First card description\"}, {\"title\":\"Second card title\",\"description\":\"Second card description\"},{\"title\":\"Last card title\",\"description\":\"Last card description\"} ]", "create": "Create", "createBoardPopup-title": "Create Board", + "createTemplateContainerPopup-title": "Add Template Container", "chooseBoardSourcePopup-title": "Import board", "createLabelPopup-title": "Create Label", "createCustomField": "Create Field", diff --git a/imports/i18n/data/ar.i18n.json b/imports/i18n/data/ar.i18n.json index 5c7227056..5ba52f954 100644 --- a/imports/i18n/data/ar.i18n.json +++ b/imports/i18n/data/ar.i18n.json @@ -78,6 +78,18 @@ "activity-deleteComment": "تعليق محذوف %s", "activity-receivedDate": "edited received date to %s of %s", "activity-startDate": "edited start date to %s of %s", + "allboards.starred": "Starred", + "allboards.templates": "نماذج", + "allboards.remaining": "Remaining", + "allboards.workspaces": "Workspaces", + "allboards.add-workspace": "Add Workspace", + "allboards.add-workspace-prompt": "Workspace name", + "allboards.add-subworkspace": "Add Subworkspace", + "allboards.add-subworkspace-prompt": "Subworkspace name", + "allboards.edit-workspace": "Edit workspace", + "allboards.edit-workspace-name": "Workspace name", + "allboards.edit-workspace-icon": "Workspace icon (markdown)", + "multi-selection-active": "Click checkboxes to select boards", "activity-dueDate": "edited due date to %s of %s", "activity-endDate": "edited end date to %s of %s", "add-attachment": "إضافة مرفق", @@ -337,6 +349,7 @@ "copyManyCardsPopup-format": "[ {\"title\": \"First card title\", \"description\":\"First card description\"}, {\"title\":\"Second card title\",\"description\":\"Second card description\"},{\"title\":\"Last card title\",\"description\":\"Last card description\"} ]", "create": "إنشاء", "createBoardPopup-title": "إنشاء لوحة", + "createTemplateContainerPopup-title": "Add Template Container", "chooseBoardSourcePopup-title": "استيراد لوحة", "createLabelPopup-title": "إنشاء علامة", "createCustomField": "انشاء حقل", diff --git a/imports/i18n/data/ary.i18n.json b/imports/i18n/data/ary.i18n.json index 4804e6dff..b6aac02e5 100644 --- a/imports/i18n/data/ary.i18n.json +++ b/imports/i18n/data/ary.i18n.json @@ -78,6 +78,18 @@ "activity-deleteComment": "deleted comment %s", "activity-receivedDate": "edited received date to %s of %s", "activity-startDate": "edited start date to %s of %s", + "allboards.starred": "Starred", + "allboards.templates": "Templates", + "allboards.remaining": "Remaining", + "allboards.workspaces": "Workspaces", + "allboards.add-workspace": "Add Workspace", + "allboards.add-workspace-prompt": "Workspace name", + "allboards.add-subworkspace": "Add Subworkspace", + "allboards.add-subworkspace-prompt": "Subworkspace name", + "allboards.edit-workspace": "Edit workspace", + "allboards.edit-workspace-name": "Workspace name", + "allboards.edit-workspace-icon": "Workspace icon (markdown)", + "multi-selection-active": "Click checkboxes to select boards", "activity-dueDate": "edited due date to %s of %s", "activity-endDate": "edited end date to %s of %s", "add-attachment": "Add Attachment", @@ -337,6 +349,7 @@ "copyManyCardsPopup-format": "[ {\"title\": \"First card title\", \"description\":\"First card description\"}, {\"title\":\"Second card title\",\"description\":\"Second card description\"},{\"title\":\"Last card title\",\"description\":\"Last card description\"} ]", "create": "Create", "createBoardPopup-title": "Create Board", + "createTemplateContainerPopup-title": "Add Template Container", "chooseBoardSourcePopup-title": "Import board", "createLabelPopup-title": "Create Label", "createCustomField": "Create Field", diff --git a/imports/i18n/data/ast-ES.i18n.json b/imports/i18n/data/ast-ES.i18n.json index 4804e6dff..b6aac02e5 100644 --- a/imports/i18n/data/ast-ES.i18n.json +++ b/imports/i18n/data/ast-ES.i18n.json @@ -78,6 +78,18 @@ "activity-deleteComment": "deleted comment %s", "activity-receivedDate": "edited received date to %s of %s", "activity-startDate": "edited start date to %s of %s", + "allboards.starred": "Starred", + "allboards.templates": "Templates", + "allboards.remaining": "Remaining", + "allboards.workspaces": "Workspaces", + "allboards.add-workspace": "Add Workspace", + "allboards.add-workspace-prompt": "Workspace name", + "allboards.add-subworkspace": "Add Subworkspace", + "allboards.add-subworkspace-prompt": "Subworkspace name", + "allboards.edit-workspace": "Edit workspace", + "allboards.edit-workspace-name": "Workspace name", + "allboards.edit-workspace-icon": "Workspace icon (markdown)", + "multi-selection-active": "Click checkboxes to select boards", "activity-dueDate": "edited due date to %s of %s", "activity-endDate": "edited end date to %s of %s", "add-attachment": "Add Attachment", @@ -337,6 +349,7 @@ "copyManyCardsPopup-format": "[ {\"title\": \"First card title\", \"description\":\"First card description\"}, {\"title\":\"Second card title\",\"description\":\"Second card description\"},{\"title\":\"Last card title\",\"description\":\"Last card description\"} ]", "create": "Create", "createBoardPopup-title": "Create Board", + "createTemplateContainerPopup-title": "Add Template Container", "chooseBoardSourcePopup-title": "Import board", "createLabelPopup-title": "Create Label", "createCustomField": "Create Field", diff --git a/imports/i18n/data/az-AZ.i18n.json b/imports/i18n/data/az-AZ.i18n.json index 4804e6dff..b6aac02e5 100644 --- a/imports/i18n/data/az-AZ.i18n.json +++ b/imports/i18n/data/az-AZ.i18n.json @@ -78,6 +78,18 @@ "activity-deleteComment": "deleted comment %s", "activity-receivedDate": "edited received date to %s of %s", "activity-startDate": "edited start date to %s of %s", + "allboards.starred": "Starred", + "allboards.templates": "Templates", + "allboards.remaining": "Remaining", + "allboards.workspaces": "Workspaces", + "allboards.add-workspace": "Add Workspace", + "allboards.add-workspace-prompt": "Workspace name", + "allboards.add-subworkspace": "Add Subworkspace", + "allboards.add-subworkspace-prompt": "Subworkspace name", + "allboards.edit-workspace": "Edit workspace", + "allboards.edit-workspace-name": "Workspace name", + "allboards.edit-workspace-icon": "Workspace icon (markdown)", + "multi-selection-active": "Click checkboxes to select boards", "activity-dueDate": "edited due date to %s of %s", "activity-endDate": "edited end date to %s of %s", "add-attachment": "Add Attachment", @@ -337,6 +349,7 @@ "copyManyCardsPopup-format": "[ {\"title\": \"First card title\", \"description\":\"First card description\"}, {\"title\":\"Second card title\",\"description\":\"Second card description\"},{\"title\":\"Last card title\",\"description\":\"Last card description\"} ]", "create": "Create", "createBoardPopup-title": "Create Board", + "createTemplateContainerPopup-title": "Add Template Container", "chooseBoardSourcePopup-title": "Import board", "createLabelPopup-title": "Create Label", "createCustomField": "Create Field", diff --git a/imports/i18n/data/az-LA.i18n.json b/imports/i18n/data/az-LA.i18n.json index 4804e6dff..b6aac02e5 100644 --- a/imports/i18n/data/az-LA.i18n.json +++ b/imports/i18n/data/az-LA.i18n.json @@ -78,6 +78,18 @@ "activity-deleteComment": "deleted comment %s", "activity-receivedDate": "edited received date to %s of %s", "activity-startDate": "edited start date to %s of %s", + "allboards.starred": "Starred", + "allboards.templates": "Templates", + "allboards.remaining": "Remaining", + "allboards.workspaces": "Workspaces", + "allboards.add-workspace": "Add Workspace", + "allboards.add-workspace-prompt": "Workspace name", + "allboards.add-subworkspace": "Add Subworkspace", + "allboards.add-subworkspace-prompt": "Subworkspace name", + "allboards.edit-workspace": "Edit workspace", + "allboards.edit-workspace-name": "Workspace name", + "allboards.edit-workspace-icon": "Workspace icon (markdown)", + "multi-selection-active": "Click checkboxes to select boards", "activity-dueDate": "edited due date to %s of %s", "activity-endDate": "edited end date to %s of %s", "add-attachment": "Add Attachment", @@ -337,6 +349,7 @@ "copyManyCardsPopup-format": "[ {\"title\": \"First card title\", \"description\":\"First card description\"}, {\"title\":\"Second card title\",\"description\":\"Second card description\"},{\"title\":\"Last card title\",\"description\":\"Last card description\"} ]", "create": "Create", "createBoardPopup-title": "Create Board", + "createTemplateContainerPopup-title": "Add Template Container", "chooseBoardSourcePopup-title": "Import board", "createLabelPopup-title": "Create Label", "createCustomField": "Create Field", diff --git a/imports/i18n/data/az.i18n.json b/imports/i18n/data/az.i18n.json index 4804e6dff..b6aac02e5 100644 --- a/imports/i18n/data/az.i18n.json +++ b/imports/i18n/data/az.i18n.json @@ -78,6 +78,18 @@ "activity-deleteComment": "deleted comment %s", "activity-receivedDate": "edited received date to %s of %s", "activity-startDate": "edited start date to %s of %s", + "allboards.starred": "Starred", + "allboards.templates": "Templates", + "allboards.remaining": "Remaining", + "allboards.workspaces": "Workspaces", + "allboards.add-workspace": "Add Workspace", + "allboards.add-workspace-prompt": "Workspace name", + "allboards.add-subworkspace": "Add Subworkspace", + "allboards.add-subworkspace-prompt": "Subworkspace name", + "allboards.edit-workspace": "Edit workspace", + "allboards.edit-workspace-name": "Workspace name", + "allboards.edit-workspace-icon": "Workspace icon (markdown)", + "multi-selection-active": "Click checkboxes to select boards", "activity-dueDate": "edited due date to %s of %s", "activity-endDate": "edited end date to %s of %s", "add-attachment": "Add Attachment", @@ -337,6 +349,7 @@ "copyManyCardsPopup-format": "[ {\"title\": \"First card title\", \"description\":\"First card description\"}, {\"title\":\"Second card title\",\"description\":\"Second card description\"},{\"title\":\"Last card title\",\"description\":\"Last card description\"} ]", "create": "Create", "createBoardPopup-title": "Create Board", + "createTemplateContainerPopup-title": "Add Template Container", "chooseBoardSourcePopup-title": "Import board", "createLabelPopup-title": "Create Label", "createCustomField": "Create Field", diff --git a/imports/i18n/data/bg.i18n.json b/imports/i18n/data/bg.i18n.json index ae0151f95..f78534731 100644 --- a/imports/i18n/data/bg.i18n.json +++ b/imports/i18n/data/bg.i18n.json @@ -78,6 +78,18 @@ "activity-deleteComment": "изтрит коментар %s", "activity-receivedDate": "edited received date to %s of %s", "activity-startDate": "edited start date to %s of %s", + "allboards.starred": "Starred", + "allboards.templates": "Шаблони", + "allboards.remaining": "Remaining", + "allboards.workspaces": "Workspaces", + "allboards.add-workspace": "Add Workspace", + "allboards.add-workspace-prompt": "Workspace name", + "allboards.add-subworkspace": "Add Subworkspace", + "allboards.add-subworkspace-prompt": "Subworkspace name", + "allboards.edit-workspace": "Edit workspace", + "allboards.edit-workspace-name": "Workspace name", + "allboards.edit-workspace-icon": "Workspace icon (markdown)", + "multi-selection-active": "Click checkboxes to select boards", "activity-dueDate": "edited due date to %s of %s", "activity-endDate": "edited end date to %s of %s", "add-attachment": "Добави прикачен файл", @@ -337,6 +349,7 @@ "copyManyCardsPopup-format": "[ {\"title\": \"First card title\", \"description\":\"First card description\"}, {\"title\":\"Second card title\",\"description\":\"Second card description\"},{\"title\":\"Last card title\",\"description\":\"Last card description\"} ]", "create": "Създай", "createBoardPopup-title": "Създай Табло", + "createTemplateContainerPopup-title": "Add Template Container", "chooseBoardSourcePopup-title": "Импортирай Табло", "createLabelPopup-title": "Създай Табло", "createCustomField": "Създай Поле", diff --git a/imports/i18n/data/br.i18n.json b/imports/i18n/data/br.i18n.json index 0402c7064..6cbe2a70b 100644 --- a/imports/i18n/data/br.i18n.json +++ b/imports/i18n/data/br.i18n.json @@ -78,6 +78,18 @@ "activity-deleteComment": "deleted comment %s", "activity-receivedDate": "edited received date to %s of %s", "activity-startDate": "edited start date to %s of %s", + "allboards.starred": "Starred", + "allboards.templates": "Templates", + "allboards.remaining": "Remaining", + "allboards.workspaces": "Workspaces", + "allboards.add-workspace": "Add Workspace", + "allboards.add-workspace-prompt": "Workspace name", + "allboards.add-subworkspace": "Add Subworkspace", + "allboards.add-subworkspace-prompt": "Subworkspace name", + "allboards.edit-workspace": "Edit workspace", + "allboards.edit-workspace-name": "Workspace name", + "allboards.edit-workspace-icon": "Workspace icon (markdown)", + "multi-selection-active": "Click checkboxes to select boards", "activity-dueDate": "edited due date to %s of %s", "activity-endDate": "edited end date to %s of %s", "add-attachment": "Add Attachment", @@ -337,6 +349,7 @@ "copyManyCardsPopup-format": "[ {\"title\": \"First card title\", \"description\":\"First card description\"}, {\"title\":\"Second card title\",\"description\":\"Second card description\"},{\"title\":\"Last card title\",\"description\":\"Last card description\"} ]", "create": "Krouiñ", "createBoardPopup-title": "Create Board", + "createTemplateContainerPopup-title": "Add Template Container", "chooseBoardSourcePopup-title": "Import board", "createLabelPopup-title": "Create Label", "createCustomField": "Create Field", diff --git a/imports/i18n/data/ca.i18n.json b/imports/i18n/data/ca.i18n.json index dddf9afc4..349a3e4b0 100644 --- a/imports/i18n/data/ca.i18n.json +++ b/imports/i18n/data/ca.i18n.json @@ -78,6 +78,18 @@ "activity-deleteComment": "Ha esborrar el comentari %s", "activity-receivedDate": "editat la data de recepció a %s de %s", "activity-startDate": "data d'inici editada a %s de %s", + "allboards.starred": "Starred", + "allboards.templates": "Plantilles", + "allboards.remaining": "Remaining", + "allboards.workspaces": "Workspaces", + "allboards.add-workspace": "Add Workspace", + "allboards.add-workspace-prompt": "Workspace name", + "allboards.add-subworkspace": "Add Subworkspace", + "allboards.add-subworkspace-prompt": "Subworkspace name", + "allboards.edit-workspace": "Edit workspace", + "allboards.edit-workspace-name": "Workspace name", + "allboards.edit-workspace-icon": "Workspace icon (markdown)", + "multi-selection-active": "Click checkboxes to select boards", "activity-dueDate": "data de venciment editada a %s de %s", "activity-endDate": "data de finalització editada a %s de %s", "add-attachment": "Afegeix adjunt", @@ -337,6 +349,7 @@ "copyManyCardsPopup-format": "[ {\"title\": \"Títol de la primera fitxa\", \"description\":\"Descripció de la primera fitxa\"}, {\"title\":\"Títol de la segona fitxa\",\"description\":\"Descripció de la segona fitxa \"},{\"title\":\"Títol de l'última fitxa\",\"description\":\"Descripció de l'última fitxa\"} ]", "create": "Crea", "createBoardPopup-title": "Crea tauler", + "createTemplateContainerPopup-title": "Afegeix un Contenidor de plantilles", "chooseBoardSourcePopup-title": "Importa tauler", "createLabelPopup-title": "Crea etiqueta", "createCustomField": "Crear campament", diff --git a/imports/i18n/data/ca@valencia.i18n.json b/imports/i18n/data/ca@valencia.i18n.json index 9af93bcf3..e8d3a8b09 100644 --- a/imports/i18n/data/ca@valencia.i18n.json +++ b/imports/i18n/data/ca@valencia.i18n.json @@ -78,6 +78,18 @@ "activity-deleteComment": "deleted comment %s", "activity-receivedDate": "edited received date to %s of %s", "activity-startDate": "edited start date to %s of %s", + "allboards.starred": "Starred", + "allboards.templates": "Templates", + "allboards.remaining": "Remaining", + "allboards.workspaces": "Workspaces", + "allboards.add-workspace": "Add Workspace", + "allboards.add-workspace-prompt": "Workspace name", + "allboards.add-subworkspace": "Add Subworkspace", + "allboards.add-subworkspace-prompt": "Subworkspace name", + "allboards.edit-workspace": "Edit workspace", + "allboards.edit-workspace-name": "Workspace name", + "allboards.edit-workspace-icon": "Workspace icon (markdown)", + "multi-selection-active": "Click checkboxes to select boards", "activity-dueDate": "edited due date to %s of %s", "activity-endDate": "edited end date to %s of %s", "add-attachment": "Add Attachment", @@ -337,6 +349,7 @@ "copyManyCardsPopup-format": "[ {\"title\": \"First card title\", \"description\":\"First card description\"}, {\"title\":\"Second card title\",\"description\":\"Second card description\"},{\"title\":\"Last card title\",\"description\":\"Last card description\"} ]", "create": "Create", "createBoardPopup-title": "Create Board", + "createTemplateContainerPopup-title": "Add Template Container", "chooseBoardSourcePopup-title": "Import board", "createLabelPopup-title": "Create Label", "createCustomField": "Create Field", diff --git a/imports/i18n/data/ca_ES.i18n.json b/imports/i18n/data/ca_ES.i18n.json index 9f4e10cb1..5cecaa279 100644 --- a/imports/i18n/data/ca_ES.i18n.json +++ b/imports/i18n/data/ca_ES.i18n.json @@ -78,6 +78,18 @@ "activity-deleteComment": "deleted comment %s", "activity-receivedDate": "edited received date to %s of %s", "activity-startDate": "edited start date to %s of %s", + "allboards.starred": "Starred", + "allboards.templates": "Templates", + "allboards.remaining": "Remaining", + "allboards.workspaces": "Workspaces", + "allboards.add-workspace": "Add Workspace", + "allboards.add-workspace-prompt": "Workspace name", + "allboards.add-subworkspace": "Add Subworkspace", + "allboards.add-subworkspace-prompt": "Subworkspace name", + "allboards.edit-workspace": "Edit workspace", + "allboards.edit-workspace-name": "Workspace name", + "allboards.edit-workspace-icon": "Workspace icon (markdown)", + "multi-selection-active": "Click checkboxes to select boards", "activity-dueDate": "edited due date to %s of %s", "activity-endDate": "edited end date to %s of %s", "add-attachment": "Add Attachment", @@ -337,6 +349,7 @@ "copyManyCardsPopup-format": "[ {\"title\": \"First card title\", \"description\":\"First card description\"}, {\"title\":\"Second card title\",\"description\":\"Second card description\"},{\"title\":\"Last card title\",\"description\":\"Last card description\"} ]", "create": "Create", "createBoardPopup-title": "Create Board", + "createTemplateContainerPopup-title": "Add Template Container", "chooseBoardSourcePopup-title": "Import board", "createLabelPopup-title": "Create Label", "createCustomField": "Create Field", diff --git a/imports/i18n/data/cmn.i18n.json b/imports/i18n/data/cmn.i18n.json index 5f18926f7..45bc58d1b 100644 --- a/imports/i18n/data/cmn.i18n.json +++ b/imports/i18n/data/cmn.i18n.json @@ -78,6 +78,18 @@ "activity-deleteComment": "deleted comment %s", "activity-receivedDate": "edited received date to %s of %s", "activity-startDate": "edited start date to %s of %s", + "allboards.starred": "Starred", + "allboards.templates": "Templates", + "allboards.remaining": "Remaining", + "allboards.workspaces": "Workspaces", + "allboards.add-workspace": "Add Workspace", + "allboards.add-workspace-prompt": "Workspace name", + "allboards.add-subworkspace": "Add Subworkspace", + "allboards.add-subworkspace-prompt": "Subworkspace name", + "allboards.edit-workspace": "Edit workspace", + "allboards.edit-workspace-name": "Workspace name", + "allboards.edit-workspace-icon": "Workspace icon (markdown)", + "multi-selection-active": "Click checkboxes to select boards", "activity-dueDate": "edited due date to %s of %s", "activity-endDate": "edited end date to %s of %s", "add-attachment": "Add Attachment", @@ -337,6 +349,7 @@ "copyManyCardsPopup-format": "[ {\"title\": \"First card title\", \"description\":\"First card description\"}, {\"title\":\"Second card title\",\"description\":\"Second card description\"},{\"title\":\"Last card title\",\"description\":\"Last card description\"} ]", "create": "Create", "createBoardPopup-title": "Create Board", + "createTemplateContainerPopup-title": "Add Template Container", "chooseBoardSourcePopup-title": "Import board", "createLabelPopup-title": "Create Label", "createCustomField": "Create Field", diff --git a/imports/i18n/data/cs-CZ.i18n.json b/imports/i18n/data/cs-CZ.i18n.json index 6a5ce87d7..e330e3f2a 100644 --- a/imports/i18n/data/cs-CZ.i18n.json +++ b/imports/i18n/data/cs-CZ.i18n.json @@ -78,6 +78,18 @@ "activity-deleteComment": "smazat komentář %s", "activity-receivedDate": "editoval(a) datum přijetí na %s z %s", "activity-startDate": "editoval(a) datum zahájení na %s z %s", + "allboards.starred": "Starred", + "allboards.templates": "Šablony", + "allboards.remaining": "Remaining", + "allboards.workspaces": "Workspaces", + "allboards.add-workspace": "Add Workspace", + "allboards.add-workspace-prompt": "Workspace name", + "allboards.add-subworkspace": "Add Subworkspace", + "allboards.add-subworkspace-prompt": "Subworkspace name", + "allboards.edit-workspace": "Edit workspace", + "allboards.edit-workspace-name": "Workspace name", + "allboards.edit-workspace-icon": "Workspace icon (markdown)", + "multi-selection-active": "Click checkboxes to select boards", "activity-dueDate": "editoval(a) termín dokončení na %s z %s", "activity-endDate": "editoval(a) datum ukončení na %s z %s", "add-attachment": "Přidat přílohu", @@ -337,6 +349,7 @@ "copyManyCardsPopup-format": "[ {\"title\": \"Nadpis první karty\", \"description\":\"Popis první karty\"}, {\"title\":\"Nadpis druhé karty\",\"description\":\"Popis druhé karty\"},{\"title\":\"Nadpis poslední kary\",\"description\":\"Popis poslední karty\"} ]", "create": "Vytvořit", "createBoardPopup-title": "Vytvořit tablo", + "createTemplateContainerPopup-title": "Přidat kontejner šablony", "chooseBoardSourcePopup-title": "Importovat tablo", "createLabelPopup-title": "Vytvořit štítek", "createCustomField": "Vytvořit pole", diff --git a/imports/i18n/data/cs.i18n.json b/imports/i18n/data/cs.i18n.json index 60f7db17b..84cc45b7e 100644 --- a/imports/i18n/data/cs.i18n.json +++ b/imports/i18n/data/cs.i18n.json @@ -78,6 +78,18 @@ "activity-deleteComment": "smazat komentář %s", "activity-receivedDate": "editoval(a) datum přijetí na %s z %s", "activity-startDate": "editoval(a) datum zahájení na %s z %s", + "allboards.starred": "Starred", + "allboards.templates": "Šablony", + "allboards.remaining": "Remaining", + "allboards.workspaces": "Workspaces", + "allboards.add-workspace": "Add Workspace", + "allboards.add-workspace-prompt": "Workspace name", + "allboards.add-subworkspace": "Add Subworkspace", + "allboards.add-subworkspace-prompt": "Subworkspace name", + "allboards.edit-workspace": "Edit workspace", + "allboards.edit-workspace-name": "Workspace name", + "allboards.edit-workspace-icon": "Workspace icon (markdown)", + "multi-selection-active": "Click checkboxes to select boards", "activity-dueDate": "editoval(a) termín dokončení na %s z %s", "activity-endDate": "editoval(a) datum ukončení na %s z %s", "add-attachment": "Přidat přílohu", @@ -337,6 +349,7 @@ "copyManyCardsPopup-format": "[ {\"title\": \"Nadpis první karty\", \"description\":\"Popis první karty\"}, {\"title\":\"Nadpis druhé karty\",\"description\":\"Popis druhé karty\"},{\"title\":\"Nadpis poslední kary\",\"description\":\"Popis poslední karty\"} ]", "create": "Vytvořit", "createBoardPopup-title": "Vytvořit tablo", + "createTemplateContainerPopup-title": "Přidat kontejner šablony", "chooseBoardSourcePopup-title": "Importovat tablo", "createLabelPopup-title": "Vytvořit štítek", "createCustomField": "Vytvořit pole", diff --git a/imports/i18n/data/cy-GB.i18n.json b/imports/i18n/data/cy-GB.i18n.json index 4804e6dff..b6aac02e5 100644 --- a/imports/i18n/data/cy-GB.i18n.json +++ b/imports/i18n/data/cy-GB.i18n.json @@ -78,6 +78,18 @@ "activity-deleteComment": "deleted comment %s", "activity-receivedDate": "edited received date to %s of %s", "activity-startDate": "edited start date to %s of %s", + "allboards.starred": "Starred", + "allboards.templates": "Templates", + "allboards.remaining": "Remaining", + "allboards.workspaces": "Workspaces", + "allboards.add-workspace": "Add Workspace", + "allboards.add-workspace-prompt": "Workspace name", + "allboards.add-subworkspace": "Add Subworkspace", + "allboards.add-subworkspace-prompt": "Subworkspace name", + "allboards.edit-workspace": "Edit workspace", + "allboards.edit-workspace-name": "Workspace name", + "allboards.edit-workspace-icon": "Workspace icon (markdown)", + "multi-selection-active": "Click checkboxes to select boards", "activity-dueDate": "edited due date to %s of %s", "activity-endDate": "edited end date to %s of %s", "add-attachment": "Add Attachment", @@ -337,6 +349,7 @@ "copyManyCardsPopup-format": "[ {\"title\": \"First card title\", \"description\":\"First card description\"}, {\"title\":\"Second card title\",\"description\":\"Second card description\"},{\"title\":\"Last card title\",\"description\":\"Last card description\"} ]", "create": "Create", "createBoardPopup-title": "Create Board", + "createTemplateContainerPopup-title": "Add Template Container", "chooseBoardSourcePopup-title": "Import board", "createLabelPopup-title": "Create Label", "createCustomField": "Create Field", diff --git a/imports/i18n/data/cy.i18n.json b/imports/i18n/data/cy.i18n.json index 4804e6dff..b6aac02e5 100644 --- a/imports/i18n/data/cy.i18n.json +++ b/imports/i18n/data/cy.i18n.json @@ -78,6 +78,18 @@ "activity-deleteComment": "deleted comment %s", "activity-receivedDate": "edited received date to %s of %s", "activity-startDate": "edited start date to %s of %s", + "allboards.starred": "Starred", + "allboards.templates": "Templates", + "allboards.remaining": "Remaining", + "allboards.workspaces": "Workspaces", + "allboards.add-workspace": "Add Workspace", + "allboards.add-workspace-prompt": "Workspace name", + "allboards.add-subworkspace": "Add Subworkspace", + "allboards.add-subworkspace-prompt": "Subworkspace name", + "allboards.edit-workspace": "Edit workspace", + "allboards.edit-workspace-name": "Workspace name", + "allboards.edit-workspace-icon": "Workspace icon (markdown)", + "multi-selection-active": "Click checkboxes to select boards", "activity-dueDate": "edited due date to %s of %s", "activity-endDate": "edited end date to %s of %s", "add-attachment": "Add Attachment", @@ -337,6 +349,7 @@ "copyManyCardsPopup-format": "[ {\"title\": \"First card title\", \"description\":\"First card description\"}, {\"title\":\"Second card title\",\"description\":\"Second card description\"},{\"title\":\"Last card title\",\"description\":\"Last card description\"} ]", "create": "Create", "createBoardPopup-title": "Create Board", + "createTemplateContainerPopup-title": "Add Template Container", "chooseBoardSourcePopup-title": "Import board", "createLabelPopup-title": "Create Label", "createCustomField": "Create Field", diff --git a/imports/i18n/data/da.i18n.json b/imports/i18n/data/da.i18n.json index d6a3e0e15..4edbf48bc 100644 --- a/imports/i18n/data/da.i18n.json +++ b/imports/i18n/data/da.i18n.json @@ -78,6 +78,18 @@ "activity-deleteComment": "slettede kommentar %s", "activity-receivedDate": "edited received date to %s of %s", "activity-startDate": "edited start date to %s of %s", + "allboards.starred": "Starred", + "allboards.templates": "Skabeloner", + "allboards.remaining": "Remaining", + "allboards.workspaces": "Workspaces", + "allboards.add-workspace": "Add Workspace", + "allboards.add-workspace-prompt": "Workspace name", + "allboards.add-subworkspace": "Add Subworkspace", + "allboards.add-subworkspace-prompt": "Subworkspace name", + "allboards.edit-workspace": "Edit workspace", + "allboards.edit-workspace-name": "Workspace name", + "allboards.edit-workspace-icon": "Workspace icon (markdown)", + "multi-selection-active": "Click checkboxes to select boards", "activity-dueDate": "edited due date to %s of %s", "activity-endDate": "edited end date to %s of %s", "add-attachment": "Tilføj vedhæftning", @@ -337,6 +349,7 @@ "copyManyCardsPopup-format": "[ {\"title\": \"First card title\", \"description\":\"First card description\"}, {\"title\":\"Second card title\",\"description\":\"Second card description\"},{\"title\":\"Last card title\",\"description\":\"Last card description\"} ]", "create": "Opret", "createBoardPopup-title": "Opret tavle", + "createTemplateContainerPopup-title": "Add Template Container", "chooseBoardSourcePopup-title": "Importér tavle", "createLabelPopup-title": "Opret etikette", "createCustomField": "Opret felt", diff --git a/imports/i18n/data/de-AT.i18n.json b/imports/i18n/data/de-AT.i18n.json index 03616ba4c..eeac5f73b 100644 --- a/imports/i18n/data/de-AT.i18n.json +++ b/imports/i18n/data/de-AT.i18n.json @@ -78,6 +78,18 @@ "activity-deleteComment": "löschte Kommentar %s", "activity-receivedDate": "hat Empfangsdatum zu %s geändert auf %s", "activity-startDate": "hat Startdatum zu %s geändert auf %s", + "allboards.starred": "Starred", + "allboards.templates": "Vorlagen", + "allboards.remaining": "Remaining", + "allboards.workspaces": "Workspaces", + "allboards.add-workspace": "Add Workspace", + "allboards.add-workspace-prompt": "Workspace name", + "allboards.add-subworkspace": "Add Subworkspace", + "allboards.add-subworkspace-prompt": "Subworkspace name", + "allboards.edit-workspace": "Edit workspace", + "allboards.edit-workspace-name": "Workspace name", + "allboards.edit-workspace-icon": "Workspace icon (markdown)", + "multi-selection-active": "Click checkboxes to select boards", "activity-dueDate": "hat Fälligkeitsdatum zu %s geändert auf %s", "activity-endDate": "hat Enddatum zu %s geändert auf %s", "add-attachment": "Datei anhängen", @@ -337,6 +349,7 @@ "copyManyCardsPopup-format": "[ {\"title\": \"Titel der ersten Karte\", \"description\":\"Beschreibung der ersten Karte\"}, {\"title\":\"Titel der zweiten Karte\",\"description\":\"Beschreibung der zweiten Karte\"},{\"title\":\"Titel der letzten Karte\",\"description\":\"Beschreibung der letzten Karte\"} ]", "create": "Erstellen", "createBoardPopup-title": "Board erstellen", + "createTemplateContainerPopup-title": "Vorlagen-Container hinzufügen", "chooseBoardSourcePopup-title": "Board importieren", "createLabelPopup-title": "Label erstellen", "createCustomField": "Feld erstellen", diff --git a/imports/i18n/data/de-CH.i18n.json b/imports/i18n/data/de-CH.i18n.json index 16a06886c..afdea98f1 100644 --- a/imports/i18n/data/de-CH.i18n.json +++ b/imports/i18n/data/de-CH.i18n.json @@ -78,6 +78,18 @@ "activity-deleteComment": "löschte Kommentar %s", "activity-receivedDate": "hat Empfangsdatum zu %s geändert auf %s", "activity-startDate": "hat Startdatum zu %s geändert auf %s", + "allboards.starred": "Starred", + "allboards.templates": "Vorlagen", + "allboards.remaining": "Remaining", + "allboards.workspaces": "Workspaces", + "allboards.add-workspace": "Add Workspace", + "allboards.add-workspace-prompt": "Workspace name", + "allboards.add-subworkspace": "Add Subworkspace", + "allboards.add-subworkspace-prompt": "Subworkspace name", + "allboards.edit-workspace": "Edit workspace", + "allboards.edit-workspace-name": "Workspace name", + "allboards.edit-workspace-icon": "Workspace icon (markdown)", + "multi-selection-active": "Click checkboxes to select boards", "activity-dueDate": "hat Fälligkeitsdatum zu %s geändert auf %s", "activity-endDate": "hat Enddatum zu %s geändert auf %s", "add-attachment": "Datei anhängen", @@ -337,6 +349,7 @@ "copyManyCardsPopup-format": "[ {\"title\": \"Titel der ersten Karte\", \"description\":\"Beschreibung der ersten Karte\"}, {\"title\":\"Titel der zweiten Karte\",\"description\":\"Beschreibung der zweiten Karte\"},{\"title\":\"Titel der letzten Karte\",\"description\":\"Beschreibung der letzten Karte\"} ]", "create": "Erstellen", "createBoardPopup-title": "Board erstellen", + "createTemplateContainerPopup-title": "Vorlagen-Container hinzufügen", "chooseBoardSourcePopup-title": "Board importieren", "createLabelPopup-title": "Label erstellen", "createCustomField": "Feld erstellen", diff --git a/imports/i18n/data/de.i18n.json b/imports/i18n/data/de.i18n.json index 0ca4846d8..cfb4ea635 100644 --- a/imports/i18n/data/de.i18n.json +++ b/imports/i18n/data/de.i18n.json @@ -78,6 +78,18 @@ "activity-deleteComment": "löschte Kommentar %s", "activity-receivedDate": "hat Empfangsdatum zu %s geändert auf %s", "activity-startDate": "hat Startdatum zu %s geändert auf %s", + "allboards.starred": "Starred", + "allboards.templates": "Vorlagen", + "allboards.remaining": "Remaining", + "allboards.workspaces": "Workspaces", + "allboards.add-workspace": "Add Workspace", + "allboards.add-workspace-prompt": "Workspace name", + "allboards.add-subworkspace": "Add Subworkspace", + "allboards.add-subworkspace-prompt": "Subworkspace name", + "allboards.edit-workspace": "Edit workspace", + "allboards.edit-workspace-name": "Workspace name", + "allboards.edit-workspace-icon": "Workspace icon (markdown)", + "multi-selection-active": "Click checkboxes to select boards", "activity-dueDate": "hat Fälligkeitsdatum zu %s geändert auf %s", "activity-endDate": "hat Enddatum zu %s geändert auf %s", "add-attachment": "Datei anhängen", @@ -337,6 +349,7 @@ "copyManyCardsPopup-format": "[ {\"title\": \"Titel der ersten Karte\", \"description\":\"Beschreibung der ersten Karte\"}, {\"title\":\"Titel der zweiten Karte\",\"description\":\"Beschreibung der zweiten Karte\"},{\"title\":\"Titel der letzten Karte\",\"description\":\"Beschreibung der letzten Karte\"} ]", "create": "Erstellen", "createBoardPopup-title": "Board erstellen", + "createTemplateContainerPopup-title": "Vorlagen-Container hinzufügen", "chooseBoardSourcePopup-title": "Board importieren", "createLabelPopup-title": "Label erstellen", "createCustomField": "Feld erstellen", diff --git a/imports/i18n/data/de_DE.i18n.json b/imports/i18n/data/de_DE.i18n.json index cc1b326ae..792d13633 100644 --- a/imports/i18n/data/de_DE.i18n.json +++ b/imports/i18n/data/de_DE.i18n.json @@ -78,6 +78,18 @@ "activity-deleteComment": "löschte Kommentar %s", "activity-receivedDate": "hat Empfangsdatum zu %s geändert auf %s", "activity-startDate": "hat Startdatum zu %s geändert auf %s", + "allboards.starred": "Starred", + "allboards.templates": "Vorlagen", + "allboards.remaining": "Remaining", + "allboards.workspaces": "Workspaces", + "allboards.add-workspace": "Add Workspace", + "allboards.add-workspace-prompt": "Workspace name", + "allboards.add-subworkspace": "Add Subworkspace", + "allboards.add-subworkspace-prompt": "Subworkspace name", + "allboards.edit-workspace": "Edit workspace", + "allboards.edit-workspace-name": "Workspace name", + "allboards.edit-workspace-icon": "Workspace icon (markdown)", + "multi-selection-active": "Click checkboxes to select boards", "activity-dueDate": "hat Fälligkeitsdatum zu %s geändert auf %s", "activity-endDate": "hat Enddatum zu %s geändert auf %s", "add-attachment": "Datei anhängen", @@ -337,6 +349,7 @@ "copyManyCardsPopup-format": "[ {\"title\": \"Titel der ersten Karte\", \"description\":\"Beschreibung der ersten Karte\"}, {\"title\":\"Titel der zweiten Karte\",\"description\":\"Beschreibung der zweiten Karte\"},{\"title\":\"Titel der letzten Karte\",\"description\":\"Beschreibung der letzten Karte\"} ]", "create": "Erstellen", "createBoardPopup-title": "Board erstellen", + "createTemplateContainerPopup-title": "Vorlagen-Container hinzufügen", "chooseBoardSourcePopup-title": "Board importieren", "createLabelPopup-title": "Label erstellen", "createCustomField": "Feld erstellen", diff --git a/imports/i18n/data/el-GR.i18n.json b/imports/i18n/data/el-GR.i18n.json index 17367197b..efd48c486 100644 --- a/imports/i18n/data/el-GR.i18n.json +++ b/imports/i18n/data/el-GR.i18n.json @@ -78,6 +78,18 @@ "activity-deleteComment": "διεγράφη το σχόλιο %s", "activity-receivedDate": "η ημερομηνία λήψης άλλαξε σε %s από %s", "activity-startDate": "η ημερομηνία έναρξης άλλαξε σε %s από %s", + "allboards.starred": "Starred", + "allboards.templates": "Πρότυπα", + "allboards.remaining": "Remaining", + "allboards.workspaces": "Workspaces", + "allboards.add-workspace": "Add Workspace", + "allboards.add-workspace-prompt": "Workspace name", + "allboards.add-subworkspace": "Add Subworkspace", + "allboards.add-subworkspace-prompt": "Subworkspace name", + "allboards.edit-workspace": "Edit workspace", + "allboards.edit-workspace-name": "Workspace name", + "allboards.edit-workspace-icon": "Workspace icon (markdown)", + "multi-selection-active": "Click checkboxes to select boards", "activity-dueDate": "υπέστη επεξεργασία η τιμή της προθεσμίας σε %s από %s", "activity-endDate": "η ημερομηνία λήξης άλλαξε σε %s από %s", "add-attachment": "Προσθήκη Συνημμένου", @@ -337,6 +349,7 @@ "copyManyCardsPopup-format": "[ {\"title\": \"Τίτλος πρώτης κάρτας\", \"description\":\"Περιγραφή πρώτης κάρτας\"}, {\"title\":\"Τίτλος δεύτερης κάρτας\",\"description\":\"Περιγραφή δεύτερης κάρτας\"},{\"title\":\"Τίτλος τελευταίας κάρτας\",\"description\":\"Περιγραφή τελευταίας κάρτας\"} ]", "create": "Δημιουργία", "createBoardPopup-title": "Δημιουργία Πίνακα", + "createTemplateContainerPopup-title": "Add Template Container", "chooseBoardSourcePopup-title": "Εισαγωγή πίνακα", "createLabelPopup-title": "Δημιουργία Ετικέτας", "createCustomField": "Δημιουργία Πεδίου", diff --git a/imports/i18n/data/el.i18n.json b/imports/i18n/data/el.i18n.json index 3bd3440e6..0ce0024ac 100644 --- a/imports/i18n/data/el.i18n.json +++ b/imports/i18n/data/el.i18n.json @@ -78,6 +78,18 @@ "activity-deleteComment": "διεγράφη το σχόλιο %s", "activity-receivedDate": "η ημερομηνία λήψης άλλαξε σε %s από %s", "activity-startDate": "η ημερομηνία έναρξης άλλαξε σε %s από %s", + "allboards.starred": "Starred", + "allboards.templates": "Πρότυπα", + "allboards.remaining": "Remaining", + "allboards.workspaces": "Workspaces", + "allboards.add-workspace": "Add Workspace", + "allboards.add-workspace-prompt": "Workspace name", + "allboards.add-subworkspace": "Add Subworkspace", + "allboards.add-subworkspace-prompt": "Subworkspace name", + "allboards.edit-workspace": "Edit workspace", + "allboards.edit-workspace-name": "Workspace name", + "allboards.edit-workspace-icon": "Workspace icon (markdown)", + "multi-selection-active": "Click checkboxes to select boards", "activity-dueDate": "υπέστη επεξεργασία η τιμή της προθεσμίας σε %s από %s", "activity-endDate": "η ημερομηνία λήξης άλλαξε σε %s από %s", "add-attachment": "Προσθήκη Συνημμένου", @@ -337,6 +349,7 @@ "copyManyCardsPopup-format": "[ {\"title\": \"Τίτλος πρώτης κάρτας\", \"description\":\"Περιγραφή πρώτης κάρτας\"}, {\"title\":\"Τίτλος δεύτερης κάρτας\",\"description\":\"Περιγραφή δεύτερης κάρτας\"},{\"title\":\"Τίτλος τελευταίας κάρτας\",\"description\":\"Περιγραφή τελευταίας κάρτας\"} ]", "create": "Δημιουργία", "createBoardPopup-title": "Δημιουργία Πίνακα", + "createTemplateContainerPopup-title": "Add Template Container", "chooseBoardSourcePopup-title": "Εισαγωγή πίνακα", "createLabelPopup-title": "Δημιουργία Ετικέτας", "createCustomField": "Δημιουργία Πεδίου", diff --git a/imports/i18n/data/en-BR.i18n.json b/imports/i18n/data/en-BR.i18n.json index 4804e6dff..b6aac02e5 100644 --- a/imports/i18n/data/en-BR.i18n.json +++ b/imports/i18n/data/en-BR.i18n.json @@ -78,6 +78,18 @@ "activity-deleteComment": "deleted comment %s", "activity-receivedDate": "edited received date to %s of %s", "activity-startDate": "edited start date to %s of %s", + "allboards.starred": "Starred", + "allboards.templates": "Templates", + "allboards.remaining": "Remaining", + "allboards.workspaces": "Workspaces", + "allboards.add-workspace": "Add Workspace", + "allboards.add-workspace-prompt": "Workspace name", + "allboards.add-subworkspace": "Add Subworkspace", + "allboards.add-subworkspace-prompt": "Subworkspace name", + "allboards.edit-workspace": "Edit workspace", + "allboards.edit-workspace-name": "Workspace name", + "allboards.edit-workspace-icon": "Workspace icon (markdown)", + "multi-selection-active": "Click checkboxes to select boards", "activity-dueDate": "edited due date to %s of %s", "activity-endDate": "edited end date to %s of %s", "add-attachment": "Add Attachment", @@ -337,6 +349,7 @@ "copyManyCardsPopup-format": "[ {\"title\": \"First card title\", \"description\":\"First card description\"}, {\"title\":\"Second card title\",\"description\":\"Second card description\"},{\"title\":\"Last card title\",\"description\":\"Last card description\"} ]", "create": "Create", "createBoardPopup-title": "Create Board", + "createTemplateContainerPopup-title": "Add Template Container", "chooseBoardSourcePopup-title": "Import board", "createLabelPopup-title": "Create Label", "createCustomField": "Create Field", diff --git a/imports/i18n/data/en-DE.i18n.json b/imports/i18n/data/en-DE.i18n.json index 6f376a14c..7a135972e 100644 --- a/imports/i18n/data/en-DE.i18n.json +++ b/imports/i18n/data/en-DE.i18n.json @@ -78,6 +78,18 @@ "activity-deleteComment": "deleted comment %s", "activity-receivedDate": "edited received date to %s of %s", "activity-startDate": "edited start date to %s of %s", + "allboards.starred": "Starred", + "allboards.templates": "Templates", + "allboards.remaining": "Remaining", + "allboards.workspaces": "Workspaces", + "allboards.add-workspace": "Add Workspace", + "allboards.add-workspace-prompt": "Workspace name", + "allboards.add-subworkspace": "Add Subworkspace", + "allboards.add-subworkspace-prompt": "Subworkspace name", + "allboards.edit-workspace": "Edit workspace", + "allboards.edit-workspace-name": "Workspace name", + "allboards.edit-workspace-icon": "Workspace icon (markdown)", + "multi-selection-active": "Click checkboxes to select boards", "activity-dueDate": "edited due date to %s of %s", "activity-endDate": "edited end date to %s of %s", "add-attachment": "Add Attachment", @@ -337,6 +349,7 @@ "copyManyCardsPopup-format": "[ {\"title\": \"First card title\", \"description\":\"First card description\"}, {\"title\":\"Second card title\",\"description\":\"Second card description\"},{\"title\":\"Last card title\",\"description\":\"Last card description\"} ]", "create": "Create", "createBoardPopup-title": "Create Board", + "createTemplateContainerPopup-title": "Add Template Container", "chooseBoardSourcePopup-title": "Import board", "createLabelPopup-title": "Create Label", "createCustomField": "Create Field", diff --git a/imports/i18n/data/en-GB.i18n.json b/imports/i18n/data/en-GB.i18n.json index 958d8313a..acf3c6934 100644 --- a/imports/i18n/data/en-GB.i18n.json +++ b/imports/i18n/data/en-GB.i18n.json @@ -78,6 +78,18 @@ "activity-deleteComment": "deleted comment %s", "activity-receivedDate": "edited received date to %s of %s", "activity-startDate": "edited start date to %s of %s", + "allboards.starred": "Starred", + "allboards.templates": "Templates", + "allboards.remaining": "Remaining", + "allboards.workspaces": "Workspaces", + "allboards.add-workspace": "Add Workspace", + "allboards.add-workspace-prompt": "Workspace name", + "allboards.add-subworkspace": "Add Subworkspace", + "allboards.add-subworkspace-prompt": "Subworkspace name", + "allboards.edit-workspace": "Edit workspace", + "allboards.edit-workspace-name": "Workspace name", + "allboards.edit-workspace-icon": "Workspace icon (markdown)", + "multi-selection-active": "Click checkboxes to select boards", "activity-dueDate": "edited due date to %s of %s", "activity-endDate": "edited end date to %s of %s", "add-attachment": "Add Attachment", @@ -337,6 +349,7 @@ "copyManyCardsPopup-format": "[ {\"title\": \"First card title\", \"description\":\"First card description\"}, {\"title\":\"Second card title\",\"description\":\"Second card description\"},{\"title\":\"Last card title\",\"description\":\"Last card description\"} ]", "create": "Create", "createBoardPopup-title": "Create Board", + "createTemplateContainerPopup-title": "Add Template Container", "chooseBoardSourcePopup-title": "Import board", "createLabelPopup-title": "Create Label", "createCustomField": "Create Field", diff --git a/imports/i18n/data/en-IT.i18n.json b/imports/i18n/data/en-IT.i18n.json index 4804e6dff..b6aac02e5 100644 --- a/imports/i18n/data/en-IT.i18n.json +++ b/imports/i18n/data/en-IT.i18n.json @@ -78,6 +78,18 @@ "activity-deleteComment": "deleted comment %s", "activity-receivedDate": "edited received date to %s of %s", "activity-startDate": "edited start date to %s of %s", + "allboards.starred": "Starred", + "allboards.templates": "Templates", + "allboards.remaining": "Remaining", + "allboards.workspaces": "Workspaces", + "allboards.add-workspace": "Add Workspace", + "allboards.add-workspace-prompt": "Workspace name", + "allboards.add-subworkspace": "Add Subworkspace", + "allboards.add-subworkspace-prompt": "Subworkspace name", + "allboards.edit-workspace": "Edit workspace", + "allboards.edit-workspace-name": "Workspace name", + "allboards.edit-workspace-icon": "Workspace icon (markdown)", + "multi-selection-active": "Click checkboxes to select boards", "activity-dueDate": "edited due date to %s of %s", "activity-endDate": "edited end date to %s of %s", "add-attachment": "Add Attachment", @@ -337,6 +349,7 @@ "copyManyCardsPopup-format": "[ {\"title\": \"First card title\", \"description\":\"First card description\"}, {\"title\":\"Second card title\",\"description\":\"Second card description\"},{\"title\":\"Last card title\",\"description\":\"Last card description\"} ]", "create": "Create", "createBoardPopup-title": "Create Board", + "createTemplateContainerPopup-title": "Add Template Container", "chooseBoardSourcePopup-title": "Import board", "createLabelPopup-title": "Create Label", "createCustomField": "Create Field", diff --git a/imports/i18n/data/en-MY.i18n.json b/imports/i18n/data/en-MY.i18n.json index 4804e6dff..b6aac02e5 100644 --- a/imports/i18n/data/en-MY.i18n.json +++ b/imports/i18n/data/en-MY.i18n.json @@ -78,6 +78,18 @@ "activity-deleteComment": "deleted comment %s", "activity-receivedDate": "edited received date to %s of %s", "activity-startDate": "edited start date to %s of %s", + "allboards.starred": "Starred", + "allboards.templates": "Templates", + "allboards.remaining": "Remaining", + "allboards.workspaces": "Workspaces", + "allboards.add-workspace": "Add Workspace", + "allboards.add-workspace-prompt": "Workspace name", + "allboards.add-subworkspace": "Add Subworkspace", + "allboards.add-subworkspace-prompt": "Subworkspace name", + "allboards.edit-workspace": "Edit workspace", + "allboards.edit-workspace-name": "Workspace name", + "allboards.edit-workspace-icon": "Workspace icon (markdown)", + "multi-selection-active": "Click checkboxes to select boards", "activity-dueDate": "edited due date to %s of %s", "activity-endDate": "edited end date to %s of %s", "add-attachment": "Add Attachment", @@ -337,6 +349,7 @@ "copyManyCardsPopup-format": "[ {\"title\": \"First card title\", \"description\":\"First card description\"}, {\"title\":\"Second card title\",\"description\":\"Second card description\"},{\"title\":\"Last card title\",\"description\":\"Last card description\"} ]", "create": "Create", "createBoardPopup-title": "Create Board", + "createTemplateContainerPopup-title": "Add Template Container", "chooseBoardSourcePopup-title": "Import board", "createLabelPopup-title": "Create Label", "createCustomField": "Create Field", diff --git a/imports/i18n/data/en-YS.i18n.json b/imports/i18n/data/en-YS.i18n.json index 4804e6dff..b6aac02e5 100644 --- a/imports/i18n/data/en-YS.i18n.json +++ b/imports/i18n/data/en-YS.i18n.json @@ -78,6 +78,18 @@ "activity-deleteComment": "deleted comment %s", "activity-receivedDate": "edited received date to %s of %s", "activity-startDate": "edited start date to %s of %s", + "allboards.starred": "Starred", + "allboards.templates": "Templates", + "allboards.remaining": "Remaining", + "allboards.workspaces": "Workspaces", + "allboards.add-workspace": "Add Workspace", + "allboards.add-workspace-prompt": "Workspace name", + "allboards.add-subworkspace": "Add Subworkspace", + "allboards.add-subworkspace-prompt": "Subworkspace name", + "allboards.edit-workspace": "Edit workspace", + "allboards.edit-workspace-name": "Workspace name", + "allboards.edit-workspace-icon": "Workspace icon (markdown)", + "multi-selection-active": "Click checkboxes to select boards", "activity-dueDate": "edited due date to %s of %s", "activity-endDate": "edited end date to %s of %s", "add-attachment": "Add Attachment", @@ -337,6 +349,7 @@ "copyManyCardsPopup-format": "[ {\"title\": \"First card title\", \"description\":\"First card description\"}, {\"title\":\"Second card title\",\"description\":\"Second card description\"},{\"title\":\"Last card title\",\"description\":\"Last card description\"} ]", "create": "Create", "createBoardPopup-title": "Create Board", + "createTemplateContainerPopup-title": "Add Template Container", "chooseBoardSourcePopup-title": "Import board", "createLabelPopup-title": "Create Label", "createCustomField": "Create Field", diff --git a/imports/i18n/data/en.i18n.json b/imports/i18n/data/en.i18n.json index 958d8313a..acf3c6934 100644 --- a/imports/i18n/data/en.i18n.json +++ b/imports/i18n/data/en.i18n.json @@ -78,6 +78,18 @@ "activity-deleteComment": "deleted comment %s", "activity-receivedDate": "edited received date to %s of %s", "activity-startDate": "edited start date to %s of %s", + "allboards.starred": "Starred", + "allboards.templates": "Templates", + "allboards.remaining": "Remaining", + "allboards.workspaces": "Workspaces", + "allboards.add-workspace": "Add Workspace", + "allboards.add-workspace-prompt": "Workspace name", + "allboards.add-subworkspace": "Add Subworkspace", + "allboards.add-subworkspace-prompt": "Subworkspace name", + "allboards.edit-workspace": "Edit workspace", + "allboards.edit-workspace-name": "Workspace name", + "allboards.edit-workspace-icon": "Workspace icon (markdown)", + "multi-selection-active": "Click checkboxes to select boards", "activity-dueDate": "edited due date to %s of %s", "activity-endDate": "edited end date to %s of %s", "add-attachment": "Add Attachment", @@ -337,6 +349,7 @@ "copyManyCardsPopup-format": "[ {\"title\": \"First card title\", \"description\":\"First card description\"}, {\"title\":\"Second card title\",\"description\":\"Second card description\"},{\"title\":\"Last card title\",\"description\":\"Last card description\"} ]", "create": "Create", "createBoardPopup-title": "Create Board", + "createTemplateContainerPopup-title": "Add Template Container", "chooseBoardSourcePopup-title": "Import board", "createLabelPopup-title": "Create Label", "createCustomField": "Create Field", diff --git a/imports/i18n/data/en_AU.i18n.json b/imports/i18n/data/en_AU.i18n.json index 958d8313a..acf3c6934 100644 --- a/imports/i18n/data/en_AU.i18n.json +++ b/imports/i18n/data/en_AU.i18n.json @@ -78,6 +78,18 @@ "activity-deleteComment": "deleted comment %s", "activity-receivedDate": "edited received date to %s of %s", "activity-startDate": "edited start date to %s of %s", + "allboards.starred": "Starred", + "allboards.templates": "Templates", + "allboards.remaining": "Remaining", + "allboards.workspaces": "Workspaces", + "allboards.add-workspace": "Add Workspace", + "allboards.add-workspace-prompt": "Workspace name", + "allboards.add-subworkspace": "Add Subworkspace", + "allboards.add-subworkspace-prompt": "Subworkspace name", + "allboards.edit-workspace": "Edit workspace", + "allboards.edit-workspace-name": "Workspace name", + "allboards.edit-workspace-icon": "Workspace icon (markdown)", + "multi-selection-active": "Click checkboxes to select boards", "activity-dueDate": "edited due date to %s of %s", "activity-endDate": "edited end date to %s of %s", "add-attachment": "Add Attachment", @@ -337,6 +349,7 @@ "copyManyCardsPopup-format": "[ {\"title\": \"First card title\", \"description\":\"First card description\"}, {\"title\":\"Second card title\",\"description\":\"Second card description\"},{\"title\":\"Last card title\",\"description\":\"Last card description\"} ]", "create": "Create", "createBoardPopup-title": "Create Board", + "createTemplateContainerPopup-title": "Add Template Container", "chooseBoardSourcePopup-title": "Import board", "createLabelPopup-title": "Create Label", "createCustomField": "Create Field", diff --git a/imports/i18n/data/en_ID.i18n.json b/imports/i18n/data/en_ID.i18n.json index 958d8313a..acf3c6934 100644 --- a/imports/i18n/data/en_ID.i18n.json +++ b/imports/i18n/data/en_ID.i18n.json @@ -78,6 +78,18 @@ "activity-deleteComment": "deleted comment %s", "activity-receivedDate": "edited received date to %s of %s", "activity-startDate": "edited start date to %s of %s", + "allboards.starred": "Starred", + "allboards.templates": "Templates", + "allboards.remaining": "Remaining", + "allboards.workspaces": "Workspaces", + "allboards.add-workspace": "Add Workspace", + "allboards.add-workspace-prompt": "Workspace name", + "allboards.add-subworkspace": "Add Subworkspace", + "allboards.add-subworkspace-prompt": "Subworkspace name", + "allboards.edit-workspace": "Edit workspace", + "allboards.edit-workspace-name": "Workspace name", + "allboards.edit-workspace-icon": "Workspace icon (markdown)", + "multi-selection-active": "Click checkboxes to select boards", "activity-dueDate": "edited due date to %s of %s", "activity-endDate": "edited end date to %s of %s", "add-attachment": "Add Attachment", @@ -337,6 +349,7 @@ "copyManyCardsPopup-format": "[ {\"title\": \"First card title\", \"description\":\"First card description\"}, {\"title\":\"Second card title\",\"description\":\"Second card description\"},{\"title\":\"Last card title\",\"description\":\"Last card description\"} ]", "create": "Create", "createBoardPopup-title": "Create Board", + "createTemplateContainerPopup-title": "Add Template Container", "chooseBoardSourcePopup-title": "Import board", "createLabelPopup-title": "Create Label", "createCustomField": "Create Field", diff --git a/imports/i18n/data/en_SG.i18n.json b/imports/i18n/data/en_SG.i18n.json index 958d8313a..acf3c6934 100644 --- a/imports/i18n/data/en_SG.i18n.json +++ b/imports/i18n/data/en_SG.i18n.json @@ -78,6 +78,18 @@ "activity-deleteComment": "deleted comment %s", "activity-receivedDate": "edited received date to %s of %s", "activity-startDate": "edited start date to %s of %s", + "allboards.starred": "Starred", + "allboards.templates": "Templates", + "allboards.remaining": "Remaining", + "allboards.workspaces": "Workspaces", + "allboards.add-workspace": "Add Workspace", + "allboards.add-workspace-prompt": "Workspace name", + "allboards.add-subworkspace": "Add Subworkspace", + "allboards.add-subworkspace-prompt": "Subworkspace name", + "allboards.edit-workspace": "Edit workspace", + "allboards.edit-workspace-name": "Workspace name", + "allboards.edit-workspace-icon": "Workspace icon (markdown)", + "multi-selection-active": "Click checkboxes to select boards", "activity-dueDate": "edited due date to %s of %s", "activity-endDate": "edited end date to %s of %s", "add-attachment": "Add Attachment", @@ -337,6 +349,7 @@ "copyManyCardsPopup-format": "[ {\"title\": \"First card title\", \"description\":\"First card description\"}, {\"title\":\"Second card title\",\"description\":\"Second card description\"},{\"title\":\"Last card title\",\"description\":\"Last card description\"} ]", "create": "Create", "createBoardPopup-title": "Create Board", + "createTemplateContainerPopup-title": "Add Template Container", "chooseBoardSourcePopup-title": "Import board", "createLabelPopup-title": "Create Label", "createCustomField": "Create Field", diff --git a/imports/i18n/data/en_TR.i18n.json b/imports/i18n/data/en_TR.i18n.json index 958d8313a..acf3c6934 100644 --- a/imports/i18n/data/en_TR.i18n.json +++ b/imports/i18n/data/en_TR.i18n.json @@ -78,6 +78,18 @@ "activity-deleteComment": "deleted comment %s", "activity-receivedDate": "edited received date to %s of %s", "activity-startDate": "edited start date to %s of %s", + "allboards.starred": "Starred", + "allboards.templates": "Templates", + "allboards.remaining": "Remaining", + "allboards.workspaces": "Workspaces", + "allboards.add-workspace": "Add Workspace", + "allboards.add-workspace-prompt": "Workspace name", + "allboards.add-subworkspace": "Add Subworkspace", + "allboards.add-subworkspace-prompt": "Subworkspace name", + "allboards.edit-workspace": "Edit workspace", + "allboards.edit-workspace-name": "Workspace name", + "allboards.edit-workspace-icon": "Workspace icon (markdown)", + "multi-selection-active": "Click checkboxes to select boards", "activity-dueDate": "edited due date to %s of %s", "activity-endDate": "edited end date to %s of %s", "add-attachment": "Add Attachment", @@ -337,6 +349,7 @@ "copyManyCardsPopup-format": "[ {\"title\": \"First card title\", \"description\":\"First card description\"}, {\"title\":\"Second card title\",\"description\":\"Second card description\"},{\"title\":\"Last card title\",\"description\":\"Last card description\"} ]", "create": "Create", "createBoardPopup-title": "Create Board", + "createTemplateContainerPopup-title": "Add Template Container", "chooseBoardSourcePopup-title": "Import board", "createLabelPopup-title": "Create Label", "createCustomField": "Create Field", diff --git a/imports/i18n/data/en_ZA.i18n.json b/imports/i18n/data/en_ZA.i18n.json index 958d8313a..acf3c6934 100644 --- a/imports/i18n/data/en_ZA.i18n.json +++ b/imports/i18n/data/en_ZA.i18n.json @@ -78,6 +78,18 @@ "activity-deleteComment": "deleted comment %s", "activity-receivedDate": "edited received date to %s of %s", "activity-startDate": "edited start date to %s of %s", + "allboards.starred": "Starred", + "allboards.templates": "Templates", + "allboards.remaining": "Remaining", + "allboards.workspaces": "Workspaces", + "allboards.add-workspace": "Add Workspace", + "allboards.add-workspace-prompt": "Workspace name", + "allboards.add-subworkspace": "Add Subworkspace", + "allboards.add-subworkspace-prompt": "Subworkspace name", + "allboards.edit-workspace": "Edit workspace", + "allboards.edit-workspace-name": "Workspace name", + "allboards.edit-workspace-icon": "Workspace icon (markdown)", + "multi-selection-active": "Click checkboxes to select boards", "activity-dueDate": "edited due date to %s of %s", "activity-endDate": "edited end date to %s of %s", "add-attachment": "Add Attachment", @@ -337,6 +349,7 @@ "copyManyCardsPopup-format": "[ {\"title\": \"First card title\", \"description\":\"First card description\"}, {\"title\":\"Second card title\",\"description\":\"Second card description\"},{\"title\":\"Last card title\",\"description\":\"Last card description\"} ]", "create": "Create", "createBoardPopup-title": "Create Board", + "createTemplateContainerPopup-title": "Add Template Container", "chooseBoardSourcePopup-title": "Import board", "createLabelPopup-title": "Create Label", "createCustomField": "Create Field", diff --git a/imports/i18n/data/eo.i18n.json b/imports/i18n/data/eo.i18n.json index f26ef6c5e..edd730ca6 100644 --- a/imports/i18n/data/eo.i18n.json +++ b/imports/i18n/data/eo.i18n.json @@ -78,6 +78,18 @@ "activity-deleteComment": "deleted comment %s", "activity-receivedDate": "edited received date to %s of %s", "activity-startDate": "edited start date to %s of %s", + "allboards.starred": "Starred", + "allboards.templates": "Templates", + "allboards.remaining": "Remaining", + "allboards.workspaces": "Workspaces", + "allboards.add-workspace": "Add Workspace", + "allboards.add-workspace-prompt": "Workspace name", + "allboards.add-subworkspace": "Add Subworkspace", + "allboards.add-subworkspace-prompt": "Subworkspace name", + "allboards.edit-workspace": "Edit workspace", + "allboards.edit-workspace-name": "Workspace name", + "allboards.edit-workspace-icon": "Workspace icon (markdown)", + "multi-selection-active": "Click checkboxes to select boards", "activity-dueDate": "edited due date to %s of %s", "activity-endDate": "edited end date to %s of %s", "add-attachment": "Add Attachment", @@ -337,6 +349,7 @@ "copyManyCardsPopup-format": "[ {\"title\": \"First card title\", \"description\":\"First card description\"}, {\"title\":\"Second card title\",\"description\":\"Second card description\"},{\"title\":\"Last card title\",\"description\":\"Last card description\"} ]", "create": "Krei", "createBoardPopup-title": "Krei tavolon", + "createTemplateContainerPopup-title": "Add Template Container", "chooseBoardSourcePopup-title": "Import board", "createLabelPopup-title": "Create Label", "createCustomField": "Create Field", diff --git a/imports/i18n/data/es-AR.i18n.json b/imports/i18n/data/es-AR.i18n.json index 9b87e09ab..2965baf43 100644 --- a/imports/i18n/data/es-AR.i18n.json +++ b/imports/i18n/data/es-AR.i18n.json @@ -78,6 +78,18 @@ "activity-deleteComment": "comentario %s eliminado", "activity-receivedDate": "edited received date to %s of %s", "activity-startDate": "edited start date to %s of %s", + "allboards.starred": "Starred", + "allboards.templates": "Plantillas", + "allboards.remaining": "Remaining", + "allboards.workspaces": "Workspaces", + "allboards.add-workspace": "Add Workspace", + "allboards.add-workspace-prompt": "Workspace name", + "allboards.add-subworkspace": "Add Subworkspace", + "allboards.add-subworkspace-prompt": "Subworkspace name", + "allboards.edit-workspace": "Edit workspace", + "allboards.edit-workspace-name": "Workspace name", + "allboards.edit-workspace-icon": "Workspace icon (markdown)", + "multi-selection-active": "Click checkboxes to select boards", "activity-dueDate": "edited due date to %s of %s", "activity-endDate": "edited end date to %s of %s", "add-attachment": "Agregar Adjunto", @@ -337,6 +349,7 @@ "copyManyCardsPopup-format": "[ {\"title\": \"Título de primera tarjeta\", \"description\":\"Descripción de primera tarjeta\"}, {\"title\":\"Título de segunda tarjeta\",\"description\":\"Descripción de segunda tarjeta\"},{\"title\":\"Título de última tarjeta\",\"description\":\"Descripción de última tarjeta\"} ]", "create": "Crear", "createBoardPopup-title": "Crear Tablero", + "createTemplateContainerPopup-title": "Add Template Container", "chooseBoardSourcePopup-title": "Importar tablero", "createLabelPopup-title": "Crear Etiqueta", "createCustomField": "Crear Campo", diff --git a/imports/i18n/data/es-CL.i18n.json b/imports/i18n/data/es-CL.i18n.json index 6adcd95b6..8b2a335e9 100644 --- a/imports/i18n/data/es-CL.i18n.json +++ b/imports/i18n/data/es-CL.i18n.json @@ -78,6 +78,18 @@ "activity-deleteComment": "comentario eliminado", "activity-receivedDate": "edited received date to %s of %s", "activity-startDate": "edited start date to %s of %s", + "allboards.starred": "Starred", + "allboards.templates": "Plantillas", + "allboards.remaining": "Remaining", + "allboards.workspaces": "Workspaces", + "allboards.add-workspace": "Add Workspace", + "allboards.add-workspace-prompt": "Workspace name", + "allboards.add-subworkspace": "Add Subworkspace", + "allboards.add-subworkspace-prompt": "Subworkspace name", + "allboards.edit-workspace": "Edit workspace", + "allboards.edit-workspace-name": "Workspace name", + "allboards.edit-workspace-icon": "Workspace icon (markdown)", + "multi-selection-active": "Click checkboxes to select boards", "activity-dueDate": "edited due date to %s of %s", "activity-endDate": "edited end date to %s of %s", "add-attachment": "Añadir adjunto", @@ -337,6 +349,7 @@ "copyManyCardsPopup-format": "[ {\"title\": \"Título de la primera tarjeta\", \"description\":\"Descripción de la primera tarjeta\"}, {\"title\":\"Título de la segunda tarjeta\",\"description\":\"Descripción de la segunda tarjeta\"},{\"title\":\"Título de la última tarjeta\",\"description\":\"Descripción de la última tarjeta\"} ]", "create": "Crear", "createBoardPopup-title": "Crear tablero", + "createTemplateContainerPopup-title": "Add Template Container", "chooseBoardSourcePopup-title": "Importar un tablero", "createLabelPopup-title": "Crear una etiqueta", "createCustomField": "Crear un campo", diff --git a/imports/i18n/data/es-LA.i18n.json b/imports/i18n/data/es-LA.i18n.json index 958d8313a..acf3c6934 100644 --- a/imports/i18n/data/es-LA.i18n.json +++ b/imports/i18n/data/es-LA.i18n.json @@ -78,6 +78,18 @@ "activity-deleteComment": "deleted comment %s", "activity-receivedDate": "edited received date to %s of %s", "activity-startDate": "edited start date to %s of %s", + "allboards.starred": "Starred", + "allboards.templates": "Templates", + "allboards.remaining": "Remaining", + "allboards.workspaces": "Workspaces", + "allboards.add-workspace": "Add Workspace", + "allboards.add-workspace-prompt": "Workspace name", + "allboards.add-subworkspace": "Add Subworkspace", + "allboards.add-subworkspace-prompt": "Subworkspace name", + "allboards.edit-workspace": "Edit workspace", + "allboards.edit-workspace-name": "Workspace name", + "allboards.edit-workspace-icon": "Workspace icon (markdown)", + "multi-selection-active": "Click checkboxes to select boards", "activity-dueDate": "edited due date to %s of %s", "activity-endDate": "edited end date to %s of %s", "add-attachment": "Add Attachment", @@ -337,6 +349,7 @@ "copyManyCardsPopup-format": "[ {\"title\": \"First card title\", \"description\":\"First card description\"}, {\"title\":\"Second card title\",\"description\":\"Second card description\"},{\"title\":\"Last card title\",\"description\":\"Last card description\"} ]", "create": "Create", "createBoardPopup-title": "Create Board", + "createTemplateContainerPopup-title": "Add Template Container", "chooseBoardSourcePopup-title": "Import board", "createLabelPopup-title": "Create Label", "createCustomField": "Create Field", diff --git a/imports/i18n/data/es-MX.i18n.json b/imports/i18n/data/es-MX.i18n.json index 97a451610..b2039e95e 100644 --- a/imports/i18n/data/es-MX.i18n.json +++ b/imports/i18n/data/es-MX.i18n.json @@ -78,6 +78,18 @@ "activity-deleteComment": "comentario eliminado %s", "activity-receivedDate": "fecha de recepción editada para %s de %s", "activity-startDate": "fecha de inicio editada a %s de %s", + "allboards.starred": "Starred", + "allboards.templates": "Templates", + "allboards.remaining": "Remaining", + "allboards.workspaces": "Workspaces", + "allboards.add-workspace": "Add Workspace", + "allboards.add-workspace-prompt": "Workspace name", + "allboards.add-subworkspace": "Add Subworkspace", + "allboards.add-subworkspace-prompt": "Subworkspace name", + "allboards.edit-workspace": "Edit workspace", + "allboards.edit-workspace-name": "Workspace name", + "allboards.edit-workspace-icon": "Workspace icon (markdown)", + "multi-selection-active": "Click checkboxes to select boards", "activity-dueDate": "fecha de vencimiento editada a %s de %s", "activity-endDate": "editada la fecha de finalización a %s de %s", "add-attachment": "Agregar adjunto", @@ -337,6 +349,7 @@ "copyManyCardsPopup-format": "[ {\"title\": \"First card title\", \"description\":\"First card description\"}, {\"title\":\"Second card title\",\"description\":\"Second card description\"},{\"title\":\"Last card title\",\"description\":\"Last card description\"} ]", "create": "Create", "createBoardPopup-title": "Create Board", + "createTemplateContainerPopup-title": "Add Template Container", "chooseBoardSourcePopup-title": "Import board", "createLabelPopup-title": "Create Label", "createCustomField": "Create Field", diff --git a/imports/i18n/data/es-PE.i18n.json b/imports/i18n/data/es-PE.i18n.json index 4deb3a96d..8d0e4c283 100644 --- a/imports/i18n/data/es-PE.i18n.json +++ b/imports/i18n/data/es-PE.i18n.json @@ -78,6 +78,18 @@ "activity-deleteComment": "comentario eliminado", "activity-receivedDate": "editada la fecha de recepción a %s de %s", "activity-startDate": "editada la fecha de inicio a %s de %s", + "allboards.starred": "Starred", + "allboards.templates": "Plantillas", + "allboards.remaining": "Remaining", + "allboards.workspaces": "Workspaces", + "allboards.add-workspace": "Add Workspace", + "allboards.add-workspace-prompt": "Workspace name", + "allboards.add-subworkspace": "Add Subworkspace", + "allboards.add-subworkspace-prompt": "Subworkspace name", + "allboards.edit-workspace": "Edit workspace", + "allboards.edit-workspace-name": "Workspace name", + "allboards.edit-workspace-icon": "Workspace icon (markdown)", + "multi-selection-active": "Click checkboxes to select boards", "activity-dueDate": "editada la fecha de vencimiento a %s de %s", "activity-endDate": "editada la fecha de finalización a %s de %s", "add-attachment": "Agregar adjunto", @@ -337,6 +349,7 @@ "copyManyCardsPopup-format": "[ {\"title\": \"Título de la primera tarjeta\", \"description\":\"Descripción de la primera tarjeta\"}, {\"title\":\"Título de la segunda tarjeta\",\"description\":\"Descripción de la segunda tarjeta\"},{\"title\":\"Título de la última tarjeta\",\"description\":\"Descripción de la última tarjeta\"} ]", "create": "Crear", "createBoardPopup-title": "Crear tablero", + "createTemplateContainerPopup-title": "Add Template Container", "chooseBoardSourcePopup-title": "Importar un tablero", "createLabelPopup-title": "Crear una etiqueta", "createCustomField": "Crear un campo", diff --git a/imports/i18n/data/es-PY.i18n.json b/imports/i18n/data/es-PY.i18n.json index 958d8313a..acf3c6934 100644 --- a/imports/i18n/data/es-PY.i18n.json +++ b/imports/i18n/data/es-PY.i18n.json @@ -78,6 +78,18 @@ "activity-deleteComment": "deleted comment %s", "activity-receivedDate": "edited received date to %s of %s", "activity-startDate": "edited start date to %s of %s", + "allboards.starred": "Starred", + "allboards.templates": "Templates", + "allboards.remaining": "Remaining", + "allboards.workspaces": "Workspaces", + "allboards.add-workspace": "Add Workspace", + "allboards.add-workspace-prompt": "Workspace name", + "allboards.add-subworkspace": "Add Subworkspace", + "allboards.add-subworkspace-prompt": "Subworkspace name", + "allboards.edit-workspace": "Edit workspace", + "allboards.edit-workspace-name": "Workspace name", + "allboards.edit-workspace-icon": "Workspace icon (markdown)", + "multi-selection-active": "Click checkboxes to select boards", "activity-dueDate": "edited due date to %s of %s", "activity-endDate": "edited end date to %s of %s", "add-attachment": "Add Attachment", @@ -337,6 +349,7 @@ "copyManyCardsPopup-format": "[ {\"title\": \"First card title\", \"description\":\"First card description\"}, {\"title\":\"Second card title\",\"description\":\"Second card description\"},{\"title\":\"Last card title\",\"description\":\"Last card description\"} ]", "create": "Create", "createBoardPopup-title": "Create Board", + "createTemplateContainerPopup-title": "Add Template Container", "chooseBoardSourcePopup-title": "Import board", "createLabelPopup-title": "Create Label", "createCustomField": "Create Field", diff --git a/imports/i18n/data/es.i18n.json b/imports/i18n/data/es.i18n.json index ad4d63edb..27be64c0c 100644 --- a/imports/i18n/data/es.i18n.json +++ b/imports/i18n/data/es.i18n.json @@ -78,6 +78,18 @@ "activity-deleteComment": "comentario eliminado", "activity-receivedDate": "editada la fecha de recepción a %s de %s", "activity-startDate": "editada la fecha de inicio a %s de %s", + "allboards.starred": "Starred", + "allboards.templates": "Plantillas", + "allboards.remaining": "Remaining", + "allboards.workspaces": "Workspaces", + "allboards.add-workspace": "Add Workspace", + "allboards.add-workspace-prompt": "Workspace name", + "allboards.add-subworkspace": "Add Subworkspace", + "allboards.add-subworkspace-prompt": "Subworkspace name", + "allboards.edit-workspace": "Edit workspace", + "allboards.edit-workspace-name": "Workspace name", + "allboards.edit-workspace-icon": "Workspace icon (markdown)", + "multi-selection-active": "Click checkboxes to select boards", "activity-dueDate": "editada la fecha de vencimiento a %s de %s", "activity-endDate": "editada la fecha de finalización a %s de %s", "add-attachment": "Añadir adjunto", @@ -337,6 +349,7 @@ "copyManyCardsPopup-format": "[ {\"title\": \"Título de la primera tarjeta\", \"description\":\"Descripción de la primera tarjeta\"}, {\"title\":\"Título de la segunda tarjeta\",\"description\":\"Descripción de la segunda tarjeta\"},{\"title\":\"Título de la última tarjeta\",\"description\":\"Descripción de la última tarjeta\"} ]", "create": "Crear", "createBoardPopup-title": "Crear tablero", + "createTemplateContainerPopup-title": "añadir plantilla de contenedor", "chooseBoardSourcePopup-title": "Importar un tablero", "createLabelPopup-title": "Crear una etiqueta", "createCustomField": "Crear un campo", diff --git a/imports/i18n/data/es_CO.i18n.json b/imports/i18n/data/es_CO.i18n.json index 958d8313a..acf3c6934 100644 --- a/imports/i18n/data/es_CO.i18n.json +++ b/imports/i18n/data/es_CO.i18n.json @@ -78,6 +78,18 @@ "activity-deleteComment": "deleted comment %s", "activity-receivedDate": "edited received date to %s of %s", "activity-startDate": "edited start date to %s of %s", + "allboards.starred": "Starred", + "allboards.templates": "Templates", + "allboards.remaining": "Remaining", + "allboards.workspaces": "Workspaces", + "allboards.add-workspace": "Add Workspace", + "allboards.add-workspace-prompt": "Workspace name", + "allboards.add-subworkspace": "Add Subworkspace", + "allboards.add-subworkspace-prompt": "Subworkspace name", + "allboards.edit-workspace": "Edit workspace", + "allboards.edit-workspace-name": "Workspace name", + "allboards.edit-workspace-icon": "Workspace icon (markdown)", + "multi-selection-active": "Click checkboxes to select boards", "activity-dueDate": "edited due date to %s of %s", "activity-endDate": "edited end date to %s of %s", "add-attachment": "Add Attachment", @@ -337,6 +349,7 @@ "copyManyCardsPopup-format": "[ {\"title\": \"First card title\", \"description\":\"First card description\"}, {\"title\":\"Second card title\",\"description\":\"Second card description\"},{\"title\":\"Last card title\",\"description\":\"Last card description\"} ]", "create": "Create", "createBoardPopup-title": "Create Board", + "createTemplateContainerPopup-title": "Add Template Container", "chooseBoardSourcePopup-title": "Import board", "createLabelPopup-title": "Create Label", "createCustomField": "Create Field", diff --git a/imports/i18n/data/et-EE.i18n.json b/imports/i18n/data/et-EE.i18n.json index 64bb83a99..d16e2a033 100644 --- a/imports/i18n/data/et-EE.i18n.json +++ b/imports/i18n/data/et-EE.i18n.json @@ -78,6 +78,18 @@ "activity-deleteComment": "kustutatud kommentaar %s", "activity-receivedDate": "redigeeritud saabunud kuupäev %s-i %s-i", "activity-startDate": "redigeeritud alguskuupäev %s-i %s-i alguskuupäevaks", + "allboards.starred": "Starred", + "allboards.templates": "Mallid", + "allboards.remaining": "Remaining", + "allboards.workspaces": "Workspaces", + "allboards.add-workspace": "Add Workspace", + "allboards.add-workspace-prompt": "Workspace name", + "allboards.add-subworkspace": "Add Subworkspace", + "allboards.add-subworkspace-prompt": "Subworkspace name", + "allboards.edit-workspace": "Edit workspace", + "allboards.edit-workspace-name": "Workspace name", + "allboards.edit-workspace-icon": "Workspace icon (markdown)", + "multi-selection-active": "Click checkboxes to select boards", "activity-dueDate": "redigeeritud tähtaeg on %s of %s", "activity-endDate": "redigeeritud lõpukuupäev %s-i %s-i lõpukuupäevaks", "add-attachment": "Lisa lisa", @@ -337,6 +349,7 @@ "copyManyCardsPopup-format": "[ {\"title\": {\"title\": \"First card title\", \"description\": \"First card description\"}, {\"title\": \"Second card title\", \"description\": \"Second card description\"},{\"title\": \"Last card title\", \"description\": \"Last card description\"}, {\"title\": \"Last card title\", \"description\": \"Last card description\"} ]", "create": "Loo", "createBoardPopup-title": "Loo juhatus", + "createTemplateContainerPopup-title": "Malli konteineri lisamine", "chooseBoardSourcePopup-title": "Impordilaua", "createLabelPopup-title": "Loo silt", "createCustomField": "Loo väli", diff --git a/imports/i18n/data/eu.i18n.json b/imports/i18n/data/eu.i18n.json index 91e716031..228365b30 100644 --- a/imports/i18n/data/eu.i18n.json +++ b/imports/i18n/data/eu.i18n.json @@ -78,6 +78,18 @@ "activity-deleteComment": "%s iruzkina ezabatu da", "activity-receivedDate": "edited received date to %s of %s", "activity-startDate": "edited start date to %s of %s", + "allboards.starred": "Starred", + "allboards.templates": "Txantiloiak", + "allboards.remaining": "Remaining", + "allboards.workspaces": "Workspaces", + "allboards.add-workspace": "Add Workspace", + "allboards.add-workspace-prompt": "Workspace name", + "allboards.add-subworkspace": "Add Subworkspace", + "allboards.add-subworkspace-prompt": "Subworkspace name", + "allboards.edit-workspace": "Edit workspace", + "allboards.edit-workspace-name": "Workspace name", + "allboards.edit-workspace-icon": "Workspace icon (markdown)", + "multi-selection-active": "Click checkboxes to select boards", "activity-dueDate": "edited due date to %s of %s", "activity-endDate": "edited end date to %s of %s", "add-attachment": "Gehitu eranskina", @@ -337,6 +349,7 @@ "copyManyCardsPopup-format": "[ {\"title\": \"First card title\", \"description\":\"First card description\"}, {\"title\":\"Second card title\",\"description\":\"Second card description\"},{\"title\":\"Last card title\",\"description\":\"Last card description\"} ]", "create": "Sortu", "createBoardPopup-title": "Sortu arbela", + "createTemplateContainerPopup-title": "Gehitu txantiloien edukiontzia", "chooseBoardSourcePopup-title": "Inportatu arbela", "createLabelPopup-title": "Sortu etiketa", "createCustomField": "Sortu eremua", diff --git a/imports/i18n/data/fa-IR.i18n.json b/imports/i18n/data/fa-IR.i18n.json index 8662becce..049e398af 100644 --- a/imports/i18n/data/fa-IR.i18n.json +++ b/imports/i18n/data/fa-IR.i18n.json @@ -78,6 +78,18 @@ "activity-deleteComment": "%s نظر حذف شد", "activity-receivedDate": "edited received date to %s of %s", "activity-startDate": "edited start date to %s of %s", + "allboards.starred": "Starred", + "allboards.templates": "قالب‌ها", + "allboards.remaining": "Remaining", + "allboards.workspaces": "Workspaces", + "allboards.add-workspace": "Add Workspace", + "allboards.add-workspace-prompt": "Workspace name", + "allboards.add-subworkspace": "Add Subworkspace", + "allboards.add-subworkspace-prompt": "Subworkspace name", + "allboards.edit-workspace": "Edit workspace", + "allboards.edit-workspace-name": "Workspace name", + "allboards.edit-workspace-icon": "Workspace icon (markdown)", + "multi-selection-active": "Click checkboxes to select boards", "activity-dueDate": "edited due date to %s of %s", "activity-endDate": "edited end date to %s of %s", "add-attachment": "افزودن ضمیمه", @@ -337,6 +349,7 @@ "copyManyCardsPopup-format": "[ {\"title\": \"First card title\", \"description\":\"First card description\"}, {\"title\":\"Second card title\",\"description\":\"Second card description\"},{\"title\":\"Last card title\",\"description\":\"Last card description\"} ]", "create": "ایجاد", "createBoardPopup-title": "ایجاد برد", + "createTemplateContainerPopup-title": "Add Template Container", "chooseBoardSourcePopup-title": "وارد کردن برد", "createLabelPopup-title": "ایجاد لیبل", "createCustomField": "ایجاد فیلد", diff --git a/imports/i18n/data/fa.i18n.json b/imports/i18n/data/fa.i18n.json index 6c44ea6f2..0d48e1299 100644 --- a/imports/i18n/data/fa.i18n.json +++ b/imports/i18n/data/fa.i18n.json @@ -78,6 +78,18 @@ "activity-deleteComment": "%s نظر حذف شد", "activity-receivedDate": "edited received date to %s of %s", "activity-startDate": "edited start date to %s of %s", + "allboards.starred": "Starred", + "allboards.templates": "قالب‌ها", + "allboards.remaining": "Remaining", + "allboards.workspaces": "Workspaces", + "allboards.add-workspace": "Add Workspace", + "allboards.add-workspace-prompt": "Workspace name", + "allboards.add-subworkspace": "Add Subworkspace", + "allboards.add-subworkspace-prompt": "Subworkspace name", + "allboards.edit-workspace": "Edit workspace", + "allboards.edit-workspace-name": "Workspace name", + "allboards.edit-workspace-icon": "Workspace icon (markdown)", + "multi-selection-active": "Click checkboxes to select boards", "activity-dueDate": "edited due date to %s of %s", "activity-endDate": "edited end date to %s of %s", "add-attachment": "افزودن ضمیمه", @@ -337,6 +349,7 @@ "copyManyCardsPopup-format": "[ {\"title\": \"First card title\", \"description\":\"First card description\"}, {\"title\":\"Second card title\",\"description\":\"Second card description\"},{\"title\":\"Last card title\",\"description\":\"Last card description\"} ]", "create": "ایجاد", "createBoardPopup-title": "ایجاد برد", + "createTemplateContainerPopup-title": "Add Template Container", "chooseBoardSourcePopup-title": "وارد کردن برد", "createLabelPopup-title": "ایجاد لیبل", "createCustomField": "ایجاد فیلد", diff --git a/imports/i18n/data/fi.i18n.json b/imports/i18n/data/fi.i18n.json index 54b041ab3..6a4c693e1 100644 --- a/imports/i18n/data/fi.i18n.json +++ b/imports/i18n/data/fi.i18n.json @@ -78,6 +78,18 @@ "activity-deleteComment": "poisti kommentin %s", "activity-receivedDate": "muokkasi vastaanotettu päiväksi %s / %s", "activity-startDate": "muokkasi aloituspäiväksi %s / %s", + "allboards.starred": "Suosikki", + "allboards.templates": "Mallit", + "allboards.remaining": "Jäljellä", + "allboards.workspaces": "Työtilat", + "allboards.add-workspace": "Lisää työtila", + "allboards.add-workspace-prompt": "Työtilan nimi", + "allboards.add-subworkspace": "Lisää alityötila", + "allboards.add-subworkspace-prompt": "Alityötilan nimi", + "allboards.edit-workspace": "Muokkaa työtilaa", + "allboards.edit-workspace-name": "Työtilan nimi", + "allboards.edit-workspace-icon": "Työtilan ikoni (markdown)", + "multi-selection-active": "Valitse taulut napsauttamalla valintaruutuja", "activity-dueDate": "muokkasi eräpäiväksi %s / %s", "activity-endDate": "muokkasi loppumispäiväksi %s / %s", "add-attachment": "Lisää liite", @@ -337,6 +349,7 @@ "copyManyCardsPopup-format": "[ {\"title\": \"Ensimmäisen kortin otsikko\", \"description\":\"Ensimmäisen kortin kuvaus\"}, {\"title\":\"Toisen kortin otsikko\",\"description\":\"Toisen kortin kuvaus\"},{\"title\":\"Viimeisen kortin otsikko\",\"description\":\"Viimeisen kortin kuvaus\"} ]", "create": "Luo", "createBoardPopup-title": "Luo taulu", + "createTemplateContainerPopup-title": "Lisää mallikontti", "chooseBoardSourcePopup-title": "Tuo taulu", "createLabelPopup-title": "Luo nimilappu", "createCustomField": "Luo kenttä", diff --git a/imports/i18n/data/fr-CH.i18n.json b/imports/i18n/data/fr-CH.i18n.json index ad02429ed..5fa3fa719 100644 --- a/imports/i18n/data/fr-CH.i18n.json +++ b/imports/i18n/data/fr-CH.i18n.json @@ -78,6 +78,18 @@ "activity-deleteComment": "deleted comment %s", "activity-receivedDate": "edited received date to %s of %s", "activity-startDate": "edited start date to %s of %s", + "allboards.starred": "Starred", + "allboards.templates": "Templates", + "allboards.remaining": "Remaining", + "allboards.workspaces": "Workspaces", + "allboards.add-workspace": "Add Workspace", + "allboards.add-workspace-prompt": "Workspace name", + "allboards.add-subworkspace": "Add Subworkspace", + "allboards.add-subworkspace-prompt": "Subworkspace name", + "allboards.edit-workspace": "Edit workspace", + "allboards.edit-workspace-name": "Workspace name", + "allboards.edit-workspace-icon": "Workspace icon (markdown)", + "multi-selection-active": "Click checkboxes to select boards", "activity-dueDate": "edited due date to %s of %s", "activity-endDate": "edited end date to %s of %s", "add-attachment": "Add Attachment", @@ -337,6 +349,7 @@ "copyManyCardsPopup-format": "[ {\"title\": \"First card title\", \"description\":\"First card description\"}, {\"title\":\"Second card title\",\"description\":\"Second card description\"},{\"title\":\"Last card title\",\"description\":\"Last card description\"} ]", "create": "Create", "createBoardPopup-title": "Create Board", + "createTemplateContainerPopup-title": "Add Template Container", "chooseBoardSourcePopup-title": "Import board", "createLabelPopup-title": "Create Label", "createCustomField": "Create Field", diff --git a/imports/i18n/data/fr-FR.i18n.json b/imports/i18n/data/fr-FR.i18n.json index 721c8be40..54b4536c1 100644 --- a/imports/i18n/data/fr-FR.i18n.json +++ b/imports/i18n/data/fr-FR.i18n.json @@ -78,6 +78,18 @@ "activity-deleteComment": "commentaire supprimé %s", "activity-receivedDate": "date de réception éditée de %s à %s", "activity-startDate": "date de début éditée de %s à %s", + "allboards.starred": "Starred", + "allboards.templates": "Modèles", + "allboards.remaining": "Remaining", + "allboards.workspaces": "Workspaces", + "allboards.add-workspace": "Add Workspace", + "allboards.add-workspace-prompt": "Workspace name", + "allboards.add-subworkspace": "Add Subworkspace", + "allboards.add-subworkspace-prompt": "Subworkspace name", + "allboards.edit-workspace": "Edit workspace", + "allboards.edit-workspace-name": "Workspace name", + "allboards.edit-workspace-icon": "Workspace icon (markdown)", + "multi-selection-active": "Click checkboxes to select boards", "activity-dueDate": "date d'échéance éditée de %s à %s", "activity-endDate": "date de fin éditée de %s à %s", "add-attachment": "Ajouter une pièce jointe", @@ -337,6 +349,7 @@ "copyManyCardsPopup-format": "[ {\"title\": \"Titre de la première carte\", \"description\":\"Description de la première carte\"}, {\"title\":\"Titre de la seconde carte\",\"description\":\"Description de la seconde carte\"},{\"title\":\"Titre de la dernière carte\",\"description\":\"Description de la dernière carte\"} ]", "create": "Créer", "createBoardPopup-title": "Créer un tableau", + "createTemplateContainerPopup-title": "Ajouter un conteneur de modèles", "chooseBoardSourcePopup-title": "Importer un tableau", "createLabelPopup-title": "Créer une étiquette", "createCustomField": "Créer un champ personnalisé", diff --git a/imports/i18n/data/fr.i18n.json b/imports/i18n/data/fr.i18n.json index cd3fdecf6..9944c844d 100644 --- a/imports/i18n/data/fr.i18n.json +++ b/imports/i18n/data/fr.i18n.json @@ -78,6 +78,18 @@ "activity-deleteComment": "commentaire supprimé %s", "activity-receivedDate": "date de réception éditée de %s à %s", "activity-startDate": "date de début éditée de %s à %s", + "allboards.starred": "Starred", + "allboards.templates": "Modèles", + "allboards.remaining": "Remaining", + "allboards.workspaces": "Workspaces", + "allboards.add-workspace": "Add Workspace", + "allboards.add-workspace-prompt": "Workspace name", + "allboards.add-subworkspace": "Add Subworkspace", + "allboards.add-subworkspace-prompt": "Subworkspace name", + "allboards.edit-workspace": "Edit workspace", + "allboards.edit-workspace-name": "Workspace name", + "allboards.edit-workspace-icon": "Workspace icon (markdown)", + "multi-selection-active": "Click checkboxes to select boards", "activity-dueDate": "date d'échéance éditée de %s à %s", "activity-endDate": "date de fin éditée de %s à %s", "add-attachment": "Ajouter une pièce jointe", @@ -337,6 +349,7 @@ "copyManyCardsPopup-format": "[ {\"title\": \"Titre de la première carte\", \"description\":\"Description de la première carte\"}, {\"title\":\"Titre de la seconde carte\",\"description\":\"Description de la seconde carte\"},{\"title\":\"Titre de la dernière carte\",\"description\":\"Description de la dernière carte\"} ]", "create": "Créer", "createBoardPopup-title": "Créer un tableau", + "createTemplateContainerPopup-title": "Ajouter un conteneur de modèles", "chooseBoardSourcePopup-title": "Importer un tableau", "createLabelPopup-title": "Créer une étiquette", "createCustomField": "Créer un champ personnalisé", diff --git a/imports/i18n/data/fy-NL.i18n.json b/imports/i18n/data/fy-NL.i18n.json index 958d8313a..acf3c6934 100644 --- a/imports/i18n/data/fy-NL.i18n.json +++ b/imports/i18n/data/fy-NL.i18n.json @@ -78,6 +78,18 @@ "activity-deleteComment": "deleted comment %s", "activity-receivedDate": "edited received date to %s of %s", "activity-startDate": "edited start date to %s of %s", + "allboards.starred": "Starred", + "allboards.templates": "Templates", + "allboards.remaining": "Remaining", + "allboards.workspaces": "Workspaces", + "allboards.add-workspace": "Add Workspace", + "allboards.add-workspace-prompt": "Workspace name", + "allboards.add-subworkspace": "Add Subworkspace", + "allboards.add-subworkspace-prompt": "Subworkspace name", + "allboards.edit-workspace": "Edit workspace", + "allboards.edit-workspace-name": "Workspace name", + "allboards.edit-workspace-icon": "Workspace icon (markdown)", + "multi-selection-active": "Click checkboxes to select boards", "activity-dueDate": "edited due date to %s of %s", "activity-endDate": "edited end date to %s of %s", "add-attachment": "Add Attachment", @@ -337,6 +349,7 @@ "copyManyCardsPopup-format": "[ {\"title\": \"First card title\", \"description\":\"First card description\"}, {\"title\":\"Second card title\",\"description\":\"Second card description\"},{\"title\":\"Last card title\",\"description\":\"Last card description\"} ]", "create": "Create", "createBoardPopup-title": "Create Board", + "createTemplateContainerPopup-title": "Add Template Container", "chooseBoardSourcePopup-title": "Import board", "createLabelPopup-title": "Create Label", "createCustomField": "Create Field", diff --git a/imports/i18n/data/fy.i18n.json b/imports/i18n/data/fy.i18n.json index 958d8313a..acf3c6934 100644 --- a/imports/i18n/data/fy.i18n.json +++ b/imports/i18n/data/fy.i18n.json @@ -78,6 +78,18 @@ "activity-deleteComment": "deleted comment %s", "activity-receivedDate": "edited received date to %s of %s", "activity-startDate": "edited start date to %s of %s", + "allboards.starred": "Starred", + "allboards.templates": "Templates", + "allboards.remaining": "Remaining", + "allboards.workspaces": "Workspaces", + "allboards.add-workspace": "Add Workspace", + "allboards.add-workspace-prompt": "Workspace name", + "allboards.add-subworkspace": "Add Subworkspace", + "allboards.add-subworkspace-prompt": "Subworkspace name", + "allboards.edit-workspace": "Edit workspace", + "allboards.edit-workspace-name": "Workspace name", + "allboards.edit-workspace-icon": "Workspace icon (markdown)", + "multi-selection-active": "Click checkboxes to select boards", "activity-dueDate": "edited due date to %s of %s", "activity-endDate": "edited end date to %s of %s", "add-attachment": "Add Attachment", @@ -337,6 +349,7 @@ "copyManyCardsPopup-format": "[ {\"title\": \"First card title\", \"description\":\"First card description\"}, {\"title\":\"Second card title\",\"description\":\"Second card description\"},{\"title\":\"Last card title\",\"description\":\"Last card description\"} ]", "create": "Create", "createBoardPopup-title": "Create Board", + "createTemplateContainerPopup-title": "Add Template Container", "chooseBoardSourcePopup-title": "Import board", "createLabelPopup-title": "Create Label", "createCustomField": "Create Field", diff --git a/imports/i18n/data/gl-ES.i18n.json b/imports/i18n/data/gl-ES.i18n.json index 37d1e5efd..ab6c97129 100644 --- a/imports/i18n/data/gl-ES.i18n.json +++ b/imports/i18n/data/gl-ES.i18n.json @@ -78,6 +78,18 @@ "activity-deleteComment": "deleted comment %s", "activity-receivedDate": "edited received date to %s of %s", "activity-startDate": "edited start date to %s of %s", + "allboards.starred": "Starred", + "allboards.templates": "Templates", + "allboards.remaining": "Remaining", + "allboards.workspaces": "Workspaces", + "allboards.add-workspace": "Add Workspace", + "allboards.add-workspace-prompt": "Workspace name", + "allboards.add-subworkspace": "Add Subworkspace", + "allboards.add-subworkspace-prompt": "Subworkspace name", + "allboards.edit-workspace": "Edit workspace", + "allboards.edit-workspace-name": "Workspace name", + "allboards.edit-workspace-icon": "Workspace icon (markdown)", + "multi-selection-active": "Click checkboxes to select boards", "activity-dueDate": "edited due date to %s of %s", "activity-endDate": "edited end date to %s of %s", "add-attachment": "Engadir anexo", @@ -337,6 +349,7 @@ "copyManyCardsPopup-format": "[ {\"title\": \"First card title\", \"description\":\"First card description\"}, {\"title\":\"Second card title\",\"description\":\"Second card description\"},{\"title\":\"Last card title\",\"description\":\"Last card description\"} ]", "create": "Crear", "createBoardPopup-title": "Crear taboleiro", + "createTemplateContainerPopup-title": "Add Template Container", "chooseBoardSourcePopup-title": "Importar taboleiro", "createLabelPopup-title": "Crear etiqueta", "createCustomField": "Create Field", diff --git a/imports/i18n/data/gl.i18n.json b/imports/i18n/data/gl.i18n.json index 08f3aa450..088d60d69 100644 --- a/imports/i18n/data/gl.i18n.json +++ b/imports/i18n/data/gl.i18n.json @@ -78,6 +78,18 @@ "activity-deleteComment": "deleted comment %s", "activity-receivedDate": "edited received date to %s of %s", "activity-startDate": "edited start date to %s of %s", + "allboards.starred": "Starred", + "allboards.templates": "Templates", + "allboards.remaining": "Remaining", + "allboards.workspaces": "Workspaces", + "allboards.add-workspace": "Add Workspace", + "allboards.add-workspace-prompt": "Workspace name", + "allboards.add-subworkspace": "Add Subworkspace", + "allboards.add-subworkspace-prompt": "Subworkspace name", + "allboards.edit-workspace": "Edit workspace", + "allboards.edit-workspace-name": "Workspace name", + "allboards.edit-workspace-icon": "Workspace icon (markdown)", + "multi-selection-active": "Click checkboxes to select boards", "activity-dueDate": "edited due date to %s of %s", "activity-endDate": "edited end date to %s of %s", "add-attachment": "Engadir anexo", @@ -337,6 +349,7 @@ "copyManyCardsPopup-format": "[ {\"title\": \"First card title\", \"description\":\"First card description\"}, {\"title\":\"Second card title\",\"description\":\"Second card description\"},{\"title\":\"Last card title\",\"description\":\"Last card description\"} ]", "create": "Crear", "createBoardPopup-title": "Crear taboleiro", + "createTemplateContainerPopup-title": "Add Template Container", "chooseBoardSourcePopup-title": "Importar taboleiro", "createLabelPopup-title": "Crear etiqueta", "createCustomField": "Create Field", diff --git a/imports/i18n/data/gu-IN.i18n.json b/imports/i18n/data/gu-IN.i18n.json index 4804e6dff..b6aac02e5 100644 --- a/imports/i18n/data/gu-IN.i18n.json +++ b/imports/i18n/data/gu-IN.i18n.json @@ -78,6 +78,18 @@ "activity-deleteComment": "deleted comment %s", "activity-receivedDate": "edited received date to %s of %s", "activity-startDate": "edited start date to %s of %s", + "allboards.starred": "Starred", + "allboards.templates": "Templates", + "allboards.remaining": "Remaining", + "allboards.workspaces": "Workspaces", + "allboards.add-workspace": "Add Workspace", + "allboards.add-workspace-prompt": "Workspace name", + "allboards.add-subworkspace": "Add Subworkspace", + "allboards.add-subworkspace-prompt": "Subworkspace name", + "allboards.edit-workspace": "Edit workspace", + "allboards.edit-workspace-name": "Workspace name", + "allboards.edit-workspace-icon": "Workspace icon (markdown)", + "multi-selection-active": "Click checkboxes to select boards", "activity-dueDate": "edited due date to %s of %s", "activity-endDate": "edited end date to %s of %s", "add-attachment": "Add Attachment", @@ -337,6 +349,7 @@ "copyManyCardsPopup-format": "[ {\"title\": \"First card title\", \"description\":\"First card description\"}, {\"title\":\"Second card title\",\"description\":\"Second card description\"},{\"title\":\"Last card title\",\"description\":\"Last card description\"} ]", "create": "Create", "createBoardPopup-title": "Create Board", + "createTemplateContainerPopup-title": "Add Template Container", "chooseBoardSourcePopup-title": "Import board", "createLabelPopup-title": "Create Label", "createCustomField": "Create Field", diff --git a/imports/i18n/data/he-IL.i18n.json b/imports/i18n/data/he-IL.i18n.json index 2b1393267..0484f7b10 100644 --- a/imports/i18n/data/he-IL.i18n.json +++ b/imports/i18n/data/he-IL.i18n.json @@ -78,6 +78,18 @@ "activity-deleteComment": "deleted comment %s", "activity-receivedDate": "edited received date to %s of %s", "activity-startDate": "edited start date to %s of %s", + "allboards.starred": "Starred", + "allboards.templates": "Templates", + "allboards.remaining": "Remaining", + "allboards.workspaces": "Workspaces", + "allboards.add-workspace": "Add Workspace", + "allboards.add-workspace-prompt": "Workspace name", + "allboards.add-subworkspace": "Add Subworkspace", + "allboards.add-subworkspace-prompt": "Subworkspace name", + "allboards.edit-workspace": "Edit workspace", + "allboards.edit-workspace-name": "Workspace name", + "allboards.edit-workspace-icon": "Workspace icon (markdown)", + "multi-selection-active": "Click checkboxes to select boards", "activity-dueDate": "edited due date to %s of %s", "activity-endDate": "edited end date to %s of %s", "add-attachment": "Add Attachment", @@ -337,6 +349,7 @@ "copyManyCardsPopup-format": "[ {\"title\": \"First card title\", \"description\":\"First card description\"}, {\"title\":\"Second card title\",\"description\":\"Second card description\"},{\"title\":\"Last card title\",\"description\":\"Last card description\"} ]", "create": "Create", "createBoardPopup-title": "Create Board", + "createTemplateContainerPopup-title": "Add Template Container", "chooseBoardSourcePopup-title": "Import board", "createLabelPopup-title": "Create Label", "createCustomField": "Create Field", diff --git a/imports/i18n/data/he.i18n.json b/imports/i18n/data/he.i18n.json index 831768065..0592d208b 100644 --- a/imports/i18n/data/he.i18n.json +++ b/imports/i18n/data/he.i18n.json @@ -78,6 +78,18 @@ "activity-deleteComment": "התגובה %s נמחקה", "activity-receivedDate": "תאריך הקבלה השתנה מ־%s ל־%s", "activity-startDate": "תאריך ההתחלה השתנה מ־%s ל־%s", + "allboards.starred": "Starred", + "allboards.templates": "תבניות", + "allboards.remaining": "Remaining", + "allboards.workspaces": "Workspaces", + "allboards.add-workspace": "Add Workspace", + "allboards.add-workspace-prompt": "Workspace name", + "allboards.add-subworkspace": "Add Subworkspace", + "allboards.add-subworkspace-prompt": "Subworkspace name", + "allboards.edit-workspace": "Edit workspace", + "allboards.edit-workspace-name": "Workspace name", + "allboards.edit-workspace-icon": "Workspace icon (markdown)", + "multi-selection-active": "Click checkboxes to select boards", "activity-dueDate": "תאריך היעד השתנה מ־%s ל־%s", "activity-endDate": "תאריך הסיום השתנה מ־%s ל־%s", "add-attachment": "הוספת קובץ מצורף", @@ -337,6 +349,7 @@ "copyManyCardsPopup-format": "[ {\"title\": \"כותרת כרטיס ראשון\", \"description\":\"תיאור כרטיס ראשון\"}, {\"title\":\"כותרת כרטיס שני\",\"description\":\"תיאור כרטיס שני\"},{\"title\":\"כותרת כרטיס אחרון\",\"description\":\"תיאור כרטיס אחרון\"} ]", "create": "יצירה", "createBoardPopup-title": "יצירת לוח", + "createTemplateContainerPopup-title": "הוספת מכולה לתבנית", "chooseBoardSourcePopup-title": "ייבוא לוח", "createLabelPopup-title": "יצירת תווית", "createCustomField": "יצירת שדה", diff --git a/imports/i18n/data/hi-IN.i18n.json b/imports/i18n/data/hi-IN.i18n.json index aa6f7f298..727186475 100644 --- a/imports/i18n/data/hi-IN.i18n.json +++ b/imports/i18n/data/hi-IN.i18n.json @@ -78,6 +78,18 @@ "activity-deleteComment": "टिप्पणी हटा दी गई", "activity-receivedDate": "edited received date to %s of %s", "activity-startDate": "edited start date to %s of %s", + "allboards.starred": "Starred", + "allboards.templates": "खाका", + "allboards.remaining": "Remaining", + "allboards.workspaces": "Workspaces", + "allboards.add-workspace": "Add Workspace", + "allboards.add-workspace-prompt": "Workspace name", + "allboards.add-subworkspace": "Add Subworkspace", + "allboards.add-subworkspace-prompt": "Subworkspace name", + "allboards.edit-workspace": "Edit workspace", + "allboards.edit-workspace-name": "Workspace name", + "allboards.edit-workspace-icon": "Workspace icon (markdown)", + "multi-selection-active": "Click checkboxes to select boards", "activity-dueDate": "edited due date to %s of %s", "activity-endDate": "edited end date to %s of %s", "add-attachment": "संलग्न करें", @@ -337,6 +349,7 @@ "copyManyCardsPopup-format": "[{\"title\":\"पहला कार्ड शीर्षक\",\"description\":\"पहला कार्ड विवरण\"},{\"title\":\"दूसरा कार्ड शीर्षक\",\"description\":\"दूसरा कार्ड विवरण\"},{\"title\":\"अंतिम कार्ड शीर्षक\",\"description\":\"अंतिम कार्ड विवरण\" }]", "create": "निर्माण करना", "createBoardPopup-title": "बोर्ड निर्माण करना", + "createTemplateContainerPopup-title": "Add Template Container", "chooseBoardSourcePopup-title": "बोर्ड आयात", "createLabelPopup-title": "नामपत्र निर्माण", "createCustomField": "क्षेत्र निर्माण", diff --git a/imports/i18n/data/hi.i18n.json b/imports/i18n/data/hi.i18n.json index 4f7be7672..854bfc00d 100644 --- a/imports/i18n/data/hi.i18n.json +++ b/imports/i18n/data/hi.i18n.json @@ -78,6 +78,18 @@ "activity-deleteComment": "टिप्पणी हटा दी गई", "activity-receivedDate": "edited received date to %s of %s", "activity-startDate": "edited start date to %s of %s", + "allboards.starred": "Starred", + "allboards.templates": "खाका", + "allboards.remaining": "Remaining", + "allboards.workspaces": "Workspaces", + "allboards.add-workspace": "Add Workspace", + "allboards.add-workspace-prompt": "Workspace name", + "allboards.add-subworkspace": "Add Subworkspace", + "allboards.add-subworkspace-prompt": "Subworkspace name", + "allboards.edit-workspace": "Edit workspace", + "allboards.edit-workspace-name": "Workspace name", + "allboards.edit-workspace-icon": "Workspace icon (markdown)", + "multi-selection-active": "Click checkboxes to select boards", "activity-dueDate": "edited due date to %s of %s", "activity-endDate": "edited end date to %s of %s", "add-attachment": "संलग्न करें", @@ -337,6 +349,7 @@ "copyManyCardsPopup-format": "[{\"title\":\"पहला कार्ड शीर्षक\",\"description\":\"पहला कार्ड विवरण\"},{\"title\":\"दूसरा कार्ड शीर्षक\",\"description\":\"दूसरा कार्ड विवरण\"},{\"title\":\"अंतिम कार्ड शीर्षक\",\"description\":\"अंतिम कार्ड विवरण\" }]", "create": "निर्माण करना", "createBoardPopup-title": "बोर्ड निर्माण करना", + "createTemplateContainerPopup-title": "Add Template Container", "chooseBoardSourcePopup-title": "बोर्ड आयात", "createLabelPopup-title": "नामपत्र निर्माण", "createCustomField": "क्षेत्र निर्माण", diff --git a/imports/i18n/data/hr.i18n.json b/imports/i18n/data/hr.i18n.json index 8c1125a31..475e99049 100644 --- a/imports/i18n/data/hr.i18n.json +++ b/imports/i18n/data/hr.i18n.json @@ -78,6 +78,18 @@ "activity-deleteComment": "obrisan komentar %s", "activity-receivedDate": "edited received date to %s of %s", "activity-startDate": "edited start date to %s of %s", + "allboards.starred": "Starred", + "allboards.templates": "Predlošci", + "allboards.remaining": "Remaining", + "allboards.workspaces": "Workspaces", + "allboards.add-workspace": "Add Workspace", + "allboards.add-workspace-prompt": "Workspace name", + "allboards.add-subworkspace": "Add Subworkspace", + "allboards.add-subworkspace-prompt": "Subworkspace name", + "allboards.edit-workspace": "Edit workspace", + "allboards.edit-workspace-name": "Workspace name", + "allboards.edit-workspace-icon": "Workspace icon (markdown)", + "multi-selection-active": "Click checkboxes to select boards", "activity-dueDate": "edited due date to %s of %s", "activity-endDate": "edited end date to %s of %s", "add-attachment": "Dodaj privitak", @@ -337,6 +349,7 @@ "copyManyCardsPopup-format": "[ {\"title\": \"First card title\", \"description\":\"First card description\"}, {\"title\":\"Second card title\",\"description\":\"Second card description\"},{\"title\":\"Last card title\",\"description\":\"Last card description\"} ]", "create": "Dodaj", "createBoardPopup-title": "Dodaj ploču", + "createTemplateContainerPopup-title": "Add Template Container", "chooseBoardSourcePopup-title": "Uvezi ploču", "createLabelPopup-title": "Dodaj oznaku", "createCustomField": "Dodaj polje", diff --git a/imports/i18n/data/hu.i18n.json b/imports/i18n/data/hu.i18n.json index cc54d40f4..1f2e257be 100644 --- a/imports/i18n/data/hu.i18n.json +++ b/imports/i18n/data/hu.i18n.json @@ -78,6 +78,18 @@ "activity-deleteComment": "törölte ezt a megjegyzést: %s", "activity-receivedDate": "átírta az \"érkezett\" dátumot erről: %s erre: %s", "activity-startDate": "átírta az \"elkezdve\" dátumot erről: %s erre: %s", + "allboards.starred": "Starred", + "allboards.templates": "Sablonok", + "allboards.remaining": "Remaining", + "allboards.workspaces": "Workspaces", + "allboards.add-workspace": "Add Workspace", + "allboards.add-workspace-prompt": "Workspace name", + "allboards.add-subworkspace": "Add Subworkspace", + "allboards.add-subworkspace-prompt": "Subworkspace name", + "allboards.edit-workspace": "Edit workspace", + "allboards.edit-workspace-name": "Workspace name", + "allboards.edit-workspace-icon": "Workspace icon (markdown)", + "multi-selection-active": "Click checkboxes to select boards", "activity-dueDate": "átírta a *határidő* dátumát erről: %s erre: %s", "activity-endDate": "átírta a \"befejezés\" dátumát erről: %s erre: %s", "add-attachment": "Melléklet hozzáadása", @@ -337,6 +349,7 @@ "copyManyCardsPopup-format": "[ {\"title\": \"Első kártya címe\", \"description\":\"Első kártya leírása\"}, {\"title\":\"Második kártya címe\",\"description\":\"Második kártya leírása\"},{\"title\":\"Utolsó kártya címe\",\"description\":\"Utolsó kártya leírása\"} ]", "create": "Létrehozás", "createBoardPopup-title": "Tábla létrehozása", + "createTemplateContainerPopup-title": "Sablon Tároló hozzáadása", "chooseBoardSourcePopup-title": "Tábla importálása", "createLabelPopup-title": "Címke létrehozása", "createCustomField": "Mező létrehozása", diff --git a/imports/i18n/data/hy.i18n.json b/imports/i18n/data/hy.i18n.json index 3ac06cc65..c16f6a92f 100644 --- a/imports/i18n/data/hy.i18n.json +++ b/imports/i18n/data/hy.i18n.json @@ -78,6 +78,18 @@ "activity-deleteComment": "deleted comment %s", "activity-receivedDate": "edited received date to %s of %s", "activity-startDate": "edited start date to %s of %s", + "allboards.starred": "Starred", + "allboards.templates": "Templates", + "allboards.remaining": "Remaining", + "allboards.workspaces": "Workspaces", + "allboards.add-workspace": "Add Workspace", + "allboards.add-workspace-prompt": "Workspace name", + "allboards.add-subworkspace": "Add Subworkspace", + "allboards.add-subworkspace-prompt": "Subworkspace name", + "allboards.edit-workspace": "Edit workspace", + "allboards.edit-workspace-name": "Workspace name", + "allboards.edit-workspace-icon": "Workspace icon (markdown)", + "multi-selection-active": "Click checkboxes to select boards", "activity-dueDate": "edited due date to %s of %s", "activity-endDate": "edited end date to %s of %s", "add-attachment": "Add Attachment", @@ -337,6 +349,7 @@ "copyManyCardsPopup-format": "[ {\"title\": \"First card title\", \"description\":\"First card description\"}, {\"title\":\"Second card title\",\"description\":\"Second card description\"},{\"title\":\"Last card title\",\"description\":\"Last card description\"} ]", "create": "Create", "createBoardPopup-title": "Create Board", + "createTemplateContainerPopup-title": "Add Template Container", "chooseBoardSourcePopup-title": "Import board", "createLabelPopup-title": "Create Label", "createCustomField": "Create Field", diff --git a/imports/i18n/data/id.i18n.json b/imports/i18n/data/id.i18n.json index d49641ae7..ce83f1f26 100644 --- a/imports/i18n/data/id.i18n.json +++ b/imports/i18n/data/id.i18n.json @@ -78,6 +78,18 @@ "activity-deleteComment": "komentar dihapus %s", "activity-receivedDate": "edited received date to %s of %s", "activity-startDate": "edited start date to %s of %s", + "allboards.starred": "Starred", + "allboards.templates": "Klise", + "allboards.remaining": "Remaining", + "allboards.workspaces": "Workspaces", + "allboards.add-workspace": "Add Workspace", + "allboards.add-workspace-prompt": "Workspace name", + "allboards.add-subworkspace": "Add Subworkspace", + "allboards.add-subworkspace-prompt": "Subworkspace name", + "allboards.edit-workspace": "Edit workspace", + "allboards.edit-workspace-name": "Workspace name", + "allboards.edit-workspace-icon": "Workspace icon (markdown)", + "multi-selection-active": "Click checkboxes to select boards", "activity-dueDate": "edited due date to %s of %s", "activity-endDate": "edited end date to %s of %s", "add-attachment": "Tambah Lampiran", @@ -337,6 +349,7 @@ "copyManyCardsPopup-format": "[ {\"title\": \"First card title\", \"description\":\"First card description\"}, {\"title\":\"Second card title\",\"description\":\"Second card description\"},{\"title\":\"Last card title\",\"description\":\"Last card description\"} ]", "create": "Buat", "createBoardPopup-title": "Buat Panel", + "createTemplateContainerPopup-title": "Add Template Container", "chooseBoardSourcePopup-title": "Import board", "createLabelPopup-title": "Buat Label", "createCustomField": "Create Field", diff --git a/imports/i18n/data/ig.i18n.json b/imports/i18n/data/ig.i18n.json index acce99747..580fbb197 100644 --- a/imports/i18n/data/ig.i18n.json +++ b/imports/i18n/data/ig.i18n.json @@ -78,6 +78,18 @@ "activity-deleteComment": "deleted comment %s", "activity-receivedDate": "edited received date to %s of %s", "activity-startDate": "edited start date to %s of %s", + "allboards.starred": "Starred", + "allboards.templates": "Templates", + "allboards.remaining": "Remaining", + "allboards.workspaces": "Workspaces", + "allboards.add-workspace": "Add Workspace", + "allboards.add-workspace-prompt": "Workspace name", + "allboards.add-subworkspace": "Add Subworkspace", + "allboards.add-subworkspace-prompt": "Subworkspace name", + "allboards.edit-workspace": "Edit workspace", + "allboards.edit-workspace-name": "Workspace name", + "allboards.edit-workspace-icon": "Workspace icon (markdown)", + "multi-selection-active": "Click checkboxes to select boards", "activity-dueDate": "edited due date to %s of %s", "activity-endDate": "edited end date to %s of %s", "add-attachment": "Add Attachment", @@ -337,6 +349,7 @@ "copyManyCardsPopup-format": "[ {\"title\": \"First card title\", \"description\":\"First card description\"}, {\"title\":\"Second card title\",\"description\":\"Second card description\"},{\"title\":\"Last card title\",\"description\":\"Last card description\"} ]", "create": "Create", "createBoardPopup-title": "Create Board", + "createTemplateContainerPopup-title": "Add Template Container", "chooseBoardSourcePopup-title": "Import board", "createLabelPopup-title": "Create Label", "createCustomField": "Create Field", diff --git a/imports/i18n/data/it.i18n.json b/imports/i18n/data/it.i18n.json index e6f9f1f62..e82709ca0 100644 --- a/imports/i18n/data/it.i18n.json +++ b/imports/i18n/data/it.i18n.json @@ -78,6 +78,18 @@ "activity-deleteComment": "commento eliminato %s", "activity-receivedDate": "ha modificato la data di ricevuta a %s di %s", "activity-startDate": "ha modificato la data di inizio a %s di %s", + "allboards.starred": "Starred", + "allboards.templates": "Template", + "allboards.remaining": "Remaining", + "allboards.workspaces": "Workspaces", + "allboards.add-workspace": "Add Workspace", + "allboards.add-workspace-prompt": "Workspace name", + "allboards.add-subworkspace": "Add Subworkspace", + "allboards.add-subworkspace-prompt": "Subworkspace name", + "allboards.edit-workspace": "Edit workspace", + "allboards.edit-workspace-name": "Workspace name", + "allboards.edit-workspace-icon": "Workspace icon (markdown)", + "multi-selection-active": "Click checkboxes to select boards", "activity-dueDate": "ha modificato la data di scadenza a %s di %s", "activity-endDate": "ha modificato la data di fine a %s di %s", "add-attachment": "Aggiungi allegato", @@ -337,6 +349,7 @@ "copyManyCardsPopup-format": "[ {\"title\": \"Titolo prima scheda\", \"description\":\"Descrizione prima scheda\"}, {\"title\":\"Titolo seconda scheda\",\"description\":\"Descrizione seconda scheda\"},{\"title\":\"Titolo ultima scheda\",\"description\":\"Descrizione ultima scheda\"} ]", "create": "Crea", "createBoardPopup-title": "Crea bacheca", + "createTemplateContainerPopup-title": "Aggiungi contenitore di Template", "chooseBoardSourcePopup-title": "Importa bacheca", "createLabelPopup-title": "Crea etichetta", "createCustomField": "Crea campo", diff --git a/imports/i18n/data/ja-HI.i18n.json b/imports/i18n/data/ja-HI.i18n.json index 4ca8bbb16..b97ac7887 100644 --- a/imports/i18n/data/ja-HI.i18n.json +++ b/imports/i18n/data/ja-HI.i18n.json @@ -78,6 +78,18 @@ "activity-deleteComment": "deleted comment %s", "activity-receivedDate": "edited received date to %s of %s", "activity-startDate": "edited start date to %s of %s", + "allboards.starred": "Starred", + "allboards.templates": "Templates", + "allboards.remaining": "Remaining", + "allboards.workspaces": "Workspaces", + "allboards.add-workspace": "Add Workspace", + "allboards.add-workspace-prompt": "Workspace name", + "allboards.add-subworkspace": "Add Subworkspace", + "allboards.add-subworkspace-prompt": "Subworkspace name", + "allboards.edit-workspace": "Edit workspace", + "allboards.edit-workspace-name": "Workspace name", + "allboards.edit-workspace-icon": "Workspace icon (markdown)", + "multi-selection-active": "Click checkboxes to select boards", "activity-dueDate": "edited due date to %s of %s", "activity-endDate": "edited end date to %s of %s", "add-attachment": "Add Attachment", @@ -337,6 +349,7 @@ "copyManyCardsPopup-format": "[ {\"title\": \"First card title\", \"description\":\"First card description\"}, {\"title\":\"Second card title\",\"description\":\"Second card description\"},{\"title\":\"Last card title\",\"description\":\"Last card description\"} ]", "create": "Create", "createBoardPopup-title": "Create Board", + "createTemplateContainerPopup-title": "Add Template Container", "chooseBoardSourcePopup-title": "Import board", "createLabelPopup-title": "Create Label", "createCustomField": "Create Field", diff --git a/imports/i18n/data/ja.i18n.json b/imports/i18n/data/ja.i18n.json index a6ec7dedb..3eb2c6a4c 100644 --- a/imports/i18n/data/ja.i18n.json +++ b/imports/i18n/data/ja.i18n.json @@ -78,6 +78,18 @@ "activity-deleteComment": "コメント %s を削除しました", "activity-receivedDate": "受付日を %s に変更しました / %s", "activity-startDate": "開始日を %s に変更しました / %s", + "allboards.starred": "Starred", + "allboards.templates": "テンプレート", + "allboards.remaining": "Remaining", + "allboards.workspaces": "Workspaces", + "allboards.add-workspace": "Add Workspace", + "allboards.add-workspace-prompt": "Workspace name", + "allboards.add-subworkspace": "Add Subworkspace", + "allboards.add-subworkspace-prompt": "Subworkspace name", + "allboards.edit-workspace": "Edit workspace", + "allboards.edit-workspace-name": "Workspace name", + "allboards.edit-workspace-icon": "Workspace icon (markdown)", + "multi-selection-active": "Click checkboxes to select boards", "activity-dueDate": "期限日を %s に変更しました / %s", "activity-endDate": "終了日を %s に変更しました / %s", "add-attachment": "添付ファイルを追加", @@ -337,6 +349,7 @@ "copyManyCardsPopup-format": "[ {\"title\": \"1つ目のカードタイトル\", \"description\":\"1つ目のカードの説明\"}, {\"title\":\"2つ目のカードタイトル\",\"description\":\"2つ目のカードの説明\"},{\"title\":\"最後のカードタイトル\",\"description\":\"最後のカードの説明\"} ]", "create": "作成", "createBoardPopup-title": "ボードの作成", + "createTemplateContainerPopup-title": "テンプレートコンテナを追加", "chooseBoardSourcePopup-title": "ボードをインポート", "createLabelPopup-title": "ラベルの作成", "createCustomField": "フィールドを作成", diff --git a/imports/i18n/data/ka.i18n.json b/imports/i18n/data/ka.i18n.json index ea24aa1ef..2c6c8ebfa 100644 --- a/imports/i18n/data/ka.i18n.json +++ b/imports/i18n/data/ka.i18n.json @@ -78,6 +78,18 @@ "activity-deleteComment": "deleted comment %s", "activity-receivedDate": "edited received date to %s of %s", "activity-startDate": "edited start date to %s of %s", + "allboards.starred": "Starred", + "allboards.templates": "Templates", + "allboards.remaining": "Remaining", + "allboards.workspaces": "Workspaces", + "allboards.add-workspace": "Add Workspace", + "allboards.add-workspace-prompt": "Workspace name", + "allboards.add-subworkspace": "Add Subworkspace", + "allboards.add-subworkspace-prompt": "Subworkspace name", + "allboards.edit-workspace": "Edit workspace", + "allboards.edit-workspace-name": "Workspace name", + "allboards.edit-workspace-icon": "Workspace icon (markdown)", + "multi-selection-active": "Click checkboxes to select boards", "activity-dueDate": "edited due date to %s of %s", "activity-endDate": "edited end date to %s of %s", "add-attachment": "მიბმული ფაილის დამატება", @@ -337,6 +349,7 @@ "copyManyCardsPopup-format": "[ {\"სათაური\": \"პირველი ბარათის სათაური\", \"აღწერა\":\"პირველი ბარათის აღწერა\"}, {\"სათაური\":\"მეორე ბარათის სათაური\",\"აღწერა\":\"მეორე ბარათის აღწერა\"},{\"სათაური\":\"ბოლო ბარათის სათაური\",\"აღწერა\":\"ბოლო ბარათის აღწერა\"} ]", "create": "შექმნა", "createBoardPopup-title": "დაფის შექმნა", + "createTemplateContainerPopup-title": "Add Template Container", "chooseBoardSourcePopup-title": "დაფის იმპორტი", "createLabelPopup-title": "ნიშნის შექმნა", "createCustomField": "ველის შექმნა", diff --git a/imports/i18n/data/km.i18n.json b/imports/i18n/data/km.i18n.json index ec5b3989c..a88c7f2c1 100644 --- a/imports/i18n/data/km.i18n.json +++ b/imports/i18n/data/km.i18n.json @@ -78,6 +78,18 @@ "activity-deleteComment": "deleted comment %s", "activity-receivedDate": "edited received date to %s of %s", "activity-startDate": "edited start date to %s of %s", + "allboards.starred": "Starred", + "allboards.templates": "Templates", + "allboards.remaining": "Remaining", + "allboards.workspaces": "Workspaces", + "allboards.add-workspace": "Add Workspace", + "allboards.add-workspace-prompt": "Workspace name", + "allboards.add-subworkspace": "Add Subworkspace", + "allboards.add-subworkspace-prompt": "Subworkspace name", + "allboards.edit-workspace": "Edit workspace", + "allboards.edit-workspace-name": "Workspace name", + "allboards.edit-workspace-icon": "Workspace icon (markdown)", + "multi-selection-active": "Click checkboxes to select boards", "activity-dueDate": "edited due date to %s of %s", "activity-endDate": "edited end date to %s of %s", "add-attachment": "Add Attachment", @@ -337,6 +349,7 @@ "copyManyCardsPopup-format": "[ {\"title\": \"First card title\", \"description\":\"First card description\"}, {\"title\":\"Second card title\",\"description\":\"Second card description\"},{\"title\":\"Last card title\",\"description\":\"Last card description\"} ]", "create": "Create", "createBoardPopup-title": "Create Board", + "createTemplateContainerPopup-title": "Add Template Container", "chooseBoardSourcePopup-title": "Import board", "createLabelPopup-title": "Create Label", "createCustomField": "Create Field", diff --git a/imports/i18n/data/ko-KR.i18n.json b/imports/i18n/data/ko-KR.i18n.json index e17b1dc3b..c0dbd8074 100644 --- a/imports/i18n/data/ko-KR.i18n.json +++ b/imports/i18n/data/ko-KR.i18n.json @@ -78,6 +78,18 @@ "activity-deleteComment": "deleted comment %s", "activity-receivedDate": "edited received date to %s of %s", "activity-startDate": "edited start date to %s of %s", + "allboards.starred": "Starred", + "allboards.templates": "Templates", + "allboards.remaining": "Remaining", + "allboards.workspaces": "Workspaces", + "allboards.add-workspace": "Add Workspace", + "allboards.add-workspace-prompt": "Workspace name", + "allboards.add-subworkspace": "Add Subworkspace", + "allboards.add-subworkspace-prompt": "Subworkspace name", + "allboards.edit-workspace": "Edit workspace", + "allboards.edit-workspace-name": "Workspace name", + "allboards.edit-workspace-icon": "Workspace icon (markdown)", + "multi-selection-active": "Click checkboxes to select boards", "activity-dueDate": "edited due date to %s of %s", "activity-endDate": "edited end date to %s of %s", "add-attachment": "Add Attachment", @@ -337,6 +349,7 @@ "copyManyCardsPopup-format": "[ {\"title\": \"First card title\", \"description\":\"First card description\"}, {\"title\":\"Second card title\",\"description\":\"Second card description\"},{\"title\":\"Last card title\",\"description\":\"Last card description\"} ]", "create": "Create", "createBoardPopup-title": "Create Board", + "createTemplateContainerPopup-title": "Add Template Container", "chooseBoardSourcePopup-title": "Import board", "createLabelPopup-title": "Create Label", "createCustomField": "Create Field", diff --git a/imports/i18n/data/ko.i18n.json b/imports/i18n/data/ko.i18n.json index 501cd1d47..0dea02432 100644 --- a/imports/i18n/data/ko.i18n.json +++ b/imports/i18n/data/ko.i18n.json @@ -78,6 +78,18 @@ "activity-deleteComment": "삭제된 댓글", "activity-receivedDate": "수신 날짜를 %s의 %s로 수정함", "activity-startDate": "시작 날짜가 %s의 %s로 수정됨", + "allboards.starred": "Starred", + "allboards.templates": "템플릿", + "allboards.remaining": "Remaining", + "allboards.workspaces": "Workspaces", + "allboards.add-workspace": "Add Workspace", + "allboards.add-workspace-prompt": "Workspace name", + "allboards.add-subworkspace": "Add Subworkspace", + "allboards.add-subworkspace-prompt": "Subworkspace name", + "allboards.edit-workspace": "Edit workspace", + "allboards.edit-workspace-name": "Workspace name", + "allboards.edit-workspace-icon": "Workspace icon (markdown)", + "multi-selection-active": "Click checkboxes to select boards", "activity-dueDate": "마감 날짜가 %s의 %s로 수정됨", "activity-endDate": "종료 날짜가 %s의 %s로 수정됨", "add-attachment": "첨부파일 추가", @@ -337,6 +349,7 @@ "copyManyCardsPopup-format": "[ {\"title\": \"First card title\", \"description\":\"First card description\"}, {\"title\":\"Second card title\",\"description\":\"Second card description\"},{\"title\":\"Last card title\",\"description\":\"Last card description\"} ]", "create": "생성", "createBoardPopup-title": "보드 생성", + "createTemplateContainerPopup-title": "템플릿 컨테이너 추가", "chooseBoardSourcePopup-title": "보드 가져오기", "createLabelPopup-title": "라벨 생성", "createCustomField": "필드 생성", diff --git a/imports/i18n/data/lt.i18n.json b/imports/i18n/data/lt.i18n.json index 4804e6dff..b6aac02e5 100644 --- a/imports/i18n/data/lt.i18n.json +++ b/imports/i18n/data/lt.i18n.json @@ -78,6 +78,18 @@ "activity-deleteComment": "deleted comment %s", "activity-receivedDate": "edited received date to %s of %s", "activity-startDate": "edited start date to %s of %s", + "allboards.starred": "Starred", + "allboards.templates": "Templates", + "allboards.remaining": "Remaining", + "allboards.workspaces": "Workspaces", + "allboards.add-workspace": "Add Workspace", + "allboards.add-workspace-prompt": "Workspace name", + "allboards.add-subworkspace": "Add Subworkspace", + "allboards.add-subworkspace-prompt": "Subworkspace name", + "allboards.edit-workspace": "Edit workspace", + "allboards.edit-workspace-name": "Workspace name", + "allboards.edit-workspace-icon": "Workspace icon (markdown)", + "multi-selection-active": "Click checkboxes to select boards", "activity-dueDate": "edited due date to %s of %s", "activity-endDate": "edited end date to %s of %s", "add-attachment": "Add Attachment", @@ -337,6 +349,7 @@ "copyManyCardsPopup-format": "[ {\"title\": \"First card title\", \"description\":\"First card description\"}, {\"title\":\"Second card title\",\"description\":\"Second card description\"},{\"title\":\"Last card title\",\"description\":\"Last card description\"} ]", "create": "Create", "createBoardPopup-title": "Create Board", + "createTemplateContainerPopup-title": "Add Template Container", "chooseBoardSourcePopup-title": "Import board", "createLabelPopup-title": "Create Label", "createCustomField": "Create Field", diff --git a/imports/i18n/data/lv.i18n.json b/imports/i18n/data/lv.i18n.json index 51e6984c0..b68d6e47f 100644 --- a/imports/i18n/data/lv.i18n.json +++ b/imports/i18n/data/lv.i18n.json @@ -78,6 +78,18 @@ "activity-deleteComment": "dzēsa komentāru %s", "activity-receivedDate": "edited received date to %s of %s", "activity-startDate": "edited start date to %s of %s", + "allboards.starred": "Starred", + "allboards.templates": "Sagataves", + "allboards.remaining": "Remaining", + "allboards.workspaces": "Workspaces", + "allboards.add-workspace": "Add Workspace", + "allboards.add-workspace-prompt": "Workspace name", + "allboards.add-subworkspace": "Add Subworkspace", + "allboards.add-subworkspace-prompt": "Subworkspace name", + "allboards.edit-workspace": "Edit workspace", + "allboards.edit-workspace-name": "Workspace name", + "allboards.edit-workspace-icon": "Workspace icon (markdown)", + "multi-selection-active": "Click checkboxes to select boards", "activity-dueDate": "edited due date to %s of %s", "activity-endDate": "edited end date to %s of %s", "add-attachment": "Pievienot failu", @@ -337,6 +349,7 @@ "copyManyCardsPopup-format": "[ {\"title\": \"Pirmās kartiņas virsraksts\", \"description\":\"Pirmās kartiņas apraksts}, {\"title\":\"Otrās kartiņas virsraksts\",\"description\":\"Otrās kartiņas apraksts\"},{\"title\":\"Pēdējās kartiņas virsraksts\",\"description\":\"Pēdējās kartiņas apraksts\"} ]", "create": "Izveidot", "createBoardPopup-title": "Izveidot dēli", + "createTemplateContainerPopup-title": "Add Template Container", "chooseBoardSourcePopup-title": "Importēt dēli", "createLabelPopup-title": "Izveidot birku", "createCustomField": "Izveidot lauku", diff --git a/imports/i18n/data/mk.i18n.json b/imports/i18n/data/mk.i18n.json index 6cc517c8b..5ac7384f0 100644 --- a/imports/i18n/data/mk.i18n.json +++ b/imports/i18n/data/mk.i18n.json @@ -78,6 +78,18 @@ "activity-deleteComment": "deleted comment %s", "activity-receivedDate": "edited received date to %s of %s", "activity-startDate": "edited start date to %s of %s", + "allboards.starred": "Starred", + "allboards.templates": "Templates", + "allboards.remaining": "Remaining", + "allboards.workspaces": "Workspaces", + "allboards.add-workspace": "Add Workspace", + "allboards.add-workspace-prompt": "Workspace name", + "allboards.add-subworkspace": "Add Subworkspace", + "allboards.add-subworkspace-prompt": "Subworkspace name", + "allboards.edit-workspace": "Edit workspace", + "allboards.edit-workspace-name": "Workspace name", + "allboards.edit-workspace-icon": "Workspace icon (markdown)", + "multi-selection-active": "Click checkboxes to select boards", "activity-dueDate": "edited due date to %s of %s", "activity-endDate": "edited end date to %s of %s", "add-attachment": "Додај прилог", @@ -337,6 +349,7 @@ "copyManyCardsPopup-format": "[ {\"title\": \"First card title\", \"description\":\"First card description\"}, {\"title\":\"Second card title\",\"description\":\"Second card description\"},{\"title\":\"Last card title\",\"description\":\"Last card description\"} ]", "create": "Креирај", "createBoardPopup-title": "Креирај Табло", + "createTemplateContainerPopup-title": "Add Template Container", "chooseBoardSourcePopup-title": "Импортирай Табло", "createLabelPopup-title": "Креирај Табло", "createCustomField": "Креирај Поле", diff --git a/imports/i18n/data/mn.i18n.json b/imports/i18n/data/mn.i18n.json index ed21faac3..eb1dbc66d 100644 --- a/imports/i18n/data/mn.i18n.json +++ b/imports/i18n/data/mn.i18n.json @@ -78,6 +78,18 @@ "activity-deleteComment": "deleted comment %s", "activity-receivedDate": "edited received date to %s of %s", "activity-startDate": "edited start date to %s of %s", + "allboards.starred": "Starred", + "allboards.templates": "Templates", + "allboards.remaining": "Remaining", + "allboards.workspaces": "Workspaces", + "allboards.add-workspace": "Add Workspace", + "allboards.add-workspace-prompt": "Workspace name", + "allboards.add-subworkspace": "Add Subworkspace", + "allboards.add-subworkspace-prompt": "Subworkspace name", + "allboards.edit-workspace": "Edit workspace", + "allboards.edit-workspace-name": "Workspace name", + "allboards.edit-workspace-icon": "Workspace icon (markdown)", + "multi-selection-active": "Click checkboxes to select boards", "activity-dueDate": "edited due date to %s of %s", "activity-endDate": "edited end date to %s of %s", "add-attachment": "Хавсралт нэмэх", @@ -337,6 +349,7 @@ "copyManyCardsPopup-format": "[ {\"title\": \"First card title\", \"description\":\"First card description\"}, {\"title\":\"Second card title\",\"description\":\"Second card description\"},{\"title\":\"Last card title\",\"description\":\"Last card description\"} ]", "create": "Үүсгэх", "createBoardPopup-title": "Самбар үүсгэх", + "createTemplateContainerPopup-title": "Add Template Container", "chooseBoardSourcePopup-title": "Import board", "createLabelPopup-title": "Шошго үүсгэх", "createCustomField": "Create Field", diff --git a/imports/i18n/data/ms-MY.i18n.json b/imports/i18n/data/ms-MY.i18n.json index c5a4d2586..7c102b2ef 100644 --- a/imports/i18n/data/ms-MY.i18n.json +++ b/imports/i18n/data/ms-MY.i18n.json @@ -78,6 +78,18 @@ "activity-deleteComment": "deleted comment %s", "activity-receivedDate": "edited received date to %s of %s", "activity-startDate": "edited start date to %s of %s", + "allboards.starred": "Starred", + "allboards.templates": "Templates", + "allboards.remaining": "Remaining", + "allboards.workspaces": "Workspaces", + "allboards.add-workspace": "Add Workspace", + "allboards.add-workspace-prompt": "Workspace name", + "allboards.add-subworkspace": "Add Subworkspace", + "allboards.add-subworkspace-prompt": "Subworkspace name", + "allboards.edit-workspace": "Edit workspace", + "allboards.edit-workspace-name": "Workspace name", + "allboards.edit-workspace-icon": "Workspace icon (markdown)", + "multi-selection-active": "Click checkboxes to select boards", "activity-dueDate": "edited due date to %s of %s", "activity-endDate": "edited end date to %s of %s", "add-attachment": "Add Attachment", @@ -337,6 +349,7 @@ "copyManyCardsPopup-format": "[ {\"title\": \"First card title\", \"description\":\"First card description\"}, {\"title\":\"Second card title\",\"description\":\"Second card description\"},{\"title\":\"Last card title\",\"description\":\"Last card description\"} ]", "create": "Create", "createBoardPopup-title": "Create Board", + "createTemplateContainerPopup-title": "Add Template Container", "chooseBoardSourcePopup-title": "Import board", "createLabelPopup-title": "Create Label", "createCustomField": "Create Field", diff --git a/imports/i18n/data/ms.i18n.json b/imports/i18n/data/ms.i18n.json index fc9e8b938..966475e88 100644 --- a/imports/i18n/data/ms.i18n.json +++ b/imports/i18n/data/ms.i18n.json @@ -78,6 +78,18 @@ "activity-deleteComment": "deleted comment %s", "activity-receivedDate": "edited received date to %s of %s", "activity-startDate": "edited start date to %s of %s", + "allboards.starred": "Starred", + "allboards.templates": "Semua Templat", + "allboards.remaining": "Remaining", + "allboards.workspaces": "Workspaces", + "allboards.add-workspace": "Add Workspace", + "allboards.add-workspace-prompt": "Workspace name", + "allboards.add-subworkspace": "Add Subworkspace", + "allboards.add-subworkspace-prompt": "Subworkspace name", + "allboards.edit-workspace": "Edit workspace", + "allboards.edit-workspace-name": "Workspace name", + "allboards.edit-workspace-icon": "Workspace icon (markdown)", + "multi-selection-active": "Click checkboxes to select boards", "activity-dueDate": "edited due date to %s of %s", "activity-endDate": "edited end date to %s of %s", "add-attachment": "Add Attachment", @@ -337,6 +349,7 @@ "copyManyCardsPopup-format": "[ {\"title\": \"First card title\", \"description\":\"First card description\"}, {\"title\":\"Second card title\",\"description\":\"Second card description\"},{\"title\":\"Last card title\",\"description\":\"Last card description\"} ]", "create": "Cipta", "createBoardPopup-title": "Cipta Papan", + "createTemplateContainerPopup-title": "Tambah Bekas Templat", "chooseBoardSourcePopup-title": "Import Papan", "createLabelPopup-title": "Cipta Label", "createCustomField": "Cipta ruangan", diff --git a/imports/i18n/data/nb.i18n.json b/imports/i18n/data/nb.i18n.json index 998b6973f..4557e6542 100644 --- a/imports/i18n/data/nb.i18n.json +++ b/imports/i18n/data/nb.i18n.json @@ -78,6 +78,18 @@ "activity-deleteComment": "slettet kommentar %s", "activity-receivedDate": "redigert mottatt dato til %s av %s", "activity-startDate": "redigert startdato til %s av %s", + "allboards.starred": "Starred", + "allboards.templates": "Maler", + "allboards.remaining": "Remaining", + "allboards.workspaces": "Workspaces", + "allboards.add-workspace": "Add Workspace", + "allboards.add-workspace-prompt": "Workspace name", + "allboards.add-subworkspace": "Add Subworkspace", + "allboards.add-subworkspace-prompt": "Subworkspace name", + "allboards.edit-workspace": "Edit workspace", + "allboards.edit-workspace-name": "Workspace name", + "allboards.edit-workspace-icon": "Workspace icon (markdown)", + "multi-selection-active": "Click checkboxes to select boards", "activity-dueDate": "redigert forfallsdato til %s av %s", "activity-endDate": "redigert sluttdato %s av %s", "add-attachment": "Legg til Vedlegg", @@ -337,6 +349,7 @@ "copyManyCardsPopup-format": "[ {\"Tittel\": \"Tittel første kort\", \"Beskrivelse\":\"Beskrivelse første kort\"}, {\"Tittel\":\"Tittel andre kort\",\"Beskrivelse\":\"Beskrivelse andre kort\"},{\"Tittel\":\"Tittel siste kort\",\"Beskrivelse\":\"Beskrivelse siste kort\"} ]", "create": "Opprett", "createBoardPopup-title": "Opprett Tavle", + "createTemplateContainerPopup-title": "Legg til Malgruppe", "chooseBoardSourcePopup-title": "Importer tavle", "createLabelPopup-title": "Opprett Etikett", "createCustomField": "Opprett Felt", diff --git a/imports/i18n/data/nl-NL.i18n.json b/imports/i18n/data/nl-NL.i18n.json index 5ff6008ed..46eb730b3 100644 --- a/imports/i18n/data/nl-NL.i18n.json +++ b/imports/i18n/data/nl-NL.i18n.json @@ -78,6 +78,18 @@ "activity-deleteComment": "aantekening verwijderd %s", "activity-receivedDate": "ontvangst datum gewijzigd naar %s van %s", "activity-startDate": "start datum gewijzigd naar %s van %s", + "allboards.starred": "Starred", + "allboards.templates": "Templates", + "allboards.remaining": "Remaining", + "allboards.workspaces": "Workspaces", + "allboards.add-workspace": "Add Workspace", + "allboards.add-workspace-prompt": "Workspace name", + "allboards.add-subworkspace": "Add Subworkspace", + "allboards.add-subworkspace-prompt": "Subworkspace name", + "allboards.edit-workspace": "Edit workspace", + "allboards.edit-workspace-name": "Workspace name", + "allboards.edit-workspace-icon": "Workspace icon (markdown)", + "multi-selection-active": "Click checkboxes to select boards", "activity-dueDate": "vervaldatum gewijzigd naar %s van %s", "activity-endDate": "einddatum gewijzigd naar %s van %s", "add-attachment": "Bijlage Toevoegen", @@ -337,6 +349,7 @@ "copyManyCardsPopup-format": "[ {\"title\": \"Titel eerste kaart\", \"description\":\"Omschrijving eerste kaart\"}, {\"title\":\"Titel tweede kaart\",\"description\":\"Omschrijving tweede kaart\"},{\"title\":\"Titel laatste kaart\",\"description\":\"Omschrijving laatste kaart\"} ]", "create": "Aanmaken", "createBoardPopup-title": "Bord aanmaken", + "createTemplateContainerPopup-title": "Template Container Toevoegen", "chooseBoardSourcePopup-title": "Importeer bord", "createLabelPopup-title": "Label aanmaken", "createCustomField": "Veld aanmaken", diff --git a/imports/i18n/data/nl.i18n.json b/imports/i18n/data/nl.i18n.json index 2d5624db5..1e3cda7f5 100644 --- a/imports/i18n/data/nl.i18n.json +++ b/imports/i18n/data/nl.i18n.json @@ -78,6 +78,18 @@ "activity-deleteComment": "aantekening verwijderd %s", "activity-receivedDate": "ontvangst datum gewijzigd naar %s van %s", "activity-startDate": "start datum gewijzigd naar %s van %s", + "allboards.starred": "Starred", + "allboards.templates": "Templates", + "allboards.remaining": "Remaining", + "allboards.workspaces": "Workspaces", + "allboards.add-workspace": "Add Workspace", + "allboards.add-workspace-prompt": "Workspace name", + "allboards.add-subworkspace": "Add Subworkspace", + "allboards.add-subworkspace-prompt": "Subworkspace name", + "allboards.edit-workspace": "Edit workspace", + "allboards.edit-workspace-name": "Workspace name", + "allboards.edit-workspace-icon": "Workspace icon (markdown)", + "multi-selection-active": "Click checkboxes to select boards", "activity-dueDate": "vervaldatum gewijzigd naar %s van %s", "activity-endDate": "einddatum gewijzigd naar %s van %s", "add-attachment": "Bijlage Toevoegen", @@ -337,6 +349,7 @@ "copyManyCardsPopup-format": "[ {\"title\": \"Titel eerste kaart\", \"description\":\"Omschrijving eerste kaart\"}, {\"title\":\"Titel tweede kaart\",\"description\":\"Omschrijving tweede kaart\"},{\"title\":\"Titel laatste kaart\",\"description\":\"Omschrijving laatste kaart\"} ]", "create": "Aanmaken", "createBoardPopup-title": "Bord aanmaken", + "createTemplateContainerPopup-title": "Template Container Toevoegen", "chooseBoardSourcePopup-title": "Importeer bord", "createLabelPopup-title": "Label aanmaken", "createCustomField": "Veld aanmaken", diff --git a/imports/i18n/data/oc.i18n.json b/imports/i18n/data/oc.i18n.json index 1acaa8d5d..9c3543796 100644 --- a/imports/i18n/data/oc.i18n.json +++ b/imports/i18n/data/oc.i18n.json @@ -78,6 +78,18 @@ "activity-deleteComment": "deleted comment %s", "activity-receivedDate": "edited received date to %s of %s", "activity-startDate": "edited start date to %s of %s", + "allboards.starred": "Starred", + "allboards.templates": "Modèls", + "allboards.remaining": "Remaining", + "allboards.workspaces": "Workspaces", + "allboards.add-workspace": "Add Workspace", + "allboards.add-workspace-prompt": "Workspace name", + "allboards.add-subworkspace": "Add Subworkspace", + "allboards.add-subworkspace-prompt": "Subworkspace name", + "allboards.edit-workspace": "Edit workspace", + "allboards.edit-workspace-name": "Workspace name", + "allboards.edit-workspace-icon": "Workspace icon (markdown)", + "multi-selection-active": "Click checkboxes to select boards", "activity-dueDate": "edited due date to %s of %s", "activity-endDate": "edited end date to %s of %s", "add-attachment": "Apondre una pèça joncha", @@ -337,6 +349,7 @@ "copyManyCardsPopup-format": "[ {\"title\": \"Títol de la primièra carta\", \"description\":\"Descripcion de la primièra carta\"}, {\"title\":\"Títol de la segonda carta\",\"description\":\"Descripcion de la segonda carta\"},{\"title\":\"Títol de la darrièra carta\",\"description\":\"Descripcion de la darrièra carta\"} ]", "create": "Crear", "createBoardPopup-title": "Crear un tablèu", + "createTemplateContainerPopup-title": "Add Template Container", "chooseBoardSourcePopup-title": "Importar un tablèu", "createLabelPopup-title": "Crear una etiqueta", "createCustomField": "Crear un camp", diff --git a/imports/i18n/data/or_IN.i18n.json b/imports/i18n/data/or_IN.i18n.json index 958d8313a..acf3c6934 100644 --- a/imports/i18n/data/or_IN.i18n.json +++ b/imports/i18n/data/or_IN.i18n.json @@ -78,6 +78,18 @@ "activity-deleteComment": "deleted comment %s", "activity-receivedDate": "edited received date to %s of %s", "activity-startDate": "edited start date to %s of %s", + "allboards.starred": "Starred", + "allboards.templates": "Templates", + "allboards.remaining": "Remaining", + "allboards.workspaces": "Workspaces", + "allboards.add-workspace": "Add Workspace", + "allboards.add-workspace-prompt": "Workspace name", + "allboards.add-subworkspace": "Add Subworkspace", + "allboards.add-subworkspace-prompt": "Subworkspace name", + "allboards.edit-workspace": "Edit workspace", + "allboards.edit-workspace-name": "Workspace name", + "allboards.edit-workspace-icon": "Workspace icon (markdown)", + "multi-selection-active": "Click checkboxes to select boards", "activity-dueDate": "edited due date to %s of %s", "activity-endDate": "edited end date to %s of %s", "add-attachment": "Add Attachment", @@ -337,6 +349,7 @@ "copyManyCardsPopup-format": "[ {\"title\": \"First card title\", \"description\":\"First card description\"}, {\"title\":\"Second card title\",\"description\":\"Second card description\"},{\"title\":\"Last card title\",\"description\":\"Last card description\"} ]", "create": "Create", "createBoardPopup-title": "Create Board", + "createTemplateContainerPopup-title": "Add Template Container", "chooseBoardSourcePopup-title": "Import board", "createLabelPopup-title": "Create Label", "createCustomField": "Create Field", diff --git a/imports/i18n/data/pa.i18n.json b/imports/i18n/data/pa.i18n.json index 4804e6dff..b6aac02e5 100644 --- a/imports/i18n/data/pa.i18n.json +++ b/imports/i18n/data/pa.i18n.json @@ -78,6 +78,18 @@ "activity-deleteComment": "deleted comment %s", "activity-receivedDate": "edited received date to %s of %s", "activity-startDate": "edited start date to %s of %s", + "allboards.starred": "Starred", + "allboards.templates": "Templates", + "allboards.remaining": "Remaining", + "allboards.workspaces": "Workspaces", + "allboards.add-workspace": "Add Workspace", + "allboards.add-workspace-prompt": "Workspace name", + "allboards.add-subworkspace": "Add Subworkspace", + "allboards.add-subworkspace-prompt": "Subworkspace name", + "allboards.edit-workspace": "Edit workspace", + "allboards.edit-workspace-name": "Workspace name", + "allboards.edit-workspace-icon": "Workspace icon (markdown)", + "multi-selection-active": "Click checkboxes to select boards", "activity-dueDate": "edited due date to %s of %s", "activity-endDate": "edited end date to %s of %s", "add-attachment": "Add Attachment", @@ -337,6 +349,7 @@ "copyManyCardsPopup-format": "[ {\"title\": \"First card title\", \"description\":\"First card description\"}, {\"title\":\"Second card title\",\"description\":\"Second card description\"},{\"title\":\"Last card title\",\"description\":\"Last card description\"} ]", "create": "Create", "createBoardPopup-title": "Create Board", + "createTemplateContainerPopup-title": "Add Template Container", "chooseBoardSourcePopup-title": "Import board", "createLabelPopup-title": "Create Label", "createCustomField": "Create Field", diff --git a/imports/i18n/data/pl-PL.i18n.json b/imports/i18n/data/pl-PL.i18n.json index 313e3db40..c7ea811a7 100644 --- a/imports/i18n/data/pl-PL.i18n.json +++ b/imports/i18n/data/pl-PL.i18n.json @@ -78,6 +78,18 @@ "activity-deleteComment": "usunął komentarz %s", "activity-receivedDate": "zmienił datę otrzymania na %s z %s", "activity-startDate": "zmienił datę rozpoczęcia na %s z %s", + "allboards.starred": "Starred", + "allboards.templates": "Szablony", + "allboards.remaining": "Remaining", + "allboards.workspaces": "Workspaces", + "allboards.add-workspace": "Add Workspace", + "allboards.add-workspace-prompt": "Workspace name", + "allboards.add-subworkspace": "Add Subworkspace", + "allboards.add-subworkspace-prompt": "Subworkspace name", + "allboards.edit-workspace": "Edit workspace", + "allboards.edit-workspace-name": "Workspace name", + "allboards.edit-workspace-icon": "Workspace icon (markdown)", + "multi-selection-active": "Click checkboxes to select boards", "activity-dueDate": "zmienił datę wykonania na %s z %s", "activity-endDate": "zmienił datę zakończenia na %s z %s", "add-attachment": "Dodaj załącznik", @@ -337,6 +349,7 @@ "copyManyCardsPopup-format": "[ {\"title\": \"Tytuł pierwszej karty\", \"description\":\"Opis pierwszej karty\"}, {\"title\":\"Tytuł drugiej karty\",\"description\":\"Opis drugiej karty\"},{\"title\":\"Tytuł ostatniej karty\",\"description\":\"Opis ostatniej karty\"} ]", "create": "Utwórz", "createBoardPopup-title": "Utwórz tablicę", + "createTemplateContainerPopup-title": "Dodaj Kontener Szablonów", "chooseBoardSourcePopup-title": "Import tablicy", "createLabelPopup-title": "Utwórz etykietę", "createCustomField": "Utwórz pole", diff --git a/imports/i18n/data/pl.i18n.json b/imports/i18n/data/pl.i18n.json index 5d38e2647..c01e9746d 100644 --- a/imports/i18n/data/pl.i18n.json +++ b/imports/i18n/data/pl.i18n.json @@ -78,6 +78,18 @@ "activity-deleteComment": "usunięto komentarz %s", "activity-receivedDate": "zmienił datę otrzymania na %s z %s", "activity-startDate": "zmienił datę rozpoczęcia na %s z %s", + "allboards.starred": "Starred", + "allboards.templates": "Szablony", + "allboards.remaining": "Remaining", + "allboards.workspaces": "Workspaces", + "allboards.add-workspace": "Add Workspace", + "allboards.add-workspace-prompt": "Workspace name", + "allboards.add-subworkspace": "Add Subworkspace", + "allboards.add-subworkspace-prompt": "Subworkspace name", + "allboards.edit-workspace": "Edit workspace", + "allboards.edit-workspace-name": "Workspace name", + "allboards.edit-workspace-icon": "Workspace icon (markdown)", + "multi-selection-active": "Click checkboxes to select boards", "activity-dueDate": "zmienił datę wykonania na %s z %s", "activity-endDate": "zmienił datę zakończenia na %s z %s", "add-attachment": "Dodaj załącznik", @@ -337,6 +349,7 @@ "copyManyCardsPopup-format": "[ {\"title\": \"Tytuł pierwszej karty\", \"description\":\"Opis pierwszej karty\"}, {\"title\":\"Tytuł drugiej karty\",\"description\":\"Opis drugiej karty\"},{\"title\":\"Tytuł ostatniej karty\",\"description\":\"Opis ostatniej karty\"} ]", "create": "Utwórz", "createBoardPopup-title": "Utwórz tablicę", + "createTemplateContainerPopup-title": "Dodaj Kontener Szablonów", "chooseBoardSourcePopup-title": "Import tablicy", "createLabelPopup-title": "Utwórz etykietę", "createCustomField": "Utwórz pole", diff --git a/imports/i18n/data/pt-BR.i18n.json b/imports/i18n/data/pt-BR.i18n.json index 89965f75b..6c2d764ed 100644 --- a/imports/i18n/data/pt-BR.i18n.json +++ b/imports/i18n/data/pt-BR.i18n.json @@ -78,6 +78,18 @@ "activity-deleteComment": "comentário excluído %s", "activity-receivedDate": "editou recebido para %s de %s", "activity-startDate": "editou data início para %s de %s", + "allboards.starred": "Starred", + "allboards.templates": "Modelos", + "allboards.remaining": "Remaining", + "allboards.workspaces": "Workspaces", + "allboards.add-workspace": "Add Workspace", + "allboards.add-workspace-prompt": "Workspace name", + "allboards.add-subworkspace": "Add Subworkspace", + "allboards.add-subworkspace-prompt": "Subworkspace name", + "allboards.edit-workspace": "Edit workspace", + "allboards.edit-workspace-name": "Workspace name", + "allboards.edit-workspace-icon": "Workspace icon (markdown)", + "multi-selection-active": "Click checkboxes to select boards", "activity-dueDate": "editou prazo final para %s de %s", "activity-endDate": "editou concluído para %s de %s", "add-attachment": "Adicionar Anexos", @@ -337,6 +349,7 @@ "copyManyCardsPopup-format": "[ {\"title\": \"Título do primeiro cartão\", \"description\":\"Descrição do primeiro cartão\"}, {\"title\":\"Título do segundo cartão\",\"description\":\"Descrição do segundo cartão\"},{\"title\":\"Título do último cartão\",\"description\":\"Descrição do último cartão\"} ]", "create": "Criar", "createBoardPopup-title": "Criar Quadro", + "createTemplateContainerPopup-title": "Adicionar Contêiner de Modelo", "chooseBoardSourcePopup-title": "Importar quadro", "createLabelPopup-title": "Criar Etiqueta", "createCustomField": "Criar campo", @@ -1404,7 +1417,7 @@ "back-to-settings": "Voltar às Configurações", "board-id": "ID do Quadro", "board-migration": "Migração de Quadro", - "board-migrations": "Board Migrations", + "board-migrations": "Migração de Quadros", "card-show-lists-on-minicard": "Mostrar Listas no Mini cartão", "comprehensive-board-migration": "Comprehensive Board Migration", "comprehensive-board-migration-description": "Performs comprehensive checks and fixes for board data integrity, including list ordering, card positions, and swimlane structure.", diff --git a/imports/i18n/data/pt.i18n.json b/imports/i18n/data/pt.i18n.json index 61052311f..2aef98106 100644 --- a/imports/i18n/data/pt.i18n.json +++ b/imports/i18n/data/pt.i18n.json @@ -78,6 +78,18 @@ "activity-deleteComment": "apagou o comentário %s", "activity-receivedDate": "editou a data recebida para %s de %s", "activity-startDate": "editou a data de início para %s de %s", + "allboards.starred": "Starred", + "allboards.templates": "Modelos", + "allboards.remaining": "Remaining", + "allboards.workspaces": "Workspaces", + "allboards.add-workspace": "Add Workspace", + "allboards.add-workspace-prompt": "Workspace name", + "allboards.add-subworkspace": "Add Subworkspace", + "allboards.add-subworkspace-prompt": "Subworkspace name", + "allboards.edit-workspace": "Edit workspace", + "allboards.edit-workspace-name": "Workspace name", + "allboards.edit-workspace-icon": "Workspace icon (markdown)", + "multi-selection-active": "Click checkboxes to select boards", "activity-dueDate": "editou a data limite para %s de %s", "activity-endDate": "editou a data de fim para %s de %s", "add-attachment": "Add Attachment", @@ -337,6 +349,7 @@ "copyManyCardsPopup-format": "[ {\"title\": \"Título do primeiro cartão\", \"description\":\"Descrição do primeiro cartão\"}, {\"title\":\"Título do segundo cartão\",\"description\":\"Descrição do segundo cartão\"},{\"title\":\"Título do último cartão\",\"description\":\"Descrição do último cartão\"} ]", "create": "Criar", "createBoardPopup-title": "Criar Quadro", + "createTemplateContainerPopup-title": "Add Template Container", "chooseBoardSourcePopup-title": "Importar quadro", "createLabelPopup-title": "Criar Etiqueta", "createCustomField": "Criar Campo", diff --git a/imports/i18n/data/pt_PT.i18n.json b/imports/i18n/data/pt_PT.i18n.json index 260261018..f355d4cb7 100644 --- a/imports/i18n/data/pt_PT.i18n.json +++ b/imports/i18n/data/pt_PT.i18n.json @@ -78,6 +78,18 @@ "activity-deleteComment": "apagou o comentário %s", "activity-receivedDate": "editou a data recebida para %s de %s", "activity-startDate": "editou a data de início para %s de %s", + "allboards.starred": "Starred", + "allboards.templates": "Modelos", + "allboards.remaining": "Remaining", + "allboards.workspaces": "Workspaces", + "allboards.add-workspace": "Add Workspace", + "allboards.add-workspace-prompt": "Workspace name", + "allboards.add-subworkspace": "Add Subworkspace", + "allboards.add-subworkspace-prompt": "Subworkspace name", + "allboards.edit-workspace": "Edit workspace", + "allboards.edit-workspace-name": "Workspace name", + "allboards.edit-workspace-icon": "Workspace icon (markdown)", + "multi-selection-active": "Click checkboxes to select boards", "activity-dueDate": "editou a data limite para %s de %s", "activity-endDate": "editou a data de fim para %s de %s", "add-attachment": "Adicionar Anexo", @@ -337,6 +349,7 @@ "copyManyCardsPopup-format": "[ {\"title\": \"Título do primeiro cartão\", \"description\":\"Descrição do primeiro cartão\"}, {\"title\":\"Título do segundo cartão\",\"description\":\"Descrição do segundo cartão\"},{\"title\":\"Título do último cartão\",\"description\":\"Descrição do último cartão\"} ]", "create": "Criar", "createBoardPopup-title": "Criar Quadro", + "createTemplateContainerPopup-title": "Add Template Container", "chooseBoardSourcePopup-title": "Importar quadro", "createLabelPopup-title": "Criar Etiqueta", "createCustomField": "Criar Campo", diff --git a/imports/i18n/data/ro-RO.i18n.json b/imports/i18n/data/ro-RO.i18n.json index 11fa7fcf3..4558b375a 100644 --- a/imports/i18n/data/ro-RO.i18n.json +++ b/imports/i18n/data/ro-RO.i18n.json @@ -78,6 +78,18 @@ "activity-deleteComment": "deleted comment %s", "activity-receivedDate": "edited received date to %s of %s", "activity-startDate": "edited start date to %s of %s", + "allboards.starred": "Starred", + "allboards.templates": "Templates", + "allboards.remaining": "Remaining", + "allboards.workspaces": "Workspaces", + "allboards.add-workspace": "Add Workspace", + "allboards.add-workspace-prompt": "Workspace name", + "allboards.add-subworkspace": "Add Subworkspace", + "allboards.add-subworkspace-prompt": "Subworkspace name", + "allboards.edit-workspace": "Edit workspace", + "allboards.edit-workspace-name": "Workspace name", + "allboards.edit-workspace-icon": "Workspace icon (markdown)", + "multi-selection-active": "Click checkboxes to select boards", "activity-dueDate": "edited due date to %s of %s", "activity-endDate": "edited end date to %s of %s", "add-attachment": "Add Attachment", @@ -337,6 +349,7 @@ "copyManyCardsPopup-format": "[ {\"title\": \"First card title\", \"description\":\"First card description\"}, {\"title\":\"Second card title\",\"description\":\"Second card description\"},{\"title\":\"Last card title\",\"description\":\"Last card description\"} ]", "create": "Create", "createBoardPopup-title": "Create Board", + "createTemplateContainerPopup-title": "Add Template Container", "chooseBoardSourcePopup-title": "Import board", "createLabelPopup-title": "Create Label", "createCustomField": "Create Field", diff --git a/imports/i18n/data/ro.i18n.json b/imports/i18n/data/ro.i18n.json index d41b0abaa..289a3a45e 100644 --- a/imports/i18n/data/ro.i18n.json +++ b/imports/i18n/data/ro.i18n.json @@ -78,6 +78,18 @@ "activity-deleteComment": "deleted comment %s", "activity-receivedDate": "edited received date to %s of %s", "activity-startDate": "edited start date to %s of %s", + "allboards.starred": "Starred", + "allboards.templates": "Templates", + "allboards.remaining": "Remaining", + "allboards.workspaces": "Workspaces", + "allboards.add-workspace": "Add Workspace", + "allboards.add-workspace-prompt": "Workspace name", + "allboards.add-subworkspace": "Add Subworkspace", + "allboards.add-subworkspace-prompt": "Subworkspace name", + "allboards.edit-workspace": "Edit workspace", + "allboards.edit-workspace-name": "Workspace name", + "allboards.edit-workspace-icon": "Workspace icon (markdown)", + "multi-selection-active": "Click checkboxes to select boards", "activity-dueDate": "edited due date to %s of %s", "activity-endDate": "edited end date to %s of %s", "add-attachment": "Add Attachment", @@ -337,6 +349,7 @@ "copyManyCardsPopup-format": "[ {\"title\": \"First card title\", \"description\":\"First card description\"}, {\"title\":\"Second card title\",\"description\":\"Second card description\"},{\"title\":\"Last card title\",\"description\":\"Last card description\"} ]", "create": "Create", "createBoardPopup-title": "Create Board", + "createTemplateContainerPopup-title": "Add Template Container", "chooseBoardSourcePopup-title": "Import board", "createLabelPopup-title": "Create Label", "createCustomField": "Create Field", diff --git a/imports/i18n/data/ru-UA.i18n.json b/imports/i18n/data/ru-UA.i18n.json index 65a3eec81..ad237aa07 100644 --- a/imports/i18n/data/ru-UA.i18n.json +++ b/imports/i18n/data/ru-UA.i18n.json @@ -78,6 +78,18 @@ "activity-deleteComment": "deleted comment %s", "activity-receivedDate": "edited received date to %s of %s", "activity-startDate": "edited start date to %s of %s", + "allboards.starred": "Starred", + "allboards.templates": "Шаблоны", + "allboards.remaining": "Remaining", + "allboards.workspaces": "Workspaces", + "allboards.add-workspace": "Add Workspace", + "allboards.add-workspace-prompt": "Workspace name", + "allboards.add-subworkspace": "Add Subworkspace", + "allboards.add-subworkspace-prompt": "Subworkspace name", + "allboards.edit-workspace": "Edit workspace", + "allboards.edit-workspace-name": "Workspace name", + "allboards.edit-workspace-icon": "Workspace icon (markdown)", + "multi-selection-active": "Click checkboxes to select boards", "activity-dueDate": "edited due date to %s of %s", "activity-endDate": "edited end date to %s of %s", "add-attachment": "Add Attachment", @@ -337,6 +349,7 @@ "copyManyCardsPopup-format": "[ {\"title\": \"First card title\", \"description\":\"First card description\"}, {\"title\":\"Second card title\",\"description\":\"Second card description\"},{\"title\":\"Last card title\",\"description\":\"Last card description\"} ]", "create": "Create", "createBoardPopup-title": "Create Board", + "createTemplateContainerPopup-title": "Добавить Шаблон Контейнера", "chooseBoardSourcePopup-title": "Import board", "createLabelPopup-title": "Create Label", "createCustomField": "Create Field", diff --git a/imports/i18n/data/ru.i18n.json b/imports/i18n/data/ru.i18n.json index 1b70576d7..04a83c4c7 100644 --- a/imports/i18n/data/ru.i18n.json +++ b/imports/i18n/data/ru.i18n.json @@ -78,6 +78,18 @@ "activity-deleteComment": "удалил комментарий %s", "activity-receivedDate": "отредактировал дату получения на %sс %s", "activity-startDate": "отредактировал дату начала на %sс %s", + "allboards.starred": "Starred", + "allboards.templates": "Шаблоны", + "allboards.remaining": "Remaining", + "allboards.workspaces": "Workspaces", + "allboards.add-workspace": "Add Workspace", + "allboards.add-workspace-prompt": "Workspace name", + "allboards.add-subworkspace": "Add Subworkspace", + "allboards.add-subworkspace-prompt": "Subworkspace name", + "allboards.edit-workspace": "Edit workspace", + "allboards.edit-workspace-name": "Workspace name", + "allboards.edit-workspace-icon": "Workspace icon (markdown)", + "multi-selection-active": "Click checkboxes to select boards", "activity-dueDate": "отредактировал срок исполнения на %s с %s", "activity-endDate": "отредактировал дату завершения на %s с %s", "add-attachment": "Добавить вложение", @@ -337,6 +349,7 @@ "copyManyCardsPopup-format": "[ {\"title\": \"Название первой карточки\", \"description\":\"Описание первой карточки\"}, {\"title\":\"Название второй карточки\",\"description\":\"Описание второй карточки\"},{\"title\":\"Название последней карточки\",\"description\":\"Описание последней карточки\"} ]", "create": "Создать", "createBoardPopup-title": "Создать доску", + "createTemplateContainerPopup-title": "Добавить Шаблон Контейнера", "chooseBoardSourcePopup-title": "Импортировать доску", "createLabelPopup-title": "Создать метку", "createCustomField": "Создать поле", diff --git a/imports/i18n/data/sk.i18n.json b/imports/i18n/data/sk.i18n.json index 6285db533..8dc0e0b52 100644 --- a/imports/i18n/data/sk.i18n.json +++ b/imports/i18n/data/sk.i18n.json @@ -78,6 +78,18 @@ "activity-deleteComment": "deleted comment %s", "activity-receivedDate": "edited received date to %s of %s", "activity-startDate": "edited start date to %s of %s", + "allboards.starred": "Starred", + "allboards.templates": "Templates", + "allboards.remaining": "Remaining", + "allboards.workspaces": "Workspaces", + "allboards.add-workspace": "Add Workspace", + "allboards.add-workspace-prompt": "Workspace name", + "allboards.add-subworkspace": "Add Subworkspace", + "allboards.add-subworkspace-prompt": "Subworkspace name", + "allboards.edit-workspace": "Edit workspace", + "allboards.edit-workspace-name": "Workspace name", + "allboards.edit-workspace-icon": "Workspace icon (markdown)", + "multi-selection-active": "Click checkboxes to select boards", "activity-dueDate": "edited due date to %s of %s", "activity-endDate": "edited end date to %s of %s", "add-attachment": "Pridať prílohu", @@ -337,6 +349,7 @@ "copyManyCardsPopup-format": "[ {\"title\": \"First card title\", \"description\":\"First card description\"}, {\"title\":\"Second card title\",\"description\":\"Second card description\"},{\"title\":\"Last card title\",\"description\":\"Last card description\"} ]", "create": "Vytvoriť", "createBoardPopup-title": "Vytvoriť nástenku", + "createTemplateContainerPopup-title": "Add Template Container", "chooseBoardSourcePopup-title": "Import board", "createLabelPopup-title": "Vytvoriť štítok", "createCustomField": "Vytvoriť pole", diff --git a/imports/i18n/data/sl.i18n.json b/imports/i18n/data/sl.i18n.json index b9a1b7522..6b28083b9 100644 --- a/imports/i18n/data/sl.i18n.json +++ b/imports/i18n/data/sl.i18n.json @@ -78,6 +78,18 @@ "activity-deleteComment": "izbrisal komentar %s", "activity-receivedDate": "edited received date to %s of %s", "activity-startDate": "edited start date to %s of %s", + "allboards.starred": "Starred", + "allboards.templates": "Predloge", + "allboards.remaining": "Remaining", + "allboards.workspaces": "Workspaces", + "allboards.add-workspace": "Add Workspace", + "allboards.add-workspace-prompt": "Workspace name", + "allboards.add-subworkspace": "Add Subworkspace", + "allboards.add-subworkspace-prompt": "Subworkspace name", + "allboards.edit-workspace": "Edit workspace", + "allboards.edit-workspace-name": "Workspace name", + "allboards.edit-workspace-icon": "Workspace icon (markdown)", + "multi-selection-active": "Click checkboxes to select boards", "activity-dueDate": "edited due date to %s of %s", "activity-endDate": "edited end date to %s of %s", "add-attachment": "Dodaj priponko", @@ -337,6 +349,7 @@ "copyManyCardsPopup-format": "[ {\"naslov\": \"Naslov prve kartice\", \"opis\":\"Opis prve kartice\"}, {\"naslov\":\"Opis druge kartice\",\"opis\":\"Opis druge kartice\"},{\"naslov\":\"Naslov zadnje kartice\",\"opis\":\"Opis zadnje kartice\"} ]", "create": "Ustvari", "createBoardPopup-title": "Ustvari tablo", + "createTemplateContainerPopup-title": "Add Template Container", "chooseBoardSourcePopup-title": "Uvozi tablo", "createLabelPopup-title": "Ustvari oznako", "createCustomField": "Ustvari polje", diff --git a/imports/i18n/data/sr.i18n.json b/imports/i18n/data/sr.i18n.json index 302397003..1fc401068 100644 --- a/imports/i18n/data/sr.i18n.json +++ b/imports/i18n/data/sr.i18n.json @@ -78,6 +78,18 @@ "activity-deleteComment": "повучено мишљење", "activity-receivedDate": "измењен датум пријема на %s са %s", "activity-startDate": "измењен почетни датум на %s са %s", + "allboards.starred": "Starred", + "allboards.templates": "Предлошци", + "allboards.remaining": "Remaining", + "allboards.workspaces": "Workspaces", + "allboards.add-workspace": "Add Workspace", + "allboards.add-workspace-prompt": "Workspace name", + "allboards.add-subworkspace": "Add Subworkspace", + "allboards.add-subworkspace-prompt": "Subworkspace name", + "allboards.edit-workspace": "Edit workspace", + "allboards.edit-workspace-name": "Workspace name", + "allboards.edit-workspace-icon": "Workspace icon (markdown)", + "multi-selection-active": "Click checkboxes to select boards", "activity-dueDate": "измењен крајњи рок на %s са %s", "activity-endDate": "измењено време завршетка на %s са %s", "add-attachment": "Додај прилог", @@ -337,6 +349,7 @@ "copyManyCardsPopup-format": "[ {\"наслов\": \"Наслов прве картице\", \"опис\":\"Опис прве картице\"}, {\"наслов\":\"Наслов друге картице\",\"опис\":\"Опис друге картице\"},{\"наслов\":\"Наслов последње картице\",\"опис\":\"Опис последње картице\"} ]", "create": "Уведи", "createBoardPopup-title": "Уведи нову пословну књигу", + "createTemplateContainerPopup-title": "Додај сандук са предлошцима", "chooseBoardSourcePopup-title": "Уведи пословну књигу", "createLabelPopup-title": "Нова налепница", "createCustomField": "Направи сасвим ново поље", diff --git a/imports/i18n/data/sv.i18n.json b/imports/i18n/data/sv.i18n.json index 177a6c147..8b6fd6061 100644 --- a/imports/i18n/data/sv.i18n.json +++ b/imports/i18n/data/sv.i18n.json @@ -78,6 +78,18 @@ "activity-deleteComment": "raderade kommentar %s", "activity-receivedDate": "redigerade mottaget datum till %s av %s", "activity-startDate": "redigerade startdatum till %s av %s", + "allboards.starred": "Starred", + "allboards.templates": "Mallar", + "allboards.remaining": "Remaining", + "allboards.workspaces": "Workspaces", + "allboards.add-workspace": "Add Workspace", + "allboards.add-workspace-prompt": "Workspace name", + "allboards.add-subworkspace": "Add Subworkspace", + "allboards.add-subworkspace-prompt": "Subworkspace name", + "allboards.edit-workspace": "Edit workspace", + "allboards.edit-workspace-name": "Workspace name", + "allboards.edit-workspace-icon": "Workspace icon (markdown)", + "multi-selection-active": "Click checkboxes to select boards", "activity-dueDate": "redigerade förfallodag till %s av %s", "activity-endDate": "redigerade slutdatum till %s av %s", "add-attachment": "Lägg till bilaga", @@ -337,6 +349,7 @@ "copyManyCardsPopup-format": "[ {\"title\": \"Första kortets titel\", \"description\":\"Första kortets beskrivning\"}, {\"title\":\"Andra kortets titel\",\"description\":\"Andra kortets beskrivning\"},{\"title\":\"Sista kortets titel\",\"description\":\"Sista kortets beskrivning\"} ]", "create": "Skapa", "createBoardPopup-title": "Skapa tavla", + "createTemplateContainerPopup-title": "Lägg till från mall", "chooseBoardSourcePopup-title": "Importera tavla", "createLabelPopup-title": "Skapa etikett", "createCustomField": "Skapa fält", diff --git a/imports/i18n/data/sw.i18n.json b/imports/i18n/data/sw.i18n.json index c5ea981db..1e7bce7ae 100644 --- a/imports/i18n/data/sw.i18n.json +++ b/imports/i18n/data/sw.i18n.json @@ -78,6 +78,18 @@ "activity-deleteComment": "deleted comment %s", "activity-receivedDate": "edited received date to %s of %s", "activity-startDate": "edited start date to %s of %s", + "allboards.starred": "Starred", + "allboards.templates": "Templates", + "allboards.remaining": "Remaining", + "allboards.workspaces": "Workspaces", + "allboards.add-workspace": "Add Workspace", + "allboards.add-workspace-prompt": "Workspace name", + "allboards.add-subworkspace": "Add Subworkspace", + "allboards.add-subworkspace-prompt": "Subworkspace name", + "allboards.edit-workspace": "Edit workspace", + "allboards.edit-workspace-name": "Workspace name", + "allboards.edit-workspace-icon": "Workspace icon (markdown)", + "multi-selection-active": "Click checkboxes to select boards", "activity-dueDate": "edited due date to %s of %s", "activity-endDate": "edited end date to %s of %s", "add-attachment": "Add Attachment", @@ -337,6 +349,7 @@ "copyManyCardsPopup-format": "[ {\"title\": \"First card title\", \"description\":\"First card description\"}, {\"title\":\"Second card title\",\"description\":\"Second card description\"},{\"title\":\"Last card title\",\"description\":\"Last card description\"} ]", "create": "Create", "createBoardPopup-title": "Create Board", + "createTemplateContainerPopup-title": "Add Template Container", "chooseBoardSourcePopup-title": "Import board", "createLabelPopup-title": "Create Label", "createCustomField": "Create Field", diff --git a/imports/i18n/data/ta.i18n.json b/imports/i18n/data/ta.i18n.json index 2c7332730..fd711f921 100644 --- a/imports/i18n/data/ta.i18n.json +++ b/imports/i18n/data/ta.i18n.json @@ -78,6 +78,18 @@ "activity-deleteComment": "deleted comment %s", "activity-receivedDate": "edited received date to %s of %s", "activity-startDate": "edited start date to %s of %s", + "allboards.starred": "Starred", + "allboards.templates": "Templates", + "allboards.remaining": "Remaining", + "allboards.workspaces": "Workspaces", + "allboards.add-workspace": "Add Workspace", + "allboards.add-workspace-prompt": "Workspace name", + "allboards.add-subworkspace": "Add Subworkspace", + "allboards.add-subworkspace-prompt": "Subworkspace name", + "allboards.edit-workspace": "Edit workspace", + "allboards.edit-workspace-name": "Workspace name", + "allboards.edit-workspace-icon": "Workspace icon (markdown)", + "multi-selection-active": "Click checkboxes to select boards", "activity-dueDate": "edited due date to %s of %s", "activity-endDate": "edited end date to %s of %s", "add-attachment": "Add Attachment", @@ -337,6 +349,7 @@ "copyManyCardsPopup-format": "[ {\"title\": \"First card title\", \"description\":\"First card description\"}, {\"title\":\"Second card title\",\"description\":\"Second card description\"},{\"title\":\"Last card title\",\"description\":\"Last card description\"} ]", "create": "உருவாக்கு", "createBoardPopup-title": "Create Board", + "createTemplateContainerPopup-title": "Add Template Container", "chooseBoardSourcePopup-title": "Import board", "createLabelPopup-title": "Create Label", "createCustomField": "Create Field", diff --git a/imports/i18n/data/te-IN.i18n.json b/imports/i18n/data/te-IN.i18n.json index 958d8313a..acf3c6934 100644 --- a/imports/i18n/data/te-IN.i18n.json +++ b/imports/i18n/data/te-IN.i18n.json @@ -78,6 +78,18 @@ "activity-deleteComment": "deleted comment %s", "activity-receivedDate": "edited received date to %s of %s", "activity-startDate": "edited start date to %s of %s", + "allboards.starred": "Starred", + "allboards.templates": "Templates", + "allboards.remaining": "Remaining", + "allboards.workspaces": "Workspaces", + "allboards.add-workspace": "Add Workspace", + "allboards.add-workspace-prompt": "Workspace name", + "allboards.add-subworkspace": "Add Subworkspace", + "allboards.add-subworkspace-prompt": "Subworkspace name", + "allboards.edit-workspace": "Edit workspace", + "allboards.edit-workspace-name": "Workspace name", + "allboards.edit-workspace-icon": "Workspace icon (markdown)", + "multi-selection-active": "Click checkboxes to select boards", "activity-dueDate": "edited due date to %s of %s", "activity-endDate": "edited end date to %s of %s", "add-attachment": "Add Attachment", @@ -337,6 +349,7 @@ "copyManyCardsPopup-format": "[ {\"title\": \"First card title\", \"description\":\"First card description\"}, {\"title\":\"Second card title\",\"description\":\"Second card description\"},{\"title\":\"Last card title\",\"description\":\"Last card description\"} ]", "create": "Create", "createBoardPopup-title": "Create Board", + "createTemplateContainerPopup-title": "Add Template Container", "chooseBoardSourcePopup-title": "Import board", "createLabelPopup-title": "Create Label", "createCustomField": "Create Field", diff --git a/imports/i18n/data/th.i18n.json b/imports/i18n/data/th.i18n.json index 2b9c09b1c..0ec733bbe 100644 --- a/imports/i18n/data/th.i18n.json +++ b/imports/i18n/data/th.i18n.json @@ -78,6 +78,18 @@ "activity-deleteComment": "deleted comment %s", "activity-receivedDate": "edited received date to %s of %s", "activity-startDate": "edited start date to %s of %s", + "allboards.starred": "Starred", + "allboards.templates": "Templates", + "allboards.remaining": "Remaining", + "allboards.workspaces": "Workspaces", + "allboards.add-workspace": "Add Workspace", + "allboards.add-workspace-prompt": "Workspace name", + "allboards.add-subworkspace": "Add Subworkspace", + "allboards.add-subworkspace-prompt": "Subworkspace name", + "allboards.edit-workspace": "Edit workspace", + "allboards.edit-workspace-name": "Workspace name", + "allboards.edit-workspace-icon": "Workspace icon (markdown)", + "multi-selection-active": "Click checkboxes to select boards", "activity-dueDate": "edited due date to %s of %s", "activity-endDate": "edited end date to %s of %s", "add-attachment": "Add Attachment", @@ -337,6 +349,7 @@ "copyManyCardsPopup-format": "[ {\"title\": \"First card title\", \"description\":\"First card description\"}, {\"title\":\"Second card title\",\"description\":\"Second card description\"},{\"title\":\"Last card title\",\"description\":\"Last card description\"} ]", "create": "สร้าง", "createBoardPopup-title": "สร้างบอร์ด", + "createTemplateContainerPopup-title": "Add Template Container", "chooseBoardSourcePopup-title": "Import board", "createLabelPopup-title": "สร้างป้ายกำกับ", "createCustomField": "Create Field", diff --git a/imports/i18n/data/tk_TM.i18n.json b/imports/i18n/data/tk_TM.i18n.json index 958d8313a..acf3c6934 100644 --- a/imports/i18n/data/tk_TM.i18n.json +++ b/imports/i18n/data/tk_TM.i18n.json @@ -78,6 +78,18 @@ "activity-deleteComment": "deleted comment %s", "activity-receivedDate": "edited received date to %s of %s", "activity-startDate": "edited start date to %s of %s", + "allboards.starred": "Starred", + "allboards.templates": "Templates", + "allboards.remaining": "Remaining", + "allboards.workspaces": "Workspaces", + "allboards.add-workspace": "Add Workspace", + "allboards.add-workspace-prompt": "Workspace name", + "allboards.add-subworkspace": "Add Subworkspace", + "allboards.add-subworkspace-prompt": "Subworkspace name", + "allboards.edit-workspace": "Edit workspace", + "allboards.edit-workspace-name": "Workspace name", + "allboards.edit-workspace-icon": "Workspace icon (markdown)", + "multi-selection-active": "Click checkboxes to select boards", "activity-dueDate": "edited due date to %s of %s", "activity-endDate": "edited end date to %s of %s", "add-attachment": "Add Attachment", @@ -337,6 +349,7 @@ "copyManyCardsPopup-format": "[ {\"title\": \"First card title\", \"description\":\"First card description\"}, {\"title\":\"Second card title\",\"description\":\"Second card description\"},{\"title\":\"Last card title\",\"description\":\"Last card description\"} ]", "create": "Create", "createBoardPopup-title": "Create Board", + "createTemplateContainerPopup-title": "Add Template Container", "chooseBoardSourcePopup-title": "Import board", "createLabelPopup-title": "Create Label", "createCustomField": "Create Field", diff --git a/imports/i18n/data/tlh.i18n.json b/imports/i18n/data/tlh.i18n.json index 958d8313a..acf3c6934 100644 --- a/imports/i18n/data/tlh.i18n.json +++ b/imports/i18n/data/tlh.i18n.json @@ -78,6 +78,18 @@ "activity-deleteComment": "deleted comment %s", "activity-receivedDate": "edited received date to %s of %s", "activity-startDate": "edited start date to %s of %s", + "allboards.starred": "Starred", + "allboards.templates": "Templates", + "allboards.remaining": "Remaining", + "allboards.workspaces": "Workspaces", + "allboards.add-workspace": "Add Workspace", + "allboards.add-workspace-prompt": "Workspace name", + "allboards.add-subworkspace": "Add Subworkspace", + "allboards.add-subworkspace-prompt": "Subworkspace name", + "allboards.edit-workspace": "Edit workspace", + "allboards.edit-workspace-name": "Workspace name", + "allboards.edit-workspace-icon": "Workspace icon (markdown)", + "multi-selection-active": "Click checkboxes to select boards", "activity-dueDate": "edited due date to %s of %s", "activity-endDate": "edited end date to %s of %s", "add-attachment": "Add Attachment", @@ -337,6 +349,7 @@ "copyManyCardsPopup-format": "[ {\"title\": \"First card title\", \"description\":\"First card description\"}, {\"title\":\"Second card title\",\"description\":\"Second card description\"},{\"title\":\"Last card title\",\"description\":\"Last card description\"} ]", "create": "Create", "createBoardPopup-title": "Create Board", + "createTemplateContainerPopup-title": "Add Template Container", "chooseBoardSourcePopup-title": "Import board", "createLabelPopup-title": "Create Label", "createCustomField": "Create Field", diff --git a/imports/i18n/data/tr.i18n.json b/imports/i18n/data/tr.i18n.json index b4690871e..d413e318b 100644 --- a/imports/i18n/data/tr.i18n.json +++ b/imports/i18n/data/tr.i18n.json @@ -78,6 +78,18 @@ "activity-deleteComment": "%s yorum silindi", "activity-receivedDate": "alma tarihi için düzenlendi%s", "activity-startDate": "başlangıç tarihi %s, %s olarak düzenlendi", + "allboards.starred": "Starred", + "allboards.templates": "Şablonlar", + "allboards.remaining": "Remaining", + "allboards.workspaces": "Workspaces", + "allboards.add-workspace": "Add Workspace", + "allboards.add-workspace-prompt": "Workspace name", + "allboards.add-subworkspace": "Add Subworkspace", + "allboards.add-subworkspace-prompt": "Subworkspace name", + "allboards.edit-workspace": "Edit workspace", + "allboards.edit-workspace-name": "Workspace name", + "allboards.edit-workspace-icon": "Workspace icon (markdown)", + "multi-selection-active": "Click checkboxes to select boards", "activity-dueDate": "bitiş tarihi %s / %s olarak düzenlendi", "activity-endDate": "son bitiş tarihi %s, %s olarak düzenlendi", "add-attachment": "Ek Ekle", @@ -337,6 +349,7 @@ "copyManyCardsPopup-format": "[ {\"title\": \"İlk kart başlığı\", \"description\":\"İlk kart açıklaması\"}, {\"title\":\"İkinci kart başlığı\",\"description\":\"İkinci kart açıklaması\"},{\"title\":\"Son kart başlığı\",\"description\":\"Son kart açıklaması\"} ]", "create": "Oluştur", "createBoardPopup-title": "Pano Oluşturma", + "createTemplateContainerPopup-title": "Şablon Konteyner Ekle", "chooseBoardSourcePopup-title": "Panoyu içe aktar", "createLabelPopup-title": "Etiket Oluşturma", "createCustomField": "Alanı yarat", diff --git a/imports/i18n/data/ug.i18n.json b/imports/i18n/data/ug.i18n.json index 958d8313a..acf3c6934 100644 --- a/imports/i18n/data/ug.i18n.json +++ b/imports/i18n/data/ug.i18n.json @@ -78,6 +78,18 @@ "activity-deleteComment": "deleted comment %s", "activity-receivedDate": "edited received date to %s of %s", "activity-startDate": "edited start date to %s of %s", + "allboards.starred": "Starred", + "allboards.templates": "Templates", + "allboards.remaining": "Remaining", + "allboards.workspaces": "Workspaces", + "allboards.add-workspace": "Add Workspace", + "allboards.add-workspace-prompt": "Workspace name", + "allboards.add-subworkspace": "Add Subworkspace", + "allboards.add-subworkspace-prompt": "Subworkspace name", + "allboards.edit-workspace": "Edit workspace", + "allboards.edit-workspace-name": "Workspace name", + "allboards.edit-workspace-icon": "Workspace icon (markdown)", + "multi-selection-active": "Click checkboxes to select boards", "activity-dueDate": "edited due date to %s of %s", "activity-endDate": "edited end date to %s of %s", "add-attachment": "Add Attachment", @@ -337,6 +349,7 @@ "copyManyCardsPopup-format": "[ {\"title\": \"First card title\", \"description\":\"First card description\"}, {\"title\":\"Second card title\",\"description\":\"Second card description\"},{\"title\":\"Last card title\",\"description\":\"Last card description\"} ]", "create": "Create", "createBoardPopup-title": "Create Board", + "createTemplateContainerPopup-title": "Add Template Container", "chooseBoardSourcePopup-title": "Import board", "createLabelPopup-title": "Create Label", "createCustomField": "Create Field", diff --git a/imports/i18n/data/uk-UA.i18n.json b/imports/i18n/data/uk-UA.i18n.json index e4a704602..c1c9dac68 100644 --- a/imports/i18n/data/uk-UA.i18n.json +++ b/imports/i18n/data/uk-UA.i18n.json @@ -78,6 +78,18 @@ "activity-deleteComment": "видалено коментар %s", "activity-receivedDate": "edited received date to %s of %s", "activity-startDate": "edited start date to %s of %s", + "allboards.starred": "Starred", + "allboards.templates": "Шаблони", + "allboards.remaining": "Remaining", + "allboards.workspaces": "Workspaces", + "allboards.add-workspace": "Add Workspace", + "allboards.add-workspace-prompt": "Workspace name", + "allboards.add-subworkspace": "Add Subworkspace", + "allboards.add-subworkspace-prompt": "Subworkspace name", + "allboards.edit-workspace": "Edit workspace", + "allboards.edit-workspace-name": "Workspace name", + "allboards.edit-workspace-icon": "Workspace icon (markdown)", + "multi-selection-active": "Click checkboxes to select boards", "activity-dueDate": "edited due date to %s of %s", "activity-endDate": "edited end date to %s of %s", "add-attachment": "Додати вкладення", @@ -337,6 +349,7 @@ "copyManyCardsPopup-format": "[{\"title\": \"Перший заголовок картки\", \"description\":\"Перший опис картки\"}, {\"title\":\"Другий заголовок картки\",\"description\":\"Другий опис картки\"},{\"title\":\"Останній заголовок картки\",\"description\":\"Останній опис картки\"} ]", "create": "Створити", "createBoardPopup-title": "Створити дошку", + "createTemplateContainerPopup-title": "Додати шаблон контейнера", "chooseBoardSourcePopup-title": "Імпортувати дошку", "createLabelPopup-title": "Створити мітку", "createCustomField": "Створити поле", diff --git a/imports/i18n/data/uk.i18n.json b/imports/i18n/data/uk.i18n.json index 1efcb3f10..03e0ac75a 100644 --- a/imports/i18n/data/uk.i18n.json +++ b/imports/i18n/data/uk.i18n.json @@ -78,6 +78,18 @@ "activity-deleteComment": "видалено коментар %s", "activity-receivedDate": "edited received date to %s of %s", "activity-startDate": "edited start date to %s of %s", + "allboards.starred": "Starred", + "allboards.templates": "Шаблони", + "allboards.remaining": "Remaining", + "allboards.workspaces": "Workspaces", + "allboards.add-workspace": "Add Workspace", + "allboards.add-workspace-prompt": "Workspace name", + "allboards.add-subworkspace": "Add Subworkspace", + "allboards.add-subworkspace-prompt": "Subworkspace name", + "allboards.edit-workspace": "Edit workspace", + "allboards.edit-workspace-name": "Workspace name", + "allboards.edit-workspace-icon": "Workspace icon (markdown)", + "multi-selection-active": "Click checkboxes to select boards", "activity-dueDate": "edited due date to %s of %s", "activity-endDate": "edited end date to %s of %s", "add-attachment": "Додати вкладення", @@ -337,6 +349,7 @@ "copyManyCardsPopup-format": "[{\"title\": \"Перший заголовок картки\", \"description\":\"Перший опис картки\"}, {\"title\":\"Другий заголовок картки\",\"description\":\"Другий опис картки\"},{\"title\":\"Останній заголовок картки\",\"description\":\"Останній опис картки\"} ]", "create": "Створити", "createBoardPopup-title": "Створити дошку", + "createTemplateContainerPopup-title": "Додати шаблон контейнера", "chooseBoardSourcePopup-title": "Імпортувати дошку", "createLabelPopup-title": "Створити мітку", "createCustomField": "Створити поле", diff --git a/imports/i18n/data/uz-AR.i18n.json b/imports/i18n/data/uz-AR.i18n.json index 958d8313a..acf3c6934 100644 --- a/imports/i18n/data/uz-AR.i18n.json +++ b/imports/i18n/data/uz-AR.i18n.json @@ -78,6 +78,18 @@ "activity-deleteComment": "deleted comment %s", "activity-receivedDate": "edited received date to %s of %s", "activity-startDate": "edited start date to %s of %s", + "allboards.starred": "Starred", + "allboards.templates": "Templates", + "allboards.remaining": "Remaining", + "allboards.workspaces": "Workspaces", + "allboards.add-workspace": "Add Workspace", + "allboards.add-workspace-prompt": "Workspace name", + "allboards.add-subworkspace": "Add Subworkspace", + "allboards.add-subworkspace-prompt": "Subworkspace name", + "allboards.edit-workspace": "Edit workspace", + "allboards.edit-workspace-name": "Workspace name", + "allboards.edit-workspace-icon": "Workspace icon (markdown)", + "multi-selection-active": "Click checkboxes to select boards", "activity-dueDate": "edited due date to %s of %s", "activity-endDate": "edited end date to %s of %s", "add-attachment": "Add Attachment", @@ -337,6 +349,7 @@ "copyManyCardsPopup-format": "[ {\"title\": \"First card title\", \"description\":\"First card description\"}, {\"title\":\"Second card title\",\"description\":\"Second card description\"},{\"title\":\"Last card title\",\"description\":\"Last card description\"} ]", "create": "Create", "createBoardPopup-title": "Create Board", + "createTemplateContainerPopup-title": "Add Template Container", "chooseBoardSourcePopup-title": "Import board", "createLabelPopup-title": "Create Label", "createCustomField": "Create Field", diff --git a/imports/i18n/data/uz-LA.i18n.json b/imports/i18n/data/uz-LA.i18n.json index 958d8313a..acf3c6934 100644 --- a/imports/i18n/data/uz-LA.i18n.json +++ b/imports/i18n/data/uz-LA.i18n.json @@ -78,6 +78,18 @@ "activity-deleteComment": "deleted comment %s", "activity-receivedDate": "edited received date to %s of %s", "activity-startDate": "edited start date to %s of %s", + "allboards.starred": "Starred", + "allboards.templates": "Templates", + "allboards.remaining": "Remaining", + "allboards.workspaces": "Workspaces", + "allboards.add-workspace": "Add Workspace", + "allboards.add-workspace-prompt": "Workspace name", + "allboards.add-subworkspace": "Add Subworkspace", + "allboards.add-subworkspace-prompt": "Subworkspace name", + "allboards.edit-workspace": "Edit workspace", + "allboards.edit-workspace-name": "Workspace name", + "allboards.edit-workspace-icon": "Workspace icon (markdown)", + "multi-selection-active": "Click checkboxes to select boards", "activity-dueDate": "edited due date to %s of %s", "activity-endDate": "edited end date to %s of %s", "add-attachment": "Add Attachment", @@ -337,6 +349,7 @@ "copyManyCardsPopup-format": "[ {\"title\": \"First card title\", \"description\":\"First card description\"}, {\"title\":\"Second card title\",\"description\":\"Second card description\"},{\"title\":\"Last card title\",\"description\":\"Last card description\"} ]", "create": "Create", "createBoardPopup-title": "Create Board", + "createTemplateContainerPopup-title": "Add Template Container", "chooseBoardSourcePopup-title": "Import board", "createLabelPopup-title": "Create Label", "createCustomField": "Create Field", diff --git a/imports/i18n/data/uz-UZ.i18n.json b/imports/i18n/data/uz-UZ.i18n.json index 958d8313a..acf3c6934 100644 --- a/imports/i18n/data/uz-UZ.i18n.json +++ b/imports/i18n/data/uz-UZ.i18n.json @@ -78,6 +78,18 @@ "activity-deleteComment": "deleted comment %s", "activity-receivedDate": "edited received date to %s of %s", "activity-startDate": "edited start date to %s of %s", + "allboards.starred": "Starred", + "allboards.templates": "Templates", + "allboards.remaining": "Remaining", + "allboards.workspaces": "Workspaces", + "allboards.add-workspace": "Add Workspace", + "allboards.add-workspace-prompt": "Workspace name", + "allboards.add-subworkspace": "Add Subworkspace", + "allboards.add-subworkspace-prompt": "Subworkspace name", + "allboards.edit-workspace": "Edit workspace", + "allboards.edit-workspace-name": "Workspace name", + "allboards.edit-workspace-icon": "Workspace icon (markdown)", + "multi-selection-active": "Click checkboxes to select boards", "activity-dueDate": "edited due date to %s of %s", "activity-endDate": "edited end date to %s of %s", "add-attachment": "Add Attachment", @@ -337,6 +349,7 @@ "copyManyCardsPopup-format": "[ {\"title\": \"First card title\", \"description\":\"First card description\"}, {\"title\":\"Second card title\",\"description\":\"Second card description\"},{\"title\":\"Last card title\",\"description\":\"Last card description\"} ]", "create": "Create", "createBoardPopup-title": "Create Board", + "createTemplateContainerPopup-title": "Add Template Container", "chooseBoardSourcePopup-title": "Import board", "createLabelPopup-title": "Create Label", "createCustomField": "Create Field", diff --git a/imports/i18n/data/uz.i18n.json b/imports/i18n/data/uz.i18n.json index 958d8313a..acf3c6934 100644 --- a/imports/i18n/data/uz.i18n.json +++ b/imports/i18n/data/uz.i18n.json @@ -78,6 +78,18 @@ "activity-deleteComment": "deleted comment %s", "activity-receivedDate": "edited received date to %s of %s", "activity-startDate": "edited start date to %s of %s", + "allboards.starred": "Starred", + "allboards.templates": "Templates", + "allboards.remaining": "Remaining", + "allboards.workspaces": "Workspaces", + "allboards.add-workspace": "Add Workspace", + "allboards.add-workspace-prompt": "Workspace name", + "allboards.add-subworkspace": "Add Subworkspace", + "allboards.add-subworkspace-prompt": "Subworkspace name", + "allboards.edit-workspace": "Edit workspace", + "allboards.edit-workspace-name": "Workspace name", + "allboards.edit-workspace-icon": "Workspace icon (markdown)", + "multi-selection-active": "Click checkboxes to select boards", "activity-dueDate": "edited due date to %s of %s", "activity-endDate": "edited end date to %s of %s", "add-attachment": "Add Attachment", @@ -337,6 +349,7 @@ "copyManyCardsPopup-format": "[ {\"title\": \"First card title\", \"description\":\"First card description\"}, {\"title\":\"Second card title\",\"description\":\"Second card description\"},{\"title\":\"Last card title\",\"description\":\"Last card description\"} ]", "create": "Create", "createBoardPopup-title": "Create Board", + "createTemplateContainerPopup-title": "Add Template Container", "chooseBoardSourcePopup-title": "Import board", "createLabelPopup-title": "Create Label", "createCustomField": "Create Field", diff --git a/imports/i18n/data/ve-CC.i18n.json b/imports/i18n/data/ve-CC.i18n.json index 958d8313a..acf3c6934 100644 --- a/imports/i18n/data/ve-CC.i18n.json +++ b/imports/i18n/data/ve-CC.i18n.json @@ -78,6 +78,18 @@ "activity-deleteComment": "deleted comment %s", "activity-receivedDate": "edited received date to %s of %s", "activity-startDate": "edited start date to %s of %s", + "allboards.starred": "Starred", + "allboards.templates": "Templates", + "allboards.remaining": "Remaining", + "allboards.workspaces": "Workspaces", + "allboards.add-workspace": "Add Workspace", + "allboards.add-workspace-prompt": "Workspace name", + "allboards.add-subworkspace": "Add Subworkspace", + "allboards.add-subworkspace-prompt": "Subworkspace name", + "allboards.edit-workspace": "Edit workspace", + "allboards.edit-workspace-name": "Workspace name", + "allboards.edit-workspace-icon": "Workspace icon (markdown)", + "multi-selection-active": "Click checkboxes to select boards", "activity-dueDate": "edited due date to %s of %s", "activity-endDate": "edited end date to %s of %s", "add-attachment": "Add Attachment", @@ -337,6 +349,7 @@ "copyManyCardsPopup-format": "[ {\"title\": \"First card title\", \"description\":\"First card description\"}, {\"title\":\"Second card title\",\"description\":\"Second card description\"},{\"title\":\"Last card title\",\"description\":\"Last card description\"} ]", "create": "Create", "createBoardPopup-title": "Create Board", + "createTemplateContainerPopup-title": "Add Template Container", "chooseBoardSourcePopup-title": "Import board", "createLabelPopup-title": "Create Label", "createCustomField": "Create Field", diff --git a/imports/i18n/data/ve-PP.i18n.json b/imports/i18n/data/ve-PP.i18n.json index 958d8313a..acf3c6934 100644 --- a/imports/i18n/data/ve-PP.i18n.json +++ b/imports/i18n/data/ve-PP.i18n.json @@ -78,6 +78,18 @@ "activity-deleteComment": "deleted comment %s", "activity-receivedDate": "edited received date to %s of %s", "activity-startDate": "edited start date to %s of %s", + "allboards.starred": "Starred", + "allboards.templates": "Templates", + "allboards.remaining": "Remaining", + "allboards.workspaces": "Workspaces", + "allboards.add-workspace": "Add Workspace", + "allboards.add-workspace-prompt": "Workspace name", + "allboards.add-subworkspace": "Add Subworkspace", + "allboards.add-subworkspace-prompt": "Subworkspace name", + "allboards.edit-workspace": "Edit workspace", + "allboards.edit-workspace-name": "Workspace name", + "allboards.edit-workspace-icon": "Workspace icon (markdown)", + "multi-selection-active": "Click checkboxes to select boards", "activity-dueDate": "edited due date to %s of %s", "activity-endDate": "edited end date to %s of %s", "add-attachment": "Add Attachment", @@ -337,6 +349,7 @@ "copyManyCardsPopup-format": "[ {\"title\": \"First card title\", \"description\":\"First card description\"}, {\"title\":\"Second card title\",\"description\":\"Second card description\"},{\"title\":\"Last card title\",\"description\":\"Last card description\"} ]", "create": "Create", "createBoardPopup-title": "Create Board", + "createTemplateContainerPopup-title": "Add Template Container", "chooseBoardSourcePopup-title": "Import board", "createLabelPopup-title": "Create Label", "createCustomField": "Create Field", diff --git a/imports/i18n/data/ve.i18n.json b/imports/i18n/data/ve.i18n.json index 958d8313a..acf3c6934 100644 --- a/imports/i18n/data/ve.i18n.json +++ b/imports/i18n/data/ve.i18n.json @@ -78,6 +78,18 @@ "activity-deleteComment": "deleted comment %s", "activity-receivedDate": "edited received date to %s of %s", "activity-startDate": "edited start date to %s of %s", + "allboards.starred": "Starred", + "allboards.templates": "Templates", + "allboards.remaining": "Remaining", + "allboards.workspaces": "Workspaces", + "allboards.add-workspace": "Add Workspace", + "allboards.add-workspace-prompt": "Workspace name", + "allboards.add-subworkspace": "Add Subworkspace", + "allboards.add-subworkspace-prompt": "Subworkspace name", + "allboards.edit-workspace": "Edit workspace", + "allboards.edit-workspace-name": "Workspace name", + "allboards.edit-workspace-icon": "Workspace icon (markdown)", + "multi-selection-active": "Click checkboxes to select boards", "activity-dueDate": "edited due date to %s of %s", "activity-endDate": "edited end date to %s of %s", "add-attachment": "Add Attachment", @@ -337,6 +349,7 @@ "copyManyCardsPopup-format": "[ {\"title\": \"First card title\", \"description\":\"First card description\"}, {\"title\":\"Second card title\",\"description\":\"Second card description\"},{\"title\":\"Last card title\",\"description\":\"Last card description\"} ]", "create": "Create", "createBoardPopup-title": "Create Board", + "createTemplateContainerPopup-title": "Add Template Container", "chooseBoardSourcePopup-title": "Import board", "createLabelPopup-title": "Create Label", "createCustomField": "Create Field", diff --git a/imports/i18n/data/vi-VN.i18n.json b/imports/i18n/data/vi-VN.i18n.json index 14e4ab2c2..feb1c09e8 100644 --- a/imports/i18n/data/vi-VN.i18n.json +++ b/imports/i18n/data/vi-VN.i18n.json @@ -78,6 +78,18 @@ "activity-deleteComment": "deleted comment %s", "activity-receivedDate": "edited received date to %s of %s", "activity-startDate": "edited start date to %s of %s", + "allboards.starred": "Starred", + "allboards.templates": "Templates", + "allboards.remaining": "Remaining", + "allboards.workspaces": "Workspaces", + "allboards.add-workspace": "Add Workspace", + "allboards.add-workspace-prompt": "Workspace name", + "allboards.add-subworkspace": "Add Subworkspace", + "allboards.add-subworkspace-prompt": "Subworkspace name", + "allboards.edit-workspace": "Edit workspace", + "allboards.edit-workspace-name": "Workspace name", + "allboards.edit-workspace-icon": "Workspace icon (markdown)", + "multi-selection-active": "Click checkboxes to select boards", "activity-dueDate": "edited due date to %s of %s", "activity-endDate": "edited end date to %s of %s", "add-attachment": "Add Attachment", @@ -337,6 +349,7 @@ "copyManyCardsPopup-format": "[ {\"title\": \"First card title\", \"description\":\"First card description\"}, {\"title\":\"Second card title\",\"description\":\"Second card description\"},{\"title\":\"Last card title\",\"description\":\"Last card description\"} ]", "create": "Create", "createBoardPopup-title": "Create Board", + "createTemplateContainerPopup-title": "Add Template Container", "chooseBoardSourcePopup-title": "Import board", "createLabelPopup-title": "Create Label", "createCustomField": "Create Field", diff --git a/imports/i18n/data/vi.i18n.json b/imports/i18n/data/vi.i18n.json index d7b031a95..90628f3aa 100644 --- a/imports/i18n/data/vi.i18n.json +++ b/imports/i18n/data/vi.i18n.json @@ -78,6 +78,18 @@ "activity-deleteComment": "đã xoá lời bình %s", "activity-receivedDate": "đã sửa ngày nhận đến %s của %s", "activity-startDate": "đã sửa ngày bắt đầu thành %s của %s", + "allboards.starred": "Starred", + "allboards.templates": "Các mẫu", + "allboards.remaining": "Remaining", + "allboards.workspaces": "Workspaces", + "allboards.add-workspace": "Add Workspace", + "allboards.add-workspace-prompt": "Workspace name", + "allboards.add-subworkspace": "Add Subworkspace", + "allboards.add-subworkspace-prompt": "Subworkspace name", + "allboards.edit-workspace": "Edit workspace", + "allboards.edit-workspace-name": "Workspace name", + "allboards.edit-workspace-icon": "Workspace icon (markdown)", + "multi-selection-active": "Click checkboxes to select boards", "activity-dueDate": "đã sửa ngày hết hạn thành %s của %s", "activity-endDate": "đã sửa ngày kết thúc thành %s của %s", "add-attachment": "Thêm Bản Đính Kèm", @@ -337,6 +349,7 @@ "copyManyCardsPopup-format": "[ {\"title\": \"Tiêu đề thẻ đầu tiên\", \"description\":\"Mô tả thẻ đầu tiên\"}, {\"title\":\"Tiêu đề thẻ thứ hai\",\"description\":\"Mô tả thẻ thứ hai\"},{\"title\":\"Tiêu đề thẻ cuối cùng\",\"description\":\"Mô tả thẻ cuối cùng\"} ]", "create": "Tạo", "createBoardPopup-title": "Tạo Bảng", + "createTemplateContainerPopup-title": "Add Template Container", "chooseBoardSourcePopup-title": "Nhập bảng", "createLabelPopup-title": "Tạo nhãn", "createCustomField": "Tạo Trường", diff --git a/imports/i18n/data/vl-SS.i18n.json b/imports/i18n/data/vl-SS.i18n.json index 958d8313a..acf3c6934 100644 --- a/imports/i18n/data/vl-SS.i18n.json +++ b/imports/i18n/data/vl-SS.i18n.json @@ -78,6 +78,18 @@ "activity-deleteComment": "deleted comment %s", "activity-receivedDate": "edited received date to %s of %s", "activity-startDate": "edited start date to %s of %s", + "allboards.starred": "Starred", + "allboards.templates": "Templates", + "allboards.remaining": "Remaining", + "allboards.workspaces": "Workspaces", + "allboards.add-workspace": "Add Workspace", + "allboards.add-workspace-prompt": "Workspace name", + "allboards.add-subworkspace": "Add Subworkspace", + "allboards.add-subworkspace-prompt": "Subworkspace name", + "allboards.edit-workspace": "Edit workspace", + "allboards.edit-workspace-name": "Workspace name", + "allboards.edit-workspace-icon": "Workspace icon (markdown)", + "multi-selection-active": "Click checkboxes to select boards", "activity-dueDate": "edited due date to %s of %s", "activity-endDate": "edited end date to %s of %s", "add-attachment": "Add Attachment", @@ -337,6 +349,7 @@ "copyManyCardsPopup-format": "[ {\"title\": \"First card title\", \"description\":\"First card description\"}, {\"title\":\"Second card title\",\"description\":\"Second card description\"},{\"title\":\"Last card title\",\"description\":\"Last card description\"} ]", "create": "Create", "createBoardPopup-title": "Create Board", + "createTemplateContainerPopup-title": "Add Template Container", "chooseBoardSourcePopup-title": "Import board", "createLabelPopup-title": "Create Label", "createCustomField": "Create Field", diff --git a/imports/i18n/data/vo.i18n.json b/imports/i18n/data/vo.i18n.json index 958d8313a..acf3c6934 100644 --- a/imports/i18n/data/vo.i18n.json +++ b/imports/i18n/data/vo.i18n.json @@ -78,6 +78,18 @@ "activity-deleteComment": "deleted comment %s", "activity-receivedDate": "edited received date to %s of %s", "activity-startDate": "edited start date to %s of %s", + "allboards.starred": "Starred", + "allboards.templates": "Templates", + "allboards.remaining": "Remaining", + "allboards.workspaces": "Workspaces", + "allboards.add-workspace": "Add Workspace", + "allboards.add-workspace-prompt": "Workspace name", + "allboards.add-subworkspace": "Add Subworkspace", + "allboards.add-subworkspace-prompt": "Subworkspace name", + "allboards.edit-workspace": "Edit workspace", + "allboards.edit-workspace-name": "Workspace name", + "allboards.edit-workspace-icon": "Workspace icon (markdown)", + "multi-selection-active": "Click checkboxes to select boards", "activity-dueDate": "edited due date to %s of %s", "activity-endDate": "edited end date to %s of %s", "add-attachment": "Add Attachment", @@ -337,6 +349,7 @@ "copyManyCardsPopup-format": "[ {\"title\": \"First card title\", \"description\":\"First card description\"}, {\"title\":\"Second card title\",\"description\":\"Second card description\"},{\"title\":\"Last card title\",\"description\":\"Last card description\"} ]", "create": "Create", "createBoardPopup-title": "Create Board", + "createTemplateContainerPopup-title": "Add Template Container", "chooseBoardSourcePopup-title": "Import board", "createLabelPopup-title": "Create Label", "createCustomField": "Create Field", diff --git a/imports/i18n/data/wa-RR.i18n.json b/imports/i18n/data/wa-RR.i18n.json index 958d8313a..acf3c6934 100644 --- a/imports/i18n/data/wa-RR.i18n.json +++ b/imports/i18n/data/wa-RR.i18n.json @@ -78,6 +78,18 @@ "activity-deleteComment": "deleted comment %s", "activity-receivedDate": "edited received date to %s of %s", "activity-startDate": "edited start date to %s of %s", + "allboards.starred": "Starred", + "allboards.templates": "Templates", + "allboards.remaining": "Remaining", + "allboards.workspaces": "Workspaces", + "allboards.add-workspace": "Add Workspace", + "allboards.add-workspace-prompt": "Workspace name", + "allboards.add-subworkspace": "Add Subworkspace", + "allboards.add-subworkspace-prompt": "Subworkspace name", + "allboards.edit-workspace": "Edit workspace", + "allboards.edit-workspace-name": "Workspace name", + "allboards.edit-workspace-icon": "Workspace icon (markdown)", + "multi-selection-active": "Click checkboxes to select boards", "activity-dueDate": "edited due date to %s of %s", "activity-endDate": "edited end date to %s of %s", "add-attachment": "Add Attachment", @@ -337,6 +349,7 @@ "copyManyCardsPopup-format": "[ {\"title\": \"First card title\", \"description\":\"First card description\"}, {\"title\":\"Second card title\",\"description\":\"Second card description\"},{\"title\":\"Last card title\",\"description\":\"Last card description\"} ]", "create": "Create", "createBoardPopup-title": "Create Board", + "createTemplateContainerPopup-title": "Add Template Container", "chooseBoardSourcePopup-title": "Import board", "createLabelPopup-title": "Create Label", "createCustomField": "Create Field", diff --git a/imports/i18n/data/wa.i18n.json b/imports/i18n/data/wa.i18n.json index 958d8313a..acf3c6934 100644 --- a/imports/i18n/data/wa.i18n.json +++ b/imports/i18n/data/wa.i18n.json @@ -78,6 +78,18 @@ "activity-deleteComment": "deleted comment %s", "activity-receivedDate": "edited received date to %s of %s", "activity-startDate": "edited start date to %s of %s", + "allboards.starred": "Starred", + "allboards.templates": "Templates", + "allboards.remaining": "Remaining", + "allboards.workspaces": "Workspaces", + "allboards.add-workspace": "Add Workspace", + "allboards.add-workspace-prompt": "Workspace name", + "allboards.add-subworkspace": "Add Subworkspace", + "allboards.add-subworkspace-prompt": "Subworkspace name", + "allboards.edit-workspace": "Edit workspace", + "allboards.edit-workspace-name": "Workspace name", + "allboards.edit-workspace-icon": "Workspace icon (markdown)", + "multi-selection-active": "Click checkboxes to select boards", "activity-dueDate": "edited due date to %s of %s", "activity-endDate": "edited end date to %s of %s", "add-attachment": "Add Attachment", @@ -337,6 +349,7 @@ "copyManyCardsPopup-format": "[ {\"title\": \"First card title\", \"description\":\"First card description\"}, {\"title\":\"Second card title\",\"description\":\"Second card description\"},{\"title\":\"Last card title\",\"description\":\"Last card description\"} ]", "create": "Create", "createBoardPopup-title": "Create Board", + "createTemplateContainerPopup-title": "Add Template Container", "chooseBoardSourcePopup-title": "Import board", "createLabelPopup-title": "Create Label", "createCustomField": "Create Field", diff --git a/imports/i18n/data/wo.i18n.json b/imports/i18n/data/wo.i18n.json index 958d8313a..acf3c6934 100644 --- a/imports/i18n/data/wo.i18n.json +++ b/imports/i18n/data/wo.i18n.json @@ -78,6 +78,18 @@ "activity-deleteComment": "deleted comment %s", "activity-receivedDate": "edited received date to %s of %s", "activity-startDate": "edited start date to %s of %s", + "allboards.starred": "Starred", + "allboards.templates": "Templates", + "allboards.remaining": "Remaining", + "allboards.workspaces": "Workspaces", + "allboards.add-workspace": "Add Workspace", + "allboards.add-workspace-prompt": "Workspace name", + "allboards.add-subworkspace": "Add Subworkspace", + "allboards.add-subworkspace-prompt": "Subworkspace name", + "allboards.edit-workspace": "Edit workspace", + "allboards.edit-workspace-name": "Workspace name", + "allboards.edit-workspace-icon": "Workspace icon (markdown)", + "multi-selection-active": "Click checkboxes to select boards", "activity-dueDate": "edited due date to %s of %s", "activity-endDate": "edited end date to %s of %s", "add-attachment": "Add Attachment", @@ -337,6 +349,7 @@ "copyManyCardsPopup-format": "[ {\"title\": \"First card title\", \"description\":\"First card description\"}, {\"title\":\"Second card title\",\"description\":\"Second card description\"},{\"title\":\"Last card title\",\"description\":\"Last card description\"} ]", "create": "Create", "createBoardPopup-title": "Create Board", + "createTemplateContainerPopup-title": "Add Template Container", "chooseBoardSourcePopup-title": "Import board", "createLabelPopup-title": "Create Label", "createCustomField": "Create Field", diff --git a/imports/i18n/data/wuu-Hans.i18n.json b/imports/i18n/data/wuu-Hans.i18n.json index 958d8313a..acf3c6934 100644 --- a/imports/i18n/data/wuu-Hans.i18n.json +++ b/imports/i18n/data/wuu-Hans.i18n.json @@ -78,6 +78,18 @@ "activity-deleteComment": "deleted comment %s", "activity-receivedDate": "edited received date to %s of %s", "activity-startDate": "edited start date to %s of %s", + "allboards.starred": "Starred", + "allboards.templates": "Templates", + "allboards.remaining": "Remaining", + "allboards.workspaces": "Workspaces", + "allboards.add-workspace": "Add Workspace", + "allboards.add-workspace-prompt": "Workspace name", + "allboards.add-subworkspace": "Add Subworkspace", + "allboards.add-subworkspace-prompt": "Subworkspace name", + "allboards.edit-workspace": "Edit workspace", + "allboards.edit-workspace-name": "Workspace name", + "allboards.edit-workspace-icon": "Workspace icon (markdown)", + "multi-selection-active": "Click checkboxes to select boards", "activity-dueDate": "edited due date to %s of %s", "activity-endDate": "edited end date to %s of %s", "add-attachment": "Add Attachment", @@ -337,6 +349,7 @@ "copyManyCardsPopup-format": "[ {\"title\": \"First card title\", \"description\":\"First card description\"}, {\"title\":\"Second card title\",\"description\":\"Second card description\"},{\"title\":\"Last card title\",\"description\":\"Last card description\"} ]", "create": "Create", "createBoardPopup-title": "Create Board", + "createTemplateContainerPopup-title": "Add Template Container", "chooseBoardSourcePopup-title": "Import board", "createLabelPopup-title": "Create Label", "createCustomField": "Create Field", diff --git a/imports/i18n/data/xh.i18n.json b/imports/i18n/data/xh.i18n.json index 958d8313a..acf3c6934 100644 --- a/imports/i18n/data/xh.i18n.json +++ b/imports/i18n/data/xh.i18n.json @@ -78,6 +78,18 @@ "activity-deleteComment": "deleted comment %s", "activity-receivedDate": "edited received date to %s of %s", "activity-startDate": "edited start date to %s of %s", + "allboards.starred": "Starred", + "allboards.templates": "Templates", + "allboards.remaining": "Remaining", + "allboards.workspaces": "Workspaces", + "allboards.add-workspace": "Add Workspace", + "allboards.add-workspace-prompt": "Workspace name", + "allboards.add-subworkspace": "Add Subworkspace", + "allboards.add-subworkspace-prompt": "Subworkspace name", + "allboards.edit-workspace": "Edit workspace", + "allboards.edit-workspace-name": "Workspace name", + "allboards.edit-workspace-icon": "Workspace icon (markdown)", + "multi-selection-active": "Click checkboxes to select boards", "activity-dueDate": "edited due date to %s of %s", "activity-endDate": "edited end date to %s of %s", "add-attachment": "Add Attachment", @@ -337,6 +349,7 @@ "copyManyCardsPopup-format": "[ {\"title\": \"First card title\", \"description\":\"First card description\"}, {\"title\":\"Second card title\",\"description\":\"Second card description\"},{\"title\":\"Last card title\",\"description\":\"Last card description\"} ]", "create": "Create", "createBoardPopup-title": "Create Board", + "createTemplateContainerPopup-title": "Add Template Container", "chooseBoardSourcePopup-title": "Import board", "createLabelPopup-title": "Create Label", "createCustomField": "Create Field", diff --git a/imports/i18n/data/yi.i18n.json b/imports/i18n/data/yi.i18n.json index 958d8313a..acf3c6934 100644 --- a/imports/i18n/data/yi.i18n.json +++ b/imports/i18n/data/yi.i18n.json @@ -78,6 +78,18 @@ "activity-deleteComment": "deleted comment %s", "activity-receivedDate": "edited received date to %s of %s", "activity-startDate": "edited start date to %s of %s", + "allboards.starred": "Starred", + "allboards.templates": "Templates", + "allboards.remaining": "Remaining", + "allboards.workspaces": "Workspaces", + "allboards.add-workspace": "Add Workspace", + "allboards.add-workspace-prompt": "Workspace name", + "allboards.add-subworkspace": "Add Subworkspace", + "allboards.add-subworkspace-prompt": "Subworkspace name", + "allboards.edit-workspace": "Edit workspace", + "allboards.edit-workspace-name": "Workspace name", + "allboards.edit-workspace-icon": "Workspace icon (markdown)", + "multi-selection-active": "Click checkboxes to select boards", "activity-dueDate": "edited due date to %s of %s", "activity-endDate": "edited end date to %s of %s", "add-attachment": "Add Attachment", @@ -337,6 +349,7 @@ "copyManyCardsPopup-format": "[ {\"title\": \"First card title\", \"description\":\"First card description\"}, {\"title\":\"Second card title\",\"description\":\"Second card description\"},{\"title\":\"Last card title\",\"description\":\"Last card description\"} ]", "create": "Create", "createBoardPopup-title": "Create Board", + "createTemplateContainerPopup-title": "Add Template Container", "chooseBoardSourcePopup-title": "Import board", "createLabelPopup-title": "Create Label", "createCustomField": "Create Field", diff --git a/imports/i18n/data/yo.i18n.json b/imports/i18n/data/yo.i18n.json index 958d8313a..acf3c6934 100644 --- a/imports/i18n/data/yo.i18n.json +++ b/imports/i18n/data/yo.i18n.json @@ -78,6 +78,18 @@ "activity-deleteComment": "deleted comment %s", "activity-receivedDate": "edited received date to %s of %s", "activity-startDate": "edited start date to %s of %s", + "allboards.starred": "Starred", + "allboards.templates": "Templates", + "allboards.remaining": "Remaining", + "allboards.workspaces": "Workspaces", + "allboards.add-workspace": "Add Workspace", + "allboards.add-workspace-prompt": "Workspace name", + "allboards.add-subworkspace": "Add Subworkspace", + "allboards.add-subworkspace-prompt": "Subworkspace name", + "allboards.edit-workspace": "Edit workspace", + "allboards.edit-workspace-name": "Workspace name", + "allboards.edit-workspace-icon": "Workspace icon (markdown)", + "multi-selection-active": "Click checkboxes to select boards", "activity-dueDate": "edited due date to %s of %s", "activity-endDate": "edited end date to %s of %s", "add-attachment": "Add Attachment", @@ -337,6 +349,7 @@ "copyManyCardsPopup-format": "[ {\"title\": \"First card title\", \"description\":\"First card description\"}, {\"title\":\"Second card title\",\"description\":\"Second card description\"},{\"title\":\"Last card title\",\"description\":\"Last card description\"} ]", "create": "Create", "createBoardPopup-title": "Create Board", + "createTemplateContainerPopup-title": "Add Template Container", "chooseBoardSourcePopup-title": "Import board", "createLabelPopup-title": "Create Label", "createCustomField": "Create Field", diff --git a/imports/i18n/data/yue_CN.i18n.json b/imports/i18n/data/yue_CN.i18n.json index 958d8313a..acf3c6934 100644 --- a/imports/i18n/data/yue_CN.i18n.json +++ b/imports/i18n/data/yue_CN.i18n.json @@ -78,6 +78,18 @@ "activity-deleteComment": "deleted comment %s", "activity-receivedDate": "edited received date to %s of %s", "activity-startDate": "edited start date to %s of %s", + "allboards.starred": "Starred", + "allboards.templates": "Templates", + "allboards.remaining": "Remaining", + "allboards.workspaces": "Workspaces", + "allboards.add-workspace": "Add Workspace", + "allboards.add-workspace-prompt": "Workspace name", + "allboards.add-subworkspace": "Add Subworkspace", + "allboards.add-subworkspace-prompt": "Subworkspace name", + "allboards.edit-workspace": "Edit workspace", + "allboards.edit-workspace-name": "Workspace name", + "allboards.edit-workspace-icon": "Workspace icon (markdown)", + "multi-selection-active": "Click checkboxes to select boards", "activity-dueDate": "edited due date to %s of %s", "activity-endDate": "edited end date to %s of %s", "add-attachment": "Add Attachment", @@ -337,6 +349,7 @@ "copyManyCardsPopup-format": "[ {\"title\": \"First card title\", \"description\":\"First card description\"}, {\"title\":\"Second card title\",\"description\":\"Second card description\"},{\"title\":\"Last card title\",\"description\":\"Last card description\"} ]", "create": "Create", "createBoardPopup-title": "Create Board", + "createTemplateContainerPopup-title": "Add Template Container", "chooseBoardSourcePopup-title": "Import board", "createLabelPopup-title": "Create Label", "createCustomField": "Create Field", diff --git a/imports/i18n/data/zgh.i18n.json b/imports/i18n/data/zgh.i18n.json index 958d8313a..acf3c6934 100644 --- a/imports/i18n/data/zgh.i18n.json +++ b/imports/i18n/data/zgh.i18n.json @@ -78,6 +78,18 @@ "activity-deleteComment": "deleted comment %s", "activity-receivedDate": "edited received date to %s of %s", "activity-startDate": "edited start date to %s of %s", + "allboards.starred": "Starred", + "allboards.templates": "Templates", + "allboards.remaining": "Remaining", + "allboards.workspaces": "Workspaces", + "allboards.add-workspace": "Add Workspace", + "allboards.add-workspace-prompt": "Workspace name", + "allboards.add-subworkspace": "Add Subworkspace", + "allboards.add-subworkspace-prompt": "Subworkspace name", + "allboards.edit-workspace": "Edit workspace", + "allboards.edit-workspace-name": "Workspace name", + "allboards.edit-workspace-icon": "Workspace icon (markdown)", + "multi-selection-active": "Click checkboxes to select boards", "activity-dueDate": "edited due date to %s of %s", "activity-endDate": "edited end date to %s of %s", "add-attachment": "Add Attachment", @@ -337,6 +349,7 @@ "copyManyCardsPopup-format": "[ {\"title\": \"First card title\", \"description\":\"First card description\"}, {\"title\":\"Second card title\",\"description\":\"Second card description\"},{\"title\":\"Last card title\",\"description\":\"Last card description\"} ]", "create": "Create", "createBoardPopup-title": "Create Board", + "createTemplateContainerPopup-title": "Add Template Container", "chooseBoardSourcePopup-title": "Import board", "createLabelPopup-title": "Create Label", "createCustomField": "Create Field", diff --git a/imports/i18n/data/zh-CN.i18n.json b/imports/i18n/data/zh-CN.i18n.json index 76590fc80..517728311 100644 --- a/imports/i18n/data/zh-CN.i18n.json +++ b/imports/i18n/data/zh-CN.i18n.json @@ -78,6 +78,18 @@ "activity-deleteComment": "评论已删除", "activity-receivedDate": "已将接收日期从 %s 修改为 %s", "activity-startDate": "已将开始日期从 %s 修改为 %s", + "allboards.starred": "Starred", + "allboards.templates": "模板", + "allboards.remaining": "Remaining", + "allboards.workspaces": "Workspaces", + "allboards.add-workspace": "Add Workspace", + "allboards.add-workspace-prompt": "Workspace name", + "allboards.add-subworkspace": "Add Subworkspace", + "allboards.add-subworkspace-prompt": "Subworkspace name", + "allboards.edit-workspace": "Edit workspace", + "allboards.edit-workspace-name": "Workspace name", + "allboards.edit-workspace-icon": "Workspace icon (markdown)", + "multi-selection-active": "Click checkboxes to select boards", "activity-dueDate": "已将到期日期从 %s 修改为 %s", "activity-endDate": "已将结束日期从 %s 修改为 %s", "add-attachment": "添加附件", @@ -337,6 +349,7 @@ "copyManyCardsPopup-format": "[ {\"title\": \"第一个卡片的标题\", \"description\":\"第一个卡片的描述\"}, {\"title\":\"第二个卡片的标题\",\"description\":\"第二个卡片的描述\"},{\"title\":\"最后一个卡片的标题\",\"description\":\"最后一个卡片的描述\"} ]", "create": "创建", "createBoardPopup-title": "创建看板", + "createTemplateContainerPopup-title": "新增模板容器", "chooseBoardSourcePopup-title": "导入看板", "createLabelPopup-title": "创建标签", "createCustomField": "创建字段", diff --git a/imports/i18n/data/zh-GB.i18n.json b/imports/i18n/data/zh-GB.i18n.json index 3f96ee90b..e75ee1b35 100644 --- a/imports/i18n/data/zh-GB.i18n.json +++ b/imports/i18n/data/zh-GB.i18n.json @@ -78,6 +78,18 @@ "activity-deleteComment": "deleted comment %s", "activity-receivedDate": "edited received date to %s of %s", "activity-startDate": "edited start date to %s of %s", + "allboards.starred": "Starred", + "allboards.templates": "Templates", + "allboards.remaining": "Remaining", + "allboards.workspaces": "Workspaces", + "allboards.add-workspace": "Add Workspace", + "allboards.add-workspace-prompt": "Workspace name", + "allboards.add-subworkspace": "Add Subworkspace", + "allboards.add-subworkspace-prompt": "Subworkspace name", + "allboards.edit-workspace": "Edit workspace", + "allboards.edit-workspace-name": "Workspace name", + "allboards.edit-workspace-icon": "Workspace icon (markdown)", + "multi-selection-active": "Click checkboxes to select boards", "activity-dueDate": "edited due date to %s of %s", "activity-endDate": "edited end date to %s of %s", "add-attachment": "Add Attachment", @@ -337,6 +349,7 @@ "copyManyCardsPopup-format": "[ {\"title\": \"First card title\", \"description\":\"First card description\"}, {\"title\":\"Second card title\",\"description\":\"Second card description\"},{\"title\":\"Last card title\",\"description\":\"Last card description\"} ]", "create": "Create", "createBoardPopup-title": "Create Board", + "createTemplateContainerPopup-title": "Add Template Container", "chooseBoardSourcePopup-title": "Import board", "createLabelPopup-title": "Create Label", "createCustomField": "Create Field", diff --git a/imports/i18n/data/zh-HK.i18n.json b/imports/i18n/data/zh-HK.i18n.json index 577b25b58..b5506b271 100644 --- a/imports/i18n/data/zh-HK.i18n.json +++ b/imports/i18n/data/zh-HK.i18n.json @@ -78,6 +78,18 @@ "activity-deleteComment": "deleted comment %s", "activity-receivedDate": "edited received date to %s of %s", "activity-startDate": "edited start date to %s of %s", + "allboards.starred": "Starred", + "allboards.templates": "Templates", + "allboards.remaining": "Remaining", + "allboards.workspaces": "Workspaces", + "allboards.add-workspace": "Add Workspace", + "allboards.add-workspace-prompt": "Workspace name", + "allboards.add-subworkspace": "Add Subworkspace", + "allboards.add-subworkspace-prompt": "Subworkspace name", + "allboards.edit-workspace": "Edit workspace", + "allboards.edit-workspace-name": "Workspace name", + "allboards.edit-workspace-icon": "Workspace icon (markdown)", + "multi-selection-active": "Click checkboxes to select boards", "activity-dueDate": "edited due date to %s of %s", "activity-endDate": "edited end date to %s of %s", "add-attachment": "Add Attachment", @@ -337,6 +349,7 @@ "copyManyCardsPopup-format": "[ {\"title\": \"First card title\", \"description\":\"First card description\"}, {\"title\":\"Second card title\",\"description\":\"Second card description\"},{\"title\":\"Last card title\",\"description\":\"Last card description\"} ]", "create": "Create", "createBoardPopup-title": "Create Board", + "createTemplateContainerPopup-title": "Add Template Container", "chooseBoardSourcePopup-title": "Import board", "createLabelPopup-title": "Create Label", "createCustomField": "Create Field", diff --git a/imports/i18n/data/zh-Hans.i18n.json b/imports/i18n/data/zh-Hans.i18n.json index 3a9ff35d7..76f2b352c 100644 --- a/imports/i18n/data/zh-Hans.i18n.json +++ b/imports/i18n/data/zh-Hans.i18n.json @@ -78,6 +78,18 @@ "activity-deleteComment": "deleted comment %s", "activity-receivedDate": "edited received date to %s of %s", "activity-startDate": "edited start date to %s of %s", + "allboards.starred": "Starred", + "allboards.templates": "Templates", + "allboards.remaining": "Remaining", + "allboards.workspaces": "Workspaces", + "allboards.add-workspace": "Add Workspace", + "allboards.add-workspace-prompt": "Workspace name", + "allboards.add-subworkspace": "Add Subworkspace", + "allboards.add-subworkspace-prompt": "Subworkspace name", + "allboards.edit-workspace": "Edit workspace", + "allboards.edit-workspace-name": "Workspace name", + "allboards.edit-workspace-icon": "Workspace icon (markdown)", + "multi-selection-active": "Click checkboxes to select boards", "activity-dueDate": "edited due date to %s of %s", "activity-endDate": "edited end date to %s of %s", "add-attachment": "Add Attachment", @@ -337,6 +349,7 @@ "copyManyCardsPopup-format": "[ {\"title\": \"First card title\", \"description\":\"First card description\"}, {\"title\":\"Second card title\",\"description\":\"Second card description\"},{\"title\":\"Last card title\",\"description\":\"Last card description\"} ]", "create": "创建", "createBoardPopup-title": "创建看板", + "createTemplateContainerPopup-title": "Add Template Container", "chooseBoardSourcePopup-title": "导入看板", "createLabelPopup-title": "Create Label", "createCustomField": "Create Field", diff --git a/imports/i18n/data/zh-Hant.i18n.json b/imports/i18n/data/zh-Hant.i18n.json index 4f00f5998..28b61bb3b 100644 --- a/imports/i18n/data/zh-Hant.i18n.json +++ b/imports/i18n/data/zh-Hant.i18n.json @@ -78,6 +78,18 @@ "activity-deleteComment": "deleted comment %s", "activity-receivedDate": "edited received date to %s of %s", "activity-startDate": "edited start date to %s of %s", + "allboards.starred": "Starred", + "allboards.templates": "Templates", + "allboards.remaining": "Remaining", + "allboards.workspaces": "Workspaces", + "allboards.add-workspace": "Add Workspace", + "allboards.add-workspace-prompt": "Workspace name", + "allboards.add-subworkspace": "Add Subworkspace", + "allboards.add-subworkspace-prompt": "Subworkspace name", + "allboards.edit-workspace": "Edit workspace", + "allboards.edit-workspace-name": "Workspace name", + "allboards.edit-workspace-icon": "Workspace icon (markdown)", + "multi-selection-active": "Click checkboxes to select boards", "activity-dueDate": "edited due date to %s of %s", "activity-endDate": "edited end date to %s of %s", "add-attachment": "Add Attachment", @@ -337,6 +349,7 @@ "copyManyCardsPopup-format": "[ {\"title\": \"First card title\", \"description\":\"First card description\"}, {\"title\":\"Second card title\",\"description\":\"Second card description\"},{\"title\":\"Last card title\",\"description\":\"Last card description\"} ]", "create": "Create", "createBoardPopup-title": "Create Board", + "createTemplateContainerPopup-title": "Add Template Container", "chooseBoardSourcePopup-title": "Import board", "createLabelPopup-title": "Create Label", "createCustomField": "Create Field", diff --git a/imports/i18n/data/zh-TW.i18n.json b/imports/i18n/data/zh-TW.i18n.json index a8a292b44..021053b09 100644 --- a/imports/i18n/data/zh-TW.i18n.json +++ b/imports/i18n/data/zh-TW.i18n.json @@ -78,6 +78,18 @@ "activity-deleteComment": "評論已刪除", "activity-receivedDate": "已編輯收到日期為 %s %s", "activity-startDate": "已編輯起始日期為 %s %s", + "allboards.starred": "Starred", + "allboards.templates": "範本", + "allboards.remaining": "Remaining", + "allboards.workspaces": "Workspaces", + "allboards.add-workspace": "Add Workspace", + "allboards.add-workspace-prompt": "Workspace name", + "allboards.add-subworkspace": "Add Subworkspace", + "allboards.add-subworkspace-prompt": "Subworkspace name", + "allboards.edit-workspace": "Edit workspace", + "allboards.edit-workspace-name": "Workspace name", + "allboards.edit-workspace-icon": "Workspace icon (markdown)", + "multi-selection-active": "Click checkboxes to select boards", "activity-dueDate": "已編輯截止日期為 %s %s", "activity-endDate": "已編輯結束日期為 %s %s", "add-attachment": "新增附件", @@ -337,6 +349,7 @@ "copyManyCardsPopup-format": "[ {\"title\": \"第一個卡片標題\", \"description\":\"第一個卡片描述\"}, {\"title\":\"第二個卡片標題\",\"description\":\"第二個卡片描述\"},{\"title\":\"最後一個卡片標題\",\"description\":\"最後一個卡片描述\"} ]", "create": "建立", "createBoardPopup-title": "建立看板", + "createTemplateContainerPopup-title": "新增範本容器", "chooseBoardSourcePopup-title": "匯入看板", "createLabelPopup-title": "新增標籤", "createCustomField": "建立欄位", diff --git a/imports/i18n/data/zh.i18n.json b/imports/i18n/data/zh.i18n.json index 237c5e595..e7bfa720a 100644 --- a/imports/i18n/data/zh.i18n.json +++ b/imports/i18n/data/zh.i18n.json @@ -78,6 +78,18 @@ "activity-deleteComment": "deleted comment %s", "activity-receivedDate": "edited received date to %s of %s", "activity-startDate": "edited start date to %s of %s", + "allboards.starred": "Starred", + "allboards.templates": "Templates", + "allboards.remaining": "Remaining", + "allboards.workspaces": "Workspaces", + "allboards.add-workspace": "Add Workspace", + "allboards.add-workspace-prompt": "Workspace name", + "allboards.add-subworkspace": "Add Subworkspace", + "allboards.add-subworkspace-prompt": "Subworkspace name", + "allboards.edit-workspace": "Edit workspace", + "allboards.edit-workspace-name": "Workspace name", + "allboards.edit-workspace-icon": "Workspace icon (markdown)", + "multi-selection-active": "Click checkboxes to select boards", "activity-dueDate": "edited due date to %s of %s", "activity-endDate": "edited end date to %s of %s", "add-attachment": "Add Attachment", @@ -337,6 +349,7 @@ "copyManyCardsPopup-format": "[ {\"title\": \"First card title\", \"description\":\"First card description\"}, {\"title\":\"Second card title\",\"description\":\"Second card description\"},{\"title\":\"Last card title\",\"description\":\"Last card description\"} ]", "create": "Create", "createBoardPopup-title": "Create Board", + "createTemplateContainerPopup-title": "Add Template Container", "chooseBoardSourcePopup-title": "Import board", "createLabelPopup-title": "Create Label", "createCustomField": "Create Field", diff --git a/imports/i18n/data/zu-ZA.i18n.json b/imports/i18n/data/zu-ZA.i18n.json index 958d8313a..acf3c6934 100644 --- a/imports/i18n/data/zu-ZA.i18n.json +++ b/imports/i18n/data/zu-ZA.i18n.json @@ -78,6 +78,18 @@ "activity-deleteComment": "deleted comment %s", "activity-receivedDate": "edited received date to %s of %s", "activity-startDate": "edited start date to %s of %s", + "allboards.starred": "Starred", + "allboards.templates": "Templates", + "allboards.remaining": "Remaining", + "allboards.workspaces": "Workspaces", + "allboards.add-workspace": "Add Workspace", + "allboards.add-workspace-prompt": "Workspace name", + "allboards.add-subworkspace": "Add Subworkspace", + "allboards.add-subworkspace-prompt": "Subworkspace name", + "allboards.edit-workspace": "Edit workspace", + "allboards.edit-workspace-name": "Workspace name", + "allboards.edit-workspace-icon": "Workspace icon (markdown)", + "multi-selection-active": "Click checkboxes to select boards", "activity-dueDate": "edited due date to %s of %s", "activity-endDate": "edited end date to %s of %s", "add-attachment": "Add Attachment", @@ -337,6 +349,7 @@ "copyManyCardsPopup-format": "[ {\"title\": \"First card title\", \"description\":\"First card description\"}, {\"title\":\"Second card title\",\"description\":\"Second card description\"},{\"title\":\"Last card title\",\"description\":\"Last card description\"} ]", "create": "Create", "createBoardPopup-title": "Create Board", + "createTemplateContainerPopup-title": "Add Template Container", "chooseBoardSourcePopup-title": "Import board", "createLabelPopup-title": "Create Label", "createCustomField": "Create Field", diff --git a/imports/i18n/data/zu.i18n.json b/imports/i18n/data/zu.i18n.json index 958d8313a..acf3c6934 100644 --- a/imports/i18n/data/zu.i18n.json +++ b/imports/i18n/data/zu.i18n.json @@ -78,6 +78,18 @@ "activity-deleteComment": "deleted comment %s", "activity-receivedDate": "edited received date to %s of %s", "activity-startDate": "edited start date to %s of %s", + "allboards.starred": "Starred", + "allboards.templates": "Templates", + "allboards.remaining": "Remaining", + "allboards.workspaces": "Workspaces", + "allboards.add-workspace": "Add Workspace", + "allboards.add-workspace-prompt": "Workspace name", + "allboards.add-subworkspace": "Add Subworkspace", + "allboards.add-subworkspace-prompt": "Subworkspace name", + "allboards.edit-workspace": "Edit workspace", + "allboards.edit-workspace-name": "Workspace name", + "allboards.edit-workspace-icon": "Workspace icon (markdown)", + "multi-selection-active": "Click checkboxes to select boards", "activity-dueDate": "edited due date to %s of %s", "activity-endDate": "edited end date to %s of %s", "add-attachment": "Add Attachment", @@ -337,6 +349,7 @@ "copyManyCardsPopup-format": "[ {\"title\": \"First card title\", \"description\":\"First card description\"}, {\"title\":\"Second card title\",\"description\":\"Second card description\"},{\"title\":\"Last card title\",\"description\":\"Last card description\"} ]", "create": "Create", "createBoardPopup-title": "Create Board", + "createTemplateContainerPopup-title": "Add Template Container", "chooseBoardSourcePopup-title": "Import board", "createLabelPopup-title": "Create Label", "createCustomField": "Create Field", diff --git a/models/users.js b/models/users.js index bebe9b633..3885638d1 100644 --- a/models/users.js +++ b/models/users.js @@ -1,4 +1,5 @@ import { ReactiveCache, ReactiveMiniMongoIndex } from '/imports/reactiveCache'; +import { Random } from 'meteor/random'; import { SyncedCron } from 'meteor/percolate:synced-cron'; import { TAPi18n } from '/imports/i18n'; import ImpersonatedUsers from './impersonatedUsers'; @@ -200,6 +201,29 @@ Users.attachSchema( type: String, optional: true, }, + 'profile.boardWorkspacesTree': { + /** + * Per-user spaces tree for All Boards page + */ + type: Array, + optional: true, + }, + 'profile.boardWorkspacesTree.$': { + /** + * Space node: { id: String, name: String, children: Array } + */ + type: Object, + blackbox: true, + optional: true, + }, + 'profile.boardWorkspaceAssignments': { + /** + * Per-user map of boardId -> spaceId + */ + type: Object, + optional: true, + blackbox: true, + }, 'profile.invitedBoards': { /** * board IDs the user has been invited to @@ -1668,6 +1692,73 @@ Meteor.methods({ const user = ReactiveCache.getCurrentUser(); user.toggleDesktopHandles(user.hasShowDesktopDragHandles()); }, + // Spaces: create a new space under parentId (or root when null) + createWorkspace({ parentId = null, name }) { + check(name, String); + if (!this.userId) throw new Meteor.Error('not-logged-in'); + const user = Users.findOne(this.userId) || {}; + const tree = (user.profile && user.profile.boardWorkspacesTree) ? EJSON.clone(user.profile.boardWorkspacesTree) : []; + + const newNode = { id: Random.id(), name, children: [] }; + + if (!parentId) { + tree.push(newNode); + } else { + const insertInto = (nodes) => { + for (let n of nodes) { + if (n.id === parentId) { + n.children = n.children || []; + n.children.push(newNode); + return true; + } + if (n.children && n.children.length) { + if (insertInto(n.children)) return true; + } + } + return false; + }; + insertInto(tree); + } + + Users.update(this.userId, { $set: { 'profile.boardWorkspacesTree': tree } }); + return newNode; + }, + // Spaces: set entire tree (used for drag-drop reordering) + setWorkspacesTree(newTree) { + check(newTree, Array); + if (!this.userId) throw new Meteor.Error('not-logged-in'); + Users.update(this.userId, { $set: { 'profile.boardWorkspacesTree': newTree } }); + return true; + }, + // Assign a board to a space + assignBoardToWorkspace(boardId, spaceId) { + check(boardId, String); + check(spaceId, String); + if (!this.userId) throw new Meteor.Error('not-logged-in'); + + const user = Users.findOne(this.userId); + const assignments = user.profile?.boardWorkspaceAssignments || {}; + assignments[boardId] = spaceId; + + Users.update(this.userId, { + $set: { 'profile.boardWorkspaceAssignments': assignments } + }); + return true; + }, + // Remove a board assignment (moves it back to Remaining) + unassignBoardFromWorkspace(boardId) { + check(boardId, String); + if (!this.userId) throw new Meteor.Error('not-logged-in'); + + const user = Users.findOne(this.userId); + const assignments = user.profile?.boardWorkspaceAssignments || {}; + delete assignments[boardId]; + + Users.update(this.userId, { + $set: { 'profile.boardWorkspaceAssignments': assignments } + }); + return true; + }, toggleHideCheckedItems() { const user = ReactiveCache.getCurrentUser(); user.toggleHideCheckedItems(); From 42594abe4e9d4f68b5ab9fd963014c112b908009 Mon Sep 17 00:00:00 2001 From: Lauri Ojansivu Date: Thu, 6 Nov 2025 00:30:08 +0200 Subject: [PATCH 49/84] Updated ChangeLog. --- CHANGELOG.md | 7 ++++++- 1 file changed, 6 insertions(+), 1 deletion(-) diff --git a/CHANGELOG.md b/CHANGELOG.md index e10e7c894..ab35b52a3 100644 --- a/CHANGELOG.md +++ b/CHANGELOG.md @@ -24,7 +24,12 @@ Those are fixed at WeKan 8.07 where database directory is back to /var/snap/weka # Upcoming WeKan ® release -This release fixes the following bugs: +This release adds the following new feature: + +- [Feature: Workspaces, at All Boards page](https://github.com/wekan/wekan/commit/0afbdc95b49537e06b4f9cf98f51a669ef249384). + Thanks to xet7. + +and fixes the following bugs: - [Fix 8.16: Switching Board View fails with 403 error](https://github.com/wekan/wekan/commit/550d87ac6cb3ec946600616485afdbd242983ab4). Thanks to xet7. From e5e711c938edcca23c974c3eec97296898bcf24e Mon Sep 17 00:00:00 2001 From: Lauri Ojansivu Date: Thu, 6 Nov 2025 00:35:49 +0200 Subject: [PATCH 50/84] Fix Card emoji issues. Thanks to xet7 ! Fixes #5995 --- client/components/cards/cardDetails.jade | 4 ++-- 1 file changed, 2 insertions(+), 2 deletions(-) diff --git a/client/components/cards/cardDetails.jade b/client/components/cards/cardDetails.jade index 90b3fef31..cf810b38a 100644 --- a/client/components/cards/cardDetails.jade +++ b/client/components/cards/cardDetails.jade @@ -191,7 +191,7 @@ template(name="cardDetails") if currentBoard.allowsMembers .card-details-item.card-details-item-members h3.card-details-item-title - | 👤s + | 👥 | {{_ 'members'}} each userId in getMembers +userAvatar(userId=userId cardId=_id) @@ -242,7 +242,7 @@ template(name="cardDetails") if currentBoard.allowsAssignedBy .card-details-item.card-details-item-name h3.card-details-item-title - | 👤-plus + | ✍️ | {{_ 'assigned-by'}} if canModifyCard unless currentUser.isWorker From c58ab5b07d0ceae741afb192e14507f6fb9b2cb2 Mon Sep 17 00:00:00 2001 From: Lauri Ojansivu Date: Thu, 6 Nov 2025 00:37:42 +0200 Subject: [PATCH 51/84] Updated ChangeLog. --- CHANGELOG.md | 2 ++ 1 file changed, 2 insertions(+) diff --git a/CHANGELOG.md b/CHANGELOG.md index ab35b52a3..d991cba0b 100644 --- a/CHANGELOG.md +++ b/CHANGELOG.md @@ -43,6 +43,8 @@ and fixes the following bugs: Thanks to xet7. - [Fix star board](https://github.com/wekan/wekan/commit/8711b476be30496b96b845529b5717bb6e685c27). Thanks to xet7. +- [Fix Card emoji issues](https://github.com/wekan/wekan/commit/e5e711c938edcca23c974c3eec97296898bcf24e). + Thanks to xet7. Thanks to above GitHub users for their contributions and translators for their translations. From 20af0a2ef55b11e7205845859ee92a929616ce91 Mon Sep 17 00:00:00 2001 From: Lauri Ojansivu Date: Thu, 6 Nov 2025 01:04:20 +0200 Subject: [PATCH 52/84] Try to fix Edit Custom Fields button not working. Removed duplicate option from Boards Settings. Thanks to xet7 ! Fixes #5988 --- client/components/sidebar/sidebar.jade | 8 ++++---- client/components/sidebar/sidebarCustomFields.jade | 4 ++++ 2 files changed, 8 insertions(+), 4 deletions(-) diff --git a/client/components/sidebar/sidebar.jade b/client/components/sidebar/sidebar.jade index 4a216c336..e5e90e1a5 100644 --- a/client/components/sidebar/sidebar.jade +++ b/client/components/sidebar/sidebar.jade @@ -631,10 +631,10 @@ template(name="boardMenuPopup") if currentUser.isBoardAdmin hr ul.pop-over-list - li - a.js-delete-duplicate-lists - | 🗑️ - | {{_ 'delete-duplicate-lists'}} + // li + // a.js-delete-duplicate-lists + // | 🗑️ + // | {{_ 'delete-duplicate-lists'}} li a.js-archive-board | ➡️📦 diff --git a/client/components/sidebar/sidebarCustomFields.jade b/client/components/sidebar/sidebarCustomFields.jade index 1cc270681..0d16559f8 100644 --- a/client/components/sidebar/sidebarCustomFields.jade +++ b/client/components/sidebar/sidebarCustomFields.jade @@ -95,3 +95,7 @@ template(name="createCustomFieldPopup") template(name="deleteCustomFieldPopup") p {{_ "custom-field-delete-pop"}} button.js-confirm.negate.full(type="submit") {{_ 'delete'}} + +// Reuse the create form for editing to satisfy popup template lookup +template(name="editCustomFieldPopup") + +Template.dynamic(template="createCustomFieldPopup") From b02af27ac381d1423e19060660eddf170dd2f82c Mon Sep 17 00:00:00 2001 From: Lauri Ojansivu Date: Thu, 6 Nov 2025 01:06:19 +0200 Subject: [PATCH 53/84] Updated ChangeLog. --- CHANGELOG.md | 2 ++ 1 file changed, 2 insertions(+) diff --git a/CHANGELOG.md b/CHANGELOG.md index d991cba0b..6d553985c 100644 --- a/CHANGELOG.md +++ b/CHANGELOG.md @@ -45,6 +45,8 @@ and fixes the following bugs: Thanks to xet7. - [Fix Card emoji issues](https://github.com/wekan/wekan/commit/e5e711c938edcca23c974c3eec97296898bcf24e). Thanks to xet7. +- [Try to fix Edit Custom Fields button not working. Removed duplicate option from Boards Settings](https://github.com/wekan/wekan/commit/20af0a2ef55b11e7205845859ee92a929616ce91). + Thanks to xet7. Thanks to above GitHub users for their contributions and translators for their translations. From 581733d605b7e0494e72229c45947cff134f6dd6 Mon Sep 17 00:00:00 2001 From: Lauri Ojansivu Date: Thu, 6 Nov 2025 02:32:34 +0200 Subject: [PATCH 54/84] Fix Regression - Show calendar popup at set due date. Thanks to xet7 ! Fixes #5978 --- client/components/cards/cardDate.jade | 21 +++ client/components/cards/cardDate.js | 11 ++ client/components/forms/datepicker.jade | 2 +- client/components/main/popup.css | 5 +- client/lib/datepicker.js | 164 ++++++------------------ 5 files changed, 77 insertions(+), 126 deletions(-) diff --git a/client/components/cards/cardDate.jade b/client/components/cards/cardDate.jade index e387112bc..c8c6f45a3 100644 --- a/client/components/cards/cardDate.jade +++ b/client/components/cards/cardDate.jade @@ -97,6 +97,12 @@ template(name="minicardCustomFieldDate") template(name="editCardReceivedDatePopup") form.edit-card-received-date .datepicker + // Date input field (existing) + // Insert calendar selector right after date input + .calendar-selector + label(for="calendar-received") 🗓️ + input#calendar-received.js-calendar-date(type="date") + // Time input field (if present) .clear-date a.js-clear-date {{_ 'clear'}} .datepicker-actions @@ -106,6 +112,11 @@ template(name="editCardReceivedDatePopup") template(name="editCardStartDatePopup") form.edit-card-start-date .datepicker + // Date input field (existing) + .calendar-selector + label(for="calendar-start") 🗓️ + input#calendar-start.js-calendar-date(type="date") + // Time input field (if present) .clear-date a.js-clear-date {{_ 'clear'}} .datepicker-actions @@ -115,6 +126,11 @@ template(name="editCardStartDatePopup") template(name="editCardDueDatePopup") form.edit-card-due-date .datepicker + // Date input field (existing) + .calendar-selector + label(for="calendar-due") 🗓️ + input#calendar-due.js-calendar-date(type="date") + // Time input field (if present) .clear-date a.js-clear-date {{_ 'clear'}} .datepicker-actions @@ -124,6 +140,11 @@ template(name="editCardDueDatePopup") template(name="editCardEndDatePopup") form.edit-card-end-date .datepicker + // Date input field (existing) + .calendar-selector + label(for="calendar-end") 🗓️ + input#calendar-end.js-calendar-date(type="date") + // Time input field (if present) .clear-date a.js-clear-date {{_ 'clear'}} .datepicker-actions diff --git a/client/components/cards/cardDate.js b/client/components/cards/cardDate.js index bf40538db..a470bbba4 100644 --- a/client/components/cards/cardDate.js +++ b/client/components/cards/cardDate.js @@ -50,6 +50,17 @@ import { onRendered() { super.onRendered(); // DatePicker base class handles initialization with native HTML inputs + const self = this; + this.$('.js-calendar-date').on('change', function(evt) { + const currentUser = ReactiveCache.getCurrentUser && ReactiveCache.getCurrentUser(); + const dateFormat = currentUser ? currentUser.getDateFormat() : 'YYYY-MM-DD'; + const value = evt.target.value; + if (value) { + // Format date according to user preference + const formatted = formatDateByUserPreference(new Date(value), dateFormat, true); + self._storeDate(new Date(value)); + } + }); } _storeDate(date) { diff --git a/client/components/forms/datepicker.jade b/client/components/forms/datepicker.jade index 1fbdb2383..c8fb0524a 100644 --- a/client/components/forms/datepicker.jade +++ b/client/components/forms/datepicker.jade @@ -4,7 +4,7 @@ template(name="datepicker") .fields .left label(for="date") {{_ 'date'}} - input.js-date-field#date(type="text" name="date" value=showDate autofocus placeholder=dateFormat) + input.js-date-field#date(type="date" name="date" value=showDate autofocus) .right label(for="time") {{_ 'time'}} input.js-time-field#time(type="time" name="time" value=showTime) diff --git a/client/components/main/popup.css b/client/components/main/popup.css index 7ddba701c..dafbd2576 100644 --- a/client/components/main/popup.css +++ b/client/components/main/popup.css @@ -293,6 +293,8 @@ overflow-y: auto !important; } + + .pop-over[data-popup="editCardReceivedDatePopup"] .edit-date button, .pop-over[data-popup="editCardStartDatePopup"] .edit-date button, .pop-over[data-popup="editCardDueDatePopup"] .edit-date button, @@ -387,9 +389,6 @@ margin: 0; visibility: hidden; } -.pop-over .quiet { -/* padding: 6px 6px 4px;*/ -} .pop-over.search-over { background: #f0f0f0; min-height: 14vh; diff --git a/client/lib/datepicker.js b/client/lib/datepicker.js index 6a4f010e9..fa2ff8129 100644 --- a/client/lib/datepicker.js +++ b/client/lib/datepicker.js @@ -1,33 +1,27 @@ import { ReactiveCache } from '/imports/reactiveCache'; import { TAPi18n } from '/imports/i18n'; -import { - formatDateTime, - formatDate, - formatDateByUserPreference, - formatTime, - getISOWeek, - isValidDate, - isBefore, - isAfter, - isSame, - add, - subtract, - startOf, - endOf, - format, - parseDate, - now, - createDate, - fromNow, - calendar -} from '/imports/lib/dateUtils'; -// Helper function to get time format for 24 hours -function adjustedTimeFormat() { - return 'HH:mm'; +// Helper to check if a date is valid +function isValidDate(date) { + return date instanceof Date && !isNaN(date); } -// .replace(/HH/i, 'H'); +// Format date as YYYY-MM-DD +function formatDate(date) { + if (!isValidDate(date)) return ''; + const year = date.getFullYear(); + const month = String(date.getMonth() + 1).padStart(2, '0'); + const day = String(date.getDate()).padStart(2, '0'); + return `${year}-${month}-${day}`; +} + +// Format time as HH:mm +function formatTime(date) { + if (!isValidDate(date)) return ''; + const hours = String(date.getHours()).padStart(2, '0'); + const minutes = String(date.getMinutes()).padStart(2, '0'); + return `${hours}:${minutes}`; +} export class DatePicker extends BlazeComponent { template() { @@ -51,35 +45,25 @@ export class DatePicker extends BlazeComponent { } onRendered() { - // Set initial values for text and time inputs + // Set initial values for native HTML inputs if (isValidDate(this.date.get())) { const dateInput = this.find('#date'); const timeInput = this.find('#time'); if (dateInput) { - // Use user's preferred format for text input - const currentUser = ReactiveCache.getCurrentUser(); - const userFormat = currentUser ? currentUser.getDateFormat() : 'YYYY-MM-DD'; - dateInput.value = formatDateByUserPreference(this.date.get(), userFormat, false); + dateInput.value = formatDate(this.date.get()); } - if (timeInput) { - if (!timeInput.value && this.defaultTime) { - const defaultDate = new Date(this.defaultTime); - timeInput.value = formatTime(defaultDate); - } else if (isValidDate(this.date.get())) { - timeInput.value = formatTime(this.date.get()); - } + if (timeInput && !timeInput.value && this.defaultTime) { + const defaultDate = new Date(this.defaultTime); + timeInput.value = formatTime(defaultDate); + } else if (timeInput && isValidDate(this.date.get())) { + timeInput.value = formatTime(this.date.get()); } } } showDate() { - if (isValidDate(this.date.get())) { - // Use user's preferred format for display, but HTML date input needs YYYY-MM-DD - const currentUser = ReactiveCache.getCurrentUser(); - const userFormat = currentUser ? currentUser.getDateFormat() : 'YYYY-MM-DD'; - return formatDateByUserPreference(this.date.get(), userFormat, false); - } + if (isValidDate(this.date.get())) return formatDate(this.date.get()); return ''; } showTime() { @@ -87,56 +71,22 @@ export class DatePicker extends BlazeComponent { return ''; } dateFormat() { - const currentUser = ReactiveCache.getCurrentUser(); - const userFormat = currentUser ? currentUser.getDateFormat() : 'YYYY-MM-DD'; - // Convert format to localized placeholder - switch (userFormat) { - case 'DD-MM-YYYY': - return TAPi18n.__('date-format-dd-mm-yyyy') || 'PP-KK-VVVV'; - case 'MM-DD-YYYY': - return TAPi18n.__('date-format-mm-dd-yyyy') || 'KK-PP-VVVV'; - case 'YYYY-MM-DD': - default: - return TAPi18n.__('date-format-yyyy-mm-dd') || 'VVVV-KK-PP'; - } + return 'YYYY-MM-DD'; } timeFormat() { - return 'LT'; + return 'HH:mm'; } events() { return [ { 'change .js-date-field'() { - // Text input date validation - const dateInput = this.find('#date'); - if (!dateInput) return; - - const dateValue = dateInput.value; + // Native HTML date input validation + const dateValue = this.find('#date').value; if (dateValue) { - // Try to parse different date formats - const formats = [ - 'YYYY-MM-DD', - 'DD-MM-YYYY', - 'MM-DD-YYYY', - 'DD/MM/YYYY', - 'MM/DD/YYYY', - 'DD.MM.YYYY', - 'MM.DD.YYYY' - ]; - - let parsedDate = null; - for (const format of formats) { - parsedDate = parseDate(dateValue, [format], true); - if (parsedDate) break; - } - - // Fallback to native Date parsing - if (!parsedDate) { - parsedDate = new Date(dateValue); - } - - if (isValidDate(parsedDate)) { + // HTML date input format is always YYYY-MM-DD + const dateObj = new Date(dateValue + 'T12:00:00'); + if (isValidDate(dateObj)) { this.error.set(''); } else { this.error.set('invalid-date'); @@ -145,12 +95,10 @@ export class DatePicker extends BlazeComponent { }, 'change .js-time-field'() { // Native HTML time input validation - const timeInput = this.find('#time'); - if (!timeInput) return; - - const timeValue = timeInput.value; + const timeValue = this.find('#time').value; if (timeValue) { - const timeObj = new Date(`1970-01-01T${timeValue}`); + // HTML time input format is always HH:mm + const timeObj = new Date(`1970-01-01T${timeValue}:00`); if (isValidDate(timeObj)) { this.error.set(''); } else { @@ -170,44 +118,16 @@ export class DatePicker extends BlazeComponent { return; } - // Try to parse different date formats - const formats = [ - 'YYYY-MM-DD', - 'DD-MM-YYYY', - 'MM-DD-YYYY', - 'DD/MM/YYYY', - 'MM/DD/YYYY', - 'DD.MM.YYYY', - 'MM.DD.YYYY' - ]; + // Combine date and time: HTML date input is YYYY-MM-DD, time input is HH:mm + const dateTimeString = `${dateValue}T${timeValue}:00`; + const newCompleteDate = new Date(dateTimeString); - let parsedDate = null; - for (const format of formats) { - parsedDate = parseDate(dateValue, [format], true); - if (parsedDate) break; - } - - // Fallback to native Date parsing - if (!parsedDate) { - parsedDate = new Date(dateValue); - } - - if (!isValidDate(parsedDate)) { + if (!isValidDate(newCompleteDate)) { this.error.set('invalid'); return; } - // Combine with time - const timeObj = new Date(`1970-01-01T${timeValue}`); - if (!isValidDate(timeObj)) { - this.error.set('invalid-time'); - return; - } - - // Set the time on the parsed date - parsedDate.setHours(timeObj.getHours(), timeObj.getMinutes(), 0, 0); - - this._storeDate(parsedDate); + this._storeDate(newCompleteDate); Popup.back(); }, 'click .js-delete-date'(evt) { From 0772ca40364b1ae01de826b44d79ea1afe6c5f76 Mon Sep 17 00:00:00 2001 From: Lauri Ojansivu Date: Thu, 6 Nov 2025 02:36:10 +0200 Subject: [PATCH 55/84] Updated ChangeLog. --- CHANGELOG.md | 2 ++ 1 file changed, 2 insertions(+) diff --git a/CHANGELOG.md b/CHANGELOG.md index 6d553985c..b4f1b0aeb 100644 --- a/CHANGELOG.md +++ b/CHANGELOG.md @@ -47,6 +47,8 @@ and fixes the following bugs: Thanks to xet7. - [Try to fix Edit Custom Fields button not working. Removed duplicate option from Boards Settings](https://github.com/wekan/wekan/commit/20af0a2ef55b11e7205845859ee92a929616ce91). Thanks to xet7. +- [Fix Regression - calendar popup to set due date has gone](https://github.com/wekan/wekan/commit/581733d605b7e0494e72229c45947cff134f6dd6). + Thanks to xet7. Thanks to above GitHub users for their contributions and translators for their translations. From c829c073cf822e48b7cd84bbfb79d42867412517 Mon Sep 17 00:00:00 2001 From: Lauri Ojansivu Date: Thu, 6 Nov 2025 02:44:30 +0200 Subject: [PATCH 56/84] Remove not working Bookmark menu option. Thanks to xet7 ! --- client/components/main/header.jade | 4 ---- client/components/users/userHeader.jade | 4 ---- 2 files changed, 8 deletions(-) diff --git a/client/components/main/header.jade b/client/components/main/header.jade index b7e870dc2..1ac11f189 100644 --- a/client/components/main/header.jade +++ b/client/components/main/header.jade @@ -83,10 +83,6 @@ template(name="header") i.mobile-icon(class="{{#if mobileMode}}active{{/if}}") 📱 i.desktop-icon(class="{{#unless mobileMode}}active{{/unless}}") 🖥️ - // Bookmarks button - desktop opens popup, mobile routes to page - a.board-header-btn.js-open-bookmarks(title="{{_ 'bookmarks'}}") - | 🔖 - // Notifications +notifications diff --git a/client/components/users/userHeader.jade b/client/components/users/userHeader.jade index 668777dbb..7ee64d138 100644 --- a/client/components/users/userHeader.jade +++ b/client/components/users/userHeader.jade @@ -25,10 +25,6 @@ template(name="memberMenuPopup") a.js-global-search(href="{{pathFor 'global-search'}}") | 🔍 | {{_ 'globalSearch-title'}} - li - a(href="{{pathFor 'home'}}") - | ⭐ - | {{_ 'my-bookmarks'}} li a(href="{{pathFor 'home'}}") | 🏠 From 46866dac85d5a2d91f26c985eee2149ac1186d2d Mon Sep 17 00:00:00 2001 From: Lauri Ojansivu Date: Thu, 6 Nov 2025 02:46:52 +0200 Subject: [PATCH 57/84] Updated ChangeLog. --- CHANGELOG.md | 2 ++ 1 file changed, 2 insertions(+) diff --git a/CHANGELOG.md b/CHANGELOG.md index b4f1b0aeb..f11cc88f4 100644 --- a/CHANGELOG.md +++ b/CHANGELOG.md @@ -49,6 +49,8 @@ and fixes the following bugs: Thanks to xet7. - [Fix Regression - calendar popup to set due date has gone](https://github.com/wekan/wekan/commit/581733d605b7e0494e72229c45947cff134f6dd6). Thanks to xet7. +- [Remove not working Bookmark menu option](https://github.com/wekan/wekan/commit/c829c073cf822e48b7cd84bbfb79d42867412517). + Thanks to xet7. Thanks to above GitHub users for their contributions and translators for their translations. From 6244657ca53a54646ec01e702851a51d89bd0d55 Mon Sep 17 00:00:00 2001 From: Lauri Ojansivu Date: Thu, 6 Nov 2025 03:06:16 +0200 Subject: [PATCH 58/84] Fix Workspaces at All Boards to have correct count of remaining etc, while starred also at Starred/Favorites. Thanks to xet7 ! --- client/components/boards/boardsList.css | 9 +++++++-- client/components/boards/boardsList.js | 11 +++++++---- 2 files changed, 14 insertions(+), 6 deletions(-) diff --git a/client/components/boards/boardsList.css b/client/components/boards/boardsList.css index e17b77b12..dc7efdd66 100644 --- a/client/components/boards/boardsList.css +++ b/client/components/boards/boardsList.css @@ -373,6 +373,12 @@ .board-list li.starred .is-star-active, .board-list li.starred .is-not-star-active { opacity: 1; + color: #ffd700; +} +/* Show star icon on hover even for non-starred boards */ +.board-list li:hover .is-star-active, +.board-list li:hover .is-not-star-active { + opacity: 1; } .board-list .board-list-item { overflow: hidden; @@ -436,7 +442,7 @@ } .board-list .is-star-active, .board-list .is-not-star-active { - bottom: 0; + top: 0; font-size: 14px; height: 18px; line-height: 18px; @@ -444,7 +450,6 @@ padding: 9px 9px; position: absolute; right: 0; - top: 0; transition-duration: 0.15s; transition-property: color, font-size, background; } diff --git a/client/components/boards/boardsList.js b/client/components/boards/boardsList.js index 171c9196b..bb1d258d0 100644 --- a/client/components/boards/boardsList.js +++ b/client/components/boards/boardsList.js @@ -293,13 +293,15 @@ BlazeComponent.extendComponent({ } else if (sel === 'templates') { list = list.filter(b => b.type === 'template-container'); } else if (sel === 'remaining') { + // Show boards not in any workspace AND not templates + // Keep starred boards visible in Remaining too list = list.filter(b => !assignments[b._id] && - b.type !== 'template-container' && - !(currentUser && currentUser.hasStarred(b._id)) + b.type !== 'template-container' ); } else { // assume sel is a workspaceId + // Keep starred boards visible in their workspace too list = list.filter(b => assignments[b._id] === sel); } @@ -803,10 +805,11 @@ BlazeComponent.extendComponent({ } else if (type === 'templates') { return allBoards.filter(b => b.type === 'template-container').length; } else if (type === 'remaining') { + // Count boards not in any workspace AND not templates + // Include starred boards (they appear in both Starred and Remaining) return allBoards.filter(b => !assignments[b._id] && - b.type !== 'template-container' && - !(currentUser && currentUser.hasStarred(b._id)) + b.type !== 'template-container' ).length; } return 0; From fe104791b5a5e548b4727a382275fa9ac7f9840c Mon Sep 17 00:00:00 2001 From: Lauri Ojansivu Date: Thu, 6 Nov 2025 03:08:51 +0200 Subject: [PATCH 59/84] Updated ChangeLog. --- CHANGELOG.md | 2 ++ 1 file changed, 2 insertions(+) diff --git a/CHANGELOG.md b/CHANGELOG.md index f11cc88f4..20ec151b6 100644 --- a/CHANGELOG.md +++ b/CHANGELOG.md @@ -51,6 +51,8 @@ and fixes the following bugs: Thanks to xet7. - [Remove not working Bookmark menu option](https://github.com/wekan/wekan/commit/c829c073cf822e48b7cd84bbfb79d42867412517). Thanks to xet7. +- [Fix Workspaces at All Boards to have correct count of remaining etc, while starred also at Starred/Favorites](https://github.com/wekan/wekan/commit/6244657ca53a54646ec01e702851a51d89bd0d55). + Thanks to xet7. Thanks to above GitHub users for their contributions and translators for their translations. From 18003900c2d497c129793d1653d4d9872a2f19da Mon Sep 17 00:00:00 2001 From: Lauri Ojansivu Date: Thu, 6 Nov 2025 03:31:14 +0200 Subject: [PATCH 60/84] Fix Worker Permissions does not allow for cards to be moved. - v8.15. Thanks to xet7 ! Fixes #5990 --- client/components/cards/minicard.jade | 1 + client/components/lists/listHeader.jade | 6 ++- .../components/swimlanes/swimlaneHeader.jade | 39 ++++++++++--------- client/config/blazeHelpers.js | 4 ++ client/lib/utils.js | 9 +++++ 5 files changed, 38 insertions(+), 21 deletions(-) diff --git a/client/components/cards/minicard.jade b/client/components/cards/minicard.jade index 4a5040e76..b36af1ceb 100644 --- a/client/components/cards/minicard.jade +++ b/client/components/cards/minicard.jade @@ -5,6 +5,7 @@ template(name="minicard") class="{{#if colorClass}}minicard-{{colorClass}}{{/if}}") if canModifyCard a.minicard-details-menu-with-handle.js-open-minicard-details-menu(title="{{_ 'cardDetailsActionsPopup-title'}}") ☰ + if canMoveCard .handle | ↕️ .dates diff --git a/client/components/lists/listHeader.jade b/client/components/lists/listHeader.jade index df3c31f51..160be7b11 100644 --- a/client/components/lists/listHeader.jade +++ b/client/components/lists/listHeader.jade @@ -55,7 +55,8 @@ template(name="listHeader") a.js-open-list-menu(title="{{_ 'listActionPopup-title'}}") ☰ else a.list-header-menu-icon.js-select-list ▶️ - a.list-header-handle.handle.js-list-handle ↕️ + unless currentUser.isWorker + a.list-header-handle.handle.js-list-handle ↕️ else if currentUser.isBoardMember if isWatching i.list-header-watch-icon | 👁️ @@ -72,7 +73,8 @@ template(name="listHeader") a.js-open-list-menu(title="{{_ 'listActionPopup-title'}}") ☰ if currentUser.isBoardMember unless currentUser.isCommentOnly - a.list-header-handle.handle.js-list-handle ↕️ + unless currentUser.isWorker + a.list-header-handle.handle.js-list-handle ↕️ template(name="editListTitleForm") .list-composer diff --git a/client/components/swimlanes/swimlaneHeader.jade b/client/components/swimlanes/swimlaneHeader.jade index 04548ffa6..a0a44eb7f 100644 --- a/client/components/swimlanes/swimlaneHeader.jade +++ b/client/components/swimlanes/swimlaneHeader.jade @@ -25,25 +25,26 @@ template(name="swimlaneFixedHeader") .swimlane-header-menu if currentUser unless currentUser.isCommentOnly - a.js-open-add-swimlane-menu.swimlane-header-plus-icon(title="{{_ 'add-swimlane'}}") - | ➕ - a.js-open-swimlane-menu(title="{{_ 'swimlaneActionPopup-title'}}") - | ☰ - //// TODO: Collapse Swimlane: make button working, etc. - //unless collapsed - // a.js-collapse-swimlane(title="{{_ 'collapse'}}") - // i.fa.fa-arrow-down.swimlane-header-collapse-down - // ⬆️.swimlane-header-collapse-up - //if collapsed - // a.js-collapse-swimlane(title="{{_ 'uncollapse'}}") - // ⬆️.swimlane-header-collapse-up - // i.fa.fa-arrow-down.swimlane-header-collapse-down - unless isTouchScreen - a.swimlane-header-handle.handle.js-swimlane-header-handle - | ↕️ - if isTouchScreen - a.swimlane-header-miniscreen-handle.handle.js-swimlane-header-handle - | ↕️ + unless currentUser.isWorker + a.js-open-add-swimlane-menu.swimlane-header-plus-icon(title="{{_ 'add-swimlane'}}") + | ➕ + a.js-open-swimlane-menu(title="{{_ 'swimlaneActionPopup-title'}}") + | ☰ + //// TODO: Collapse Swimlane: make button working, etc. + //unless collapsed + // a.js-collapse-swimlane(title="{{_ 'collapse'}}") + // i.fa.fa-arrow-down.swimlane-header-collapse-down + // ⬆️.swimlane-header-collapse-up + //if collapsed + // a.js-collapse-swimlane(title="{{_ 'uncollapse'}}") + // ⬆️.swimlane-header-collapse-up + // i.fa.fa-arrow-down.swimlane-header-collapse-down + unless isTouchScreen + a.swimlane-header-handle.handle.js-swimlane-header-handle + | ↕️ + if isTouchScreen + a.swimlane-header-miniscreen-handle.handle.js-swimlane-header-handle + | ↕️ template(name="editSwimlaneTitleForm") .list-composer diff --git a/client/config/blazeHelpers.js b/client/config/blazeHelpers.js index 967b83059..333913dfc 100644 --- a/client/config/blazeHelpers.js +++ b/client/config/blazeHelpers.js @@ -73,6 +73,10 @@ Blaze.registerHelper('canModifyCard', () => Utils.canModifyCard(), ); +Blaze.registerHelper('canMoveCard', () => + Utils.canMoveCard(), +); + Blaze.registerHelper('canModifyBoard', () => Utils.canModifyBoard(), ); diff --git a/client/lib/utils.js b/client/lib/utils.js index a20d65f00..078dfe967 100644 --- a/client/lib/utils.js +++ b/client/lib/utils.js @@ -214,6 +214,15 @@ Utils = { ); return ret; }, + canMoveCard() { + const currentUser = ReactiveCache.getCurrentUser(); + const ret = ( + currentUser && + currentUser.isBoardMember() && + !currentUser.isCommentOnly() + ); + return ret; + }, canModifyBoard() { const currentUser = ReactiveCache.getCurrentUser(); const ret = ( From 7f53dfac3c670ed677ceaa08698335bb9c8503f6 Mon Sep 17 00:00:00 2001 From: Lauri Ojansivu Date: Thu, 6 Nov 2025 03:33:46 +0200 Subject: [PATCH 61/84] Updated ChangeLog. --- CHANGELOG.md | 2 ++ 1 file changed, 2 insertions(+) diff --git a/CHANGELOG.md b/CHANGELOG.md index 20ec151b6..9da26a7b8 100644 --- a/CHANGELOG.md +++ b/CHANGELOG.md @@ -53,6 +53,8 @@ and fixes the following bugs: Thanks to xet7. - [Fix Workspaces at All Boards to have correct count of remaining etc, while starred also at Starred/Favorites](https://github.com/wekan/wekan/commit/6244657ca53a54646ec01e702851a51d89bd0d55). Thanks to xet7. +- [Fix Worker Permissions does not allow for cards to be moved. - v8.15. Removed buttons Worker should not use](https://github.com/wekan/wekan/commit/18003900c2d497c129793d1653d4d9872a2f19da). + Thanks to xet7. Thanks to above GitHub users for their contributions and translators for their translations. From 0004ae716b525d45d6a0a7fdf018f2390f8a62ce Mon Sep 17 00:00:00 2001 From: Lauri Ojansivu Date: Thu, 6 Nov 2025 04:00:04 +0200 Subject: [PATCH 62/84] v8.17 --- CHANGELOG.md | 2 +- Dockerfile | 6 +++--- Stackerfile.yml | 2 +- docs/Platforms/Propietary/Windows/Offline.md | 4 ++-- package-lock.json | 2 +- package.json | 2 +- sandstorm-pkgdef.capnp | 4 ++-- snapcraft.yaml | 8 ++++---- 8 files changed, 15 insertions(+), 15 deletions(-) diff --git a/CHANGELOG.md b/CHANGELOG.md index 9da26a7b8..64f8ee592 100644 --- a/CHANGELOG.md +++ b/CHANGELOG.md @@ -22,7 +22,7 @@ Fixing other platforms In Progress. WeKan 8.00-8.06 had wrong raw database directory setting /var/snap/wekan/common/wekan and some cards were not visible. Those are fixed at WeKan 8.07 where database directory is back to /var/snap/wekan/common and all cards are visible. -# Upcoming WeKan ® release +# v8.17 2025-11-06 WeKan ® release This release adds the following new feature: diff --git a/Dockerfile b/Dockerfile index 2b90be928..aedb88c5b 100644 --- a/Dockerfile +++ b/Dockerfile @@ -249,9 +249,9 @@ cd /home/wekan/app # Remove legacy webbroser bundle, so that Wekan works also at Android Firefox, iOS Safari, etc. #rm -rf /home/wekan/app_build/bundle/programs/web.browser.legacy #mv /home/wekan/app_build/bundle /build -wget "https://github.com/wekan/wekan/releases/download/v8.16/wekan-8.16-amd64.zip" -unzip wekan-8.16-amd64.zip -rm wekan-8.16-amd64.zip +wget "https://github.com/wekan/wekan/releases/download/v8.17/wekan-8.17-amd64.zip" +unzip wekan-8.17-amd64.zip +rm wekan-8.17-amd64.zip mv /home/wekan/app/bundle /build # Put back the original tar diff --git a/Stackerfile.yml b/Stackerfile.yml index 544188a36..e0529f92f 100644 --- a/Stackerfile.yml +++ b/Stackerfile.yml @@ -1,5 +1,5 @@ appId: wekan-public/apps/77b94f60-dec9-0136-304e-16ff53095928 -appVersion: "v8.16.0" +appVersion: "v8.17.0" files: userUploads: - README.md diff --git a/docs/Platforms/Propietary/Windows/Offline.md b/docs/Platforms/Propietary/Windows/Offline.md index 7913faba4..4da341fd1 100644 --- a/docs/Platforms/Propietary/Windows/Offline.md +++ b/docs/Platforms/Propietary/Windows/Offline.md @@ -10,7 +10,7 @@ This is without container (without Docker or Snap). Right click and download files 1-4: -1. [wekan-8.16-amd64-windows.zip](https://github.com/wekan/wekan/releases/download/v8.16/wekan-8.16-amd64-windows.zip) +1. [wekan-8.17-amd64-windows.zip](https://github.com/wekan/wekan/releases/download/v8.17/wekan-8.17-amd64-windows.zip) 2. [node.exe](https://nodejs.org/dist/latest-v14.x/win-x64/node.exe) @@ -22,7 +22,7 @@ Right click and download files 1-4: 6. Double click `mongodb-windows-x86_64-7.0.25-signed.msi` . In installer, uncheck downloading MongoDB compass. -7. Unzip `wekan-8.16-amd64-windows.zip` , inside it is directory `bundle`, to it copy other files: +7. Unzip `wekan-8.17-amd64-windows.zip` , inside it is directory `bundle`, to it copy other files: ``` bundle (directory) diff --git a/package-lock.json b/package-lock.json index 9e338c147..c22f9d0de 100644 --- a/package-lock.json +++ b/package-lock.json @@ -1,6 +1,6 @@ { "name": "wekan", - "version": "v8.16.0", + "version": "v8.17.0", "lockfileVersion": 1, "requires": true, "dependencies": { diff --git a/package.json b/package.json index 7edd176d4..0b2a6fb67 100644 --- a/package.json +++ b/package.json @@ -1,6 +1,6 @@ { "name": "wekan", - "version": "v8.16.0", + "version": "v8.17.0", "description": "Open-Source kanban", "private": true, "repository": { diff --git a/sandstorm-pkgdef.capnp b/sandstorm-pkgdef.capnp index 21401c3ce..088111c90 100644 --- a/sandstorm-pkgdef.capnp +++ b/sandstorm-pkgdef.capnp @@ -22,10 +22,10 @@ const pkgdef :Spk.PackageDefinition = ( appTitle = (defaultText = "Wekan"), # The name of the app as it is displayed to the user. - appVersion = 816, + appVersion = 817, # Increment this for every release. - appMarketingVersion = (defaultText = "8.16.0~2025-11-02"), + appMarketingVersion = (defaultText = "8.17.0~2025-11-06"), # Human-readable presentation of the app version. minUpgradableAppVersion = 0, diff --git a/snapcraft.yaml b/snapcraft.yaml index 3e3e503bf..6cea1799a 100644 --- a/snapcraft.yaml +++ b/snapcraft.yaml @@ -1,5 +1,5 @@ name: wekan -version: '8.16' +version: '8.17' base: core24 summary: Open Source kanban description: | @@ -166,9 +166,9 @@ parts: # Cleanup mkdir .build cd .build - wget https://github.com/wekan/wekan/releases/download/v8.16/wekan-8.16-amd64.zip - unzip wekan-8.16-amd64.zip - rm wekan-8.16-amd64.zip + wget https://github.com/wekan/wekan/releases/download/v8.17/wekan-8.17-amd64.zip + unzip wekan-8.17-amd64.zip + rm wekan-8.17-amd64.zip cd .. ##cd .build/bundle ##find . -type d -name '*-garbage*' | xargs rm -rf From c5f5ce126df56cebde685c92041834185fa87e7c Mon Sep 17 00:00:00 2001 From: "Buo-ren Lin (OSSII)" Date: Mon, 10 Nov 2025 10:49:26 +0800 Subject: [PATCH 63/84] Fix Broken Strikethroughs in Markdown to HTML conversion. Allow the s tag to be rendered. Fixes #6008. Signed-off-by: Buo-ren Lin (OSSII) --- client/lib/secureDOMPurify.js | 4 ++-- 1 file changed, 2 insertions(+), 2 deletions(-) diff --git a/client/lib/secureDOMPurify.js b/client/lib/secureDOMPurify.js index 898687dad..323c3b6c0 100644 --- a/client/lib/secureDOMPurify.js +++ b/client/lib/secureDOMPurify.js @@ -4,7 +4,7 @@ import DOMPurify from 'dompurify'; export function getSecureDOMPurifyConfig() { return { // Allow common markdown elements including anchor tags - ALLOWED_TAGS: ['a', 'p', 'br', 'strong', 'em', 'u', 'h1', 'h2', 'h3', 'h4', 'h5', 'h6', 'ul', 'ol', 'li', 'blockquote', 'pre', 'code', 'img', 'table', 'thead', 'tbody', 'tr', 'th', 'td', 'hr', 'div', 'span'], + ALLOWED_TAGS: ['a', 'p', 'br', 'strong', 'em', 'u', 'h1', 'h2', 'h3', 'h4', 'h5', 'h6', 'ul', 'ol', 'li', 'blockquote', 'pre', 'code', 'img', 'table', 'thead', 'tbody', 'tr', 'th', 'td', 'hr', 'div', 'span', 's'], // Allow safe attributes including href for anchor tags ALLOWED_ATTR: ['href', 'title', 'alt', 'src', 'width', 'height', 'target', 'rel'], // Allow safe protocols for links @@ -44,7 +44,7 @@ export function getSecureDOMPurifyConfig() { } return false; } - + // Additional check for base64 encoded SVG with script tags if (src.startsWith('data:image/svg+xml;base64,')) { try { From 6302a48221eb3eaf6cc9fcc8f812554c0dfb9f2a Mon Sep 17 00:00:00 2001 From: "dependabot[bot]" <49699333+dependabot[bot]@users.noreply.github.com> Date: Mon, 10 Nov 2025 20:02:19 +0000 Subject: [PATCH 64/84] Bump docker/metadata-action from 5.8.0 to 5.9.0 Bumps [docker/metadata-action](https://github.com/docker/metadata-action) from 5.8.0 to 5.9.0. - [Release notes](https://github.com/docker/metadata-action/releases) - [Commits](https://github.com/docker/metadata-action/compare/c1e51972afc2121e065aed6d45c65596fe445f3f...318604b99e75e41977312d83839a89be02ca4893) --- updated-dependencies: - dependency-name: docker/metadata-action dependency-version: 5.9.0 dependency-type: direct:production update-type: version-update:semver-minor ... Signed-off-by: dependabot[bot] --- .github/workflows/docker-publish.yml | 2 +- 1 file changed, 1 insertion(+), 1 deletion(-) diff --git a/.github/workflows/docker-publish.yml b/.github/workflows/docker-publish.yml index 54af974ce..eab9e0fbb 100644 --- a/.github/workflows/docker-publish.yml +++ b/.github/workflows/docker-publish.yml @@ -48,7 +48,7 @@ jobs: # https://github.com/docker/metadata-action - name: Extract Docker metadata id: meta - uses: docker/metadata-action@c1e51972afc2121e065aed6d45c65596fe445f3f + uses: docker/metadata-action@318604b99e75e41977312d83839a89be02ca4893 with: images: ${{ env.REGISTRY }}/${{ env.IMAGE_NAME }} From 7ff1649d8909917cae590c68def6eecac0442f91 Mon Sep 17 00:00:00 2001 From: Lauri Ojansivu Date: Fri, 14 Nov 2025 07:47:31 +0200 Subject: [PATCH 65/84] Updated security.md --- SECURITY.md | 36 +++++++++++++++++------------------- 1 file changed, 17 insertions(+), 19 deletions(-) diff --git a/SECURITY.md b/SECURITY.md index 0ad7a0256..5cde5926b 100644 --- a/SECURITY.md +++ b/SECURITY.md @@ -1,12 +1,20 @@ -About money, see [CONTRIBUTING.md](CONTRIBUTING.md) -Security is very important to us. If you discover any issue regarding security, please disclose -the information responsibly by sending an email from Protonmail to security@wekan.fi -that is Protomail email address, or by using this PGP key -[security-at-wekan.fi.asc](security-at-wekan.fi.asc) to security@wekan.fi -and not by creating a GitHub issue. We will respond swiftly to fix verifiable security issues. +## Responsible Security Disclosure -We thank you with a place at our hall of fame page, that is at https://wekan.fi/hall-of-fame +- To send email, use [ProtonMail](https://proton.me) email address or use PGP key [security-at-wekan.fi.asc](security-at-wekan.fi.asc) +- Send info about security issue ONLY to security@wekan.fi (that is Protomail email address). NOT TO ANYWHERE ELSE. NO CC, NO BCC. +- Wait for new WeKan release that fixes security issue +- If you approve, we thank you by adding you to Hall of Fame: https://wekan.fi/hall-of-fame/ + +## Bonus Points + +- If you include code for fixing security issue + +## Losing Points + +- If you ask about [bounty](CONTRIBUTING.md). There is no bounty. WeKan is NOT Big Tech. WeKan is FLOSS. +- If you forget to include vulnerability details. +- If you send info about security issue to somewhere else than security@wekan.fi ## How should reports be formatted? @@ -26,7 +34,7 @@ CWSS (optional): %cwss Anyone who reports a unique security issue in scope and does not disclose it to a third party before we have patched and updated may be upon their approval -added to the Wekan Hall of Fame. +added to the WeKan Hall of Fame https://wekan.fi/hall-of-fame/ ## Which domains are in scope? @@ -63,11 +71,6 @@ and by by companies that have 30k users. - If you are thinking about TLS MITM, look at https://github.com/caddyserver/caddy/issues/2530 - Let's Encrypt TLS requires publicly accessible webserver, that Let's Encrypt TLS validation servers check. - If firewall limits to only allowed IP addresses, you may need non-Let's Encrypt TLS cert. -- For On Premise: - - https://caddyserver.com/docs/automatic-https#local-https - - https://github.com/wekan/wekan/wiki/Caddy-Webserver-Config - - https://github.com/wekan/wekan/wiki/Azure - - https://github.com/wekan/wekan/wiki/Traefik-and-self-signed-SSL-certs ## XSS @@ -269,9 +272,4 @@ Typical already known or "no impact" bugs such as: - Email spoofing, SPF, DMARC & DKIM. Wekan does not include email server. Wekan is Open Source with MIT license, and free to use also for commercial use. -We welcome all fixes to improve security by email to security@wekan.team - -## Bonus Points - -If your Responsible Security Disclosure includes code for fixing security issue, -you get bonus points, as seen on [Hall of Fame](https://wekan.github.io/hall-of-fame). +We welcome all fixes to improve security by email to security@wekan.fi From 37a3065f3c9f0f5c527cff40bf13ffa1058cd022 Mon Sep 17 00:00:00 2001 From: Lauri Ojansivu Date: Sat, 15 Nov 2025 16:35:31 +0200 Subject: [PATCH 66/84] Updated translations. --- imports/i18n/data/fr.i18n.json | 52 +- imports/i18n/data/ja-HI.i18n.json | 6 +- imports/i18n/data/ja.i18n.json | 6 +- imports/i18n/data/nl.i18n.json | 138 +- imports/i18n/data/pt-BR.i18n.json | 134 +- imports/i18n/data/sr.i18n.json | 2114 ++++++++++++++--------------- imports/i18n/data/sv.i18n.json | 20 +- imports/i18n/data/zh-TW.i18n.json | 154 +-- 8 files changed, 1312 insertions(+), 1312 deletions(-) diff --git a/imports/i18n/data/fr.i18n.json b/imports/i18n/data/fr.i18n.json index 9944c844d..dc83c8e89 100644 --- a/imports/i18n/data/fr.i18n.json +++ b/imports/i18n/data/fr.i18n.json @@ -78,18 +78,18 @@ "activity-deleteComment": "commentaire supprimé %s", "activity-receivedDate": "date de réception éditée de %s à %s", "activity-startDate": "date de début éditée de %s à %s", - "allboards.starred": "Starred", + "allboards.starred": "Favoris", "allboards.templates": "Modèles", - "allboards.remaining": "Remaining", - "allboards.workspaces": "Workspaces", - "allboards.add-workspace": "Add Workspace", - "allboards.add-workspace-prompt": "Workspace name", - "allboards.add-subworkspace": "Add Subworkspace", - "allboards.add-subworkspace-prompt": "Subworkspace name", - "allboards.edit-workspace": "Edit workspace", - "allboards.edit-workspace-name": "Workspace name", + "allboards.remaining": "Restant", + "allboards.workspaces": "Espaces de travail", + "allboards.add-workspace": "Ajouter un espace de travail", + "allboards.add-workspace-prompt": "Nom de l'espace de travail", + "allboards.add-subworkspace": "Ajouter un sous-espace de travail", + "allboards.add-subworkspace-prompt": "Nom du sous-espace de travail", + "allboards.edit-workspace": "Modifier l'espace de travail", + "allboards.edit-workspace-name": "Nom de l'espace de travail", "allboards.edit-workspace-icon": "Workspace icon (markdown)", - "multi-selection-active": "Click checkboxes to select boards", + "multi-selection-active": "Cliquez sur les cases à cocher pour sélectionner les tableaux", "activity-dueDate": "date d'échéance éditée de %s à %s", "activity-endDate": "date de fin éditée de %s à %s", "add-attachment": "Ajouter une pièce jointe", @@ -1416,16 +1416,16 @@ "automatic-migration": "Migration automatique", "back-to-settings": "Retour aux paramètres", "board-id": "ID du tableau", - "board-migration": "Board Migration", - "board-migrations": "Board Migrations", + "board-migration": "Migration du tableau", + "board-migrations": "Migrations de tableau", "card-show-lists-on-minicard": "Afficher les listes sur la mini-carte", - "comprehensive-board-migration": "Comprehensive Board Migration", + "comprehensive-board-migration": "Migration complète de tableau", "comprehensive-board-migration-description": "Performs comprehensive checks and fixes for board data integrity, including list ordering, card positions, and swimlane structure.", "delete-duplicate-empty-lists-migration": "Delete Duplicate Empty Lists", "delete-duplicate-empty-lists-migration-description": "Safely deletes empty duplicate lists. Only removes lists that have no cards AND have another list with the same title that contains cards.", - "lost-cards": "Lost Cards", + "lost-cards": "Cartes perdues", "lost-cards-list": "Restored Items", - "restore-lost-cards-migration": "Restore Lost Cards", + "restore-lost-cards-migration": "Restaurer les cartes perdues", "restore-lost-cards-migration-description": "Finds and restores cards and lists with missing swimlaneId or listId. Creates a 'Lost Cards' swimlane to make all lost items visible again.", "restore-all-archived-migration": "Restore All Archived", "restore-all-archived-migration-description": "Restores all archived swimlanes, lists, and cards. Automatically fixes any missing swimlaneId or listId to make items visible.", @@ -1435,16 +1435,16 @@ "fix-avatar-urls-migration-description": "Updates avatar URLs for board members to use the correct storage backend and fixes broken avatar references.", "fix-all-file-urls-migration": "Fix All File URLs", "fix-all-file-urls-migration-description": "Updates all file attachment URLs on this board to use the correct storage backend and fixes broken file references.", - "migration-needed": "Migration Needed", + "migration-needed": "Migration requise", "migration-complete": "Terminé", - "migration-running": "Running...", - "migration-successful": "Migration completed successfully", - "migration-failed": "Migration failed", + "migration-running": "En cours ...", + "migration-successful": "Migration terminée avec succès", + "migration-failed": "Migration en échec", "migrations": "Migrations", "migrations-admin-only": "Only board administrators can run migrations", "migrations-description": "Run data integrity checks and repairs for this board. Each migration can be executed individually.", - "no-issues-found": "No issues found", - "run-migration": "Run Migration", + "no-issues-found": "Aucun problème détecté", + "run-migration": "Lancer la migration", "run-comprehensive-migration-confirm": "This will perform a comprehensive migration to check and fix board data integrity. This may take a few moments. Continue?", "run-delete-duplicate-empty-lists-migration-confirm": "This will first convert any shared lists to per-swimlane lists, then delete empty lists that have a duplicate list with the same title containing cards. Only truly redundant empty lists will be removed. Continue?", "run-restore-lost-cards-migration-confirm": "This will create a 'Lost Cards' swimlane and restore all cards and lists with missing swimlaneId or listId. This only affects non-archived items. Continue?", @@ -1454,18 +1454,18 @@ "run-fix-all-file-urls-migration-confirm": "This will update all file attachment URLs on this board to use the correct storage backend. Continue?", "restore-lost-cards-nothing-to-restore": "No lost swimlanes, lists, or cards to restore", - "migration-progress-title": "Board Migration in Progress", + "migration-progress-title": "Migration du tableau en cours", "migration-progress-overall": "Overall Progress", - "migration-progress-current-step": "Current Step", + "migration-progress-current-step": "Étape courante", "migration-progress-status": "Statut", "migration-progress-details": "Détails", "migration-progress-note": "Please wait while we migrate your board to the latest structure...", "step-analyze-board-structure": "Analyze Board Structure", - "step-fix-orphaned-cards": "Fix Orphaned Cards", + "step-fix-orphaned-cards": "Corriger les cartes orphelines", "step-convert-shared-lists": "Convert Shared Lists", "step-ensure-per-swimlane-lists": "Ensure Per-Swimlane Lists", - "step-validate-migration": "Validate Migration", + "step-validate-migration": "Valider la migration", "step-fix-avatar-urls": "Fix Avatar URLs", "step-fix-attachment-urls": "Fix Attachment URLs", "step-analyze-lists": "Analyze Lists", @@ -1475,7 +1475,7 @@ "step-delete-duplicate-empty-lists": "Delete Duplicate Empty Lists", "step-ensure-lost-cards-swimlane": "Ensure Lost Cards Swimlane", "step-restore-lists": "Restore Lists", - "step-restore-cards": "Restore Cards", + "step-restore-cards": "Restaurer les cartes", "step-restore-swimlanes": "Restore Swimlanes", "step-fix-missing-ids": "Fix Missing IDs", "step-scan-users": "Checking board member avatars", diff --git a/imports/i18n/data/ja-HI.i18n.json b/imports/i18n/data/ja-HI.i18n.json index b97ac7887..ef8a053d6 100644 --- a/imports/i18n/data/ja-HI.i18n.json +++ b/imports/i18n/data/ja-HI.i18n.json @@ -1474,9 +1474,9 @@ "step-finalize": "Finalize", "step-delete-duplicate-empty-lists": "Delete Duplicate Empty Lists", "step-ensure-lost-cards-swimlane": "Ensure Lost Cards Swimlane", - "step-restore-lists": "Restore Lists", - "step-restore-cards": "Restore Cards", - "step-restore-swimlanes": "Restore Swimlanes", + "step-restore-lists": "リストをリストア", + "step-restore-cards": "カードをリストア", + "step-restore-swimlanes": "スイムレーンをリストア", "step-fix-missing-ids": "Fix Missing IDs", "step-scan-users": "Checking board member avatars", "step-scan-files": "Checking board file attachments", diff --git a/imports/i18n/data/ja.i18n.json b/imports/i18n/data/ja.i18n.json index 3eb2c6a4c..5403ebdcc 100644 --- a/imports/i18n/data/ja.i18n.json +++ b/imports/i18n/data/ja.i18n.json @@ -1474,9 +1474,9 @@ "step-finalize": "Finalize", "step-delete-duplicate-empty-lists": "Delete Duplicate Empty Lists", "step-ensure-lost-cards-swimlane": "Ensure Lost Cards Swimlane", - "step-restore-lists": "Restore Lists", - "step-restore-cards": "Restore Cards", - "step-restore-swimlanes": "Restore Swimlanes", + "step-restore-lists": "リストをリストア", + "step-restore-cards": "カードをリストア", + "step-restore-swimlanes": "スイムレーンをリストア", "step-fix-missing-ids": "Fix Missing IDs", "step-scan-users": "Checking board member avatars", "step-scan-files": "Checking board file attachments", diff --git a/imports/i18n/data/nl.i18n.json b/imports/i18n/data/nl.i18n.json index 1e3cda7f5..79b9a4b75 100644 --- a/imports/i18n/data/nl.i18n.json +++ b/imports/i18n/data/nl.i18n.json @@ -78,18 +78,18 @@ "activity-deleteComment": "aantekening verwijderd %s", "activity-receivedDate": "ontvangst datum gewijzigd naar %s van %s", "activity-startDate": "start datum gewijzigd naar %s van %s", - "allboards.starred": "Starred", + "allboards.starred": "Favorieten", "allboards.templates": "Templates", - "allboards.remaining": "Remaining", - "allboards.workspaces": "Workspaces", - "allboards.add-workspace": "Add Workspace", - "allboards.add-workspace-prompt": "Workspace name", - "allboards.add-subworkspace": "Add Subworkspace", - "allboards.add-subworkspace-prompt": "Subworkspace name", - "allboards.edit-workspace": "Edit workspace", - "allboards.edit-workspace-name": "Workspace name", - "allboards.edit-workspace-icon": "Workspace icon (markdown)", - "multi-selection-active": "Click checkboxes to select boards", + "allboards.remaining": "Resterend", + "allboards.workspaces": "Werkruimte", + "allboards.add-workspace": "Werkruimte toevoegen", + "allboards.add-workspace-prompt": "Werkruimtenaam", + "allboards.add-subworkspace": "Sub-werkruimte toevoegen", + "allboards.add-subworkspace-prompt": "Sub-werkruimtenaam", + "allboards.edit-workspace": "Wijzig werkruimte", + "allboards.edit-workspace-name": "Werkruimtenaam", + "allboards.edit-workspace-icon": "Werkruimteicoon (markdown)", + "multi-selection-active": "Vink de checkboxen om borden te selecteren", "activity-dueDate": "vervaldatum gewijzigd naar %s van %s", "activity-endDate": "einddatum gewijzigd naar %s van %s", "add-attachment": "Bijlage Toevoegen", @@ -936,7 +936,7 @@ "people-number": "Het aantal gebruikers is:", "swimlaneDeletePopup-title": "Swimlane verwijderen?", "swimlane-delete-pop": "Alle acties zullen verwijderd worden van de activiteiten feed en je kunt de swimlane niet terughalen. Er is geen herstelmogelijkheid.", - "restore-all": "Haal alles terug", + "restore-all": "Herstel alles", "delete-all": "Verwijder alles", "loading": "Laden, even geduld.", "previous_as": "laatste keer was", @@ -1417,70 +1417,70 @@ "back-to-settings": "Terug naar Instellingen", "board-id": "Bord ID", "board-migration": "Bord Migratie", - "board-migrations": "Board Migrations", + "board-migrations": "Bord Migraties", "card-show-lists-on-minicard": "Toon Lijsten op Minikaart", - "comprehensive-board-migration": "Comprehensive Board Migration", - "comprehensive-board-migration-description": "Performs comprehensive checks and fixes for board data integrity, including list ordering, card positions, and swimlane structure.", - "delete-duplicate-empty-lists-migration": "Delete Duplicate Empty Lists", - "delete-duplicate-empty-lists-migration-description": "Safely deletes empty duplicate lists. Only removes lists that have no cards AND have another list with the same title that contains cards.", - "lost-cards": "Lost Cards", - "lost-cards-list": "Restored Items", - "restore-lost-cards-migration": "Restore Lost Cards", - "restore-lost-cards-migration-description": "Finds and restores cards and lists with missing swimlaneId or listId. Creates a 'Lost Cards' swimlane to make all lost items visible again.", - "restore-all-archived-migration": "Restore All Archived", - "restore-all-archived-migration-description": "Restores all archived swimlanes, lists, and cards. Automatically fixes any missing swimlaneId or listId to make items visible.", - "fix-missing-lists-migration": "Fix Missing Lists", - "fix-missing-lists-migration-description": "Detects and repairs missing or corrupted lists in the board structure.", - "fix-avatar-urls-migration": "Fix Avatar URLs", - "fix-avatar-urls-migration-description": "Updates avatar URLs for board members to use the correct storage backend and fixes broken avatar references.", - "fix-all-file-urls-migration": "Fix All File URLs", - "fix-all-file-urls-migration-description": "Updates all file attachment URLs on this board to use the correct storage backend and fixes broken file references.", - "migration-needed": "Migration Needed", + "comprehensive-board-migration": "Uitgebreide Bord Migratie", + "comprehensive-board-migration-description": "Voert uitgebreide controles en reparaties uit voor bord data-integriteit, inclusief lijst sortering, kaart posities en swimlane-structuur.", + "delete-duplicate-empty-lists-migration": "Verwijder Dubbele Lege Lijsten", + "delete-duplicate-empty-lists-migration-description": "Verwijderd veilig lege dubbele lijsten. Verwijderd alleen lijsten die geen kaarten bevatten EN waar een andere lijst bestaat met dezelfde titel die wel kaarten bevat.", + "lost-cards": "Verloren Kaarten", + "lost-cards-list": "Herstelde Items", + "restore-lost-cards-migration": "Herstel Verloren Kaarten", + "restore-lost-cards-migration-description": "Vind en herstel kaarten en lijsten met missende swimlane-ID of list-ID. Hier wordt een 'Verloren Kaarten'-swimlane gemaakt om alles verloren items weer zichtbaar te maken.", + "restore-all-archived-migration": "Herstel Alles ui Archief", + "restore-all-archived-migration-description": "Herstel alle ge-archiveerde swimlanes, lijsten en kaarten. Hierbij worden automatisch de missende swimlane-ID of lijs-ID gerapareerd om de items weer zichtbaar te maken.", + "fix-missing-lists-migration": "Repareer Missende Lijsten", + "fix-missing-lists-migration-description": "Detecteer en repareer missende of corrupte lijsten in de bordstructuur.", + "fix-avatar-urls-migration": "Repareer Avatar URL's", + "fix-avatar-urls-migration-description": "Werkt de avatar URL's van de bord-leden bij zodat de juiste opslagmethode gebruikt wordt en het repareert defecte avatar verwijzingen.", + "fix-all-file-urls-migration": "Repareer alle bestand URL's", + "fix-all-file-urls-migration-description": "Werkt alle bestandsbijlagen URL's op dit bord bij naar de juiste opslagmethode en repareert defecte bestandsverwijzingen.", + "migration-needed": "Migratie Nodig", "migration-complete": "Voltooid", - "migration-running": "Running...", - "migration-successful": "Migration completed successfully", - "migration-failed": "Migration failed", - "migrations": "Migrations", - "migrations-admin-only": "Only board administrators can run migrations", - "migrations-description": "Run data integrity checks and repairs for this board. Each migration can be executed individually.", - "no-issues-found": "No issues found", - "run-migration": "Run Migration", - "run-comprehensive-migration-confirm": "This will perform a comprehensive migration to check and fix board data integrity. This may take a few moments. Continue?", - "run-delete-duplicate-empty-lists-migration-confirm": "This will first convert any shared lists to per-swimlane lists, then delete empty lists that have a duplicate list with the same title containing cards. Only truly redundant empty lists will be removed. Continue?", - "run-restore-lost-cards-migration-confirm": "This will create a 'Lost Cards' swimlane and restore all cards and lists with missing swimlaneId or listId. This only affects non-archived items. Continue?", - "run-restore-all-archived-migration-confirm": "This will restore ALL archived swimlanes, lists, and cards, making them visible again. Any items with missing IDs will be automatically fixed. This cannot be easily undone. Continue?", - "run-fix-missing-lists-migration-confirm": "This will detect and repair missing or corrupted lists in the board structure. Continue?", - "run-fix-avatar-urls-migration-confirm": "This will update avatar URLs for board members to use the correct storage backend. Continue?", - "run-fix-all-file-urls-migration-confirm": "This will update all file attachment URLs on this board to use the correct storage backend. Continue?", - "restore-lost-cards-nothing-to-restore": "No lost swimlanes, lists, or cards to restore", + "migration-running": "In Uitvoering...", + "migration-successful": "Migratie succesvol uitgevoerd", + "migration-failed": "Migratie mislukt", + "migrations": "Migraties", + "migrations-admin-only": "Alleen bord-beherders kunnen migraties uitvoeren", + "migrations-description": "Voer data-integriteits controles en reparaties uit op dit bord. Elke migratie kan afzonderlijk uitgevoerd worden.", + "no-issues-found": "Geen problemen gevonden", + "run-migration": "Voer Migratie Uit", + "run-comprehensive-migration-confirm": "Dit voert een uitgebreide migratie uit en controleert en repareert de data-integriteit op dit bord. Dit kan even duren. Doorgaan?", + "run-delete-duplicate-empty-lists-migration-confirm": "Dit converteert eventuele gedeelde lijsten naar lijsten per swimlane waarna lege lijsten verwijderd worden die dubbel zijn met een lijst met dezelfde titel en wel kaarten bevatten. Alleen echt dubbele lege lijsten worden verwijderd. Doorgaan?", + "run-restore-lost-cards-migration-confirm": "Dit creëert een 'Verloren Kaarten' swimlane en hierin worden kaarten en lijsten hersteld met een missende swimlane-ID of lijst-ID. Dit heeft alleen gevolgen voor niet gearchiveerde items. Doorgaan?", + "run-restore-all-archived-migration-confirm": "Dit herstelt alle gearchiveerde swimlanes, lijsten en kaarten waardoor ze weer zichtbaar worden. Items met een missende ID zulle automatisch gerepareerd worden. Doorgaan?", + "run-fix-missing-lists-migration-confirm": "Dit detecteert en repareert missende of corrupte lijsten in de bord-structuur. Doorgaan?", + "run-fix-avatar-urls-migration-confirm": "Dit werkt alle avatar URL's van de bord-leden bij naar de juiste opslagmethode. Doorgaan?", + "run-fix-all-file-urls-migration-confirm": "dit werkt alle bestandsbijlagen URL's bij naar de juiste opslagmethode. Doorgaan?", + "restore-lost-cards-nothing-to-restore": "Geen verloren swinmlanes, lijsten of kaarten om te herstellen", - "migration-progress-title": "Board Migration in Progress", + "migration-progress-title": "Bord Migratie in Uitvoering", "migration-progress-overall": "Algehele Voortgang", - "migration-progress-current-step": "Current Step", + "migration-progress-current-step": "Huidige Stap", "migration-progress-status": "Status", "migration-progress-details": "Details", - "migration-progress-note": "Please wait while we migrate your board to the latest structure...", + "migration-progress-note": "Wacht tot we jouw bord gemigreerd hebben naar de actuele structuur...", - "step-analyze-board-structure": "Analyze Board Structure", - "step-fix-orphaned-cards": "Fix Orphaned Cards", - "step-convert-shared-lists": "Convert Shared Lists", - "step-ensure-per-swimlane-lists": "Ensure Per-Swimlane Lists", - "step-validate-migration": "Validate Migration", - "step-fix-avatar-urls": "Fix Avatar URLs", - "step-fix-attachment-urls": "Fix Attachment URLs", - "step-analyze-lists": "Analyze Lists", - "step-create-missing-lists": "Create Missing Lists", - "step-update-cards": "Update Cards", - "step-finalize": "Finalize", - "step-delete-duplicate-empty-lists": "Delete Duplicate Empty Lists", - "step-ensure-lost-cards-swimlane": "Ensure Lost Cards Swimlane", - "step-restore-lists": "Restore Lists", - "step-restore-cards": "Restore Cards", - "step-restore-swimlanes": "Restore Swimlanes", - "step-fix-missing-ids": "Fix Missing IDs", - "step-scan-users": "Checking board member avatars", - "step-scan-files": "Checking board file attachments", - "step-fix-file-urls": "Fixing file URLs", + "step-analyze-board-structure": "Bordstructuur Analyseren", + "step-fix-orphaned-cards": "Repareer Verweesde Kaarten", + "step-convert-shared-lists": "Converteer Gedeelde Lijsten", + "step-ensure-per-swimlane-lists": "Maak Per-Swimlane Lijsten", + "step-validate-migration": "Valideer Migratie", + "step-fix-avatar-urls": "Repareer Avatar URL's", + "step-fix-attachment-urls": "Repareer Bijlage URL's", + "step-analyze-lists": "Analyseer Lijsten", + "step-create-missing-lists": "Maak Missende Lijsten", + "step-update-cards": "Werk Kaarten Bij", + "step-finalize": "Beëindig", + "step-delete-duplicate-empty-lists": "Verwijder Dubbele Lege Lijsten", + "step-ensure-lost-cards-swimlane": "Maak Verloren Kaarten Swimlane", + "step-restore-lists": "Herstel Lijsten", + "step-restore-cards": "Herstel Kaarten", + "step-restore-swimlanes": "Herstel Swimlanes", + "step-fix-missing-ids": "Herstel Missende ID's", + "step-scan-users": "Bord-leden avatars controleren", + "step-scan-files": "Bord bestandsbijlagen controleren", + "step-fix-file-urls": "Herstel bestand URL's", "cleanup": "Opschonen", "cleanup-old-jobs": "Schoon Oude Taken Op", "completed": "Afgewerkt", diff --git a/imports/i18n/data/pt-BR.i18n.json b/imports/i18n/data/pt-BR.i18n.json index 6c2d764ed..1a44664bb 100644 --- a/imports/i18n/data/pt-BR.i18n.json +++ b/imports/i18n/data/pt-BR.i18n.json @@ -78,18 +78,18 @@ "activity-deleteComment": "comentário excluído %s", "activity-receivedDate": "editou recebido para %s de %s", "activity-startDate": "editou data início para %s de %s", - "allboards.starred": "Starred", + "allboards.starred": "Favoritado", "allboards.templates": "Modelos", - "allboards.remaining": "Remaining", - "allboards.workspaces": "Workspaces", - "allboards.add-workspace": "Add Workspace", - "allboards.add-workspace-prompt": "Workspace name", - "allboards.add-subworkspace": "Add Subworkspace", - "allboards.add-subworkspace-prompt": "Subworkspace name", - "allboards.edit-workspace": "Edit workspace", - "allboards.edit-workspace-name": "Workspace name", - "allboards.edit-workspace-icon": "Workspace icon (markdown)", - "multi-selection-active": "Click checkboxes to select boards", + "allboards.remaining": "Restante", + "allboards.workspaces": "Áreas de trabalho", + "allboards.add-workspace": "Adicionar Área de trabalho", + "allboards.add-workspace-prompt": "Nome da Área de trabalho", + "allboards.add-subworkspace": "Adicionar Subárea de trabalho", + "allboards.add-subworkspace-prompt": "Nome da Subárea de trabalho", + "allboards.edit-workspace": "Editar Área de trabalho", + "allboards.edit-workspace-name": "Nome da Área de trabalho", + "allboards.edit-workspace-icon": "Ícone da Área de trabalho (markdown)", + "multi-selection-active": "Clique nas caixas de seleção para selecionar os quadros", "activity-dueDate": "editou prazo final para %s de %s", "activity-endDate": "editou concluído para %s de %s", "add-attachment": "Adicionar Anexos", @@ -1419,68 +1419,68 @@ "board-migration": "Migração de Quadro", "board-migrations": "Migração de Quadros", "card-show-lists-on-minicard": "Mostrar Listas no Mini cartão", - "comprehensive-board-migration": "Comprehensive Board Migration", - "comprehensive-board-migration-description": "Performs comprehensive checks and fixes for board data integrity, including list ordering, card positions, and swimlane structure.", - "delete-duplicate-empty-lists-migration": "Delete Duplicate Empty Lists", - "delete-duplicate-empty-lists-migration-description": "Safely deletes empty duplicate lists. Only removes lists that have no cards AND have another list with the same title that contains cards.", - "lost-cards": "Lost Cards", - "lost-cards-list": "Restored Items", - "restore-lost-cards-migration": "Restore Lost Cards", - "restore-lost-cards-migration-description": "Finds and restores cards and lists with missing swimlaneId or listId. Creates a 'Lost Cards' swimlane to make all lost items visible again.", - "restore-all-archived-migration": "Restore All Archived", - "restore-all-archived-migration-description": "Restores all archived swimlanes, lists, and cards. Automatically fixes any missing swimlaneId or listId to make items visible.", - "fix-missing-lists-migration": "Fix Missing Lists", - "fix-missing-lists-migration-description": "Detects and repairs missing or corrupted lists in the board structure.", - "fix-avatar-urls-migration": "Fix Avatar URLs", - "fix-avatar-urls-migration-description": "Updates avatar URLs for board members to use the correct storage backend and fixes broken avatar references.", - "fix-all-file-urls-migration": "Fix All File URLs", - "fix-all-file-urls-migration-description": "Updates all file attachment URLs on this board to use the correct storage backend and fixes broken file references.", - "migration-needed": "Migration Needed", + "comprehensive-board-migration": "Migração de Quadros abrangente", + "comprehensive-board-migration-description": "Realiza verificações e correções abrangentes para a integridade dos dados do quadro, incluindo a ordem da lista, as posições dos cartões e a estrutura das raias.", + "delete-duplicate-empty-lists-migration": "Apagar Listas Vazias Duplicadas", + "delete-duplicate-empty-lists-migration-description": "Exclui com segurança listas duplicadas vazias. Remove apenas listas que não contêm cartões E que possuem outra lista com o mesmo título que contém cartões.", + "lost-cards": "Cartões Perdidos", + "lost-cards-list": "Itens Recuperados", + "restore-lost-cards-migration": "Recuperar Cartões Perdidos", + "restore-lost-cards-migration-description": "Encontra e restaura cartões e listas com ID de raia ou ID de lista ausentes. Cria uma raia \"Cartões Perdidos\" para tornar todos os itens perdidos visíveis novamente.", + "restore-all-archived-migration": "Recuperar Todos Arquivados", + "restore-all-archived-migration-description": "Restaura todas as raias, listas e cartões arquivados. Corrige automaticamente qualquer ID de raia ou ID de lista ausente para tornar os itens visíveis.", + "fix-missing-lists-migration": "Corrigir Listas Ausentes", + "fix-missing-lists-migration-description": "Detecta e repara listas ausentes ou corrompidas na estrutura do quadro.", + "fix-avatar-urls-migration": "Corrigir URLs de Avatar", + "fix-avatar-urls-migration-description": "Atualiza os URLs dos avatares dos membros do quadro para que utilizem o backend de armazenamento correto e corrige referências de avatar quebradas.", + "fix-all-file-urls-migration": "Corrigir todas URLs de arquivos", + "fix-all-file-urls-migration-description": "Atualiza todas URLs de arquivos de anexo neste quadro para usar o backend de armazenamento correto e corrige referências de arquivos quebradas.", + "migration-needed": "Migração necessária", "migration-complete": "Concluído", - "migration-running": "Running...", - "migration-successful": "Migration completed successfully", - "migration-failed": "Migration failed", - "migrations": "Migrations", - "migrations-admin-only": "Only board administrators can run migrations", - "migrations-description": "Run data integrity checks and repairs for this board. Each migration can be executed individually.", - "no-issues-found": "No issues found", - "run-migration": "Run Migration", - "run-comprehensive-migration-confirm": "This will perform a comprehensive migration to check and fix board data integrity. This may take a few moments. Continue?", - "run-delete-duplicate-empty-lists-migration-confirm": "This will first convert any shared lists to per-swimlane lists, then delete empty lists that have a duplicate list with the same title containing cards. Only truly redundant empty lists will be removed. Continue?", - "run-restore-lost-cards-migration-confirm": "This will create a 'Lost Cards' swimlane and restore all cards and lists with missing swimlaneId or listId. This only affects non-archived items. Continue?", - "run-restore-all-archived-migration-confirm": "This will restore ALL archived swimlanes, lists, and cards, making them visible again. Any items with missing IDs will be automatically fixed. This cannot be easily undone. Continue?", - "run-fix-missing-lists-migration-confirm": "This will detect and repair missing or corrupted lists in the board structure. Continue?", - "run-fix-avatar-urls-migration-confirm": "This will update avatar URLs for board members to use the correct storage backend. Continue?", - "run-fix-all-file-urls-migration-confirm": "This will update all file attachment URLs on this board to use the correct storage backend. Continue?", - "restore-lost-cards-nothing-to-restore": "No lost swimlanes, lists, or cards to restore", + "migration-running": "Executando...", + "migration-successful": "Migração concluída com sucesso.", + "migration-failed": "A migração falhou", + "migrations": "Migrações", + "migrations-admin-only": "Somente os administradores do quadro podem executar migrações.", + "migrations-description": "Execute verificações de integridade de dados e reparos para este quadro. Cada migração pode ser executada individualmente.", + "no-issues-found": "Nenhum problema encontrado", + "run-migration": "Executar Migração", + "run-comprehensive-migration-confirm": "Isso realizará uma migração completa para verificar e corrigir a integridade dos dados do quadro. Isso pode levar alguns instantes. Continuar?", + "run-delete-duplicate-empty-lists-migration-confirm": "Primeiro, isso converterá todas as listas compartilhadas em listas por raia e, em seguida, excluirá as listas vazias que contêm listas duplicadas com o mesmo título e que possuem cartões. Somente as listas vazias realmente redundantes serão removidas. Continuar?", + "run-restore-lost-cards-migration-confirm": "Isso criará uma raia \"Cartões Perdidos\" e restaurará todos os cartões e listas com ID de raia ou ID de lista ausentes. Isso afeta apenas itens não arquivados. Continuar?", + "run-restore-all-archived-migration-confirm": "Isso restaurará TODAS as raias, listas e cartões arquivados, tornando-os visíveis novamente. Quaisquer itens com IDs ausentes serão corrigidos automaticamente. Esta ação não pode ser desfeita facilmente. Continuar?", + "run-fix-missing-lists-migration-confirm": "Isso detectará e corrigirá listas ausentes ou corrompidas na estrutura do quadro. Continuar?", + "run-fix-avatar-urls-migration-confirm": "Isso atualizará as URLs dos avatares dos membros do quadro para usar o backend de armazenamento correto. Continuar?", + "run-fix-all-file-urls-migration-confirm": "Isso atualizará todas URLs de arquivos de anexos neste quadro para usar o servidor de armazenamento correto. Continuar?", + "restore-lost-cards-nothing-to-restore": "Sem raias, listas ou cartões perdidos para restaurar.", - "migration-progress-title": "Board Migration in Progress", + "migration-progress-title": "Migração do Quadro em Andamento", "migration-progress-overall": "Progresso Geral", - "migration-progress-current-step": "Current Step", + "migration-progress-current-step": "Etapa Atual", "migration-progress-status": "Status", "migration-progress-details": "Detalhes", - "migration-progress-note": "Please wait while we migrate your board to the latest structure...", + "migration-progress-note": "Aguarde enquanto migramos seu quadro para a estrutura mais recente...", - "step-analyze-board-structure": "Analyze Board Structure", - "step-fix-orphaned-cards": "Fix Orphaned Cards", - "step-convert-shared-lists": "Convert Shared Lists", - "step-ensure-per-swimlane-lists": "Ensure Per-Swimlane Lists", - "step-validate-migration": "Validate Migration", - "step-fix-avatar-urls": "Fix Avatar URLs", - "step-fix-attachment-urls": "Fix Attachment URLs", - "step-analyze-lists": "Analyze Lists", - "step-create-missing-lists": "Create Missing Lists", - "step-update-cards": "Update Cards", - "step-finalize": "Finalize", - "step-delete-duplicate-empty-lists": "Delete Duplicate Empty Lists", - "step-ensure-lost-cards-swimlane": "Ensure Lost Cards Swimlane", - "step-restore-lists": "Restore Lists", - "step-restore-cards": "Restore Cards", - "step-restore-swimlanes": "Restore Swimlanes", - "step-fix-missing-ids": "Fix Missing IDs", - "step-scan-users": "Checking board member avatars", - "step-scan-files": "Checking board file attachments", - "step-fix-file-urls": "Fixing file URLs", + "step-analyze-board-structure": "Analisar a Estrutura do Quadro", + "step-fix-orphaned-cards": "Corrigir Cartões Órfãos", + "step-convert-shared-lists": "Converter Listas Compartilhadas", + "step-ensure-per-swimlane-lists": "Garantir listas por raia", + "step-validate-migration": "Validar Migração", + "step-fix-avatar-urls": "Corrigir URLs de Avatar", + "step-fix-attachment-urls": "Corrigir URLs de anexos", + "step-analyze-lists": "Analisar Listas", + "step-create-missing-lists": "Criar Listas Faltantes", + "step-update-cards": "Atualizar Cartões", + "step-finalize": "Finalizar", + "step-delete-duplicate-empty-lists": "Apagar Listas Vazias Duplicadas", + "step-ensure-lost-cards-swimlane": "Garantir a Raia dos Cartões Perdidos", + "step-restore-lists": "Restaurar Listas", + "step-restore-cards": "Restaurar Cartões", + "step-restore-swimlanes": "Restaurar Raias", + "step-fix-missing-ids": "Corrigir IDs ausentes", + "step-scan-users": "Verificando os avatares dos membros do quadro", + "step-scan-files": "Verificando arquivos de anexos do quadro", + "step-fix-file-urls": "Corrigindo URLs de arquivos", "cleanup": "Limpeza", "cleanup-old-jobs": "Limpar Trabalhos Antigos", "completed": "Completado", diff --git a/imports/i18n/data/sr.i18n.json b/imports/i18n/data/sr.i18n.json index 1fc401068..4fbf8a0ba 100644 --- a/imports/i18n/data/sr.i18n.json +++ b/imports/i18n/data/sr.i18n.json @@ -1,249 +1,249 @@ { "accept": "Прихвати", "act-activity-notify": "Обавештење о последњим променама", - "act-addAttachment": "додао прилог __attachment__ на картици __card__ на деоници __list__ на стази __swimlane__ у пословној књизи __board__", - "act-deleteAttachment": "уклонио прилог __attachment__ са картице __card__ на деоници __list__ стазе __swimlane__ у пословној књизи __board__", - "act-addSubtask": "додао подзадатак __subtask__ на картицу __card__ на деоници __list__ стазе __swimlane__ у пословној књизи __board__", - "act-addLabel": "Залепио налепницу __label__ на картицу __card__ на деоници __list__ стазе __swimlane__ у пословној књизи __board__", - "act-addedLabel": "Залепљена је налепница __label__ на картицу __card__ на деоници __list__ стазе __swimlane__ у пословној књизи __board__", - "act-removeLabel": "Скинуо налепницу __label__ са картице __card__ на деоници __list__ стазе __swimlane__ у пословној књизи __board__", - "act-removedLabel": "Скинута је налепница __label__ са картице __card__ на деоници __list__ стазе __swimlane__ у пословној књизи __board__", - "act-addChecklist": "додао списак за обавити __checklist__ на задатак __card__ на деоници __list__ стазе __swimlane__ у пословној књизи __board__", - "act-addChecklistItem": "додао ставку __checklistItem__ на списак за обавити __checklist__ задатка __card__ на деоници __list__ стазе __swimlane__ у пословној књизи __board__", - "act-removeChecklist": "уклонио списак за обавити __checklist__ са задатка __card__ на деоници __list__ стазе __swimlane__ у пословној књизи __board__", - "act-removeChecklistItem": "уклонио ставку __checklistItem__ са списка за обавити __checkList__ задатка __card__ на деоници __list__ стазе __swimlane__ у пословној књизи __board__", - "act-checkedItem": "обавио __checklistItem__ са списка __checklist__ на задатку __card__ на деоници __list__ стазе __swimlane__ у пословној књизи __board__", - "act-uncheckedItem": "вратио да се поново обави __checklistItem__ са списка __checklist__ на задатку __card__ на деоници __list__ стазе __swimlane__ у пословној књизи __board__", - "act-completeChecklist": "обавио све са списка __checklist__ на задатку __card__ на деоници __list__ стазе __swimlane__ у пословној књизи __board__", - "act-uncompleteChecklist": "није обављено све са списка __checklist__ на задатку __card__ на деоници __list__ стазе __swimlane__ у пословној књизи __board__", - "act-addComment": "изразио мишљење на картици __card__: __comment__ на деоници __list__ стазе __swimlane__ у пословној књизи __board__", - "act-editComment": "изменио мишљење на картици __card__:__comment__ на деоници __list__ стазе __swimlane__ у пословној књизи __board__", - "act-deleteComment": "повукао мишљење на картици __card__:__comment__ на деоници __list__ стазе __swimlane__ у пословној књизи __board__", - "act-createBoard": "Отворена је сасвим нова пословна књига __board__", - "act-createSwimlane": "Додата је сасвим нова стаза __swimlane__ у пословној књизи __board__", - "act-createCard": "додата је сасвим нова картица са задатком __card__ на деоници __list__ на стази __swimlane__ у пословној књизи __board__", - "act-createCustomField": "направљено сасвим ново поље __customField__ у пословној књизи __board__", - "act-deleteCustomField": "избрисано сасвим ново поље __customField__ у пословној књизи __board__", - "act-setCustomField": "измењено сасвим ново поље __customField__:__customFieldValue__ на картици са задатком __card__ на деоници __list__ стазе __swimlane__ у пословној књизи __board", - "act-createList": "придодата деоница __list__ у пословној књизи __board__", - "act-addBoardMember": "примљен сарадник __member__ у пословној књизи __board__", - "act-archivedBoard": "Пословна књига __board__ је премештена у архив", - "act-archivedCard": "Картица са задатком __card__ са деонице __list__ стазе __swimlane__ из пословне књиге __board__ однешена у архив", - "act-archivedList": "Деоница __list__ стазе __swimlane__ из пословне књиге __board__ премештена у архив", - "act-archivedSwimlane": "Стаза __swimlane__ из пословне књиге __board__ однешена у архив", - "act-importBoard": "увезао књигу пословања __board__", - "act-importCard": "увезао картицу са задатком __card__ на деоници __list__ на стази __swimlane__ у пословној књизи __board__", - "act-importList": "увезао деоницу __list__ на стазу __swimlane__ у пословној књизи __board__", - "act-joinMember": "примио сарадника __member__ на картицу __card__ на деоници __list__ стазе __swimlane__ у пословној књизи __board__", - "act-moveCard": "преместио картицу __card__ из пословне књиге __board__ са старе деонице __oldList__ старе стазе __oldSwimlane__ на деоницу __list__ нове стазе __swimlane__", - "act-moveCardToOtherBoard": "преместио картицу __card__ са старе деонице __oldList__ старе стазе __oldSwimlane__ из старе пословне књиге __oldBoard__ на деоницу __list__ стазе __swimlane__ у пословну књигу __board__", - "act-removeBoardMember": "удаљио сарадника __member__ из пословне књиге __board__", - "act-restoredCard": "обновио картицу са задатком __card__ на деоници __list__ у стази __swimlane__ у пословној књизи __board__", - "act-unjoinMember": "удаљио сарадника __member__ са картице __card__ на деоници __list__ стазе __swimlane__ из пословне књиге __board__", + "act-addAttachment": "унео предметну грађу __attachment__ у предмет __card__ у делу __list__ поступка __swimlane__ међу списе __board__", + "act-deleteAttachment": "уклонио предметну грађу __attachment__ из предмета __card__ у делу __list__ поступка __swimlane__ из списа __board__", + "act-addSubtask": "је издвојио посао __subtask__ из предмета __card__ у __list__ делу __swimlane__ поступка у списима __board__", + "act-addLabel": "Залепио налепницу __label__ на омот предмета __card__ у делу __list__ поступка __swimlane__ у списима __board__", + "act-addedLabel": "Залепљена је налепница __label__ на омот предмета __card__ у делу __list__ поступка __swimlane__ у списима __board__", + "act-removeLabel": "Скинуо је налепницу __label__ са омота предмета __card__ у делу __list__ поступка __swimlane__ из списа __board__", + "act-removedLabel": "Скинута је налепница __label__ са омота предмета __card__ у делу __list__ поступка __swimlane__ из списа __board__", + "act-addChecklist": "је додао списак радњи __checklist__ у предмет __card__ у __list__ делу поступка __swimlane__ заведеног у списима __board__", + "act-addChecklistItem": "је додао ставку __checklistItem__ на списак __checklist__ у предмету __card__ у __list__ поступка __swimlane__ у списима __board__", + "act-removeChecklist": "је уклонио списак радњи __checklist__ из предмета __card__ у __list__ делу поступка __swimlane__ у списима __board__", + "act-removeChecklistItem": "је уклонио ставку __checklistItem__ са списка __checkList__ предмета __card__ у __list__ делу __swimlane__ у списима __board__", + "act-checkedItem": "је прецртао ставку __checklistItem__ са списка __checklist__ у предмету __card__ у __list__ делу __swimlane__ у списима __board__", + "act-uncheckedItem": "је вратио да се поново обави __checklistItem__ радња са списка __checklist__ у предмету __card__ у __list__ делу __swimlane__ у списима __board__", + "act-completeChecklist": "је испунио све са списка __checklist__ у предмету __card__ у __list__ делу __swimlane__ у списима __board__", + "act-uncompleteChecklist": "није испунио све са списка __checklist__ у предмету __card__ у __list__ делу __swimlane__ у списима __board__", + "act-addComment": "изразио следеће мишљење у расправи у предмету __card__: __comment__ у делу __list__ поступка __swimlane__ у списима __board__", + "act-editComment": "допунио став у расправи у предмету __card__:__comment__ у делу __list__ поступка __swimlane__ у списима __board__", + "act-deleteComment": "повукао мишљење у предмету __card__:__comment__ у делу __list__ поступка __swimlane__ у списима __board__", + "act-createBoard": "је увео нове списе са деловодним бројем __board__", + "act-createSwimlane": "је додао нови предметни ток __swimlane__ међу списе __board__", + "act-createCard": "је додао нови предмет __card__ у делу __list__ поступка __swimlane__ међу списе __board__", + "act-createCustomField": "је додао нову рубрику __customField__ међу списе __board__", + "act-deleteCustomField": "је избрисао рубрику __customField__ из списа __board__", + "act-setCustomField": "је изменио запис у рубрици __customField__:__customFieldValue__ у предмету __card__ у __list__ делу поступка __swimlane__ у списима __board", + "act-createList": "је придодао део поступка __list__ међу списе __board__", + "act-addBoardMember": "примљен сарадник __member__ за рад на списима __board__", + "act-archivedBoard": "Списи __board__ су спаковани у архив", + "act-archivedCard": "Предмет __card__ из дела __list__ поступка __swimlane__ из списа __board__ је спакован у архив", + "act-archivedList": "Део __list__ поступка __swimlane__ из списа __board__ је спакован у архив", + "act-archivedSwimlane": "Врста поступка __swimlane__ из списа __board__ је спакована у архив", + "act-importBoard": "унео списе __board__", + "act-importCard": "унео предмет __card__ у делу __list__ поступка __swimlane__ у списе __board__", + "act-importList": "унео део __list__ поступка __swimlane__ у списе __board__", + "act-joinMember": "примљен сарадник __member__ да ради на предмету __card__ у делу __list__ поступка __swimlane__ у списима __board__", + "act-moveCard": "преместио предмет __card__ из списа __board__ са старог дела __oldList__ поступка __oldSwimlane__ у нови део __list__ поступка __swimlane__", + "act-moveCardToOtherBoard": "преместио предмет __card__ са старог дела __oldList__ поступка __oldSwimlane__ из старих списа __oldBoard__ у део __list__ поступка __swimlane__ међу списе __board__", + "act-removeBoardMember": "изузео сарадника __member__ из списа __board__", + "act-restoredCard": "опоравио предмет __card__ у делу поступка __list__ из поступка __swimlane__ из списа __board__", + "act-unjoinMember": "изузео сарадника __member__ са предмета __card__ у делу поступка __list__ из поступка __swimlane__ из списа __board__", "act-withBoardTitle": "__board__", "act-withCardTitle": "[__board__] __card__", "actions": "Радње", - "activities": "Последње промене", + "activities": "Дневник промена", "activity": "Последња промена", - "activity-added": "додао %s у %s", + "activity-added": "је додао %s у %s", "activity-archived": "%s премештено у архиву", - "activity-attached": "приложио %s у %s", - "activity-created": "направио %s", - "activity-changedListTitle": "renamed list to %s", - "activity-customfield-created": "направио сасвим ново поље %s", + "activity-attached": "унео предметну грађу %s у %s", + "activity-created": "је направио %s", + "activity-changedListTitle": "је преименовао део поступка у %s", + "activity-customfield-created": "је додао рубрику %s", "activity-excluded": "изузет %s из %s", "activity-imported": "увезао %s у %s из %s", "activity-imported-board": "увезао %s из %s", "activity-joined": "спојио %s", - "activity-moved": "преместио %s из %s у %s", + "activity-moved": "је преместио %s из %s у %s", "activity-on": "на %s", "activity-removed": "уклонио %s из %s", "activity-sent": "послао %s %s-у", "activity-unjoined": "раставио %s", - "activity-subtask-added": "додао подзадатак за %s", - "activity-checked-item": "обављено %s са списка %s од %s", - "activity-unchecked-item": "није обављено %s са списка %s од %s", - "activity-checklist-added": "деоница је додата у %s", - "activity-checklist-removed": "уклонио списак за обавити са %s", - "activity-checklist-completed": "испуњеност задатака са списка %s од %s", - "activity-checklist-uncompleted": "списак није испуњен %s од %s", - "activity-checklist-item-added": "придодата ставка на списак за обавити '%s' у %s", - "activity-checklist-item-removed": "избачена ставка са списка за обавити из '%s' у %s", + "activity-subtask-added": "је издвојио посао из предмета %s", + "activity-checked-item": "пријављује да је обавио задатак %s са списка %s у предмету %s", + "activity-unchecked-item": "повлачи пријаву да је обавио задатак %s са списка %s у предмету %s", + "activity-checklist-added": "је унео предметну радњу у предмет %s", + "activity-checklist-removed": "је уклонио предметну радњу из предмета %s", + "activity-checklist-completed": "је испунио све са списка предметне радње %s у предмету %s", + "activity-checklist-uncompleted": "ипак није испунио све са списка предметне радње %s у предмету %s", + "activity-checklist-item-added": "је придодао ставку '%s' у предметну радњу %s", + "activity-checklist-item-removed": "је ставку '%s' избацио из предметне радње %s", "add": "Додај", - "activity-checked-item-card": "испуњена обавеза %s у списку за обавити %s", - "activity-unchecked-item-card": "неиспуњена обавеза %s на списку за обавити %s", - "activity-checklist-completed-card": "испуњене обавезе са списка __checklist__ на картици __card__ деонице __list__ на стази __swimlane__ пословне књиге __board__", + "activity-checked-item-card": "је испунио обавезу %s као ставку предметне радње %s", + "activity-unchecked-item-card": "је означио обавезу %s ипак као неиспуњен део предметне радње %s", + "activity-checklist-completed-card": "испуњене обавезе са списка __checklist__ у оквиру предмета __card__ у __list__ делу поступка __swimlane__ заведеног у списима __board__", "activity-checklist-uncompleted-card": "нису испуњене обавезе са списка %s", "activity-editComment": "променио мишљење", "activity-deleteComment": "повучено мишљење", - "activity-receivedDate": "измењен датум пријема на %s са %s", - "activity-startDate": "измењен почетни датум на %s са %s", - "allboards.starred": "Starred", - "allboards.templates": "Предлошци", - "allboards.remaining": "Remaining", - "allboards.workspaces": "Workspaces", - "allboards.add-workspace": "Add Workspace", - "allboards.add-workspace-prompt": "Workspace name", - "allboards.add-subworkspace": "Add Subworkspace", - "allboards.add-subworkspace-prompt": "Subworkspace name", - "allboards.edit-workspace": "Edit workspace", - "allboards.edit-workspace-name": "Workspace name", - "allboards.edit-workspace-icon": "Workspace icon (markdown)", - "multi-selection-active": "Click checkboxes to select boards", - "activity-dueDate": "измењен крајњи рок на %s са %s", - "activity-endDate": "измењено време завршетка на %s са %s", - "add-attachment": "Додај прилог", - "add-board": "Уведи пословну књигу", - "add-template": "Додај предложак", - "add-card": "Додај картицу са задатком", - "add-card-to-top-of-list": "Додај картицу/задатак на врх деонице", - "add-card-to-bottom-of-list": "Додај картицу/задатак на дно деонице", - "setListWidthPopup-title": "Set Widths", - "set-list-width": "Set Widths", - "set-list-width-value": "Set Min & Max Widths (pixels)", - "list-width-error-message": "List widths must be integers greater than 100", - "keyboard-shortcuts-enabled": "Keyboard shortcuts enabled. Click to disable.", - "keyboard-shortcuts-disabled": "Keyboard shortcuts disabled. Click to enable.", - "setSwimlaneHeightPopup-title": "Подеси висину стазе", - "set-swimlane-height": "Подеси висину стазе", - "set-swimlane-height-value": "Висина стазе (у пикселима)", - "swimlane-height-error-message": "Висина стазе мора бити позитиван број", - "add-swimlane": "Додај стазу", - "add-subtask": "Додај подзадатак", - "add-checklist": "Додај списак за обавити", - "add-checklist-item": "Додај нову ставку на списак", - "close-add-checklist-item": "Затвори образац за додавање ставке на списак за обавити", - "close-edit-checklist-item": "Затвори образац за уређивање ставке списка за обавити", - "convertChecklistItemToCardPopup-title": "Претвори ставку за обавити у облик картице са задатком", - "add-cover": "Додајте насловну слику на мини картицу", + "activity-receivedDate": "је изменио датум и време пријема на %s у предмету %s", + "activity-startDate": "је изменио датум и време %s кад почиње да ради на предмету %s", + "allboards.starred": "Истакнути", + "allboards.templates": "Обрасци", + "allboards.remaining": "Нерасподељени", + "allboards.workspaces": "Радни простор", + "allboards.add-workspace": "Прошири радни простор", + "allboards.add-workspace-prompt": "Назив радног простора", + "allboards.add-subworkspace": "Додај радни кутак", + "allboards.add-subworkspace-prompt": "Назив за радни кутак", + "allboards.edit-workspace": "Преименовање радног простора", + "allboards.edit-workspace-name": "Назив радног простора", + "allboards.edit-workspace-icon": "Слика за радни простор (код)", + "multi-selection-active": "Штиклирајте да би изабрали списе", + "activity-dueDate": "је изменио крајњи рок на %s у предмету %s", + "activity-endDate": "је изменио датум и време %s кад је окончао предмет %s", + "add-attachment": "Додај предметну грађу", + "add-board": "Уведи нове списе", + "add-template": "Додај образац", + "add-card": "Додај предмет", + "add-card-to-top-of-list": "Додај предмет на врх", + "add-card-to-bottom-of-list": "Додај предмет на дно", + "setListWidthPopup-title": "Ширина дела поступка", + "set-list-width": "Постави ширину", + "set-list-width-value": "Мин/Макс. ширина (px)", + "list-width-error-message": "Ширина дела поступка мора бити цео број већи од 100", + "keyboard-shortcuts-enabled": "Пречице са тастатуре су укључене.", + "keyboard-shortcuts-disabled": "Пречице са тастатуре су искључене.", + "setSwimlaneHeightPopup-title": "Подеси висину у овој врсти поступка", + "set-swimlane-height": "Подеси висину овог поступка", + "set-swimlane-height-value": "Висина ове врсте поступка (у тачкама)", + "swimlane-height-error-message": "Висина тока поступка мора бити позитиван број", + "add-swimlane": "Додај врсту поступка", + "add-subtask": "Издвоји посао", + "add-checklist": "Додај предметну радњу", + "add-checklist-item": "Додај помоћну предметну радњу", + "close-add-checklist-item": "Окончај додавање помоћних предметних радњи", + "close-edit-checklist-item": "Прекини измену помоћне предметне радње", + "convertChecklistItemToCardPopup-title": "Материјал ⇨ нов предмет", + "add-cover": "Осликани омот предмета", "add-label": "Додај налепницу", "add-list": "Додај деоницу", "add-after-list": "Додај након листе", "add-members": "Прими сараднике", - "added": "Додао", + "added": "додао", "addMemberPopup-title": "Сарадници", "memberPopup-title": "Избор сарадника", "admin": "Управник", - "admin-desc": "Може да гледа и мења картице, удаљи сараднике и мења правила књиге пословања", - "admin-announcement": "Најава", - "admin-announcement-active": "Дозволи системска обавештења", - "admin-announcement-title": "Обавештење од управника", - "all-boards": "Све пословне књиге", - "and-n-other-card": "И __count__ осталих картица", - "and-n-other-card_plural": "И __count__ осталих картица", + "admin-desc": "Може да има и увид и пун приступ предметима, може да одабира сараднике на њима и може да постави правила рада на списима.", + "admin-announcement": "Јавни разглас", + "admin-announcement-active": "Пусти на јавни разглас", + "admin-announcement-title": "Јавно саопштење управника:", + "all-boards": "Полица са списима", + "and-n-other-card": "И __count__ други предмет", + "and-n-other-card_plural": "И __count__ других предмета", "apply": "Примени", "app-is-offline": "Исчитавам, молим да сачекате. Освежавање стране ће изазвати губитак података. Ако повлачење података не ради, молим да проверите да ли је сервер заустављен.", "app-try-reconnect": "Покушавам да се поново повежем.", - "archive": "Премести у архиву", - "archive-all": "Премести све у архиву", - "archive-board": "Премести пословну књигу у архиву", - "archive-board-confirm": "Are you sure you want to archive this board?", - "archive-card": "Премести картицу са задатком у архиву", - "archive-list": "Премести деоницу у архиву", - "archive-swimlane": "Премести стазу у архиву", - "archive-selection": "Премести изабрано у архиву", - "archiveBoardPopup-title": "Преместићете ову пословну књигу у архиву?", - "archived-items": "Архивирај", - "archived-boards": "Пословне књиге из архиве", - "restore-board": "Извуци пословну књигу из архиве", - "no-archived-boards": "Нема књига пословања у архиви.", - "archives": "Архива", - "template": "Предложак", - "templates": "Предлошци", - "template-container": "Сандук са предлошцима", - "add-template-container": "Додај сандук са предлошцима", + "archive": "Спакуј у архиву", + "archive-all": "Спакуј све у архиву", + "archive-board": "Спакуј ове списе у архиву", + "archive-board-confirm": "Да ли сте сигурни да желите да спакујете ове списе у архив?", + "archive-card": "Спакуј предмет у архиву", + "archive-list": "Спакуј овај део поступка у архиву", + "archive-swimlane": "Спакуј овај ток поступка у архиву", + "archive-selection": "Спакуј изабрано у архиву", + "archiveBoardPopup-title": "Архивираћете ове списе?", + "archived-items": "Архива", + "archived-boards": "Архивирани списи", + "restore-board": "Извуци списе из архиве", + "no-archived-boards": "Архива је празна!", + "archives": "Архиве", + "template": "Образац", + "templates": "Обрасци", + "template-container": "Сандук са обрасцима", + "add-template-container": "Додај сандук са обрасцима", "assign-member": "Додели члана", "attached": "Приложено", "attachment": "Приложени документ", "attachment-delete-pop": "Брисање приложеног документа је трајно. Опозив ове радње неће бити могућ.", "attachmentDeletePopup-title": "Обрисаћете приложени документ?", - "attachments": "Приложени документи", - "auto-watch": "Чим се књига пословања отвори почни да је пратиш", - "avatar-too-big": "Ова сличица је превелика (__size__ max)", + "attachments": "Предметна грађа", + "auto-watch": "Чим настану нови списи почни да прикупљаш све промене", + "avatar-too-big": "Слика је превелика (__size__ max)", "back": "Назад", - "board-change-color": "Обоји корице књиге пословања", - "board-change-background-image": "Насловница пословне књиге", - "board-background-image-url": "Веза до слике за насловницу пословне књиге", - "add-background-image": "Додај позадинску слику", - "remove-background-image": "Уклони позадинску слику", - "show-at-all-boards-page" : "Прикажи на полазној страни где су све пословне књиге", - "board-info-on-my-boards" : "Правила за све књиге пословања", - "boardInfoOnMyBoardsPopup-title" : "Правила за све књиге пословања", - "boardInfoOnMyBoards-title": "Правила за све пословне књиге", - "show-card-counter-per-list": "Прикажи број картица на деоници", - "show-board_members-avatar": "Прикажи сличице сарадника/корисника пословне књиге", + "board-change-color": "Обоји омот списа", + "board-change-background-image": "Осликај подлогу списа", + "board-background-image-url": "Веза до слике", + "add-background-image": "Искористи слику за подлогу", + "remove-background-image": "Уклони подлогу", + "show-at-all-boards-page" : "Смести на полицу са списима", + "board-info-on-my-boards" : "Списи у мојој надлежности", + "boardInfoOnMyBoardsPopup-title" : "Списи у мојој надлежности", + "boardInfoOnMyBoards-title": "Списи у мојој надлежности", + "show-card-counter-per-list": "Прикажи бројач предмета на сваком делу тока поступка", + "show-board_members-avatar": "Прикажи слике сарадника на омоту списа", "board-nb-stars": "%s звездица", - "board-not-found": "Књига пословања није пронађена", - "board-private-info": "Ова пословна књига ће бити приватна.", - "board-public-info": "Ова пословна књига ће бити јавна.", - "board-drag-drop-reorder-or-click-open": "Држите и пренесите да би пресложили иконице пословне књиге. Притисните на иконицу да отворите књигу пословања.", - "boardChangeColorPopup-title": "Промени позадину насловне стране књиге пословања", - "boardChangeBackgroundImagePopup-title": "Промена насловнице пословне књиге", - "allBoardsChangeColorPopup-title": "Обоји корице књиге пословања", - "allBoardsChangeBackgroundImagePopup-title": "Насловница пословне књиге", - "boardChangeTitlePopup-title": "Преименуј наслов на пословној књизи", - "boardChangeVisibilityPopup-title": "Промени видљивост", - "boardChangeWatchPopup-title": "Промени праћење", - "boardMenuPopup-title": "Правила коришћења пословне књиге", + "board-not-found": "Спис није пронађен", + "board-private-info": "Ови списи су видљиви само сарадницима.", + "board-public-info": "Ови списи су видљиви свима.", + "board-drag-drop-reorder-or-click-open": "Притисните и задржите па превуците да би пресложили списе. Притисните на сличицу да отворите списе.", + "boardChangeColorPopup-title": "Промени боју омота ових списа", + "boardChangeBackgroundImagePopup-title": "Списи на осликаној подлози", + "allBoardsChangeColorPopup-title": "Зид у боји", + "allBoardsChangeBackgroundImagePopup-title": "Осликани зид", + "boardChangeTitlePopup-title": "Деловодни број и опис", + "boardChangeVisibilityPopup-title": "Промена тајности ових списа", + "boardChangeWatchPopup-title": "Пријем обавештења", + "boardMenuPopup-title": "Рад са списима", "allBoardsMenuPopup-title": "Подешавања", - "boardChangeViewPopup-title": "Распоред у књизи пословања", - "boards": "Књиге пословања", - "board-view": "Прегледност пословне књиге", - "desktop-mode": "Desktop Mode", - "mobile-mode": "Mobile Mode", - "mobile-desktop-toggle": "Toggle between Mobile and Desktop Mode", - "zoom-in": "Zoom In", - "zoom-out": "Zoom Out", - "click-to-change-zoom": "Click to change zoom level", - "zoom-level": "Zoom Level", - "enter-zoom-level": "Enter zoom level (50-300%):", + "boardChangeViewPopup-title": "Поглед на списе", + "boards": "Списи", + "board-view": "Поглед на списе", + "desktop-mode": "Приказ прилагођен за екран рачунара", + "mobile-mode": "Приказ прилагођен за екран мобилног уређаја", + "mobile-desktop-toggle": "Замени приказ прилагођен за рачунар/мобилни", + "zoom-in": "Приближи", + "zoom-out": "Одаљи", + "click-to-change-zoom": "Лупа", + "zoom-level": "Ниво", + "enter-zoom-level": "Задајте ниво (50-300%):", "board-view-cal": "Календар", - "board-view-swimlanes": "Стазе", - "board-view-collapse": "Сажми", + "board-view-swimlanes": "Врсте поступака", + "board-view-collapse": "Скупи", "board-view-gantt": "Гант", "board-view-lists": "Деонице", - "bucket-example": "Like \"Bucket List\" for example", - "calendar-previous-month-label": "Previous Month", - "calendar-next-month-label": "Next Month", + "bucket-example": "Деловодни број (на пример 4/2025)", + "calendar-previous-month-label": "Претходни месец", + "calendar-next-month-label": "Наредни месец", "cancel": "Откажи", - "card-archived": "Ова картица је архивирана.", - "board-archived": "Ова књига пословања је премештена у архиву.", - "card-comments-title": "На овој картици је исказано %s мишљења.", - "card-delete-notice": "Брисање је трајно. Изгубићете све у вези са овом картицом са задацима.", - "card-delete-pop": "Све радње ће бити уклоњене са списка промена и картица са задатком неће моћи бити опорављена. Опозив ове радње неће бити могућ.", - "card-delete-suggest-archive": "Оно што можете је да картицу са задатком архивирате (чиме се све у вези ње очувава) а где је ипак избацујете из књиге пословања.", - "card-archive-pop": "Након њеног архивирања, картица више неће бити видљива на овој деоници.", - "card-archive-suggest-cancel": "Касније,ипак, можете извући и вратити картицу из архиве.", - "card-due": "Рок", - "card-due-on": "Рок истиче", + "card-archived": "Предмет је архивиран.", + "board-archived": "Ови списи су премештени у архиву.", + "card-comments-title": "У овом предмету у расправи је исказано %s мишљења.", + "card-delete-notice": "Брисање је трајно. Изгубићете све у вези са овим предметом.", + "card-delete-pop": "Читав записник ће бити уклоњен и предмет неће моћи бити опорављен. Опозив ове радње неће бити могућ.", + "card-delete-suggest-archive": "Оно што можете је да предмете исчупате из списа архивирањем (тада се записник чува).", + "card-archive-pop": "Након архивирања, предмет више неће бити видљив у овом делу тока поступка.", + "card-archive-suggest-cancel": "Касније ипак можете извући и вратити предмет из архиве.", + "card-due": "Орочен", + "card-due-on": "Предмету истиче рок дана", "card-spent": "Утрошено време", - "card-edit-attachments": "Уреди прилоге", - "card-edit-custom-fields": "Уреди сасвим нова поља", + "card-edit-attachments": "Рад са предметном грађом", + "card-edit-custom-fields": "Придружене рубрике", "card-edit-labels": "Уреди налепнице", "card-edit-members": "Осмисли сарадничку мрежу", - "card-labels-title": "Залепи налепницу на картицу.", - "card-members-title": "Додај или уклони сараднике из књиге пословања са картице.", - "card-start": "Почетак", - "card-start-on": "Почиње", + "card-labels-title": "Залепи налепницу на омот предмета.", + "card-members-title": "Одабир сарадника на списима за овај предмет.", + "card-start": "Започет", + "card-start-on": "Предмет започет дана", "cardAttachmentsPopup-title": "Прилог долази са", - "cardCustomField-datePopup-title": "Промени датум", - "cardCustomFieldsPopup-title": "Уреди сасвим нова поља", - "cardStartVotingPopup-title": "Ново гласање", - "positiveVoteMembersPopup-title": "Гласали За", - "negativeVoteMembersPopup-title": "Гласали Против", - "card-edit-voting": "Постави правила гласања", - "editVoteEndDatePopup-title": "Гласачко место се затвара", - "allowNonBoardMembers": "Дозволи свим корисницима са пријавом", - "vote-question": "Гласачко питање", - "vote-public": "Прикажи резултате гласања", + "cardCustomField-datePopup-title": "Промени датум у рубрици", + "cardCustomFieldsPopup-title": "Придружене рубрике", + "cardStartVotingPopup-title": "Саветодавни референдум", + "positiveVoteMembersPopup-title": "Гласали „За“", + "negativeVoteMembersPopup-title": "Гласали „Против“", + "card-edit-voting": "Јавно гласање", + "editVoteEndDatePopup-title": "Када се гласачко место затвара", + "allowNonBoardMembers": "Дај право гласа и онима који немају увид у ове списе", + "vote-question": "Питање на јавном гласању", + "vote-public": "Прикажи резултате јавног гласања", "vote-for-it": "за", "vote-against": "против", "deleteVotePopup-title": "Избрисаћете гласање?", "vote-delete-pop": "Брисање има трајни карактер. Све у вези са овим гласањем ће бити неповратно избрисано.", "cardStartPlanningPokerPopup-title": "Започни договор за партију покера", - "card-edit-planning-poker": "Договор за партију покера", + "card-edit-planning-poker": "Такмичење", "editPokerEndDatePopup-title": "Промени крајњи датум до кад траје гласање за партију покера", - "poker-question": "Договор за партију покера", + "poker-question": "Играмо карте", "poker-one": "1", "poker-two": "2", "poker-three": "3", @@ -261,101 +261,101 @@ "set-estimation": "Постави прогнозу", "deletePokerPopup-title": "Уклони договор за партију покера?", "poker-delete-pop": "Брисање има трајни карактер. Изгубићете све радње у вези овог плана за партију покера.", - "cardDeletePopup-title": "Обрисаћете картицу?", - "cardArchivePopup-title": "Архивираћете картицу?", - "cardDetailsActionsPopup-title": "Радње на картицама са задацима", + "cardDeletePopup-title": "Обрисаћете предмет?", + "cardArchivePopup-title": "Архивираћете предмет?", + "cardDetailsActionsPopup-title": "Шире радње на предмету", "cardLabelsPopup-title": "Налепнице", "cardMembersPopup-title": "Сарадничка мрежа", "cardMorePopup-title": "Више", - "cardTemplatePopup-title": "Направи предложак", - "cards": "Картице", - "cards-count": "Картице", - "cards-count-one": "Картица", + "cardTemplatePopup-title": "Направи образац", + "cards": "Предмети", + "cards-count": "Предмета", + "cards-count-one": "Предмет", "casSignIn": "Пријави се користећи CAS", - "cardType-card": "Картица са задацима", - "cardType-linkedCard": "Повезана картица са задацима", - "cardType-linkedBoard": "Повезана деоница", + "cardType-card": "Предмет", + "cardType-linkedCard": "Везани предмет", + "cardType-linkedBoard": "Везани списи", "change": "Промени", - "change-avatar": "Промени сличицу", + "change-avatar": "Моја слика", "change-password": "Промени лозинку", - "change-permissions": "Промени дозволе", - "change-settings": "Неколико важних правила", - "changeAvatarPopup-title": "Промени сличицу", + "change-permissions": "Промени улогу", + "change-settings": "Поставка предмета", + "changeAvatarPopup-title": "Моја слика", "changeLanguagePopup-title": "Избор језика", "changePasswordPopup-title": "Промена лозинке", - "changePermissionsPopup-title": "Промени дозволе", - "changeSettingsPopup-title": "Неколико важних правила", - "subtasks": "Подзадаци", - "checklists": "Спискови", - "click-to-star": "Притисни да означиш звездицом ову књигу пословања.", - "click-to-unstar": "Притисни да уклониш звездицу са ове пословне књиге.", + "changePermissionsPopup-title": "Избор улоге", + "changeSettingsPopup-title": "Општа поставка предмета", + "subtasks": "Издвојени послови", + "checklists": "Предметне радње", + "click-to-star": "Означи звездицом ове списе.", + "click-to-unstar": "Уклони звездицу са ових списа.", "click-to-enable-auto-width": "Auto list width disabled. Click to enable.", "click-to-disable-auto-width": "Auto list width enabled. Click to disable.", "auto-list-width": "Auto list width", "clipboard": "Из историје или пренеси и испусти", "close": "Заклопи", - "close-board": "Заклопи књигу пословања", - "close-board-pop": "Моћи ћете да повратите пословну књигу притиском на дугме “Архив” са почетног заглавља.", - "close-card": "Затвори картицу са задатком", + "close-board": "Заклопи списе", + "close-board-pop": "Моћи ћете да повратите списе притиском на дугме “Архив” са почетног заглавља.", + "close-card": "Затвори предмет", "color-black": "црна", "color-blue": "плава", - "color-crimson": "тамно-црвена", - "color-darkgreen": "тамно-зелена", + "color-crimson": "тамноцрвена", + "color-darkgreen": "тамнозелена", "color-gold": "златна", "color-gray": "сива", "color-green": "зелена", - "color-indigo": "индиго", - "color-lime": "креч", - "color-magenta": "пурпурно-црвена", - "color-mistyrose": "магловито роза", - "color-navy": "морнарско плава", + "color-indigo": "у боји индиго пресликача", + "color-lime": "пастелнозелена", + "color-magenta": "љубичасто-црвена", + "color-mistyrose": "прљаво роза", + "color-navy": "тамноплава", "color-orange": "наранџаста", - "color-paleturquoise": "бледо-зелена", - "color-peachpuff": "кајсија", + "color-paleturquoise": "бледозелена", + "color-peachpuff": "у боји кајсије", "color-pink": "пинк", - "color-plum": "шљива", + "color-plum": "модра", "color-purple": "љубичаста", "color-red": "црвена", - "color-saddlebrown": "кожно седло", + "color-saddlebrown": "у боји кестена", "color-silver": "сребрна", "color-sky": "небеско плава", - "color-slateblue": "загасито плава", + "color-slateblue": "загаситоплава", "color-white": "бела", "color-yellow": "жута", "unset-color": "Обриши подешавање", - "comments": "Comments", + "comments": "Ставови", "comment": "Изнеси мишљење", - "comment-placeholder": "Простор за изношење мишљења", - "comment-only": "Износи мишљење", - "comment-only-desc": "Може да изнесе мишљење само на картицама са задацима.", - "comment-delete": "Да ли сте сигурни да желите да уклоните изнешено мишљење?", + "comment-placeholder": "Место за расправу", + "comment-only": "Стручни саветник", + "comment-only-desc": "Једино може да учествује у расправи око одређеног предмета.", + "comment-delete": "Да ли сте сигурни да желите да повучете изнешено мишљење?", "deleteCommentPopup-title": "Повлачите мишљење?", - "no-comments": "Нико није дао мишљење", - "no-comments-desc": "Не може да види изнешена мишљења и догађаје.", - "worker": "Радник", - "worker-desc": "Може само да помера картице са задацима, додељује их себи и да даје мишљење. ", + "no-comments": "Посматрач", + "no-comments-desc": "Не може да види расправу и прати записник.", + "worker": "Приправник", + "worker-desc": "Може да ради помоћне послове - да премешта предмете, бирa оне које ће пратити и да учествује у расправи. ", "computer": "Рачунар", - "confirm-subtask-delete-popup": "Да ли сте сигурни да желите да избришете подзадатак?", - "confirm-checklist-delete-popup": "Да ли сте сигурни да желите да избришете овај списак за обавити?", - "subtaskDeletePopup-title": "Избрисаћете подзадатак?", - "checklistDeletePopup-title": "Избрисаћете списак за обавити?", - "copy-card-link-to-clipboard": "Причувај везу до картице у привременој меморији", - "copy-text-to-clipboard": "Причувај текст у привременој меморији", - "linkCardPopup-title": "Повежи картицу", + "confirm-subtask-delete-popup": "Да ли сте сигурни да желите да избришете овај издвојени посао?", + "confirm-checklist-delete-popup": "Да ли сте сигурни да желите да избришете ову предметну радњу?", + "subtaskDeletePopup-title": "Избрисаћете издвојени посао?", + "checklistDeletePopup-title": "Избрисаћете предметну радњу?", + "copy-card-link-to-clipboard": "Причувај везу на кратко", + "copy-text-to-clipboard": "Причувај текст на кратко", + "linkCardPopup-title": "Повежи предмет", "searchElementPopup-title": "Претрага", - "copyCardPopup-title": "Умножи картицу", - "copyManyCardsPopup-title": "Умножи предложак на више картица", - "copyManyCardsPopup-instructions": "Наслови и описи одредишних картица у следећем JSON облику", - "copyManyCardsPopup-format": "[ {\"наслов\": \"Наслов прве картице\", \"опис\":\"Опис прве картице\"}, {\"наслов\":\"Наслов друге картице\",\"опис\":\"Опис друге картице\"},{\"наслов\":\"Наслов последње картице\",\"опис\":\"Опис последње картице\"} ]", + "copyCardPopup-title": "Умножи предмет", + "copyManyCardsPopup-title": "Расади попуњен образац", + "copyManyCardsPopup-instructions": "Наслови и описи одредишних предмета треба да имају следећи JSON облик:", + "copyManyCardsPopup-format": "[ {\"наслов\": \"Наслов првог предмета\", \"опис\":\"Опис првог предмета\"}, {\"наслов\":\"Наслов другог предмета\",\"опис\":\"Опис другог предмета\"},{\"наслов\":\"Наслов последњег предмете\",\"опис\":\"Опис последњег предмета\"} ]", "create": "Уведи", - "createBoardPopup-title": "Уведи нову пословну књигу", + "createBoardPopup-title": "Увод нових списа", "createTemplateContainerPopup-title": "Додај сандук са предлошцима", - "chooseBoardSourcePopup-title": "Уведи пословну књигу", + "chooseBoardSourcePopup-title": "Унеси спољне списе", "createLabelPopup-title": "Нова налепница", - "createCustomField": "Направи сасвим ново поље", - "createCustomFieldPopup-title": "Направи сасвим ново поље", + "createCustomField": "Нова рубрика", + "createCustomFieldPopup-title": "Нова рубрика", "current": "текуће", - "custom-field-delete-pop": "Опозив ове радње неће бити могућ. Овим се уклања ово сасвим ново поље из свих картица и уништива његова историја.", + "custom-field-delete-pop": "Опозив ове радње неће бити могућ. Радњом се уклања ова придружена рубрика из свих предмета и уништива њена претходна историја употребе.", "custom-field-checkbox": "Поље за штиклирање", "custom-field-currency": "Валута", "custom-field-currency-option": "Код валуте", @@ -367,16 +367,16 @@ "custom-field-dropdown-unknown": "(непознато)", "custom-field-number": "Број", "custom-field-text": "Текст", - "custom-fields": "Сасвим нова поља", + "custom-fields": "Придружене рубрике", "date": "Датум", - "date-format": "Date Format", - "date-format-yyyy-mm-dd": "YYYY-MM-DD", - "date-format-dd-mm-yyyy": "DD-MM-YYYY", - "date-format-mm-dd-yyyy": "MM-DD-YYYY", + "date-format": "Начин записивање датума", + "date-format-yyyy-mm-dd": "година-месец-дан", + "date-format-dd-mm-yyyy": "дан-месец-година", + "date-format-mm-dd-yyyy": "месец-дан-година", "decline": "Одбиј", - "default-avatar": "Унапред изабрана сличица", + "default-avatar": "иницијали уместо слике", "delete": "Уклони", - "deleteCustomFieldPopup-title": "Обрисаћете сасвим ново поље?", + "deleteCustomFieldPopup-title": "Обрисаћете ову придружену рубрику?", "deleteLabelPopup-title": "Одлепићете све овакве налепнице?", "description": "Опис", "disambiguateMultiLabelPopup-title": "Недвосмислене радње на налепницама", @@ -386,122 +386,122 @@ "download": "Преузми", "edit": "Уреди", "edit-avatar": "Моја слика", - "edit-profile": "Моји лични подаци", - "edit-wip-limit": "Уреди ограничење броја послова", - "soft-wip-limit": "Мека граница броја послова", - "editCardStartDatePopup-title": "Кад је задатак започет", - "editCardDueDatePopup-title": "Крајњи рок за испуњење задатка", - "editCustomFieldPopup-title": "Измени поље", + "edit-profile": "Лични подаци", + "edit-wip-limit": "Затрпавање предметима", + "soft-wip-limit": "Мека граница броја предмета", + "editCardStartDatePopup-title": "Кад сте започели рад на предмету", + "editCardDueDatePopup-title": "Крајњи рок за предмет", + "editCustomFieldPopup-title": "Измени наслов рубрике", "addReactionPopup-title": "Реакција на објаву", "editCardSpentTimePopup-title": "Промени утрошено време", "editLabelPopup-title": "Постојећа налепница", "editNotificationPopup-title": "Измени обавештење", "editProfilePopup-title": "Лични подаци", "email": "Е-пошта", - "email-address": "Email Address", + "email-address": "Адреса електронске поште", "email-enrollAccount-subject": "За Вас је направљен један налог на __siteName__", "email-enrollAccount-text": "Здраво __user__,\n\nДа би почели да користите услугу, једноставно притисните на везу која је испод.\n\n__url__\n\nХвала.", - "email-fail": "Неуспело слање е-поште", + "email-fail": "Неуспело слање е-поште!", "email-fail-text": "Грешка при покушају слања е-поште", "email-invalid": "Неисправна адреса е-поште", "email-invite": "Позив преко е-поште", "email-invite-subject": "__inviter__ Вам је послао позивницу", - "email-invite-text": "Драги __user__,\n\n__inviter__ Вас позива на заједничку сарадњу кроз \"__board__\" пословну књигу.\n\nМолим да испратите везу испод:\n\n__url__\n\nХвала.", + "email-invite-text": "Поштовани __user__,\n\n__inviter__ Вам је омогућио пун увид у \"__board__\" списе и нада се заједничкој сарадњи.\n\nМолим да испратите везу испод:\n\n__url__\n\nХвала.", "email-resetPassword-subject": "Задајте поново вашу лозинку на страници __siteName__", "email-resetPassword-text": "Добар дан __user__,\n\nДа би сте поново задали вашу лозинку, једноставно притисните на везу испод.\n\n__url__\n\nХвала.", "email-sent": "Е-пошта је послана", "email-verifyEmail-subject": "Потврдите Вашу адресу е-поште на страници __siteName__", "email-verifyEmail-text": "Добар дан __user__,\n\nДа би сте потврдили ваш налог за е-пошту, једноставно притисните на везу испод.\n\n__url__\n\nХвала.", "enable-vertical-scrollbars": "Enable vertical scrollbars", - "enable-wip-limit": "Ограничи број послова", - "error-board-doesNotExist": "Ова пословна књига не постоји", - "error-board-notAdmin": "Да би то урадили, треба да будете администратор/управник ове књиге пословања", - "error-board-notAMember": "Да би то урадили треба да будете сарадник у овој пословној књизи", + "enable-wip-limit": "Ограничи број предмета", + "error-board-doesNotExist": "Овакви списи не постоје", + "error-board-notAdmin": "Да би то урадили, треба да будете управник ових списа", + "error-board-notAMember": "Да би то урадили треба да будете сарадник на овим списима", "error-json-malformed": "То што сте укуцали нема исправан JSON облик", "error-json-schema": "Ваши подаци у JSON облику не укључују исправне информације у предвиђеном облику", "error-csv-schema": "Ваш CSV(Вредности раздвојене зарезом)/TSV (Вредности раздвојене табулатором) не укључује исправну информацију у предвиђеном облику", "error-list-doesNotExist": "Ова деоница не постоји", "error-user-doesNotExist": "Сарадник не постоји", - "error-user-notAllowSelf": "Не можеш да позовеш сам себе", - "error-user-notCreated": "Сарадник не постоји", - "error-username-taken": "Корисничко име је већ у употреби", - "error-orgname-taken": "Ово име предузећа је већ у употреби", - "error-teamname-taken": "Ово име тима је већ у употреби", + "error-user-notAllowSelf": "Не можете сами себе да позовете", + "error-user-notCreated": "Сарадник није уписан", + "error-username-taken": "Кориснички налог је већ у употреби", + "error-orgname-taken": "Ово (пословно) име странке је већ у употреби", + "error-teamname-taken": "Ово име правног тима је већ у употреби", "error-email-taken": "Ова адреса е-поште је већ у употреби", - "export-board": "Преведи пословну књигу", - "export-board-json": "Преведи пословну књигу у JSON", - "export-board-csv": "Преведи пословну књигу у CSV", - "export-board-tsv": "Преведи пословну књигу у TSV", - "export-board-excel": "Преведи пословну књигу у Excel", - "user-can-not-export-excel": "Корисник не може да преводи у Excel", - "export-board-html": "Преведи пословну књигу у HTML", - "export-card": "Преведи картицу са задатком", - "export-card-pdf": "Преведи картицу са задатком у PDF", - "user-can-not-export-card-to-pdf": "Корисник не може да преведе садржај картице са задатком у PDF", - "exportBoardPopup-title": "Превођење пословне књиге", - "exportCardPopup-title": "Превођење картице са задатком", + "export-board": "Претвори списе", + "export-board-json": "Претварање списа у JSON", + "export-board-csv": "Претварање списа у CSV", + "export-board-tsv": "Претварање списа у TSV", + "export-board-excel": "Претварање списа у Excel", + "user-can-not-export-excel": "Корисник не може да претвори списе у Excel облик", + "export-board-html": "Претварање списа у HTML", + "export-card": "Претвори предмет", + "export-card-pdf": "Претвори предмет у PDF облик", + "user-can-not-export-card-to-pdf": "Корисник не може да претвори садржај предмета у PDF облик", + "exportBoardPopup-title": "Претварање списа", + "exportCardPopup-title": "Претвори предмет", "sort": "Редослед", "sorted": "Сложено", - "remove-sort": "Разбацај", - "sort-desc": "Притисните да би сложили деонице", - "list-sort-by": "Поређај деоницу по:", - "list-label-modifiedAt": "Време последње измене", - "list-label-title": "Назив деонице", - "list-label-sort": "Vaš ručni nalog", - "list-label-short-modifiedAt": "(P)", - "list-label-short-title": "(N)", - "list-label-short-sort": "(R)", + "remove-sort": "Измешај предмете", + "sort-desc": "Притисните да би сложили делове поступка", + "list-sort-by": "Поређај делове поступка по:", + "list-label-modifiedAt": "Времену последње измене", + "list-label-title": "Називу дела поступка", + "list-label-sort": "Вашем личном нахођењу", + "list-label-short-modifiedAt": "(П)", + "list-label-short-title": "(Н)", + "list-label-short-sort": "(Л)", "filter": "Издвајање", - "filter-cards": "Издвој картице или деонице", + "filter-cards": "Издвој предмете или делове поступка", "filter-dates-label": "Издвој по датуму", - "filter-no-due-date": "Нема датума истека", - "filter-overdue": "Прекорачено време", - "filter-due-today": "Рок истиче данас", - "filter-due-this-week": "Рок истиче ове недеље", - "filter-due-next-week": "Рок истиче следеће недеље", - "filter-due-tomorrow": "Рок истиче сутрадан", - "list-filter-label": "Издвој деонице по наслову", + "filter-no-due-date": "Где није постављен рок", + "filter-overdue": "Где је пробијен рок", + "filter-due-today": "Где рок истиче данас", + "filter-due-this-week": "Где рок истиче ове недеље", + "filter-due-next-week": "Где рок истиче следеће недеље", + "filter-due-tomorrow": "Где рок истиче сутрадан", + "list-filter-label": "Издвој део поступка по наслову", "filter-clear": "Прекини са издвајањем", "filter-labels-label": "Издвој по налепници", "filter-no-label": "Нема налепницу", - "filter-member-label": "Издвој по сараднику", - "filter-no-member": "Нема сарадника", - "filter-assignee-label": "Издвој по задужењу", - "filter-no-assignee": "Нико није задужен", - "filter-custom-fields-label": "Издвој по сасвим новим пољима", - "filter-no-custom-fields": "Нема сасвим нових поља", - "filter-show-archive": "Прикажи архивиране деонице", - "filter-hide-empty": "Сакриј празне деонице", + "filter-member-label": "Издвој по надлежном сараднику", + "filter-no-member": "Нико није надлежан", + "filter-assignee-label": "Издвој по пуномоћнику", + "filter-no-assignee": "Где никоме није дата пуномоћ", + "filter-custom-fields-label": "Издвој по рубрикама", + "filter-no-custom-fields": "Где нема рубрика", + "filter-show-archive": "Прикажи архивиране делове поступка", + "filter-hide-empty": "Сакриј делове поступка без предмета", "filter-on": "Издваја се", - "filter-on-desc": "Издвајате картице из ове пословне књиге. Притисните овде да одредите како се издвајање врши.", + "filter-on-desc": "Сад издвајате предмете из ових списа. Притисните овде да одредите како се издвајање врши.", "filter-to-selection": "Издвајање по изабраном", - "other-filters-label": "Другачије издвајање", + "other-filters-label": "Преостала издвајања", "advanced-filter-label": "Напредно издвајање", - "advanced-filter-description": "Напредно издвајање Вам омогућава да напишете ниску која садржи следеће оператореAdvanced: == != <= >= && || ( ) Празно место се користи да раздвоји операторе. Можете да издвајате сва поља која сте измислили пишући њихова имена и вредности. На пример: ИмеИзмишљеногПоља == Вредност. Напомена: Уколико предметна поља или њихове вредности садрже празна места треба да их обухватите у једноструке наводнике. На пример: 'Поље 1' == 'Вредност 1'. Да би избегли контролне знаке (' \\\\/) , можете да користите \\\\. Например: Поље1 == Ал\\\\' је дугачко поље. Такође можете измешати више услова. На пример: Поље1 == Вредност1 || Поље1 == Вредност2. Провера услова се обавља са лева на десно. Додавањем заграда ћете утицати на проверу. На пример: Поље1 == Вредност1 && ( Поље2 == Вредност2 || Поље2 == Вредност3 ). Такође можете претраживати поља програмерским изразима: Поље1 == /Tes.*/i", + "advanced-filter-description": "Напредно издвајање Вам омогућава да напишете ниску која садржи следеће операторе: == != <= >= && || ( ) Размак се користи да раздвоји операторе. Можете да издвајате поља из придружених рубрика позивајући се на њихова имена и вредности. На пример: ИмеПридруженогПоља == Вредност. Напомена: Уколико предметна поља или њихове вредности садрже размак треба да га обухватите у једноструке наводнике. На пример: 'Поље 1' == 'Вредност 1'. Да би избегли контролне знаке (' \\\\/) , можете да користите \\\\. Например: Поље1 == I\\\\'m . Такође можете измешати више услова. На пример: Поље1 == Вредност1 || Поље1 == Вредност2. Провера услова се обавља са лева на десно. Додавањем заграда ћете утицати на проверу. На пример: Поље1 == Вредност1 && ( Поље2 == Вредност2 || Поље2 == Вредност3 ). Такође можете претраживати поља програмерским изразима: Поље1 == /Tes.*/i", "fullname": "Име и презиме", - "header-logo-title": "Вратите се на полицу са Вашим пословним књигама.", - "show-activities": "Show Activities", - "headerBarCreateBoardPopup-title": "Отвори нову књигу пословања", + "header-logo-title": "Вратите се на полицу са Вашим списима.", + "show-activities": "Писани записник", + "headerBarCreateBoardPopup-title": "Увод нових списа", "home": "Почетна", "import": "Унеси", - "impersonate-user": "Лажно представљен корисник", + "impersonate-user": "Увуци се у налог овог корисника", "link": "Веза", - "import-board": "уведи пословну књигу", - "import-board-c": "Уведи пословну књигу", - "import-board-title-trello": "Уведи књигу пословања из Trella", - "import-board-title-wekan": "Уведи пословну књигу из претходног преноса", - "import-board-title-csv": "Уведи једну CSV/TSV пословну књигу", - "from-trello": "Из Trello", - "from-wekan": "Из претходног превода", - "from-csv": "Из CSV/TSV", - "import-board-instruction-trello": "У Вашој Trello пословној књизи, идите у 'Menu', затим 'More', 'Print and Export', 'Export JSON', и умножите представљени текст.", + "import-board": "унеси спољне списе", + "import-board-c": "Извор спољних списа", + "import-board-title-trello": "Trello", + "import-board-title-wekan": "Wekan", + "import-board-title-csv": "CSV/TSV", + "from-trello": "Извор Trello", + "from-wekan": "Извор претходни Wekan подаци", + "from-csv": "Извор CSV/TSV датотека", + "import-board-instruction-trello": "У Вашоим Trello списима, идите у 'Menu', затим 'More', 'Print and Export', 'Export JSON', и умножите представљени текст.", "import-board-instruction-csv": "Овде залепите Ваше CSV податке (вредности раздвојене зарезом) односно TSV податке (вредности раздвојене табулатором) .", - "import-board-instruction-wekan": "У Вашој књизи пословања, изаберите ставке 'Мени', затим 'Преведи пословни књигу' и на крају умножите сав садржај/текст из преузете датотеке.", - "import-board-instruction-about-errors": "Ако се јаве грешке при уношење пословне књиге то не значи да унос није успео!Проверите јел се књига појавила у одељку Моје пословне књиге.", + "import-board-instruction-wekan": "У Вашим списима, изаберите ставке 'Мени', затим 'Унеси списе' и на крају умножите сав садржај/текст из преузете датотеке.", + "import-board-instruction-about-errors": "Ако се јаве грешке при уношењу списа то не значи да унос није успео!Проверите јел се спис појавио у одељку „Списи у мојој надлежности“.", "import-json-placeholder": "Овде сместите Ваше исправне JSON податке", "import-csv-placeholder": "Овде сместите Ваше исправне CSV/TSV податке", "import-map-members": "Мапирај сараднике", - "import-members-map": "Пословна књига коју сте унели има постављену мрежу сарадника. Молим да мапирате сараднике из те мреже на Ваше сараднике", + "import-members-map": "Списи које сте унели имају постављену мрежу сарадника. Молим да мапирате сараднике из те мреже на Ваше сараднике", "import-members-map-note": "Белешка: Немапирани сарадници ће бити додељени на тренутног корисника.", "import-show-user-mapping": "Испрегледај како су сарадници измапирани", "import-user-select": "Изаберите Вашег познатог сарадника којег желите да корисите као овог", @@ -512,68 +512,68 @@ "invalid-time": "Нетачно време", "invalid-user": "Непознат корисник", "joined": "придружен", - "just-invited": "Управо сте позвани у ову књигу пословања", + "just-invited": "Управо сте позвани да радите на овим списима", "keyboard-shortcuts": "Пречице на тастатури", "label-create": "Уведи нову налепницу", "label-default": "%s налепница (подразумевано)", - "label-delete-pop": "Пажња. Опозив ове радње неће бити могућ! Изабрана налепница ће бити одлепљена/уклоњена са свих картица са задацима и њена историја ће исчезнути.", + "label-delete-pop": "Пажња. Опозив ове радње неће бити могућ! Изабрана налепница ће бити одлепљена/уклоњена из свих предмета и њена историја ће исчезнути.", "labels": "Налепнице", "language": "Језик", "last-admin-desc": "Не можете да мењате задужења јер мора постојати барем један управник/администратор.", - "leave-board": "Раздужи књигу пословања", - "leave-board-pop": "Да ли сте сигурни да желите да раздужите пословну књигу __boardTitle__? Бићете одстрањени са свих задатака из ове књиге пословања.", - "leaveBoardPopup-title": "Раздужићете пословну књигу ?", - "link-card": "Веза до ове картице", - "list-archive-cards": "Премести у архив све картице са задацима са ове деонице", - "list-archive-cards-pop": "Овим склањате све задатке на овој деоници из књиге пословања. Да испрегледате архивиране задатке и вратите их натраг у књигу пословања притисните на “Мени” > “Архив”.", - "list-move-cards": "Премести све картице на овој деоници", - "list-select-cards": "Изабери све картице на овој деоници", - "set-color-list": "Обоји деоницу", - "listActionPopup-title": "Радње на деоници", - "settingsUserPopup-title": "Корисничка подешавања", - "settingsTeamPopup-title": "Тимска подешавања", - "settingsOrgPopup-title": "Подешавања предузећа", - "swimlaneActionPopup-title": "Радње на стазама", - "swimlaneAddPopup-title": "Додаје једну стазу испод", - "listImportCardPopup-title": "Унеси једну Trello картицу", + "leave-board": "Раздужи предмете из списа", + "leave-board-pop": "Да ли сте сигурни да желите да раздужите предмете из списа __boardTitle__? Бићете одстрањени са свих тих предмета.", + "leaveBoardPopup-title": "Раздужићете списе ?", + "link-card": "Повежи предмете", + "list-archive-cards": "Архивирај све предмете из овог дела", + "list-archive-cards-pop": "Овим склањате све предмете из овог дела поступка из списа. Да испрегледате тако архивиране предмете и вратите их натраг у ове списе притисните на “Мени” > “Архива”.", + "list-move-cards": "Премести све предмете из овог дела поступка", + "list-select-cards": "Изабери све предмете", + "set-color-list": "Обоји", + "listActionPopup-title": "Радње у делу поступка", + "settingsUserPopup-title": "Рад са сарадницима", + "settingsTeamPopup-title": "Рад са правним тимовима", + "settingsOrgPopup-title": "Рад са странкама", + "swimlaneActionPopup-title": "Радње у овој врсти поступка", + "swimlaneAddPopup-title": "Додаје испод врсту поступка", + "listImportCardPopup-title": "Унеси једнан Trello предмет", "listImportCardsTsvPopup-title": "Унеси Excel CSV/TSV", "listMorePopup-title": "Више", "link-list": "Веза до ове деонице", - "list-delete-pop": "Све досадашње забележене радње на деоници биће уклоњене и нећете моћи опоравити деоницу. Опозив ове радње неће бити могућ.", - "list-delete-suggest-archive": "Можете да преместите целу неку деоницу у Архив чиме чувате историју деонице а ипак је склањате из књиге пословања.", - "lists": "Деонице", - "swimlanes": "Стазе", - "log-out": "Одјава", + "list-delete-pop": "Читав записник са овог дела поступка биће уклоњен и нећете га моћи опоравити. Опозив ове радње неће бити могућ.", + "list-delete-suggest-archive": "Боље је да преместите читав део поступка у Архив чиме чувате записник а део поступка ипак склањате из списа.", + "lists": "Делови поступка", + "swimlanes": "Врсте поступака", + "log-out": "Одјави се", "log-in": "Пријави се", - "loginPopup-title": "Пријава", + "loginPopup-title": "Пријавница", "memberMenuPopup-title": "Сарадник", "members": "Сарадници", "menu": "Мени", "move-selection": "Премести изабрано", - "moveCardPopup-title": "Премести картицу са задатком", + "moveCardPopup-title": "Премести предмет", "moveCardToBottom-title": "Премести на дно", "moveCardToTop-title": "Премести на врх", "moveSelectionPopup-title": "Премести изабрано", "multi-selection": "Вишеструк избор", - "multi-selection-label": "Изаберите налепницу", - "multi-selection-member": "Изаберите и сарадника", - "multi-selection-on": "Вишеструк избор је омогућен", - "muted": "Утишано", - "muted-info": "Нећете чути обавештење о променама у овој пословној књизи", - "my-boards": "Моје књиге пословања", + "multi-selection-label": "Залепите или одлепите налепнице на одабране предмете", + "multi-selection-member": "Одаберите и сараднике", + "multi-selection-on": "Постоји вишеструк избор", + "muted": "Не примај обававештења", + "muted-info": "Нећете чути обавештења кад наступе било какве промене у овим списима", + "my-boards": "Списи у мојој надлежности", "name": "Задајте (нови) натпис", - "no-archived-cards": "Нама архивираних картица са задацима.", - "no-archived-lists": "Нема архивираних деоница.", - "no-archived-swimlanes": "Нема архивираних стаза.", + "no-archived-cards": "Нема архивираних предмета.", + "no-archived-lists": "Нема архивираних делова поступака.", + "no-archived-swimlanes": "Нема архивираних врсти поступака.", "no-results": "Нема резултата", - "normal": "Нормално", - "normal-desc": "Може да гледа и уређује картице. Не може да мења подешавања.", + "normal": "Виши сарадник", + "normal-desc": "Може да има и увид и пун приступ предметима. Не може да поставља правила рада на списима.", "not-accepted-yet": "Позив још није прихваћен", - "notify-participate": "Примајте допунске извештаје за било које картице са задацима у којима учествујете као налогодавац или сарадник", - "notify-watch": "Примајте допунске извештаје за било које пословне књиге, деонице, или картице са задацима које пратите", + "notify-participate": "Примајте допунске извештаје при било којој измени предмета које сте сами завели или где сте сарадник", + "notify-watch": "Примајте допунске извештаје при било којој измени списа, делова поступака или предмета које пратите", "optional": "по избору", "or": "или", - "page-maybe-private": "Ова страницаје је, могуће, приватна. Можда ћете моћи да је видите кад се пријавите.", + "page-maybe-private": "Ови списи су могуће под велом тајности. Можда ћете ипак моћи да их видите кад се пријавите овде.", "page-not-found": "Страница није пронађена.", "password": "Лозинка", "paste-or-dragdrop": "налепи, или држи и испусти слику на то (важи само за слике)", @@ -581,114 +581,114 @@ "preview": "Приказ", "previewAttachedImagePopup-title": "Приказ", "previewClipboardImagePopup-title": "Приказ", - "private": "Приватно", - "private-desc": "Ова пословна књига је приватна. Само одобрени сарадници могу да је читају и уређују.", + "private": "Видљиво сарадницима", + "private-desc": "Ови списи су тајни. Само одобрени сарадници могу да их читају и уређују.", "profile": "Особина", - "public": "Јавно", - "public-desc": "Ова пословна књига је јавна. Видљива је сваком ко има везу и појавиће се у google претрагама. Једино одобрени сарадници могу бити уредници.", - "quick-access-description": "Означи књигу пословања звездицом да додаш пречицу на ову траку.", - "remove-cover": "Уклоните насловну слику са мини картице", - "remove-from-board": "Избаци из књиге пословања", + "public": "Списи видљиви свима", + "public-desc": "Ови списи су потпуно јавни. Видљиви су сваком ко има везу до њих и појавиће се и у google претрагама. Једино одобрени сарадници могу бити уредници.", + "quick-access-description": "Означите списе звездицом да би додали пречицу на ову траку.", + "remove-cover": "Уклоните осликани омот", + "remove-from-board": "Избаци из списа", "remove-label": "Скини налепницу", - "listDeletePopup-title": "Обрисаћете деоницу?", - "remove-member": "Удаљи сарадника", - "remove-member-from-card": "Удаљи га са задатка", - "remove-member-pop": "Удаљићете __name__ (__username__) из __boardTitle__? Сарадник ће бити удаљен са свих задатака из ове књиге пословања. Примиће обавештење о томе.", - "removeMemberPopup-title": "Удаљићете сарадника?", + "listDeletePopup-title": "Обрисаћете део поступка?", + "remove-member": "Скини сарадника", + "remove-member-from-card": "Скини га са предмета", + "remove-member-pop": "Изузећете __name__ (__username__) из списа __boardTitle__? Сарадник ће бити изузет са свих предмета у списима. Примиће обавештење о томе.", + "removeMemberPopup-title": "Изузећете сарадника?", "rename": "Преименуј", - "rename-board": "Преименуј књигу пословања", + "rename-board": "Преименуј списе", "restore": "Опорави", - "rescue-card-description": "Понуди још једну прилику да се забележе несачувани описи задатака пре затварања картице", - "rescue-card-description-dialogue": "Преписаћете тренутни опис задатка са Вашим описом?", + "rescue-card-description": "Понуди још једну прилику да се забележи несачувани опис предмета пре затварања", + "rescue-card-description-dialogue": "Преписаћете тренутни опис предмета са допуњеним?", "save": "Сачувај", "search": "Претрага", - "rules": "Правила", - "search-cards": "Тражи у насловима задатака/деоница, новим пољима у овој пословној књизи", - "search-example": "Укуцајте речи/текст које тражите и притисните ентер", - "select-color": "Изаберите (нову) боју", - "select-board": "Изаберите књигу пословања", - "set-wip-limit-value": "Поставите границу за максимални дозвољени број задатака на овој деоници", - "setWipLimitPopup-title": "Поставите ограничење броја послова", - "shortcut-add-self": "Add yourself to current card", - "shortcut-assign-self": "Придружите себе тренутној картици", + "rules": "Правилник", + "search-cards": "Тражи у насловима предмета, делова поступака и рубрика у овим списима", + "search-example": "Укуцајте све што тражите и притисните ентер", + "select-color": "Изабери боју за овакав поступак", + "select-board": "Изаберите списе", + "set-wip-limit-value": "Поставите границу дозвољеног броја предмета у овом делу поступка", + "setWipLimitPopup-title": "Ограничење броја предмета", + "shortcut-add-self": "Доделите надлежност себи", + "shortcut-assign-self": "Поверите пуномоћ себи", "shortcut-autocomplete-emoji": "Сам попуни emoji", "shortcut-autocomplete-members": "Сам попуни сараднике", "shortcut-clear-filters": "Прекини издвајање", "shortcut-close-dialog": "Затвори прозорче", - "shortcut-filter-my-cards": "Издвој задатке", - "shortcut-filter-my-assigned-cards": "Filter my assigned cards", - "shortcut-show-shortcuts": "Прикажи овај списак пречица", - "shortcut-toggle-filterbar": "Укључи/искључи бочни мени за издвајање", - "shortcut-toggle-searchbar": "Укључи/искључи бочни мени за претрагу", - "shortcut-toggle-sidebar": "Укључи/искључи бочни мени пословне књиге", - "show-cards-minimum-count": "Прикажи број картица са задацима ако деоница садржи више од", - "sidebar-open": "Отвори бочну површину", - "sidebar-close": "Затвори бочну површину", - "signupPopup-title": "Направите налог", - "star-board-title": "Притисни да означиш звездицом баш ову књигу пословања. Имаће предност и бити на челу.", - "starred-boards": "Пословне књиге са звездицом", - "starred-boards-description": "Књиге пословања са звездицом се појављују испред других.", + "shortcut-filter-my-cards": "Издвој предмете где имам увид", + "shortcut-filter-my-assigned-cards": "Издвој предмете где сам пуномоћник", + "shortcut-show-shortcuts": "Прикажи овај подсетник са пречицама", + "shortcut-toggle-filterbar": "Алат за издвајање", + "shortcut-toggle-searchbar": "Алат за претрагу", + "shortcut-toggle-sidebar": "Алат за рад са списима", + "show-cards-minimum-count": "Изброј предмете и прикажи тај број тек ако део поступка садржи више од", + "sidebar-open": "Извади алат", + "sidebar-close": "Спакуј алат", + "signupPopup-title": "Јединствени регистар", + "star-board-title": "Означите спис звездицом. Имаће предност и бити на челу.", + "starred-boards": "Списи са звездицом", + "starred-boards-description": "Списи означени звездицом се појављују испред других.", "subscribe": "Претплати се", - "team": "Тим", - "this-board": "ова пословна књига", - "this-card": "ова картица са задатком", + "team": "Правни тим", + "this-board": "ове списе", + "this-card": "овај предмет", "spent-time-hours": "Потрошено време (у сатима)", "overtime-hours": "Прековремени (сати)", "overtime": "Прековремено", - "has-overtime-cards": "Има задатке на којима је прековремено радио", - "has-spenttime-cards": "Има задатке са мерењем времена", + "has-overtime-cards": "Постоје предмети са прековременим радом", + "has-spenttime-cards": "Постоје предмети где сат одбројава", "time": "Време", "title": "Наслов", - "toggle-assignees": "Toggle assignees 1-9 for card (By order of addition to board).", - "toggle-labels": "Укључи/искључи натписе од 1 до 9 за задатак. Вишеструк избор додаје натпис од 1 до 9", - "remove-labels-multiselect": "Вишеструким избором се уклањају натписи од 1 до 9", - "tracking": "Праћење", - "tracking-info": "Бићете обавештени о свим променама на оним картицама са задацима где сте укључени као налогодавац или као сарадник.", + "toggle-assignees": "Одабери пуномоћника на предмету по редоследу додавања у увид у списе.", + "toggle-labels": "Лепите и одлепљујете налепнице по броју. Вишеструким избором можете само да лепите", + "remove-labels-multiselect": "Вишеструким избором одлепљујете налепнице", + "tracking": "Прикупљај обавештења из моје надлежности", + "tracking-info": "Бићете обавештени о било каквој промени у оним предметима где сте укључени као сарадник или које сте лично завели.", "type": "Врста", "unassign-member": "Недодељени сарадник", "unsaved-description": "Нисте сачували опис.", "unwatch": "Скини са мера праћења", "upload": "Подигни", - "upload-avatar": "Пошаљите Вашу сличицу на сервер", - "uploaded-avatar": "Сличица је подигнута на сервер", - "uploading-files": "Uploading files", - "upload-failed": "Upload failed", - "upload-completed": "Upload completed", - "custom-top-left-corner-logo-image-url": "Веза до места на Интернету где је слика са Вашим логом за приказивање у горњем левом углу", + "upload-avatar": "слика уместо иницијала", + "uploaded-avatar": "Слика је подигнута на сервер", + "uploading-files": "Шаљем датотеке", + "upload-failed": "Слање није успело", + "upload-completed": "Слање је успело", + "custom-top-left-corner-logo-image-url": "Веза до места на Интернету где се налази слика са Вашим логом за приказивање у горњем левом углу", "custom-top-left-corner-logo-link-url": "Веза до места на Интернету које се посећује када си кликне на Ваш лого у горњем левом углу", "custom-top-left-corner-logo-height": "Висина за Ваш лого у горњем левом углу. Уобичајена вредност: 27", - "custom-login-logo-image-url": "Веза до места на Интернету где је насловна слика код пријаве", + "custom-login-logo-image-url": "Веза до места на Интернету са ког се повлачи насловна слика код пријаве", "custom-login-logo-link-url": "Слика изнад дела за пријаву води до следећег места на Интернету", "custom-help-link-url": "Место на Интернету где се може затражити помоћ", "text-below-custom-login-logo": "Порука испод насловне слике на страни за пријаву", "automatic-linked-url-schemes": "Група места на Интернету која би могла да се посете. Једна група по линији", - "username": "Корисничко име", - "import-usernames": "Увези корисничка имена", + "username": "Кориснички налог", + "import-usernames": "Увези корисничке налоге", "view-it": "Погледај је", - "warn-list-archived": "упозорење: ова картица са задатком јесте у једној деоници али у архиви", - "watch": "Посматрај", - "watching": "Посматрање", - "watching-info": "Бићете обавештени о променама у овој пословној књизи", - "welcome-board": "Пословна књига за добродошлицу", - "welcome-swimlane": "Стаза 1", + "warn-list-archived": "упозорење: овај предмет постоји у архиви у једном делу поступка", + "watch": "Стави на мере праћења", + "watching": "Прикупљај сва обавештења", + "watching-info": "Бићете обавештени о било каквој промени у овим списима", + "welcome-board": "Списи са добродошлицом", + "welcome-swimlane": "Врста 1", "welcome-list1": "Основно", "welcome-list2": "Напредно", - "card-templates-swimlane": "Предлошци за картице са задацима", - "list-templates-swimlane": "Предложак за деонице", - "board-templates-swimlane": "Предложак за пословну књигу", + "card-templates-swimlane": "Обрасци за предмете", + "list-templates-swimlane": "Обрасци за делове поступака", + "board-templates-swimlane": "Обрасци за списе", "what-to-do": "Шта желите да урадите ?", - "wipLimitErrorPopup-title": "Досегнуто ограничење броја послова", - "wipLimitErrorPopup-dialog-pt1": "Број задатака на овој деоници је већи него ограничење које сте поставили.", - "wipLimitErrorPopup-dialog-pt2": "Молим да или обавите неке послове на овој деоници или да поставите веће ограничење дозвољеног броја послова.", - "admin-panel": "Главна управа", - "settings": "Подешавања", + "wipLimitErrorPopup-title": "Досегнуто ограничење броја предмета", + "wipLimitErrorPopup-dialog-pt1": "Број предмета у овом делу поступка је већи него ограничење које сте поставили.", + "wipLimitErrorPopup-dialog-pt2": "Молим да преместите неке предмете из овог дела поступка или да повећате границу.", + "admin-panel": "Управа", + "settings": "Општа поставка", "people": "Сарадници", - "registration": "Отварање налога", - "disable-self-registration": "Онемогући да корисници сами себи отворе налог", - "disable-forgot-password": "Онемогући да корисници сами затраже промену лозинке", - "invite": "Позив", - "invite-people": "Позови сараднике", - "to-boards": "У пословну књигу(e)", + "registration": "Упис", + "disable-self-registration": "Налог не може да се отвори без позивнице", + "disable-forgot-password": "Спречи процедуру за заборављену лозинку", + "invite": "Упути позив", + "invite-people": "Доле позваним омогући и пун увид", + "to-boards": "У списе", "email-addresses": "Адресе е-поште", "smtp-host-description": "Тачна адреса SMTP сервера који рукује вашом е-поштом.", "smtp-port-description": "Порт Вашег SMTP сервера који се користи за одлазну е-пошту.", @@ -699,23 +699,23 @@ "smtp-password": "Лозинка", "smtp-tls": "TLS подршка", "send-from": "Од", - "send-smtp-test": "Пошаљите сами себи једну пробну е-поруку", + "send-smtp-test": "Проба да ли поруке пролазе", "invitation-code": "Позивни код", "email-invite-register-subject": "__inviter__ Вам је послао позивницу", - "email-invite-register-text": "Драги __user__,\n\n__inviter__ Вас позива да користите kanban пословне књиге за заједнички рад.\n\nМолим испратите везу испод:\n__url__\n\nИ да, Ваш позивни код је: __icode__\n\nХвала.", + "email-invite-register-text": "Поштовани __user__,\n\n__inviter__ Вас позива да користите kanban списе за заједнички рад.\n\nМолим испратите везу испод:\n__url__\n\nИ да, Ваш позивни код је: __icode__\n\nХвала.", "email-smtp-test-subject": "Проба да ли ради слање е-поште", "email-smtp-test-text": "Успешно сте послали е-пошту", - "error-invitation-code-not-exist": "Позивни код не постоји", + "error-invitation-code-not-exist": "Такав позивни код не постоји", "error-notAuthorized": "Немате права да приступате овој страници.", - "webhook-title": "Назив мрежне куке", + "webhook-title": "Назив за дојаву догађаја", "webhook-token": "Token (необавезно за проверу идентитета)", - "outgoing-webhooks": "Одлазна мрежна кука", - "bidirectional-webhooks": "Двосмерне мрежне куке", - "outgoingWebhooksPopup-title": "Одлазне мрежне куке", + "outgoing-webhooks": "Дојава догађаја", + "bidirectional-webhooks": "Двосмерно понашање", + "outgoingWebhooksPopup-title": "Дојава догађаја", "boardCardTitlePopup-title": "Издвајање по наслову задатка", - "disable-webhook": "Онеспособи ову мрежну куку", - "global-webhook": "Глобална мрежна кука", - "new-outgoing-webhook": "Нова одлазна мрежна кука", + "disable-webhook": "Онеспособи ову дојаву догађаја", + "global-webhook": "Дојава догађаја", + "new-outgoing-webhook": "Нова дојава догађаја", "no-name": "(непознато)", "Node_version": "Node верзија", "Meteor_version": "Meteor верзија", @@ -735,78 +735,78 @@ "hours": "сати", "minutes": "минута", "seconds": "секунди", - "show-field-on-card": "Прикажи ово поље на картицу са задатком", - "automatically-field-on-card": "Придодај поље новим картицама", - "always-field-on-card": "Придодај поље свим картицама", - "showLabel-field-on-card": "Прикажи натпис на налепници", - "showSum-field-on-list": "Прикажи збирни број поља на врху деонице", + "show-field-on-card": "Придружи ову рубрику овом предмету", + "automatically-field-on-card": "Сви будући предмети ће имати овакву рубрику", + "always-field-on-card": "Придодај овакву рубрику и свим постојећим предметима", + "showLabel-field-on-card": "Прикажи натпис ове рубрике на омоту предмета", + "showSum-field-on-list": "Прикажи збирни број придружених рубрика на врху деонице", "yes": "Да", "no": "Не", "accounts": "Налози", "accounts-allowEmailChange": "Дозволи промену адресе е-поште", - "accounts-allowUserNameChange": "Дозволи промену корисничког имена", - "tableVisibilityMode-allowPrivateOnly": "Видљивост пословне књиге: Дозволи једино приватне пословне књиге", - "tableVisibilityMode" : "Видљивост пословне књиге", + "accounts-allowUserNameChange": "Дозволи промену имена налога", + "tableVisibilityMode-allowPrivateOnly": "Тајност списа: Дозволи приступ искључиво затвореном кругу сарадника", + "tableVisibilityMode" : "Тајност списа", "createdAt": "Направљено дана", "modifiedAt": "Време измене", - "verified": "Потрврђено", + "verified": "Потврђен налог", "active": "На окупу", - "card-received": "Пријем", - "card-received-on": "Примљен дана", - "card-end": "Завршетак", - "card-end-on": "Завршава се дана", - "editCardReceivedDatePopup-title": "Датум и време запримања задатка", - "editCardEndDatePopup-title": "Кад је задатак постао испуњен", - "setCardColorPopup-title": "Обоји картицу", - "setCardActionsColorPopup-title": "Изабери боју", - "setSwimlaneColorPopup-title": "Изабери боју стазе", - "setListColorPopup-title": "Изабери боју деонице", - "assigned-by": "Додељено од стране", + "card-received": "Запримљен", + "card-received-on": "Предмет запримљен дана", + "card-end": "Окончан", + "card-end-on": "Предмет окончан дана", + "editCardReceivedDatePopup-title": "Датум и време запримања предмета", + "editCardEndDatePopup-title": "Кад је задатак окончан", + "setCardColorPopup-title": "Боја омота предмета", + "setCardActionsColorPopup-title": "Боја за подлогу предмета", + "setSwimlaneColorPopup-title": "Боја за врсту поступка", + "setListColorPopup-title": "Боја за део поступка", + "assigned-by": "Властодавац", "requested-by": "Захтевано од", - "card-sorting-by-number": "Пресложи картице по броју", - "board-delete-notice": "Избацивање је трајно. Изгубићете све деонице, картице са задацима и радње повезане са овом књигом пословања.", - "delete-board-confirm-popup": "Све деонице, картице са задацима, налепнице и радње биће избачене и нећете моћи да повратите садржај књиге пословања. Опозив ове радње неће бити могућ.", - "boardDeletePopup-title": "Избацићете пословну књигу?", - "delete-board": "Избаци пословну књигу", - "delete-duplicate-lists": "Delete Duplicate Lists", - "delete-duplicate-lists-confirm": "Are you sure? This will delete all duplicate lists that have the same name and contain no cards.", - "default-subtasks-board": "Под задаци за __board__ књигу пословања", + "card-sorting-by-number": "Бројчана ознака за редослед", + "board-delete-notice": "Избацивање је трајно. Изгубићете све делове поступка, предмете и радње повезане са овим списима.", + "delete-board-confirm-popup": "Сви делови поступка, предмети, налепнице и записници биће избачени и нећете моћи да повратите садржај списа. Опозив ове радње неће бити могућ.", + "boardDeletePopup-title": "Избацићете ове списе?", + "delete-board": "Избаци списе", + "delete-duplicate-lists": "Уклоните истоимене делове поступка", + "delete-duplicate-lists-confirm": "Да ли сте сигурни? Овом радњом ће бити избрисани сви истоимени делови поступка где нема предмета.", + "default-subtasks-board": "Утврђени пријемни списи __board__", "default": "Подразумевано", "defaultdefault": "Подразумевано", - "queue": "Ред", - "subtask-settings": "Подешавања подзадатака", - "card-settings": "Подешавања картице са задацима", - "minicard-settings": "Подешавања малих картица", - "boardSubtaskSettingsPopup-title": "Подешавања подзадатака", - "boardCardSettingsPopup-title": "Подешавања картица са задацима", - "boardMinicardSettingsPopup-title": "Подешавања малих картица", - "deposit-subtasks-board": "Депонуј подзадатке у ову пословну књигу:", - "deposit-subtasks-list": "Деоница у којој се смештају под задаци овде депоновани:", - "show-parent-in-minicard": "Прикажи порекло у малој картици:", - "description-on-minicard": "Опис мале картице", - "cover-attachment-on-minicard": "Насловна слика на мини картици", - "badge-attachment-on-minicard": "Број прилога на мини картици", - "card-sorting-by-number-on-minicard": "Слагање картица по броју на мини картици", - "prefix-with-full-path": "Префикс са пуном путањом", - "prefix-with-parent": "Префикс са пореклом", - "subtext-with-full-path": "Subtext са пуном путањом", - "subtext-with-parent": "Subtext са пореклом", - "change-card-parent": "Промени порекло картице", - "parent-card": "Родитељска картица", - "source-board": "Изворна књига пословања", - "no-parent": "Не приказуј порекло", + "queue": "Пријемно одељење", + "subtask-settings": "Поставка издвојених послова", + "card-settings": "Појединости и омот предмета", + "minicard-settings": "Рад са омотима предмета", + "boardSubtaskSettingsPopup-title": "Руковање издвојеним пословима", + "boardCardSettingsPopup-title": "Предмети у списима", + "boardMinicardSettingsPopup-title": "Рад са омотима предмета", + "deposit-subtasks-board": "Депонуј издвојене послове у списе:", + "deposit-subtasks-list": "Пријемно одељење кад овде депонују:", + "show-parent-in-minicard": "Испис порекла на омоту предмета:", + "description-on-minicard": "Опис на омоту", + "cover-attachment-on-minicard": "Мотив из предметне грађе", + "badge-attachment-on-minicard": "Број поред спајалице", + "card-sorting-by-number-on-minicard": "Бројчана ознака за редослед", + "prefix-with-full-path": "Испиши пуну путању као уводни слог", + "prefix-with-parent": "Испиши порекло као уводни слог", + "subtext-with-full-path": "Испиши пуну путању испод ситним словима", + "subtext-with-parent": "Испиши порекло испод ситним словима", + "change-card-parent": "Промени порекло предмета", + "parent-card": "Родитељски предмет", + "source-board": "Изворни списи", + "no-parent": "Немој исписивати никакво порекло", "activity-added-label": "залепљена налепница '%s' на %s", "activity-removed-label": "уклоњена је налепница '%s' са %s", "activity-delete-attach": "уклоњен је прилог са %s", "activity-added-label-card": "залепљена је налепница '%s'", "activity-removed-label-card": "уклоњена је налепница '%s'", "activity-delete-attach-card": "избрисан је прилог", - "activity-set-customfield": "постављено је сасвим ново поље '%s' на '%s' у %s", - "activity-unset-customfield": "уклоњено је сасвим ново поље '%s' у %s", + "activity-set-customfield": "придружена је рубрика '%s' на '%s' у %s", + "activity-unset-customfield": "уклоњена је рубрика '%s' у %s", "r-rule": "Правило", - "r-add-trigger": "Додај окидач", - "r-add-action": "Додај радњу", - "r-board-rules": "Правила пословне књиге", + "r-add-trigger": "Прво додајте догађај", + "r-add-action": "Затим додајте радњу као одговор", + "r-board-rules": "Списак правила", "r-add-rule": "Додај правило", "r-view-rule": "Прегледај правило", "r-delete-rule": "Обриши правило", @@ -814,104 +814,104 @@ "r-no-rules": "Није уведено правило", "r-trigger": "Окидач", "r-action": "Радња", - "r-when-a-card": "Када картица", + "r-when-a-card": "Предмет", "r-is": "је", - "r-is-moved": "је премештена", - "r-added-to": "Додан у", - "r-removed-from": "Уклоњена из", - "r-the-board": "пословна књига", - "r-list": "деоница", + "r-is-moved": "премештен", + "r-added-to": "додана на", + "r-removed-from": "скинут са", + "r-the-board": "списа", + "r-list": "део", "set-filter": "Правило издвајања", - "r-moved-to": "Премештен у", - "r-moved-from": "Премештен из", + "r-moved-to": "премештен у", + "r-moved-from": "премештен из", "r-archived": "Премештен у архиву", "r-unarchived": "Извучен из архиве", - "r-a-card": "једна картица са задатком", - "r-when-a-label-is": "Када је налепница", - "r-when-the-label": "Када налепница", - "r-list-name": "име деонице", - "r-when-a-member": "Када је сарадник", - "r-when-the-member": "Када сарадник", + "r-a-card": "предмет", + "r-when-a-label-is": "Када је било која налепница", + "r-when-the-label": "Налепница", + "r-list-name": "име дела поступка", + "r-when-a-member": "Када је било који сарадник", + "r-when-the-member": "Сарадник по имену", "r-name": "име", - "r-when-a-attach": "Када је додатак", - "r-when-a-checklist": "Када је списак за обавити", - "r-when-the-checklist": "Када списак за обавити", - "r-completed": "Обављен", - "r-made-incomplete": "Није обављен", - "r-when-a-item": "Када је ставка на списку за обавити", - "r-when-the-item": "Када ставка на списку за обавити", - "r-checked": "Обављена", - "r-unchecked": "Необављена", - "r-move-card-to": "Премести картицу са задатком на", + "r-when-a-attach": "Када је нека предметна грађа", + "r-when-a-checklist": "Када је нека предметна радња", + "r-when-the-checklist": "Када је предметна радња", + "r-completed": "обављена", + "r-made-incomplete": "означена као необављена", + "r-when-a-item": "Када је нека ставка са списка", + "r-when-the-item": "Када је ставка са списка", + "r-checked": "прецртана", + "r-unchecked": "означена као не прецртана", + "r-move-card-to": "Премести предмет на", "r-top-of": "Врх", "r-bottom-of": "Дно", - "r-its-list": "своје деонице", - "r-archive": "Премести у архиву", - "r-unarchive": "Извуци из архиве", - "r-card": "картица", + "r-its-list": "свог дела поступка", + "r-archive": "спакован у архиву", + "r-unarchive": "извучен из архиве", + "r-card": "предмет", "r-add": "Додај", "r-remove": "Уклони", - "r-label": "налепница", - "r-member": "сарадник", - "r-remove-all": "Удаљи све сараднике са задатка", - "r-set-color": "Обоји са", - "r-checklist": "списак за обавити", - "r-check-all": "Све је обављено", - "r-uncheck-all": "Није ништа обављено", - "r-items-check": "ставке на списку за обавити", - "r-check": "Обављено", - "r-uncheck": "Необављено", - "r-item": "ставка", - "r-of-checklist": "списка за обавити", + "r-label": "налепницу", + "r-member": "сарадника по имену", + "r-remove-all": "Скини све сареднике са предмета", + "r-set-color": "Обоји", + "r-checklist": "предметну радњу", + "r-check-all": "Прецртај све", + "r-uncheck-all": "Означи све као непрецртано", + "r-items-check": "са списка", + "r-check": "Прецртај", + "r-uncheck": "Врати на дораду", + "r-item": "ставку", + "r-of-checklist": "са списка", "r-send-email": "Пошаљи е-пошту", "r-to": "за", "r-of": "од", "r-subject": "наслов", "r-rule-details": "Детаљи извођења правила", - "r-d-move-to-top-gen": "Премести картицу са задатком на врх своје деонице", - "r-d-move-to-top-spec": "Премести картицу са задатком на врх деонице", - "r-d-move-to-bottom-gen": "Премести картицу са задатком на дно своје деонице", - "r-d-move-to-bottom-spec": "Премести картицу са задатком на дно деонице", + "r-d-move-to-top-gen": "Помери предмет на врх тог дела поступка", + "r-d-move-to-top-spec": "Помери предмет на врх дела поступка", + "r-d-move-to-bottom-gen": "Помери предмет на дно тог дела поступка", + "r-d-move-to-bottom-spec": "Помери предмет на дно дела поступка", "r-d-send-email": "Пошаљи е-пошту", "r-d-send-email-to": "за", "r-d-send-email-subject": "наслов", "r-d-send-email-message": "порука", - "r-d-archive": "Премести картицу са задатком у архиву", - "r-d-unarchive": "Извуци из архиве картицу са задатком", + "r-d-archive": "Премести предмет у архиву", + "r-d-unarchive": "Извуци предмет из архиве", "r-d-add-label": "Залепи налепницу", "r-d-remove-label": "Уклони налепницу", - "r-create-card": "Направи нову картицу са задатком", - "r-in-list": "на деоници", - "r-in-swimlane": "у стази", + "r-create-card": "Заведи предмет", + "r-in-list": "у делу", + "r-in-swimlane": "поступка", "r-d-add-member": "Прими сарадника", - "r-d-remove-member": "Удаљи сарадника", - "r-d-remove-all-member": "Удаљи све сараднике", - "r-d-check-all": "Означи све ставке на списку за обавити као обављене", - "r-d-uncheck-all": "Означи све ставке на списку за обавити као необављене", - "r-d-check-one": "Обављена ставка", - "r-d-uncheck-one": "Необављена ставка", + "r-d-remove-member": "Изузми сарадника", + "r-d-remove-all-member": "Изузми све сараднике", + "r-d-check-all": "Означи све ставке са списка као обављене", + "r-d-uncheck-all": "Означи све ставке са списка као необављене", + "r-d-check-one": "Обављена радња", + "r-d-uncheck-one": "Необављена радња", "r-d-check-of-list": "списка за обавити", - "r-d-add-checklist": "Додај списак за обавити", - "r-d-remove-checklist": "Уклони списак за обавити", + "r-d-add-checklist": "Додај списак", + "r-d-remove-checklist": "Уклони списак", "r-by": "од", - "r-add-checklist": "Додај списак за обавити", + "r-add-checklist": "Кад неко дода списак", "r-with-items": "са ставкама", "r-items-list": "ставка1,ставка2,ставка3", - "r-add-swimlane": "Додај стазу", - "r-swimlane-name": "име стазе", - "r-board-note": "Белешка: Остављено празно поље одговара свакој могућој вредности.", - "r-checklist-note": "Белешка: Ставке на списку за обавити морају да буду раздвојене зарезом.", - "r-when-a-card-is-moved": "Када је картица са задатком премештена на другу деоницу", - "r-set": "Постави", - "r-update": "Освежи", - "r-datefield": "поље са датумом", - "r-df-start-at": "почетак", - "r-df-due-at": "крајњи датум", - "r-df-end-at": "крај", - "r-df-received-at": "примљен", - "r-to-current-datetime": "до тренутног датума/времена", - "r-remove-value-from": "Уклони вредност са", - "r-link-card": "Увежи картицу на", + "r-add-swimlane": "Додај врсту поступка", + "r-swimlane-name": "врсту поступка", + "r-board-note": "Белешка: Непопуњено поље одговара свакој могућој вредности.", + "r-checklist-note": "Белешка: Ставке на списку морају да буду раздвојене зарезом.", + "r-when-a-card-is-moved": "Када је предмет премештен у други део поступка", + "r-set": "Унеси", + "r-update": "Помери", + "r-datefield": "временску одредницу кад је предмет", + "r-df-start-at": "започет", + "r-df-due-at": "орочен", + "r-df-end-at": "окончан", + "r-df-received-at": "запримљен", + "r-to-current-datetime": "на данашњи датум и тренутно време", + "r-remove-value-from": "Уклони временску одредницу из поља", + "r-link-card": "Кад је предмет у вези са списима", "ldap": "LDAP", "oauth2": "OAuth2", "cas": "CAS", @@ -920,30 +920,30 @@ "custom-product-name": "Ваша реклама", "layout": "Распоред", "hide-logo": "Сакриј лого", - "hide-card-counter-list": "Сакриј бројач задатака на картици у свим пословним књигама", - "hide-board-member-list": "Сакриј списак сарадника у свим књигама пословања", - "add-custom-html-after-body-start": "Уметни овај HTML код на сваки почетак деонице", - "add-custom-html-before-body-end": "Уметни овај HTML код пред сваки крај деонице", - "error-undefined": "Негде је настао проблем", + "hide-card-counter-list": "Сакриј бројач предмета на полици са списима", + "hide-board-member-list": "Сакриј списак сарадника на полици са списима", + "add-custom-html-after-body-start": "Уметни овај HTML код где почиње деоница", + "add-custom-html-before-body-end": "Уметни овај HTML код пред крај деонице", + "error-undefined": "Имамо непознат проблем", "error-ldap-login": "Догодила се нека грешка приликом покушаја пријављивања", "display-authentication-method": "Прикажи метод за утврђивање идентитета", "oidc-button-text": "Измени OIDC текст на дугмету", "default-authentication-method": "Уобичајени метод за утврђивање идентитета", - "duplicate-board": "Умножи пословну књигу у још један примерак", + "duplicate-board": "Умножи списе у још један примерак", "duplicate-board-confirm": "Are you sure you want to duplicate this board?", - "org-number": "Број предузећа је:", - "team-number": "Број тимова је:", - "people-number": "Број сарадника је:", - "swimlaneDeletePopup-title": "Избрисаћете стазу ?", - "swimlane-delete-pop": "Све радње ће бити уклоњене из активности и нећете моћи да вратите стазу. Нема поништавања.", + "org-number": "Број евидентираних странака је:", + "team-number": "Број правних тимова је:", + "people-number": "Укупан број сарадника је:", + "swimlaneDeletePopup-title": "Избрисаћете овај поступак ?", + "swimlane-delete-pop": "Све радње ће бити уклоњене из записника и нећете моћи да опоравите поступак. Нема поништавања.", "restore-all": "Опорави све", "delete-all": "Обриши све", "loading": "Исчитавање, молим сачекајте.", "previous_as": "прошли пут је био", - "act-a-dueAt": "измењени крајњи рок \nКада:__timeValue__\nГде:__card__\nПретходни рок је био __timeOldValue__", - "act-a-endAt": "измењено време завршетка на __timeValue__ са претходно постављеног (__timeOldValue__)", - "act-a-startAt": "измењено време почетка на __timeValue__ са претходно постављеног (__timeOldValue__)", - "act-a-receivedAt": "измењено време пријема на __timeValue__ са претходно постављеног (__timeOldValue__)", + "act-a-dueAt": "измењени крајњи рок \nРок:__timeValue__\nПредмет:__card__\nПретходни крајњи рок је био __timeOldValue__", + "act-a-endAt": "измењен датум и време окончања на __timeValue__ са претходно постављеног (__timeOldValue__)", + "act-a-startAt": "измењен датум и време почетка рада на предмету на __timeValue__ са претходно постављеног (__timeOldValue__)", + "act-a-receivedAt": "измењен датум и време пријема предмета на __timeValue__ са претходно постављеног (__timeOldValue__)", "a-dueAt": "измењен крајњи рок", "a-endAt": "измењено време завршетка", "a-startAt": "измењено време почетка", @@ -956,25 +956,25 @@ "act-almostdue": "подсећам да се приближавамо последњем року (__timeValue__) за завршетак задатка __card__", "act-pastdue": "подсећам да је последњи рок (__timeValue__) задатка __card__ пробијен", "act-duenow": "подсећам да последњи рок (__timeValue__) задатка __card__ истиче данас", - "act-atUserComment": "Споменути сте у пословној књизи [__board__] тачније на деоници __list__ на картици са задатком __card__", - "delete-user-confirm-popup": "Да ли сте сигурни да желите да избришете овај кориснички налог? Опозив ове радње неће бити могућ.", - "delete-team-confirm-popup": "Да ли сте сигурни да желите да избришете овај тим? Опозив ове радње неће бити могућ.", - "delete-org-confirm-popup": "Да ли сте сигурни да желите да избришете ово предузеће? Опозив ове радње неће бити могућ.", - "accounts-allowUserDelete": "Дозволи корисницима да сами избришу свој налог", + "act-atUserComment": "Споменути сте у списима [__board__] тачније на делу поступка __list__ у предмету __card__", + "delete-user-confirm-popup": "Да ли сте сигурни да желите да уклоните овај појединачни кориснички налог сарадника? Опозив ове радње неће бити могућ.", + "delete-team-confirm-popup": "Да ли сте сигурни да желите да уклоните овај цео правни тим? Опозив ове радње неће бити могућ.", + "delete-org-confirm-popup": "Да ли сте сигурни да желите да уклоните ову странку? Опозив ове радње неће бити могућ.", + "accounts-allowUserDelete": "Дозволи сарадницима да сами испишу свој налог из јединственог регистра", "hide-minicard-label-text": "Сакриј натпис на налепници", "show-desktop-drag-handles": "Прикажи ручке за повлачење на радној површини", - "assignee": "Извршилац", - "cardAssigneesPopup-title": "Коме се додељује да изврши", + "assignee": "Пуномоћник", + "cardAssigneesPopup-title": "Коме се даје пуномоћ", "addmore-detail": "Додај детаљнији опис", - "show-on-card": "Прикажи на картици", - "show-on-minicard": "Прикажи на налепници", + "show-on-card": "Видљиво у појединостима", + "show-on-minicard": "Видљиво на омоту", "new": "Ново", - "editOrgPopup-title": "Уреди предузеће", - "newOrgPopup-title": "Ново предузеће", - "editTeamPopup-title": "Уреди податке за тим", - "newTeamPopup-title": "Нови тим", - "editUserPopup-title": "Уреди податке корисника", - "newUserPopup-title": "Нови корисник", + "editOrgPopup-title": "Опис странке", + "newOrgPopup-title": "Нова странка", + "editTeamPopup-title": "Опис правног тима", + "newTeamPopup-title": "Нови правни тим", + "editUserPopup-title": "Опис сарадника", + "newUserPopup-title": "Нови сарадник", "notifications": "Обавештења", "help": "Помоћ", "view-all": "Прикажи сва", @@ -992,109 +992,109 @@ "saturday": "Субота", "sunday": "Недеља", "status": "Стање", - "swimlane": "Стаза", + "swimlane": "Поступак", "owner": "Власник", "last-modified-at": "Време последње измене", "last-activity": "Последње промене", "voting": "Гласање", "archived": "Архивирано", - "delete-linked-card-before-this-card": "Не можете избрисати ову картицу пре него што прво избришете повезану картицу која има", - "delete-linked-cards-before-this-list": "Не можете избрисати ову деоницу -- прво избришете повезане картице које показују на картице на овој деоници", - "hide-checked-items": "Сакриј већ обављено са списка", - "hide-finished-checklist": "Hide finished checklist", + "delete-linked-card-before-this-card": "Можете избрисати овај предмет тек кад, прво, избришете предмет у вези са њим", + "delete-linked-cards-before-this-list": "Не можете избрисати овај део поступка - прво избришете предмете који су у вези са предметима у овом делу поступка", + "hide-checked-items": "Сакриј обављене помоћне предметне радње", + "hide-finished-checklist": "Сакриј обављене предметне радње", "task": "Задатак", "create-task": "Задај", - "ok": "У реду", - "organizations": "Предузећа", - "teams": "Тимови", - "displayName": "Име које се приказује", - "shortName": "Скраћеница", - "autoAddUsersWithDomainName": "Automatically add users with the domain name", - "website": "Интернет страница", - "person": "Особа", - "my-cards": "Моја задужења", - "card": "Картица", - "list": "Деоница", - "board": "Пословна књига", + "ok": "Сагласан", + "organizations": "Странке", + "teams": "Правни тимови", + "displayName": "Име (Пословно име)", + "shortName": "Скраћено пословно име", + "autoAddUsersWithDomainName": "Повежи са оним сарадницима чији је домен е-поште", + "website": "Званична интернет страница", + "person": "Лице", + "my-cards": "Предмети где имам пуномоћ", + "card": "Предмет", + "list": "Део поступка", + "board": "Списи", "context-separator": "/", - "myCardsViewChange-title": "Приказ мојих задужења", - "myCardsViewChangePopup-title": "Приказ мојих задужења", - "myCardsViewChange-choice-boards": "Размештена по пословним књигама", - "myCardsViewChange-choice-table": "Приказана у табели", - "myCardsSortChange-title": "Издвајање мојих личних картица", - "myCardsSortChangePopup-title": "Издвајање мојих личних картица", - "myCardsSortChange-choice-board": "По натпису на књизи пословања", + "myCardsViewChange-title": "Поглед на такве предмете", + "myCardsViewChangePopup-title": "Такви предмети треба да буду", + "myCardsViewChange-choice-boards": "Размештени по списима", + "myCardsViewChange-choice-table": "Приказани у табели", + "myCardsSortChange-title": "Редослед предмета где имам пуномоћ", + "myCardsSortChangePopup-title": "Поређај предмете где имам пуномоћ", + "myCardsSortChange-choice-board": "По деловодном броју списа", "myCardsSortChange-choice-dueat": "По датуму доспећа", - "dueCards-title": "Послови којима истиче рок", - "dueCardsViewChange-title": "Приказ картица које истичу", - "dueCardsViewChangePopup-title": "Приказ картица које истичу", - "dueCardsViewChange-choice-me": "Моји недовршени послови", - "dueCardsViewChange-choice-all": "Недовршени послови сарадника", - "dueCardsViewChange-choice-all-description": "Прикажи све недовршене задатке који имају постављен рок решавања из пословних књига за које корисник има одобрење.", - "dueCards-noResults-title": "No Due Cards Found", - "dueCards-noResults-description": "You don't have any cards with due dates at the moment.", - "broken-cards": "Покидане картице", - "board-title-not-found": "Књига пословања '%s' није пронађена.", - "swimlane-title-not-found": "Стаза '%s' није пронађена.", - "list-title-not-found": "Деоница '%s' није пронађена.", - "label-not-found": "Налепница '%s' није пронађена.", - "label-color-not-found": "Боја налепнице %s није пронађена.", - "user-username-not-found": "Корисничко име '%s' није пронађено.", + "dueCards-title": "Предмети са роковима", + "dueCardsViewChange-title": "Надлежност", + "dueCardsViewChangePopup-title": "Надлежност", + "dueCardsViewChange-choice-me": "где имам пуномоћ", + "dueCardsViewChange-choice-all": "где имам право увида", + "dueCardsViewChange-choice-all-description": "Приказује све нерешене предмете који имају постављен рок решавања из оних списа за које корисник има одобрење.", + "dueCards-noResults-title": "Није пронађен ниједан предмет са роком", + "dueCards-noResults-description": "Тренутно немате никакве предмете који има постављене рокове.", + "broken-cards": "Предмети где је веза покидана", + "board-title-not-found": "Списи под деловодним бројем '%s' нису пронађени.", + "swimlane-title-not-found": "Ток поступка '%s' није пронађен.", + "list-title-not-found": "Део поступка '%s' није пронађен.", + "label-not-found": "Натпис на налепници '%s' није пронађен.", + "label-color-not-found": "%s налепница у боји није пронађена.", + "user-username-not-found": "Сарадник са корисничким налогом '%s' није пронађен.", "comment-not-found": "Мишљење које садржи речи '%s' није пронађено.", - "org-name-not-found": "Предузеће '%s' није пронађено.", - "team-name-not-found": "Тим '%s' није пронађен.", - "globalSearch-title": "Претрага", - "no-cards-found": "Није пронађена ни једна картица са задацима", - "one-card-found": "Пронађена је једна картица са задацима", - "n-cards-found": "Број пронађених картица са задацима: %s", - "n-n-of-n-cards-found": "__start__-__end__ од __total__ картица је нађено", - "operator-board": "пословна књига", - "operator-board-abbrev": "п", - "operator-swimlane": "стаза", - "operator-swimlane-abbrev": "с", - "operator-list": "деонице", + "org-name-not-found": "Странка по имену '%s' није пронађена.", + "team-name-not-found": "Правни тим по имену '%s' није пронађен.", + "globalSearch-title": "Претрага по списима", + "no-cards-found": "Није пронађен ни један предмет", + "one-card-found": "Пронађен је један предмет", + "n-cards-found": "Број пронађених предмета: %s", + "n-n-of-n-cards-found": "__start__-__end__ од __total__ предмета је нађено", + "operator-board": "списи", + "operator-board-abbrev": "с", + "operator-swimlane": "поступак", + "operator-swimlane-abbrev": "п", + "operator-list": "део", "operator-list-abbrev": "д", - "operator-label": "налепнице", + "operator-label": "налепницa", "operator-label-abbrev": "#", "operator-user": "корисник", "operator-user-abbrev": "@", "operator-member": "сарадник", "operator-member-abbrev": "н", - "operator-assignee": "коме је додељено", - "operator-assignee-abbrev": "о", - "operator-creator": "налогодавац", + "operator-assignee": "пуномоћник", + "operator-assignee-abbrev": "м", + "operator-creator": "завео", "operator-status": "стање", "operator-due": "крајњи датум", - "operator-created": "постављен", + "operator-created": "заведено", "operator-modified": "измењено", "operator-sort": "редослед", "operator-comment": "мишљење", "operator-has": "има", "operator-limit": "ограничење", "operator-debug": "тражење грешака", - "operator-org": "предузеће", + "operator-org": "странка", "operator-team": "тим", "predicate-archived": "архивиран", "predicate-open": "отворен", "predicate-ended": "окончан", "predicate-all": "све", "predicate-overdue": "истекао", - "predicate-week": "недеља", - "predicate-month": "месец", - "predicate-quarter": "тромесечје", - "predicate-year": "година", + "predicate-week": "недељно", + "predicate-month": "месечно", + "predicate-quarter": "тромесечно", + "predicate-year": "годишње", "predicate-due": "рок", - "predicate-modified": "измењен", - "predicate-created": "направљен", - "predicate-attachment": "прилог", - "predicate-description": "опис", - "predicate-checklist": "списак за обавити", - "predicate-start": "почетак", - "predicate-end": "крај", - "predicate-assignee": "задужени", + "predicate-modified": "изменио", + "predicate-created": "завео", + "predicate-attachment": "приложио", + "predicate-description": "описао", + "predicate-checklist": "предметна радња", + "predicate-start": "започео", + "predicate-end": "окончао", + "predicate-assignee": "овластио", "predicate-member": "сарадник", - "predicate-public": "јавно", - "predicate-private": "приватно", + "predicate-public": "јавни", + "predicate-private": "тајни", "predicate-selector": "изборник", "predicate-projection": "очекивање", "operator-unknown-error": "%s уопште није оператор", @@ -1106,76 +1106,76 @@ "operator-debug-invalid": "%s уопште није исправан предикат", "next-page": "Следећа страна", "previous-page": "Претходна страна", - "heading-notes": "Белешке", - "globalSearch-instructions-heading": "Упутство за детаљно претраживање", - "globalSearch-instructions-description": "Да би се претрага побољшала, можете да користите операторе. Оператори се задају навођењем имена оператора и навођењем вредности где се оператор и вредност раздвајају двотачком. На пример, ако задате `деоница:блокирано` то ће сузити претрагу само на оне картице са задацима које су смештене на деоници под називом *блокирано*. Уколико вредност садржи празна места или посебне знакове морате их обухватити наводницима (нпр. `__operator_list__:\"радови у току\"`).", + "heading-notes": "Додатак", + "globalSearch-instructions-heading": "Правилник за детаљно претраживање", + "globalSearch-instructions-description": "Да би сте сузили резултате претраге предлажемо Вам да почнете да користите операнде. Операнд је математички појам којег овде дефинишемо као целину која се састоји од променљиве и њене припадајуће вредности који су међусобно раздвојени двотачком. На пример, ако задате операнд `део:одложено` то ће сузити претрагу само на оне предмете који су смештени у делу поступка под називом *одложено*. Уколико вредност садржи размак између речи или садржи посебне знакове вредност морате обухватити наводницима (нпр. `__operator_list__:\"у жалбеном поступку\"`).", "globalSearch-instructions-operators": "Оператори на располагању:", - "globalSearch-instructions-operator-board": "`__operator_board__:<наслов>` - задаци на картицама у пословним књигама које садрже *<наслов>*", - "globalSearch-instructions-operator-list": "`__operator_list__:<назив>` - задаци на деоницама које садрже *<назив>*", - "globalSearch-instructions-operator-swimlane": "`__operator_swimlane__:<назив>` - задаци на стазама које садрже *<назив>*", - "globalSearch-instructions-operator-comment": "`__operator_comment__:<речи>` - задаци где је мишљење исказано *<речима>*.", - "globalSearch-instructions-operator-label": "`__operator_label__:<боја>` `__operator_label__:<име>` - задаци чија налепница има *<боју>* or *<име>", - "globalSearch-instructions-operator-hash": "`__operator_label_abbrev__<име|боја>` - скраћеница за `__operator_label__:<боју>` или `__operator_label__:<име>`", - "globalSearch-instructions-operator-user": "`__operator_user__:<неко>` - задаци где је *<неко>* или *сарадник* или *извршилац*", + "globalSearch-instructions-operator-board": "`__operator_board__:<наслов>` - сви предмети у списима са задатим *<насловом>*", + "globalSearch-instructions-operator-list": "`__operator_list__:<назив>` - сви предмети чији део поступка има следећи *<назив>*", + "globalSearch-instructions-operator-swimlane": "`__operator_swimlane__:<врста>` - сви предмети који се налазе у задатој *<врсти>* поступка", + "globalSearch-instructions-operator-comment": "`__operator_comment__:<речи>` - сви предмети где су у расправи коришћене следеће *<речи>*.", + "globalSearch-instructions-operator-label": "`__operator_label__:<боја>` `__operator_label__:<натпис>` - сви предмети на које је залепљена налепница следеће *<боје>* а која има *<натпис>", + "globalSearch-instructions-operator-hash": "`__operator_label_abbrev__<име|боја>` - скраћеница за `__operator_label__:<боја>` или `__operator_label__:<име>`", + "globalSearch-instructions-operator-user": "`__operator_user__:<неко>` - сви предмети где је *<неко>* или *сарадник* или *пуномоћник*", "globalSearch-instructions-operator-at": "`__operator_user_abbrev__корисничко име` - скраћеница за `user:<корисничко име>`", - "globalSearch-instructions-operator-member": "`__operator_member__:<неко>` - задаци где је *<неко>* *сарадник*", - "globalSearch-instructions-operator-assignee": "`__operator_assignee__:<неко>` - задаци где је *<неком>* *додељен* задатак", - "globalSearch-instructions-operator-creator": "`__operator_creator__:<неко>` - задаци где је *<неко>* направио картицу са задатком", - "globalSearch-instructions-operator-org": "`__operator_org__:<назив предузећа|скраћени назив>` - задаци који припадају пословној књизи додељеној *<предузећу>*", - "globalSearch-instructions-operator-team": "`__operator_team__:<назив тима|скраћени назив>` - задаци који припадају пословној књизи додељеној *<тиму>*", - "globalSearch-instructions-operator-due": "`__operator_due__:` - задаци чији рок истиче за ** дана. `__operator_due__:__predicate_overdue__ приказује списак свих задатака са истеклим роком.", - "globalSearch-instructions-operator-created": "`__operator_created__:` - задаци који су постављени пре ** или мање дана", - "globalSearch-instructions-operator-modified": "`__operator_modified__:` - задаци који су измењени пре ** или мање дана", + "globalSearch-instructions-operator-member": "`__operator_member__:<неко>` - предмети где је *<неко>* сарадник", + "globalSearch-instructions-operator-assignee": "`__operator_assignee__:<неко>` - предмети где је *<неко>* пуномоћник", + "globalSearch-instructions-operator-creator": "`__operator_creator__:<неко>` - предмети које је завео *<неко>*", + "globalSearch-instructions-operator-org": "`__operator_org__:<име или пословно име|скраћено пословно име>` - предмети у вези са *<странком>*", + "globalSearch-instructions-operator-team": "`__operator_team__:<назив тима|скраћени назив>` - предмети које обрађује правни тим *<назив>*", + "globalSearch-instructions-operator-due": "`__operator_due__:` - предмети чији рок истиче за ** дана. `__operator_due__:__predicate_overdue__ предмети са истеклим роком.", + "globalSearch-instructions-operator-created": "`__operator_created__:` - предмети који су заведени у задњих ** дана", + "globalSearch-instructions-operator-modified": "`__operator_modified__:` - предмети који су имали измене у задњих ** дана", "globalSearch-instructions-operator-status": "`__operator_status__:<стање>` - где се под *<стањем>* подразумева једно од следећих:", - "globalSearch-instructions-status-archived": "`__predicate_archived__` - архивирани задаци", + "globalSearch-instructions-status-archived": "`__predicate_archived__` - архивирани предмети", "globalSearch-instructions-status-all": "`__predicate_all__` - сви архивирани и неархивирани задаци", - "globalSearch-instructions-status-ended": "`__predicate_ended__` - задаци где је постављен датум завршетка", - "globalSearch-instructions-status-public": "`__predicate_public__` - једино задаци у јавним пословним књигама", - "globalSearch-instructions-status-private": "`__predicate_private__` - једино задаци у приватним пословним књигама", - "globalSearch-instructions-operator-has": "`__operator_has__:<поље>` - где је то *<поље>* једно од `__predicate_attachment__`, `__predicate_checklist__`, `__predicate_description__`, `__predicate_start__`, `__predicate_due__`, `__predicate_end__`, `__predicate_assignee__` или `__predicate_member__`. Уколико уметнете знак минус `-` испред тог *<поља>* претражује се недостатак те вредности (нпр `има:-рок` претражује задатке без постављеног рока).", + "globalSearch-instructions-status-ended": "`__predicate_ended__` - сви окончани предмети", + "globalSearch-instructions-status-public": "`__predicate_public__` - предмети видљиви свима на интернету", + "globalSearch-instructions-status-private": "`__predicate_private__` - предмети видљиви искључиво сарадницима", + "globalSearch-instructions-operator-has": "`__operator_has__:<поље>` - где је то *<поље>* једно од `__predicate_attachment__`, `__predicate_checklist__`, `__predicate_description__`, `__predicate_start__`, `__predicate_due__`, `__predicate_end__`, `__predicate_assignee__` или `__predicate_member__`. Уколико уметнете знак минус `-` испред тог *<поља>* претражује се недостатак те вредности (нпр `има:-рок` претражује предмете без постављеног рока).", "globalSearch-instructions-operator-sort": "`__operator_sort__:<правило>` - где је *<правило>* једно од `__predicate_due__`, `__predicate_created__` или `__predicate_modified__`. За опадајући редослед уметните минус `-` испред правила.", - "globalSearch-instructions-operator-limit": "`__operator_limit__:` - позитиван цео број ** представља број задатака који се приказују по страни.", + "globalSearch-instructions-operator-limit": "`__operator_limit__:` - позитиван цео број ** представља број предмета који се приказују по страни.", "globalSearch-instructions-notes-1": "Можете истовремено задати више оператора.", - "globalSearch-instructions-notes-2": "Над поновљеним операторима се изводи операција логичко ИЛИ. Биће приказани задаци који испуњавају било који од услова.\n`__operator_list__:Доступно __operator_list__:Блокирано` као резултат приказује све картице које су било у деоници *Блокирано* било у деоници *Доступно*.", - "globalSearch-instructions-notes-3": "На различитим операторима се изводи операција логичко И. Биће приказани само они задаци који испуњавају све услове. `__operator_list__:Доступно __operator_label__:црвено` приказаће међу свим задацима на деоници *Доступно* само оне са *црвеном* налепницом.", + "globalSearch-instructions-notes-2": "Над сродним операторима се изводи операција логичко ИЛИ. Биће приказани предмети који испуњавају било који од услова.\n`__operator_list__:Припрема __operator_list__:Жалба` као резултат приказује све предмете које су било у деоници *Припрема* било у деоници *Жалба*.", + "globalSearch-instructions-notes-3": "На несродним операторима се изводи операција логичко И. Биће приказани само они задаци који испуњавају све услове. `__operator_list__:Жалба __operator_label__:црвена` приказаће међу свим предметима у *жалбеном* делу поступка само оне са *црвеном* налепницом.", "globalSearch-instructions-notes-3-2": "Дани се могу задати као позитивни или негативни цели бројеви или се користе `__predicate_week__`, `__predicate_month__`, `__predicate_quarter__` or `__predicate_year__` за текући период.", - "globalSearch-instructions-notes-4": "Претрага речи не прави разлику између малих и великих слова.", - "globalSearch-instructions-notes-5": "Преподешено је да се архивиране картице са задацима не претражују.", + "globalSearch-instructions-notes-4": "Разлика између малих и великих слова се не узима у обзир.", + "globalSearch-instructions-notes-5": "Ако нисте ништа дирали, претрагом неће бити обухваћени предмети из архиве.", "link-to-search": "Повежи до ове претраге", "excel-font": "Arial словни лик", "number": "Број", - "label-colors": "Боје налепница", + "label-colors": "Налепнице у боји", "label-names": "Натписи на налепницама", - "archived-at": "време архивирања", - "sort-cards": "Пресложи картице", - "sort-is-on": "Слагање је укључено", - "cardsSortPopup-title": "Пресложи картице", - "due-date": "Датум истека", + "archived-at": "датум и време архивирања", + "sort-cards": "Сложи предмете", + "sort-is-on": "Предмети су сложени", + "cardsSortPopup-title": "Сложи предмете", + "due-date": "По крајњем року", "server-error": "Грешка на серверу", "server-error-troubleshooting": "Молим да нам пошаљете извештај о грешци коју је изазвао сервер.\nАко је у питању snap инсталација, покрените: `sudo snap logs wekan.wekan`\nАко је у питању Docker инсталација, покрените: `sudo docker logs wekan-app`", - "title-alphabetically": "Наслов (Азбучним редом)", - "created-at-newest-first": "Кад је направљено (Најновије прво)", - "created-at-oldest-first": "Кад је направљено (Најстарије прво)", + "title-alphabetically": "По наслову абучним редом", + "created-at-newest-first": "По најновијим запримљеним", + "created-at-oldest-first": "По најстарије запримљеним)", "links-heading": "Везе", "hide-activities-of-all-boards": "Don't show the board activities on all boards", "now-activities-of-all-boards-are-hidden": "Now all activities of all boards are hidden", - "move-swimlane": "Премести стазу", - "moveSwimlanePopup-title": "Премештање стазе", - "custom-field-stringtemplate": "Предложак за словни низ", + "move-swimlane": "Премести ток поступка", + "moveSwimlanePopup-title": "Премештање тока поступка", + "custom-field-stringtemplate": "Образац за словни низ", "custom-field-stringtemplate-format": "Облик (користите %{вредност} као основу/носач)", "custom-field-stringtemplate-separator": "Раздвајач (користите или   за размак)", "custom-field-stringtemplate-item-placeholder": "Притисните ентер да додате још ставки", - "creator": "Задао", - "creator-on-minicard": "Creator on minicard", - "filesReportTitle": "Извештај везан за датотеке", + "creator": "Завео", + "creator-on-minicard": "Завео", + "filesReportTitle": "Извештај везан за предметну грађу", "reports": "Извештаји", "rulesReportTitle": "Извештај везан за правила", - "boardsReportTitle": "Извештај везан за пословне књиге", - "cardsReportTitle": "Извештај везан за картице", - "copy-swimlane": "Умножи стазу", - "copySwimlanePopup-title": "Умножавање стазе", - "display-card-creator": "Прикажи ко је направио картицу", - "wait-spinner": "Док се исчитава пословна књига", + "boardsReportTitle": "Извештај везан за списе", + "cardsReportTitle": "Извештај везан за предмете", + "copy-swimlane": "Умножи ток поступка", + "copySwimlanePopup-title": "Умножавање тока поступка", + "display-card-creator": "Прикажи ко је завео предмет", + "wait-spinner": "Док се исчитавају списи", "Bounce": "Исцртавај три тачкице", "Cube": "Исцртавај квадратиће", "Cube-Grid": "Исцртавај мрежу квадратића", @@ -1184,16 +1184,16 @@ "Rotateplane": "Исцртавај лист који се окреће", "Scaleout": "Исцртавај удаљавање", "Wave": "Исцртавај таласе", - "maximize-card": "Увећај картицу", - "minimize-card": "Умањи картицу", - "delete-org-warning-message": "Не могу да угасим ово предузеће пошто постоји барем један корисник који му припада", - "delete-team-warning-message": "Не могу да расформирам овај тим пошто постоји барем један корисник који му припада", + "maximize-card": "Рашири предметне списе", + "minimize-card": "Састави предметне списе", + "delete-org-warning-message": "Не могу да избацим ову странку пошто постоји барем један везани сарадник", + "delete-team-warning-message": "Не могу да распустим овај правни тим пошто постоји барем једно лице које му припада", "subject": "Тема", "details": "Детаљи", "carbon-copy": "још један примерак", - "ticket": "Предмет", - "tickets": "Предмети", - "ticket-number": "Број предмета", + "ticket": "Пријава квара", + "tickets": "Пријављени кварови", + "ticket-number": "Пријавни број", "open": "Отворен", "pending": "На чекању", "closed": "Затворен", @@ -1203,14 +1203,14 @@ "request": "Захтев", "requests": "Захтеви", "help-request": "Захтева се помоћ", - "editCardSortOrderPopup-title": "Промените правило слагања картица", - "cardDetailsPopup-title": "Појединости картице", + "editCardSortOrderPopup-title": "Промените редослед слагања предмета", + "cardDetailsPopup-title": "Шире радње на предмету", "add-teams": "Додај тимове", "add-teams-label": "Додати тимови су приказани доле:", - "remove-team-from-table": "Да ли сте сигурни да желите да уклоните овај тим из пословне књиге ?", + "remove-team-from-table": "Да ли сте сигурни да желите да овом правном тиму укинете увид у ове списе ?", "confirm-btn": "Потврди", "remove-btn": "Уклони", - "filter-card-title-label": "Издвој по наслову картице са задатком", + "filter-card-title-label": "Издвој по наслову предмета", "invite-people-success": "Позив за сарадњу је успешно послат", "invite-people-error": "Догодила се грешка приликом слања позива за сарадњу", "can-invite-if-same-mailDomainName": "Домен за електронску пошту", @@ -1230,350 +1230,350 @@ "Node_memory_usage_heap_total": "Node искоришћење меморије: укупан простор који је алоциран за heap", "Node_memory_usage_heap_used": "Node искоришћење меморије: колико заправо је искоришћено", "Node_memory_usage_external": "Node искоришћење меморије: спољна", - "add-organizations": "Додај предузећа", - "add-organizations-label": "Додана предузећа су приказана доле:", - "remove-organization-from-board": "Да ли сте сигурни да желите да уклоните ово предузеће из ове пословне књиге ?", - "to-create-organizations-contact-admin": "Да би могли да додајете предузећа, молим да се обратите управнику/администратору.", + "add-organizations": "Додај странке", + "add-organizations-label": "Странке које имају увид су приказана доле:", + "remove-organization-from-board": "Да ли сте сигурни да желите да овој странци забраните увид у ове списе ?", + "to-create-organizations-contact-admin": "Да би могли да додајете странке, молим да се обратите управнику/администратору.", "custom-legal-notice-link-url": "Место на Интернету са условима/уговором за коришћење", "acceptance_of_our_legalNotice": "Ако наставите даље, сагласни сте да прихватате наше", "legalNotice": "услови коришћења", "copied": "Умножено!", - "checklistActionsPopup-title": "Радње на списковима за обавити", - "moveChecklist": "Премести списак за обавити", - "moveChecklistPopup-title": "Премести списак за обавити", - "newlineBecomesNewChecklistItem": "Each line of text becomes one of the checklist items", - "newLineNewItem": "One line of text = one checklist item", - "newlineBecomesNewChecklistItemOriginOrder": "Each line of text becomes one of the checklist items, original order", - "originOrder": "original order", + "checklistActionsPopup-title": "Руковање списковима радњи", + "moveChecklist": "Премести списак радњи", + "moveChecklistPopup-title": "Премести списак радњи", + "newlineBecomesNewChecklistItem": "Сваки ред текста постаје ставка на списку", + "newLineNewItem": "Један ред = једна ставка", + "newlineBecomesNewChecklistItemOriginOrder": "Сваки ред текста постаје ставка на списку - по изворном редоследу уноса", + "originOrder": "изворни редослед", "copyChecklist": "Умножи списак", "copyChecklistPopup-title": "Умножи списак", - "card-show-lists": "Прикажи спискове на картици", - "subtaskActionsPopup-title": "Радње на под задацима", + "card-show-lists": "Прикажи делове поступка на омоту предмета", + "subtaskActionsPopup-title": "Радње на издвојеним пословима", "attachmentActionsPopup-title": "Однос према прилозима", "attachment-move-storage-fs": "Премести прилог у локални систем датотека", "attachment-move-storage-gridfs": "Премести прилог у GridFS", - "attachment-move-storage-s3": "Премести прилог у облак на S3", + "attachment-move-storage-s3": "Премести прилог у облак на Amazon S3", "attachment-move": "Премести прилог", - "move-all-attachments-to-fs": "Премести све прилоге у локални систем датотека", - "move-all-attachments-to-gridfs": "Премести све прилоге у GridFS", - "move-all-attachments-to-s3": "Премести све прилоге у облак на S3", - "move-all-attachments-of-board-to-fs": "Премести све прилоге пословне књиге у локални систем датотека", - "move-all-attachments-of-board-to-gridfs": "Премести све прилоге пословне књиге у GridFS", - "move-all-attachments-of-board-to-s3": "Премести све прилоге пословне књиге у облак на S3", + "move-all-attachments-to-fs": "Премести целокупну предметну грађу у локални систем датотека", + "move-all-attachments-to-gridfs": "Премести целокупну предметну грађу у MongoDB GridFS", + "move-all-attachments-to-s3": "Премести целокупну предметну грађу из списа у облак на Amazon S3", + "move-all-attachments-of-board-to-fs": "Премести предметну грађу ових списа у локални систем датотека", + "move-all-attachments-of-board-to-gridfs": "Премести предметну грађу ових списа у MongoDB GridFS", + "move-all-attachments-of-board-to-s3": "Премести предметну грађу ових списа у облак на Amazon S3", "path": "Путања", "version-name": "Назив издања", "size": "Величина", "storage": "Складиште", "action": "Радња", - "board-title": "Натпис на пословној књизи", + "board-title": "Деловодни број", "attachmentRenamePopup-title": "Преименуј", "uploading": "Подижем на сервер", "remaining_time": "Преостало време", "speed": "Брзина", "progress": "Напредак", - "password-again": "Лозинка (понови је)", - "if-you-already-have-an-account": "Ако већ имате један налог", - "register": "Регистрација", + "password-again": "Поновљена лозинка", + "if-you-already-have-an-account": "Ако већ поседујете налог ⇨ ", + "register": "Упиши се", "forgot-password": "Заборављена лозинка", - "minicardDetailsActionsPopup-title": "Појединости картице", + "minicardDetailsActionsPopup-title": "Предмет у омоту", "Mongo_sessions_count": "Број Mongo сесија", - "change-visibility": "Промени видљивост", + "change-visibility": "Промени тајност списа", "max-upload-filesize": "Максимална величина датотеке за слање (у бајтима):", "allowed-upload-filetypes": "Дозвољене врсте датотека за слање:", - "max-avatar-filesize": "Максимална величина сличице (у бајтима):", + "max-avatar-filesize": "Максимална величина слике (у бајтима):", "allowed-avatar-filetypes": "Дозвољене врсте слика:", - "invalid-file": "Ако са именом датотеке нешто није у реду тада ће и слање и преименовање бити отказано.", - "preview-pdf-not-supported": "Ваш уређај не подржава приказивање ПДФ докумената. Пробајте да преузмете документ.", - "drag-board": "Пренеси пословну књигу", - "translation-number": "Број прилагођених превода је:", - "delete-translation-confirm-popup": "Да ли сте сигурни да желите да избришете овај прилагођени превод? Опозив ове радње неће бити могућ.", - "newTranslationPopup-title": "Нови прилагођени превод", - "editTranslationPopup-title": "Уреди прилагођени превод", - "settingsTranslationPopup-title": "Обриши овај прилагођени превод", - "translation": "Превод", - "text": "Текст", - "translation-text": "Превод текста", - "show-subtasks-field": "Прикажи поље за подзадатке", + "invalid-file": "Ако са именом нешто није у реду тада ће и слање и преименовање бити отказано.", + "preview-pdf-not-supported": "Ваш уређај не подржава приказивање pdf докумената. Пробајте да преузмете документ.", + "drag-board": "Пребаци списе", + "translation-number": "Број исправки превода је:", + "delete-translation-confirm-popup": "Да ли сте сигурни да желите да избришете ову исправку превода? Опозив ове радње неће бити могућ.", + "newTranslationPopup-title": "Нова исправка превода", + "editTranslationPopup-title": "Исправи превод", + "settingsTranslationPopup-title": "Обриши ову исправку превода", + "translation": "Речник", + "text": "Изворна реченица", + "translation-text": "Превод", + "show-subtasks-field": "Прикажи поље издвојени послови", "show-week-of-year": "Show week of year (ISO 8601)", - "convert-to-markdown": "Претвори у маркдаун", - "import-board-zip": "Add .zip file that has board JSON files, and board name subdirectories with attachments", - "collapse": "Сажми", - "uncollapse": "Uncollapse", - "hideCheckedChecklistItems": "Hide checked checklist items", - "hideAllChecklistItems": "Hide all checklist items", - "support": "Support", - "supportPopup-title": "Support", - "accessibility": "Accessibility", - "accessibility-page-enabled": "Accessibility page enabled", - "accessibility-info-not-added-yet": "Accessibility info has not been added yet", - "accessibility-title": "Accessibility title", - "accessibility-content": "Accessibility content", - "accounts-lockout-settings": "Brute Force Protection Settings", - "accounts-lockout-info": "These settings control how login attempts are protected against brute force attacks.", - "accounts-lockout-known-users": "Settings for known users (correct username, wrong password)", - "accounts-lockout-unknown-users": "Settings for unknown users (non-existent username)", - "accounts-lockout-failures-before": "Failures before lockout", - "accounts-lockout-period": "Lockout period (seconds)", - "accounts-lockout-failure-window": "Failure window (seconds)", - "accounts-lockout-settings-updated": "Brute force protection settings have been updated", - "accounts-lockout-locked-users": "Locked Users", - "accounts-lockout-locked-users-info": "Users currently locked out due to too many failed login attempts", - "accounts-lockout-no-locked-users": "There are currently no locked users", - "accounts-lockout-failed-attempts": "Failed Attempts", - "accounts-lockout-remaining-time": "Remaining Time", - "accounts-lockout-user-unlocked": "User has been unlocked successfully", - "accounts-lockout-confirm-unlock": "Are you sure you want to unlock this user?", - "accounts-lockout-confirm-unlock-all": "Are you sure you want to unlock all locked users?", - "accounts-lockout-show-locked-users": "Show locked users only", - "accounts-lockout-user-locked": "User is locked", - "accounts-lockout-click-to-unlock": "Click to unlock this user", - "accounts-lockout-status": "Стање", - "admin-people-filter-show": "Show:", - "admin-people-filter-all": "Недовршени послови сарадника", - "admin-people-filter-locked": "Locked Users Only", - "admin-people-filter-active": "На окупу", - "admin-people-filter-inactive": "Not Active", - "admin-people-active-status": "Active Status", - "admin-people-user-active": "User is active - click to deactivate", - "admin-people-user-inactive": "User is inactive - click to activate", - "accounts-lockout-all-users-unlocked": "All locked users have been unlocked", - "accounts-lockout-unlock-all": "Unlock All", - "active-cron-jobs": "Active Scheduled Jobs", - "add-cron-job": "Add Scheduled Job", - "add-cron-job-placeholder": "Add Scheduled Job functionality coming soon", - "attachment-storage-configuration": "Attachment Storage Configuration", - "attachments-path": "Attachments Path", - "attachments-path-description": "Path where attachment files are stored", - "avatars-path": "Avatars Path", - "avatars-path-description": "Path where avatar files are stored", - "board-archive-failed": "Failed to schedule board archive", - "board-archive-scheduled": "Board archive scheduled successfully", - "board-backup-failed": "Failed to schedule board backup", - "board-backup-scheduled": "Board backup scheduled successfully", - "board-cleanup-failed": "Failed to schedule board cleanup", - "board-cleanup-scheduled": "Board cleanup scheduled successfully", - "board-operations": "Board Operations", - "cron-jobs": "Scheduled Jobs", - "cron-migrations": "Scheduled Migrations", - "cron-job-delete-confirm": "Are you sure you want to delete this scheduled job?", - "cron-job-delete-failed": "Failed to delete scheduled job", - "cron-job-deleted": "Scheduled job deleted successfully", - "cron-job-pause-failed": "Failed to pause scheduled job", - "cron-job-paused": "Scheduled job paused successfully", - "filesystem-path-description": "Base path for file storage", - "gridfs-enabled": "GridFS Enabled", - "gridfs-enabled-description": "Use MongoDB GridFS for file storage", - "migration-pause-failed": "Failed to pause migrations", - "migration-paused": "Migrations paused successfully", - "migration-progress": "Migration Progress", - "migration-start-failed": "Failed to start migrations", - "migration-started": "Migrations started successfully", - "migration-status": "Migration Status", - "migration-stop-confirm": "Are you sure you want to stop all migrations?", - "migration-stop-failed": "Failed to stop migrations", - "migration-stopped": "Migrations stopped successfully", - "mongodb-gridfs-storage": "MongoDB GridFS Storage", - "pause-all-migrations": "Pause All Migrations", - "s3-access-key": "S3 Access Key", - "s3-access-key-description": "AWS S3 access key for authentication", - "s3-access-key-placeholder": "Enter S3 access key", + "convert-to-markdown": "Претвори у структуирани текст", + "import-board-zip": "Додај .zip архиву која има списе у json облику и фасцикле по имену списа где је предметна грађа", + "collapse": "Скупи", + "uncollapse": "Рашири", + "hideCheckedChecklistItems": "Сакриј све обављене помоћне предметне радње", + "hideAllChecklistItems": "Сакриј све помоћне предметне радње", + "support": "Подршка", + "supportPopup-title": "Подршка", + "accessibility": "Особе са посебним потешкоћама", + "accessibility-page-enabled": "Омогући страницу за особе са посебним потешкоћама", + "accessibility-info-not-added-yet": "Информације намењене особама са посебним потешкоћама, за сада, нису додате", + "accessibility-title": "Наслов такве странице", + "accessibility-content": "Садржај такве странице", + "accounts-lockout-settings": "Заштитне мере од насилног упада", + "accounts-lockout-info": "Овим мерама се штитимо од насилних покушаја пријављивања погађањем лозинкеf.", + "accounts-lockout-known-users": "Мере за познате налоге (где налог постоји али се лозинка не подудара)", + "accounts-lockout-unknown-users": "Мере за непознате налоге (где сарадник са таквим корисничким именом не постоји)", + "accounts-lockout-failures-before": "Граница неуспешних покушаја пријаве (након које се сараднику одузима право приступа)", + "accounts-lockout-period": "Трајање мере забране (у секундама)", + "accounts-lockout-failure-window": "Временски оквир у којем се одиграва насилан упад (у секундама)", + "accounts-lockout-settings-updated": "Мере против насилног упада су допуњене", + "accounts-lockout-locked-users": "Налози на мерама", + "accounts-lockout-locked-users-info": "Сарадници којима је одузето право приступа због превеликог броја неуспешних покушаја пријаве", + "accounts-lockout-no-locked-users": "Тренутно не постоје налози са мером забране права приступа", + "accounts-lockout-failed-attempts": "Број неуспешних покушаја", + "accounts-lockout-remaining-time": "Преостало време", + "accounts-lockout-user-unlocked": "Налог је успешно скинут са мера забране приступа", + "accounts-lockout-confirm-unlock": "Да ли сте сигурни да желите да скинете овај налог са мера?", + "accounts-lockout-confirm-unlock-all": "Да ли сте сигурни да желите да скинете све налоге са мера?", + "accounts-lockout-show-locked-users": "Прикажи само налоге на мерама", + "accounts-lockout-user-locked": "Налог је на мерама", + "accounts-lockout-click-to-unlock": "Притисните да налог добије право приступа", + "accounts-lockout-status": "Право приступа", + "admin-people-filter-show": "Издвој:", + "admin-people-filter-all": "Обухвати све расположиве сараднике", + "admin-people-filter-locked": "Налоге на мерама забране права приступа", + "admin-people-filter-active": "Сарадник је на платном списку", + "admin-people-filter-inactive": "Сарадник више није на платном списку", + "admin-people-active-status": "Радни однос", + "admin-people-user-active": "Сарадник је на платном списку - притисните да би прекинули сарадњу", + "admin-people-user-inactive": "Сарадник није на платном списку - притисните да би га увели на платни списак", + "accounts-lockout-all-users-unlocked": "Сви сарадници којима је претходно било одузето право пријаве су скинути са ове мере", + "accounts-lockout-unlock-all": "Скини све са мера", + "active-cron-jobs": "Већ унапред заказани послови:", + "add-cron-job": "Закажи нови посао", + "add-cron-job-placeholder": "Ова функционалност биће ускоро додата", + "attachment-storage-configuration": "Поставка складишта предметне грађе", + "attachments-path": "Пуна путања до складишта", + "attachments-path-description": "Путања до складишта где се чува предметна грађа", + "avatars-path": "Путања до слика сарадника", + "avatars-path-description": "Путања где се складиште слике сарадника", + "board-archive-failed": "Није успело заказивање архивирања списа", + "board-archive-scheduled": "Заказано је паковање списа у архиву", + "board-backup-failed": "Није успело заказивање израде резервног примерка списа", + "board-backup-scheduled": "Заказана је израда резервног примерка списа", + "board-cleanup-failed": "Није успело заказивање чишћења списа", + "board-cleanup-scheduled": "Заказано је чишћење списа", + "board-operations": "Радње на списима", + "cron-jobs": "Заказани послови", + "cron-migrations": "Заказани поступци обнове оштећених списа", + "cron-job-delete-confirm": "Да ли сте сигурни да желите да уклоните заказано?", + "cron-job-delete-failed": "Није успело уклањање заказаног посла", + "cron-job-deleted": "Успешно је уклоњен заказани посао", + "cron-job-pause-failed": "Није успело паузирање заказаног посла", + "cron-job-paused": "Заказани посао је успешно паузиран", + "filesystem-path-description": "Почетак пута до складишта предметне грађе", + "gridfs-enabled": "GridFS ради и можете га користити", + "gridfs-enabled-description": "Можете користити MongoDB GridFS за складиште предметне грађе", + "migration-pause-failed": "Није било могуће направити предах у поступку обнове оштећених списа", + "migration-paused": "Направљен је предах у поступку обнове оштећених списа", + "migration-progress": "Напредак у току обнове", + "migration-start-failed": "Није било могуће покренути обнову оштећених списа", + "migration-started": "Обнова оштећених списа је започета", + "migration-status": "Пресек стања", + "migration-stop-confirm": "Да ли сте сигурни да желите да зауставите све поступке обнове оштећених предметних списа?", + "migration-stop-failed": "Није било могуће прекинути поступке обнове оштећених списа", + "migration-stopped": "Поступци обнове оштећених списа су управо заустављени", + "mongodb-gridfs-storage": "MongoDB GridFS складиште", + "pause-all-migrations": "Предах за све", + "s3-access-key": "S3 приступни кључ", + "s3-access-key-description": "AWS S3 приступни кључ за пријаву", + "s3-access-key-placeholder": "Унесите S3 приступни кључ", "s3-bucket": "S3 Bucket", - "s3-bucket-description": "S3 bucket name for storing files", - "s3-connection-failed": "S3 connection failed", - "s3-connection-success": "S3 connection successful", - "s3-enabled": "S3 Enabled", - "s3-enabled-description": "Use AWS S3 or MinIO for file storage", - "s3-endpoint": "S3 Endpoint", - "s3-endpoint-description": "S3 endpoint URL (e.g., s3.amazonaws.com or minio.example.com)", - "s3-minio-storage": "S3/MinIO Storage", + "s3-bucket-description": "S3 bucket име где чувате предметну грађу", + "s3-connection-failed": "S3 веза је у прекиду", + "s3-connection-success": "Управо је остварена веза до S3", + "s3-enabled": "S3 ради и може да се користи", + "s3-enabled-description": "Можете да користите AWS S3 или MinIO као складиште предметне грађе", + "s3-endpoint": "S3 Endpoint тачка", + "s3-endpoint-description": "S3 endpoint тачка (нпр. s3.amazonaws.com или minio.example.com)", + "s3-minio-storage": "S3/MinIO складиште", "s3-port": "S3 Port", - "s3-port-description": "S3 endpoint port number", + "s3-port-description": "S3 endpoint port број", "s3-region": "S3 Region", - "s3-region-description": "AWS S3 region (e.g., us-east-1)", - "s3-secret-key": "S3 Secret Key", - "s3-secret-key-description": "AWS S3 secret key for authentication", - "s3-secret-key-placeholder": "Enter S3 secret key", - "s3-secret-key-required": "S3 secret key is required", - "s3-settings-save-failed": "Failed to save S3 settings", - "s3-settings-saved": "S3 settings saved successfully", - "s3-ssl-enabled": "S3 SSL Enabled", - "s3-ssl-enabled-description": "Use SSL/TLS for S3 connections", - "save-s3-settings": "Save S3 Settings", - "schedule-board-archive": "Schedule Board Archive", - "schedule-board-backup": "Schedule Board Backup", - "schedule-board-cleanup": "Schedule Board Cleanup", - "scheduled-board-operations": "Scheduled Board Operations", - "start-all-migrations": "Start All Migrations", - "stop-all-migrations": "Stop All Migrations", - "test-s3-connection": "Test S3 Connection", - "writable-path": "Writable Path", - "writable-path-description": "Base directory path for file storage", - "add-job": "Add Job", - "attachment-migration": "Attachment Migration", - "attachment-monitoring": "Attachment Monitoring", - "attachment-settings": "Attachment Settings", - "attachment-storage-settings": "Storage Settings", - "automatic-migration": "Automatic Migration", - "back-to-settings": "Back to Settings", - "board-id": "Board ID", - "board-migration": "Board Migration", - "board-migrations": "Board Migrations", - "card-show-lists-on-minicard": "Show Lists on Minicard", - "comprehensive-board-migration": "Comprehensive Board Migration", - "comprehensive-board-migration-description": "Performs comprehensive checks and fixes for board data integrity, including list ordering, card positions, and swimlane structure.", - "delete-duplicate-empty-lists-migration": "Delete Duplicate Empty Lists", - "delete-duplicate-empty-lists-migration-description": "Safely deletes empty duplicate lists. Only removes lists that have no cards AND have another list with the same title that contains cards.", - "lost-cards": "Lost Cards", - "lost-cards-list": "Restored Items", - "restore-lost-cards-migration": "Restore Lost Cards", - "restore-lost-cards-migration-description": "Finds and restores cards and lists with missing swimlaneId or listId. Creates a 'Lost Cards' swimlane to make all lost items visible again.", - "restore-all-archived-migration": "Restore All Archived", - "restore-all-archived-migration-description": "Restores all archived swimlanes, lists, and cards. Automatically fixes any missing swimlaneId or listId to make items visible.", - "fix-missing-lists-migration": "Fix Missing Lists", - "fix-missing-lists-migration-description": "Detects and repairs missing or corrupted lists in the board structure.", - "fix-avatar-urls-migration": "Fix Avatar URLs", - "fix-avatar-urls-migration-description": "Updates avatar URLs for board members to use the correct storage backend and fixes broken avatar references.", - "fix-all-file-urls-migration": "Fix All File URLs", - "fix-all-file-urls-migration-description": "Updates all file attachment URLs on this board to use the correct storage backend and fixes broken file references.", - "migration-needed": "Migration Needed", - "migration-complete": "Complete", - "migration-running": "Running...", - "migration-successful": "Migration completed successfully", - "migration-failed": "Migration failed", - "migrations": "Migrations", - "migrations-admin-only": "Only board administrators can run migrations", - "migrations-description": "Run data integrity checks and repairs for this board. Each migration can be executed individually.", - "no-issues-found": "No issues found", - "run-migration": "Run Migration", - "run-comprehensive-migration-confirm": "This will perform a comprehensive migration to check and fix board data integrity. This may take a few moments. Continue?", - "run-delete-duplicate-empty-lists-migration-confirm": "This will first convert any shared lists to per-swimlane lists, then delete empty lists that have a duplicate list with the same title containing cards. Only truly redundant empty lists will be removed. Continue?", - "run-restore-lost-cards-migration-confirm": "This will create a 'Lost Cards' swimlane and restore all cards and lists with missing swimlaneId or listId. This only affects non-archived items. Continue?", - "run-restore-all-archived-migration-confirm": "This will restore ALL archived swimlanes, lists, and cards, making them visible again. Any items with missing IDs will be automatically fixed. This cannot be easily undone. Continue?", - "run-fix-missing-lists-migration-confirm": "This will detect and repair missing or corrupted lists in the board structure. Continue?", - "run-fix-avatar-urls-migration-confirm": "This will update avatar URLs for board members to use the correct storage backend. Continue?", - "run-fix-all-file-urls-migration-confirm": "This will update all file attachment URLs on this board to use the correct storage backend. Continue?", - "restore-lost-cards-nothing-to-restore": "No lost swimlanes, lists, or cards to restore", + "s3-region-description": "AWS S3 регион (нпр. us-east-1)", + "s3-secret-key": "S3 тајни кључ", + "s3-secret-key-description": "AWS S3 тајни кључ за пријаву", + "s3-secret-key-placeholder": "Унесите S3 тајни кључ", + "s3-secret-key-required": "Тражи се и S3 тајни кључ", + "s3-settings-save-failed": "Није могла бити сачувана S3 поставка", + "s3-settings-saved": "S3 поставка је сачувана", + "s3-ssl-enabled": "S3 SSL шифровање је укључено", + "s3-ssl-enabled-description": "Користи SSL/TLS за везу до Amazon S3 складишта", + "save-s3-settings": "Сачувај ову S3 поставку", + "schedule-board-archive": "Закажи архивирање списа", + "schedule-board-backup": "Закажи израду резервног примерка", + "schedule-board-cleanup": "Закажи чишћење", + "scheduled-board-operations": "Заказане радње на списима", + "start-all-migrations": "Пуна обнова", + "stop-all-migrations": "Прекини поступке", + "test-s3-connection": "Проба везе до Amazon S3", + "writable-path": "Проходна путања", + "writable-path-description": "Почетак путање према складишту предметне грађе", + "add-job": "Додај посао", + "attachment-migration": "Пресељење предметне грађе", + "attachment-monitoring": "Надзор над предметном грађом", + "attachment-settings": "Рад са предметном грађом", + "attachment-storage-settings": "Складиште предметне грађе", + "automatic-migration": "Обнављај иза у тишини", + "back-to-settings": "Натраг на поставку", + "board-id": "BoardID", + "board-migration": "Обнова оштећених списа", + "board-migrations": "Поступци обнове оштећених списа", + "card-show-lists-on-minicard": "Прикажи део поступка на омоту", + "comprehensive-board-migration": "Свеобухватна обнова", + "comprehensive-board-migration-description": "Изводе се свеобухватне провера и врше поправке целовитости података у списима - што укључује провере и поправке редоследа делова поступака, места где се тачно предмети налазе и структуре поступака.", + "delete-duplicate-empty-lists-migration": "Брисање истоимених делова поступка", + "delete-duplicate-empty-lists-migration-description": "На један безбедан начин брише празне истоимене делове поступка. Бришу се сви делови поступка у којима нема предмета ако постоји део поступка са истим тим насловом где већ постоје предмети.", + "lost-cards": "Загубљени предмети", + "lost-cards-list": "Опорављени предмети", + "restore-lost-cards-migration": "Опоравак загубљених предмета", + "restore-lost-cards-migration-description": "Открива и опоравља предмете и делове поступака којима недостају поља swimlaneId или listId и смешта у „Опорављене предмете“.", + "restore-all-archived-migration": "Опоравак свих предмета из архиве", + "restore-all-archived-migration-description": "Опоравља све архивиране делове поступака, поступке и предмете притом поправљајући недостајућа поља swimlaneId или listId.", + "fix-missing-lists-migration": "Поправка кад недостају делови поступка", + "fix-missing-lists-migration-description": "Открива и поправља сваки део поступка који недостаје или који је оштећен.", + "fix-avatar-urls-migration": "Поправка кад се не виде слике сарадника", + "fix-avatar-urls-migration-description": "Исправља везу до складишта са сликама сарадника из ових списа.", + "fix-all-file-urls-migration": "Поправка кад ишчезне предметна грађа", + "fix-all-file-urls-migration-description": "Преусмерава везу ка складишту где се стварно налази предметна грађа из ових списа.", + "migration-needed": "Потребна је обнова", + "migration-complete": "Обнова је обављена", + "migration-running": "Обнова је у току...", + "migration-successful": "Обнова је успешно окончана", + "migration-failed": "Обнова није била успешна", + "migrations": "Радионица", + "migrations-admin-only": "За обнову ових списа надлежан је једино онај ко је њихов управник", + "migrations-description": "Овде може да се изврши провера целовитости података и поправка оштећења. Сваки поступак обнове се може спровести независно један од другог", + "no-issues-found": "Нису уочена оштећења", + "run-migration": "Покрени опоравак", + "run-comprehensive-migration-confirm": "Овим ће бити изведене свеобухватне провере и поправке целовитости података у списима. То би кратко трајало уколико сте сагласни?", + "run-delete-duplicate-empty-lists-migration-confirm": "Овим ће прво сви дељени делови поступака бити везани на саме поступке, затим ће бити избрисан сваки део поступка који је празан где постоји део поступка истог имена који има предмете. Да ли сте сагласни?", + "run-restore-lost-cards-migration-confirm": "Овим се прави ток „Загубљени предмети“ и у њега смештају сви предмети и делови поступка којима недостају поља swimlaneId или listId. Мисли се на оно што је загубљено али није загубљено у архиви. Да ли сте сагласни? ", + "run-restore-all-archived-migration-confirm": "Овим се износи натраг из архиве све оно што сте до сада спаковали тамо. Поља где недостаје id биће поправљена. Упозорење! Да ли заиста то желите?", + "run-fix-missing-lists-migration-confirm": "Овим се откривају и поправљају они делови поступака који недостају или који су оштећени. Да ли сте сагласни?", + "run-fix-avatar-urls-migration-confirm": "Овим се исправља веза до складишта са сликама сарадника из ових списа. Да ли сте сагласни?", + "run-fix-all-file-urls-migration-confirm": "Овим ће бити исправљене све везе у списима да упућују на стварно складиште предметне грађе. Да ли сте сагласни?", + "restore-lost-cards-nothing-to-restore": "Нема нити загубљених поступака нити предмета за опоравак", "migration-progress-title": "Board Migration in Progress", - "migration-progress-overall": "Overall Progress", + "migration-progress-overall": "Укупни напредак", "migration-progress-current-step": "Current Step", "migration-progress-status": "Стање", - "migration-progress-details": "Детаљи", - "migration-progress-note": "Please wait while we migrate your board to the latest structure...", + "migration-progress-details": "Појединости", + "migration-progress-note": "Молимо да будете стрпљиви док траје препакивање Ваших списа...", - "step-analyze-board-structure": "Analyze Board Structure", - "step-fix-orphaned-cards": "Fix Orphaned Cards", - "step-convert-shared-lists": "Convert Shared Lists", - "step-ensure-per-swimlane-lists": "Ensure Per-Swimlane Lists", - "step-validate-migration": "Validate Migration", - "step-fix-avatar-urls": "Fix Avatar URLs", - "step-fix-attachment-urls": "Fix Attachment URLs", - "step-analyze-lists": "Analyze Lists", - "step-create-missing-lists": "Create Missing Lists", - "step-update-cards": "Update Cards", - "step-finalize": "Finalize", - "step-delete-duplicate-empty-lists": "Delete Duplicate Empty Lists", - "step-ensure-lost-cards-swimlane": "Ensure Lost Cards Swimlane", - "step-restore-lists": "Restore Lists", - "step-restore-cards": "Restore Cards", - "step-restore-swimlanes": "Restore Swimlanes", - "step-fix-missing-ids": "Fix Missing IDs", - "step-scan-users": "Checking board member avatars", - "step-scan-files": "Checking board file attachments", - "step-fix-file-urls": "Fixing file URLs", - "cleanup": "Cleanup", - "cleanup-old-jobs": "Cleanup Old Jobs", - "completed": "Обављен", - "conversion-info-text": "This conversion is performed once per board and improves performance. You can continue using the board normally.", - "converting-board": "Converting Board", - "converting-board-description": "Converting board structure for improved functionality. This may take a few moments.", - "cpu-cores": "CPU Cores", - "cpu-usage": "CPU Usage", - "current-action": "Current Action", - "database-migration": "Database Migration", - "database-migration-description": "Updating database structure for improved functionality and performance. This process may take several minutes.", - "database-migrations": "Database Migrations", - "days-old": "Days Old", - "duration": "Duration", - "errors": "Errors", - "estimated-time-remaining": "Estimated time remaining", - "every-1-day": "Every 1 day", - "every-1-hour": "Every 1 hour", - "every-1-minute": "Every 1 minute", - "every-10-minutes": "Every 10 minutes", - "every-30-minutes": "Every 30 minutes", - "every-5-minutes": "Every 5 minutes", - "every-6-hours": "Every 6 hours", - "export-monitoring": "Export Monitoring", - "filesystem-attachments": "Filesystem Attachments", - "filesystem-size": "Filesystem Size", - "filesystem-storage": "Filesystem Storage", - "force-board-scan": "Force Board Scan", - "gridfs-attachments": "GridFS Attachments", - "gridfs-size": "GridFS Size", + "step-analyze-board-structure": "Изучавам везе у списима", + "step-fix-orphaned-cards": "Поправљам одбачене предмете", + "step-convert-shared-lists": "Претварам дељене делове поступка", + "step-ensure-per-swimlane-lists": "Осигуравам да је део поступка везан за поступак", + "step-validate-migration": "Оцењујем обнову", + "step-fix-avatar-urls": "Поправљам везе до складишта слика за сараднике", + "step-fix-attachment-urls": "Поправљам везе до складишта предметне грађе", + "step-analyze-lists": "Изучавам делове поступка", + "step-create-missing-lists": "Стварам недостајуће делове", + "step-update-cards": "Допуњујем предмете", + "step-finalize": "Завршавам", + "step-delete-duplicate-empty-lists": "Бришем исте празне делове поступка", + "step-ensure-lost-cards-swimlane": "Правим место за загубљене предмете", + "step-restore-lists": "Опорављам делове поступка", + "step-restore-cards": "Опорављам предмете", + "step-restore-swimlanes": "Опорављам читав ток поступка", + "step-fix-missing-ids": "Поправљам недостајућа id поља", + "step-scan-users": "Проверавам слике сарадника на списима", + "step-scan-files": "Проверавам предметну грађу", + "step-fix-file-urls": "Поправљам везе до предметне грађе", + "cleanup": "Чистим", + "cleanup-old-jobs": "Чистим остављено", + "completed": "Прошао сам кроз све кораке", + "conversion-info-text": "Ово преслагање се изводи само једном по спису и то после олакшава читање. Без сметњи настављате да читате спис након тога.", + "converting-board": "Преслагање списа", + "converting-board-description": "Преслагање списа олакшава употребу. Ово може потрајати неколико тренутака.", + "cpu-cores": "Процесорска језгра", + "cpu-usage": "Заузеће процесора", + "current-action": "Текућа радња", + "database-migration": "Обнова базе података", + "database-migration-description": "Преслажем структуру базе података да би била боља и бржа. Процес може потрајати неколико минута.", + "database-migrations": "Преслагање базе података", + "days-old": "дана стар", + "duration": "Трајање", + "errors": "Грешке", + "estimated-time-remaining": "Процењено преостало време", + "every-1-day": "Једном дневно", + "every-1-hour": "На сваки сат", + "every-1-minute": "На сваки минут", + "every-10-minutes": "На сваких десет минута", + "every-30-minutes": "На сваких пола сата", + "every-5-minutes": "На пет минута", + "every-6-hours": "На шест сати", + "export-monitoring": "Надзор током изношења", + "filesystem-attachments": "Локална предметна грађа", + "filesystem-size": "Величина локалног складишта", + "filesystem-storage": "Локално складиште", + "force-board-scan": "Читање списа на силу", + "gridfs-attachments": "GridFS раздељена предметна грађа", + "gridfs-size": "GridFS величина", "gridfs-storage": "GridFS", - "hide-list-on-minicard": "Hide List on Minicard", - "idle-migration": "Idle Migration", - "job-description": "Job Description", - "job-details": "Job Details", - "job-name": "Job Name", - "job-queue": "Job Queue", - "last-run": "Last Run", - "max-concurrent": "Max Concurrent", - "memory-usage": "Memory Usage", - "migrate-all-to-filesystem": "Migrate All to Filesystem", - "migrate-all-to-gridfs": "Migrate All to GridFS", - "migrate-all-to-s3": "Migrate All to S3", - "migrated-attachments": "Migrated Attachments", - "migration-batch-size": "Batch Size", - "migration-batch-size-description": "Number of attachments to process in each batch (1-100)", - "migration-cpu-threshold": "CPU Threshold (%)", - "migration-cpu-threshold-description": "Pause migration when CPU usage exceeds this percentage (10-90)", - "migration-delay-ms": "Delay (ms)", - "migration-delay-ms-description": "Delay between batches in milliseconds (100-10000)", - "migration-detector": "Migration Detector", - "migration-info-text": "Database migrations are performed once and improve system performance. The process continues in the background even if you close your browser.", - "migration-log": "Migration Log", - "migration-markers": "Migration Markers", - "migration-resume-failed": "Failed to resume migration", - "migration-resumed": "Migration resumed", - "migration-steps": "Migration Steps", - "migration-warning-text": "Please do not close your browser during migration. The process will continue in the background but may take longer to complete.", - "monitoring-export-failed": "Failed to export monitoring data", - "monitoring-refresh-failed": "Failed to refresh monitoring data", - "next": "Next", - "next-run": "Next Run", + "hide-list-on-minicard": "Сакриј део поступка са омота", + "idle-migration": "Тиха обнова", + "job-description": "Опис посла", + "job-details": "Појединости посла", + "job-name": "Име посла", + "job-queue": "Посао по реду", + "last-run": "Последње покретање", + "max-concurrent": "Паралелно", + "memory-usage": "Искоришћеност меморије", + "migrate-all-to-filesystem": "Пресели све у локално складиште", + "migrate-all-to-gridfs": "Пресели све и издели у GridFS", + "migrate-all-to-s3": "Пресели све у Amazon S3 облак", + "migrated-attachments": "Пресељена предметна грађа", + "migration-batch-size": "Ширина захвата", + "migration-batch-size-description": "Број јединица предметне грађе који ће бити обрађен при сваком пролазу (1-100)", + "migration-cpu-threshold": "Граница искоришћења процесора (%)", + "migration-cpu-threshold-description": "Направи предах када заузеће процесора досегне ове проценте (10-90)", + "migration-delay-ms": "Задршка (ms)", + "migration-delay-ms-description": "Задршка између пролаза изражена у милисекундама (100-10000)", + "migration-detector": "Детектор", + "migration-info-text": "Преслагање базе података се изводи само једном и то поправља брзину. Ова радња се изводи у позадини чак иако одете.", + "migration-log": "Записник из обнове", + "migration-markers": "Обележја обнове", + "migration-resume-failed": "Није успео наставак обнове", + "migration-resumed": "Обнова је настављена", + "migration-steps": "Кораци у обнови", + "migration-warning-text": "Предлажемо да не затварате прозор током обнове. Чак и тада процес ће бити изведен у позадини али све може дуже да траје.", + "monitoring-export-failed": "Не могу да извезем надзорне податаке", + "monitoring-refresh-failed": "Не могу да освежим надзорне податке", + "next": "Следеће", + "next-run": "Наредни покрет", "of": "од", - "operation-type": "Operation Type", - "overall-progress": "Overall Progress", - "page": "Page", - "pause-migration": "Pause Migration", - "previous": "Previous", - "refresh": "Refresh", - "refresh-monitoring": "Refresh Monitoring", - "remaining-attachments": "Remaining Attachments", - "resume-migration": "Resume Migration", - "run-once": "Run once", - "s3-attachments": "S3 Attachments", - "s3-size": "S3 Size", + "operation-type": "Врста радње", + "overall-progress": "Измерени напредак", + "page": "Страна", + "pause-migration": "Направи предах", + "previous": "Претходна", + "refresh": "Освежи", + "refresh-monitoring": "Најновији подаци са надзора", + "remaining-attachments": "Преостала предметна грађа", + "resume-migration": "Настави обнову", + "run-once": "Покрени једном", + "s3-attachments": "Предметна грађа у Amazon S3 облаку", + "s3-size": "Amazon S3 заузеће", "s3-storage": "S3", - "scanning-status": "Scanning Status", - "schedule": "Schedule", - "search-boards-or-operations": "Search boards or operations...", - "show-list-on-minicard": "Show List on Minicard", - "showing": "Showing", + "scanning-status": "Изучено стање", + "schedule": "Распоред", + "search-boards-or-operations": "Претрага списа или радњи...", + "show-list-on-minicard": "Прикажи део поступка на омоту", + "showing": "Приказујем", "start-test-operation": "Start Test Operation", - "start-time": "Start Time", - "step-progress": "Step Progress", - "stop-migration": "Stop Migration", - "storage-distribution": "Storage Distribution", - "system-resources": "System Resources", - "total-attachments": "Total Attachments", - "total-operations": "Total Operations", - "total-size": "Total Size", - "unmigrated-boards": "Unmigrated Boards", - "weight": "Weight", - "idle": "Idle", - "complete": "Complete", - "cron": "Cron" + "start-time": "Покрени штоперицу", + "step-progress": "Појединачни напредак", + "stop-migration": "Заустави обнову", + "storage-distribution": "Расподела у складишту", + "system-resources": "Системска снага", + "total-attachments": "Број јединица предметне грађе", + "total-operations": "Укупан број радњи", + "total-size": "Укупна величина", + "unmigrated-boards": "Необновљени списи", + "weight": "Оптерећење", + "idle": "Стање мировања", + "complete": "Посао је успешно обављен", + "cron": "Периодични послови" } diff --git a/imports/i18n/data/sv.i18n.json b/imports/i18n/data/sv.i18n.json index 8b6fd6061..beb2d5047 100644 --- a/imports/i18n/data/sv.i18n.json +++ b/imports/i18n/data/sv.i18n.json @@ -78,17 +78,17 @@ "activity-deleteComment": "raderade kommentar %s", "activity-receivedDate": "redigerade mottaget datum till %s av %s", "activity-startDate": "redigerade startdatum till %s av %s", - "allboards.starred": "Starred", + "allboards.starred": " Stjärnmärkt", "allboards.templates": "Mallar", - "allboards.remaining": "Remaining", - "allboards.workspaces": "Workspaces", - "allboards.add-workspace": "Add Workspace", - "allboards.add-workspace-prompt": "Workspace name", - "allboards.add-subworkspace": "Add Subworkspace", - "allboards.add-subworkspace-prompt": "Subworkspace name", - "allboards.edit-workspace": "Edit workspace", - "allboards.edit-workspace-name": "Workspace name", - "allboards.edit-workspace-icon": "Workspace icon (markdown)", + "allboards.remaining": "Återstående", + "allboards.workspaces": "Arbetsytor", + "allboards.add-workspace": "Lägg till arbetsyta", + "allboards.add-workspace-prompt": "Arbetsytans namn", + "allboards.add-subworkspace": "Lägg till underarbetsyta", + "allboards.add-subworkspace-prompt": "Underarbetsytans namn", + "allboards.edit-workspace": "Redigera arbetsyta", + "allboards.edit-workspace-name": "Arbetsytans namn", + "allboards.edit-workspace-icon": "Arbetsytans ikon (markdown)", "multi-selection-active": "Click checkboxes to select boards", "activity-dueDate": "redigerade förfallodag till %s av %s", "activity-endDate": "redigerade slutdatum till %s av %s", diff --git a/imports/i18n/data/zh-TW.i18n.json b/imports/i18n/data/zh-TW.i18n.json index 021053b09..302092c7c 100644 --- a/imports/i18n/data/zh-TW.i18n.json +++ b/imports/i18n/data/zh-TW.i18n.json @@ -78,18 +78,18 @@ "activity-deleteComment": "評論已刪除", "activity-receivedDate": "已編輯收到日期為 %s %s", "activity-startDate": "已編輯起始日期為 %s %s", - "allboards.starred": "Starred", + "allboards.starred": "已加星號", "allboards.templates": "範本", - "allboards.remaining": "Remaining", - "allboards.workspaces": "Workspaces", - "allboards.add-workspace": "Add Workspace", - "allboards.add-workspace-prompt": "Workspace name", - "allboards.add-subworkspace": "Add Subworkspace", - "allboards.add-subworkspace-prompt": "Subworkspace name", - "allboards.edit-workspace": "Edit workspace", - "allboards.edit-workspace-name": "Workspace name", - "allboards.edit-workspace-icon": "Workspace icon (markdown)", - "multi-selection-active": "Click checkboxes to select boards", + "allboards.remaining": "剩餘", + "allboards.workspaces": "工作空間", + "allboards.add-workspace": "新增工作空間", + "allboards.add-workspace-prompt": "工作空間名稱", + "allboards.add-subworkspace": "新增子工作空間", + "allboards.add-subworkspace-prompt": "子工作空間名稱", + "allboards.edit-workspace": "編輯工作空間", + "allboards.edit-workspace-name": "工作空間名稱", + "allboards.edit-workspace-icon": "工作空間圖示 (markdown)", + "multi-selection-active": "點選核取方塊以選取看板", "activity-dueDate": "已編輯截止日期為 %s %s", "activity-endDate": "已編輯結束日期為 %s %s", "add-attachment": "新增附件", @@ -466,8 +466,8 @@ "filter-no-label": "沒有標籤", "filter-member-label": "按成員篩選", "filter-no-member": "沒有成員", - "filter-assignee-label": "按代理人篩選", - "filter-no-assignee": "沒有代理人", + "filter-assignee-label": "按承辦人篩選", + "filter-no-assignee": "沒有承辦人", "filter-custom-fields-label": "按自訂欄位篩選", "filter-no-custom-fields": "沒有自訂欄位", "filter-show-archive": "顯示封存的清單", @@ -639,7 +639,7 @@ "has-spenttime-cards": "耗時卡", "time": "時間", "title": "標題", - "toggle-assignees": "切換卡片的代理人 1-9(按加入看板的順序)。", + "toggle-assignees": "切換卡片的承辦人 1-9(按加入看板的順序)。", "toggle-labels": "切換卡片的標籤 1-9。多重選擇新增標籤 1-9", "remove-labels-multiselect": "多重選擇移除標籤 1-9", "tracking": "訂閱相關通知", @@ -963,8 +963,8 @@ "accounts-allowUserDelete": "允許用戶自行刪除其帳戶", "hide-minicard-label-text": "隱藏迷你卡片標籤內文", "show-desktop-drag-handles": "顯示桌面拖曳工具", - "assignee": "代理人", - "cardAssigneesPopup-title": "代理人", + "assignee": "承辦人", + "cardAssigneesPopup-title": "承辦人", "addmore-detail": "新增更多詳細描述", "show-on-card": "在卡片上顯示", "show-on-minicard": "在小卡片顯示", @@ -1060,7 +1060,7 @@ "operator-user-abbrev": "@", "operator-member": "成員", "operator-member-abbrev": "m", - "operator-assignee": "代理人", + "operator-assignee": "承辦人", "operator-assignee-abbrev": "a", "operator-creator": "建立者", "operator-status": "狀態", @@ -1091,7 +1091,7 @@ "predicate-checklist": "待辦清單", "predicate-start": "開始", "predicate-end": "完成", - "predicate-assignee": "代理人", + "predicate-assignee": "承辦人", "predicate-member": "成員", "predicate-public": "公開", "predicate-private": "私有", @@ -1116,10 +1116,10 @@ "globalSearch-instructions-operator-comment": "`__operator_comment__:` - 卡片評論包含 **.", "globalSearch-instructions-operator-label": "`__operator_label__:` `__operator_label__:` - 卡片標籤要符合 ** 或 *", "globalSearch-instructions-operator-hash": "`__operator_label_abbrev__` - 的簡寫 `__operator_label__:` 或 `__operator_label__:`", - "globalSearch-instructions-operator-user": "`__operator_user__:` - 卡片,其中 ** 是 *成員* 或 *代理人*", + "globalSearch-instructions-operator-user": "`__operator_user__:` - 卡片,其中 ** 是 *成員* 或 *承辦人*", "globalSearch-instructions-operator-at": "`__operator_user_abbrev__username` - `user:` 的簡寫", "globalSearch-instructions-operator-member": "`__operator_member__:` - 卡片,其中 ** 是i *成員*", - "globalSearch-instructions-operator-assignee": "`__operator_assignee__:` - 卡片,其中 ** 是 *代理人*", + "globalSearch-instructions-operator-assignee": "`__operator_assignee__:` - 卡片,其中 ** 是 *承辦人*", "globalSearch-instructions-operator-creator": "`__operator_creator__:` - 卡片,其中 ** 是卡片的建立者", "globalSearch-instructions-operator-org": "`__operator_org__:` - 屬於分配給組織 ** 看板的卡片", "globalSearch-instructions-operator-team": "`__operator_team__:` - 屬於分配給團隊 ** 看板的卡片", @@ -1417,70 +1417,70 @@ "back-to-settings": "回到設定", "board-id": "看板 ID", "board-migration": "看板遷移", - "board-migrations": "Board Migrations", + "board-migrations": "看板遷移", "card-show-lists-on-minicard": "在迷你卡片上顯示清單", - "comprehensive-board-migration": "Comprehensive Board Migration", - "comprehensive-board-migration-description": "Performs comprehensive checks and fixes for board data integrity, including list ordering, card positions, and swimlane structure.", - "delete-duplicate-empty-lists-migration": "Delete Duplicate Empty Lists", - "delete-duplicate-empty-lists-migration-description": "Safely deletes empty duplicate lists. Only removes lists that have no cards AND have another list with the same title that contains cards.", - "lost-cards": "Lost Cards", - "lost-cards-list": "Restored Items", - "restore-lost-cards-migration": "Restore Lost Cards", - "restore-lost-cards-migration-description": "Finds and restores cards and lists with missing swimlaneId or listId. Creates a 'Lost Cards' swimlane to make all lost items visible again.", - "restore-all-archived-migration": "Restore All Archived", - "restore-all-archived-migration-description": "Restores all archived swimlanes, lists, and cards. Automatically fixes any missing swimlaneId or listId to make items visible.", - "fix-missing-lists-migration": "Fix Missing Lists", - "fix-missing-lists-migration-description": "Detects and repairs missing or corrupted lists in the board structure.", - "fix-avatar-urls-migration": "Fix Avatar URLs", - "fix-avatar-urls-migration-description": "Updates avatar URLs for board members to use the correct storage backend and fixes broken avatar references.", - "fix-all-file-urls-migration": "Fix All File URLs", - "fix-all-file-urls-migration-description": "Updates all file attachment URLs on this board to use the correct storage backend and fixes broken file references.", - "migration-needed": "Migration Needed", + "comprehensive-board-migration": "全面看板遷移", + "comprehensive-board-migration-description": "執行全面檢查與修復,確保看板資料完整性,包括清單排序、卡片位置及泳道結構。", + "delete-duplicate-empty-lists-migration": "刪除重複的空清單", + "delete-duplicate-empty-lists-migration-description": "安全地刪除空的重複清單。僅移除既無卡片、又存在標題相同且含卡片的另一份清單的清單。", + "lost-cards": "遺失的卡片", + "lost-cards-list": "已還原的項目", + "restore-lost-cards-migration": "還原遺失的卡片", + "restore-lost-cards-migration-description": "尋找並還原缺少泳道 ID 或清單 ID 的卡片與清單。建立「遺失的卡片」泳道,使所有遺失項目重新可見。", + "restore-all-archived-migration": "還原所有封存", + "restore-all-archived-migration-description": "還原所有已封存的泳道、清單與卡片。自動修復任何缺少泳道 ID 或清單 ID 的項目以使它們重新可見。", + "fix-missing-lists-migration": "修復遺失的清單", + "fix-missing-lists-migration-description": "偵測並修復在看板結構中遺失或損毀的清單。", + "fix-avatar-urls-migration": "修復大頭照 URL", + "fix-avatar-urls-migration-description": "更新看板成員的大頭照 URL 以使用正確的儲存空間後端並修復損壞的大頭照參照。", + "fix-all-file-urls-migration": "修復所有檔案 URL", + "fix-all-file-urls-migration-description": "更新所有此看板的檔案附件 URL 以使用正確的儲存空間後端並修復損壞的檔案參照。", + "migration-needed": "需要遷移", "migration-complete": "完成", - "migration-running": "Running...", - "migration-successful": "Migration completed successfully", - "migration-failed": "Migration failed", - "migrations": "Migrations", - "migrations-admin-only": "Only board administrators can run migrations", - "migrations-description": "Run data integrity checks and repairs for this board. Each migration can be executed individually.", - "no-issues-found": "No issues found", - "run-migration": "Run Migration", - "run-comprehensive-migration-confirm": "This will perform a comprehensive migration to check and fix board data integrity. This may take a few moments. Continue?", - "run-delete-duplicate-empty-lists-migration-confirm": "This will first convert any shared lists to per-swimlane lists, then delete empty lists that have a duplicate list with the same title containing cards. Only truly redundant empty lists will be removed. Continue?", - "run-restore-lost-cards-migration-confirm": "This will create a 'Lost Cards' swimlane and restore all cards and lists with missing swimlaneId or listId. This only affects non-archived items. Continue?", - "run-restore-all-archived-migration-confirm": "This will restore ALL archived swimlanes, lists, and cards, making them visible again. Any items with missing IDs will be automatically fixed. This cannot be easily undone. Continue?", - "run-fix-missing-lists-migration-confirm": "This will detect and repair missing or corrupted lists in the board structure. Continue?", - "run-fix-avatar-urls-migration-confirm": "This will update avatar URLs for board members to use the correct storage backend. Continue?", - "run-fix-all-file-urls-migration-confirm": "This will update all file attachment URLs on this board to use the correct storage backend. Continue?", - "restore-lost-cards-nothing-to-restore": "No lost swimlanes, lists, or cards to restore", + "migration-running": "正在執行……", + "migration-successful": "遷移成功完成", + "migration-failed": "遷移失敗", + "migrations": "遷移", + "migrations-admin-only": "僅看板管理員可執行遷移", + "migrations-description": "為此看板執行資料完整性檢查並修復。每個遷移皆可單獨執行。", + "no-issues-found": "未找到問題", + "run-migration": "執行遷移", + "run-comprehensive-migration-confirm": "這將會執行全面的遷移以檢查並修復看板資料完整性。這可能需要數分鐘。要繼續嗎?", + "run-delete-duplicate-empty-lists-migration-confirm": "此操作將先將所有共享清單轉換為每個泳道專屬的清單,接著刪除那些存在標題相同且含卡片之重複清單的空清單。僅會移除真正冗餘的空清單。要繼續嗎?", + "run-restore-lost-cards-migration-confirm": "這將建立一個「遺失的卡片」泳道,並還原所有缺少泳道 ID 或清單 ID 的卡片與清單。此操作僅影響未封存項目。繼續?", + "run-restore-all-archived-migration-confirm": "此操作將還原所有已封存的泳道、清單及卡片,使其重新顯示。任何缺少 ID 的項目將自動修復。此操作無法輕易撤銷。要繼續嗎?", + "run-fix-missing-lists-migration-confirm": "這將會偵測並修復在看板結構中遺失或損毀的清單。要繼續嗎?", + "run-fix-avatar-urls-migration-confirm": "這將會更新看板成員的大頭照 URL 以使用正確的儲存空間後端。要繼續嗎?", + "run-fix-all-file-urls-migration-confirm": "這將會更新此看板上的所有檔案附件 URL 以使用正確的儲存空間後端。要繼續嗎?", + "restore-lost-cards-nothing-to-restore": "沒有需要還原的遺失泳道、清單或卡片", - "migration-progress-title": "Board Migration in Progress", + "migration-progress-title": "正在進行看板遷移", "migration-progress-overall": "整體進度", - "migration-progress-current-step": "Current Step", + "migration-progress-current-step": "目前步驟", "migration-progress-status": "狀態", "migration-progress-details": "內容", - "migration-progress-note": "Please wait while we migrate your board to the latest structure...", + "migration-progress-note": "請稍候,我們正在將您的看板遷移至最新結構……", - "step-analyze-board-structure": "Analyze Board Structure", - "step-fix-orphaned-cards": "Fix Orphaned Cards", - "step-convert-shared-lists": "Convert Shared Lists", - "step-ensure-per-swimlane-lists": "Ensure Per-Swimlane Lists", - "step-validate-migration": "Validate Migration", - "step-fix-avatar-urls": "Fix Avatar URLs", - "step-fix-attachment-urls": "Fix Attachment URLs", - "step-analyze-lists": "Analyze Lists", - "step-create-missing-lists": "Create Missing Lists", - "step-update-cards": "Update Cards", - "step-finalize": "Finalize", - "step-delete-duplicate-empty-lists": "Delete Duplicate Empty Lists", - "step-ensure-lost-cards-swimlane": "Ensure Lost Cards Swimlane", - "step-restore-lists": "Restore Lists", - "step-restore-cards": "Restore Cards", - "step-restore-swimlanes": "Restore Swimlanes", - "step-fix-missing-ids": "Fix Missing IDs", - "step-scan-users": "Checking board member avatars", - "step-scan-files": "Checking board file attachments", - "step-fix-file-urls": "Fixing file URLs", + "step-analyze-board-structure": "分析看板結構", + "step-fix-orphaned-cards": "修復孤立卡片", + "step-convert-shared-lists": "轉換共享清單", + "step-ensure-per-swimlane-lists": "確保每個泳道專屬的清單", + "step-validate-migration": "驗證遷移", + "step-fix-avatar-urls": "修復大頭照 URL", + "step-fix-attachment-urls": "修復附件 URL", + "step-analyze-lists": "分析清單", + "step-create-missing-lists": "建立遺失的清單", + "step-update-cards": "更新卡片", + "step-finalize": "完成", + "step-delete-duplicate-empty-lists": "刪除重複的空清單", + "step-ensure-lost-cards-swimlane": "確保遺失的卡片泳道", + "step-restore-lists": "還原清單", + "step-restore-cards": "還原卡片", + "step-restore-swimlanes": "還原泳道", + "step-fix-missing-ids": "修復遺失的 ID", + "step-scan-users": "正在檢查看板成員大頭照", + "step-scan-files": "正在檢查看板檔案附件", + "step-fix-file-urls": "正在修復檔案 URL", "cleanup": "清理", "cleanup-old-jobs": "清理舊工作", "completed": "已完成", From f16780b5e3d112e9d388f054a6bfd3170466d5d3 Mon Sep 17 00:00:00 2001 From: Lauri Ojansivu Date: Wed, 19 Nov 2025 09:34:57 +0200 Subject: [PATCH 67/84] Updated translations. --- imports/i18n/data/sv.i18n.json | 14 +++++++------- 1 file changed, 7 insertions(+), 7 deletions(-) diff --git a/imports/i18n/data/sv.i18n.json b/imports/i18n/data/sv.i18n.json index beb2d5047..de8ea3332 100644 --- a/imports/i18n/data/sv.i18n.json +++ b/imports/i18n/data/sv.i18n.json @@ -89,7 +89,7 @@ "allboards.edit-workspace": "Redigera arbetsyta", "allboards.edit-workspace-name": "Arbetsytans namn", "allboards.edit-workspace-icon": "Arbetsytans ikon (markdown)", - "multi-selection-active": "Click checkboxes to select boards", + "multi-selection-active": "Klicka i kryssrutor för att välja tavlor", "activity-dueDate": "redigerade förfallodag till %s av %s", "activity-endDate": "redigerade slutdatum till %s av %s", "add-attachment": "Lägg till bilaga", @@ -1411,12 +1411,12 @@ "add-job": "Lägg till jobb", "attachment-migration": "Migrering av bilagor", "attachment-monitoring": "Övervakning av bilagor", - "attachment-settings": "Attachment Settings", - "attachment-storage-settings": "Storage Settings", - "automatic-migration": "Automatic Migration", - "back-to-settings": "Back to Settings", - "board-id": "Board ID", - "board-migration": "Board Migration", + "attachment-settings": "Bilageinställningar", + "attachment-storage-settings": " Lagringsinställningar", + "automatic-migration": "Automatisk migrering", + "back-to-settings": "Tillbaka till inställningar", + "board-id": "Tavlans ID", + "board-migration": "Tavelmigration", "board-migrations": "Board Migrations", "card-show-lists-on-minicard": "Show Lists on Minicard", "comprehensive-board-migration": "Comprehensive Board Migration", From 3db1305e58168f7417023ccd8d54995026844b18 Mon Sep 17 00:00:00 2001 From: Lauri Ojansivu Date: Fri, 21 Nov 2025 02:44:50 +0200 Subject: [PATCH 68/84] Updated build script for Linux arm64 bundle. Thanks to xet7 ! --- releases/build-bundle-arm64.sh | 35 +++++++++++++++++++++++++--------- 1 file changed, 26 insertions(+), 9 deletions(-) diff --git a/releases/build-bundle-arm64.sh b/releases/build-bundle-arm64.sh index 53687545a..c9c4a25ce 100755 --- a/releases/build-bundle-arm64.sh +++ b/releases/build-bundle-arm64.sh @@ -10,21 +10,38 @@ if [ $# -ne 1 ] exit 1 fi -sudo apt -y install g++ build-essential p7zip-full +sudo apt -y install g++ build-essential p7zip-full npm +sudo npm -g install n +# Building bundle works with Node.js 14.21.3 . +# Running works also with 14.21.4, many architectures at https://github.com/wekan/node-v14-esm/releases/tag/v14.21.4 +sudo n 14.21.3 sudo npm -g uninstall node-pre-gyp sudo npm -g install @mapbox/node-pre-gyp rm -rf bundle rm wekan-$1-arm64.zip -#rm wekan-$1.zip -#wget https://releases.wekan.team/wekan-$1.zip -7z x wekan-$1-amd64.zip +rm wekan-$1.zip +wget https://github.com/wekan/wekan/releases/download/v$1/wekan-$1-amd64.zip +7z x wekan-$1-arm64.zip -(cd bundle/programs/server && chmod u+w *.json && cd node_modules/fibers && node build.js) +# Get working fibers and bcrypt from previous WeKan v7.93 https://github.com/wekan/wekan/releases/tag/v7.93 +wget https://github.com/wekan/wekan/releases/download/v7.93/wekan-7.93-arm64.zip +mkdir 7.93 +cd 7.93 +7z x ../wekan-7.93-arm64.zip +cd .. + +#wget https://releases.wekan.team/wekan-$1.zip +#7z x wekan-$1-amd64.zip + +#(cd bundle/programs/server && chmod u+w *.json && cd node_modules/fibers && node build.js) +(cd bundle/programs/server && chmod u+w *.json) +# && cd node_modules/fibers && node build.js) #cd ../../../.. -#(cd bundle/programs/server/npm/node_modules/meteor/accounts-password && npm remove bcrypt && npm install bcrypt) +(cd bundle/programs/server/npm/node_modules/meteor/accounts-password/node_modules && rm -rf bcrypt) +(cp -pR 7.93/bundle/programs/server/npm/node_modules/meteor/accounts-password/node_modules/bcrypt bundle/programs/server/npm/node_modules/meteor/accounts-password/node_modules/) # Requires building from source https://github.com/meteor/meteor/issues/11682 -(cd bundle/programs/server/npm/node_modules/meteor/accounts-password && npm rebuild --build-from-source) +#(cd bundle/programs/server/npm/node_modules/meteor/accounts-password && npm rebuild --build-from-source) cd bundle find . -type d -name '*-garbage*' | xargs rm -rf @@ -35,6 +52,6 @@ cd .. 7z a wekan-$1-arm64.zip bundle -sudo snap start juju-db +#sudo snap start juju-db -./start-wekan.sh +#./start-wekan.sh From 960e2126b487edbda74576560f04b64baac79b08 Mon Sep 17 00:00:00 2001 From: Lauri Ojansivu Date: Fri, 21 Nov 2025 03:02:41 +0200 Subject: [PATCH 69/84] Updated ChangeLog. --- CHANGELOG.md | 18 ++++++++++++++++++ 1 file changed, 18 insertions(+) diff --git a/CHANGELOG.md b/CHANGELOG.md index 64f8ee592..8b27c474d 100644 --- a/CHANGELOG.md +++ b/CHANGELOG.md @@ -22,6 +22,24 @@ Fixing other platforms In Progress. WeKan 8.00-8.06 had wrong raw database directory setting /var/snap/wekan/common/wekan and some cards were not visible. Those are fixed at WeKan 8.07 where database directory is back to /var/snap/wekan/common and all cards are visible. +# Upcoming WeKan ® release + +This release adds the following updates: + +- [Update GitHub docker/metadata-action from 5.8.0 to 5.9.0](https://github.com/wekan/wekan/pull/6012). + Thanks to dependabot. +- [Updated security.md](https://github.com/wekan/wekan/commit/7ff1649d8909917cae590c68def6eecac0442f91). + Thanks to xet7. +- [Updated build script for Linux arm64 bundle](https://github.com/wekan/wekan/commit/3db1305e58168f7417023ccd8d54995026844b18). + Thanks to xet7. + +and fixes the following bugs: + +- [Fix Broken Strikethroughs in Markdown to HTML conversion](https://github.com/wekan/wekan/pull/6009). + Thanks to brlin-tw. + +Thanks to above GitHub users for their contributions and translators for their translations. + # v8.17 2025-11-06 WeKan ® release This release adds the following new feature: From 70975c29440aed43f7d5d08bd25c7797ba2a39ec Mon Sep 17 00:00:00 2001 From: "dependabot[bot]" <49699333+dependabot[bot]@users.noreply.github.com> Date: Mon, 24 Nov 2025 20:03:19 +0000 Subject: [PATCH 70/84] Bump actions/checkout from 5 to 6 Bumps [actions/checkout](https://github.com/actions/checkout) from 5 to 6. - [Release notes](https://github.com/actions/checkout/releases) - [Changelog](https://github.com/actions/checkout/blob/main/CHANGELOG.md) - [Commits](https://github.com/actions/checkout/compare/v5...v6) --- updated-dependencies: - dependency-name: actions/checkout dependency-version: '6' dependency-type: direct:production update-type: version-update:semver-major ... Signed-off-by: dependabot[bot] --- .github/workflows/depsreview.yaml | 2 +- .github/workflows/docker-publish.yml | 2 +- .github/workflows/dockerimage.yml | 2 +- .github/workflows/release.yml | 2 +- .github/workflows/test_suite.yml | 10 +++++----- 5 files changed, 9 insertions(+), 9 deletions(-) diff --git a/.github/workflows/depsreview.yaml b/.github/workflows/depsreview.yaml index 8461b453c..ae5ae5989 100644 --- a/.github/workflows/depsreview.yaml +++ b/.github/workflows/depsreview.yaml @@ -9,6 +9,6 @@ jobs: runs-on: ubuntu-latest steps: - name: 'Checkout Repository' - uses: actions/checkout@v5 + uses: actions/checkout@v6 - name: 'Dependency Review' uses: actions/dependency-review-action@v4 diff --git a/.github/workflows/docker-publish.yml b/.github/workflows/docker-publish.yml index eab9e0fbb..ecc6c5044 100644 --- a/.github/workflows/docker-publish.yml +++ b/.github/workflows/docker-publish.yml @@ -32,7 +32,7 @@ jobs: steps: - name: Checkout repository - uses: actions/checkout@v5 + uses: actions/checkout@v6 # Login against a Docker registry except on PR # https://github.com/docker/login-action diff --git a/.github/workflows/dockerimage.yml b/.github/workflows/dockerimage.yml index 14f8dfe01..0f85c0d96 100644 --- a/.github/workflows/dockerimage.yml +++ b/.github/workflows/dockerimage.yml @@ -15,6 +15,6 @@ jobs: runs-on: ubuntu-latest steps: - - uses: actions/checkout@v5 + - uses: actions/checkout@v6 - name: Build the Docker image run: docker build . --file Dockerfile --tag wekan:$(date +%s) diff --git a/.github/workflows/release.yml b/.github/workflows/release.yml index 9d93b7588..0c05c85e6 100644 --- a/.github/workflows/release.yml +++ b/.github/workflows/release.yml @@ -15,7 +15,7 @@ jobs: runs-on: ubuntu-latest steps: - name: Checkout - uses: actions/checkout@v5 + uses: actions/checkout@v6 with: fetch-depth: 0 diff --git a/.github/workflows/test_suite.yml b/.github/workflows/test_suite.yml index 7e2e4944a..6de010b61 100644 --- a/.github/workflows/test_suite.yml +++ b/.github/workflows/test_suite.yml @@ -18,7 +18,7 @@ jobs: # runs-on: ubuntu-latest # steps: # - name: checkout -# uses: actions/checkout@v5 +# uses: actions/checkout@v6 # # - name: setup node # uses: actions/setup-node@v1 @@ -42,7 +42,7 @@ jobs: # needs: [lintcode] # steps: # - name: checkout -# uses: actions/checkout@v5 +# uses: actions/checkout@v6 # # - name: setup node # uses: actions/setup-node@v1 @@ -65,7 +65,7 @@ jobs: # needs: [lintcode,lintstyle] # steps: # - name: checkout -# uses: actions/checkout@v5 +# uses: actions/checkout@v6 # # - name: setup node # uses: actions/setup-node@v1 @@ -90,7 +90,7 @@ jobs: # CHECKOUTS - name: Checkout - uses: actions/checkout@v5 + uses: actions/checkout@v6 # CACHING - name: Install Meteor @@ -147,7 +147,7 @@ jobs: needs: [tests] steps: - name: Checkout - uses: actions/checkout@v5 + uses: actions/checkout@v6 - name: Download coverage uses: actions/download-artifact@v6 From e30ce7805391165f80a08b48e051f85b5ecb2949 Mon Sep 17 00:00:00 2001 From: Mial Lewis Date: Wed, 26 Nov 2025 23:57:49 +0000 Subject: [PATCH 71/84] add archive card to api --- models/cards.js | 38 ++++++++++++++++++++++++++++++++++++++ 1 file changed, 38 insertions(+) diff --git a/models/cards.js b/models/cards.js index 546efdfe6..fb52ea122 100644 --- a/models/cards.js +++ b/models/cards.js @@ -4440,6 +4440,44 @@ JsonRoutes.add('GET', '/api/boards/:boardId/cards_count', function( }); }, ); + + /** + * @operation archive_card + * @summary Archive a card + * + * @description Archive a card + * @param {string} boardId the board ID of the card + * @param {string} listId the list ID of the card + * @param {string} cardId the ID of the card + * @return_type {_id: string, archived: bool, archivedAt: Date} + */ + JsonRoutes.add( + 'POST', + '/api/boards/:boardId/lists/:listId/cards/:cardId/archive', + function(req, res) { + const paramBoardId = req.params.boardId; + const paramCardId = req.params.cardId; + const paramListId = req.params.listId; + Authentication.checkBoardAccess(req.userId, paramBoardId); + const card = ReactiveCache.getCard({ + _id: paramCardId, + listId: paramListId, + boardId: paramBoardId, + archived: false, + }); + if (!card) { + throw new Meteor.Error(404, 'Card not found'); + } + const archive_res = card.archive(); + JsonRoutes.sendResult(res, { + code: 200, + data: { + _id: paramCardId, + ...archive_res.$set, + }, + }); + }, + ); } // Position history tracking methods From a81a6030311654752c953b0ec9b9951f0f1ce567 Mon Sep 17 00:00:00 2001 From: Mial Lewis Date: Wed, 26 Nov 2025 23:59:00 +0000 Subject: [PATCH 72/84] update bool to boolean --- models/cards.js | 2 +- 1 file changed, 1 insertion(+), 1 deletion(-) diff --git a/models/cards.js b/models/cards.js index fb52ea122..9c48546fc 100644 --- a/models/cards.js +++ b/models/cards.js @@ -4449,7 +4449,7 @@ JsonRoutes.add('GET', '/api/boards/:boardId/cards_count', function( * @param {string} boardId the board ID of the card * @param {string} listId the list ID of the card * @param {string} cardId the ID of the card - * @return_type {_id: string, archived: bool, archivedAt: Date} + * @return_type {_id: string, archived: boolean, archivedAt: Date} */ JsonRoutes.add( 'POST', From 67c8a98f20fd610a27325f855ce53485d97c0fc6 Mon Sep 17 00:00:00 2001 From: Mial Lewis Date: Thu, 27 Nov 2025 00:05:53 +0000 Subject: [PATCH 73/84] add route to wekan.yml --- public/api/wekan.yml | 47 ++++++++++++++++++++++++++++++++++++++++++++ 1 file changed, 47 insertions(+) diff --git a/public/api/wekan.yml b/public/api/wekan.yml index cbb1a23a6..5982969dc 100644 --- a/public/api/wekan.yml +++ b/public/api/wekan.yml @@ -2594,6 +2594,53 @@ paths: properties: _id: type: string + /api/boards/{board}/lists/{list}/cards/{card}/archive: + post: + operationId: archive_card + summary: Archive a card + description: | + Archive a card + tags: + - Cards + consumes: + - multipart/form-data + - application/json + parameters: + - name: board + in: path + description: | + the board ID of the card + type: string + required: true + - name: list + in: path + description: | + the list ID of the card + type: string + required: true + - name: card + in: path + description: | + the ID of the card + type: string + required: true + produces: + - application/json + security: + - UserSecurity: [] + responses: + '200': + description: |- + 200 response + schema: + type: object + properties: + _id: + type: string + archived: + type: boolean + archivedAt: + type: string /api/boards/{board}/lists/{list}/cards/{card}/customFields/{customField}: post: operationId: edit_card_custom_field From 36d7b0f8a7b66a458eafe76f56d1f1a9b7037303 Mon Sep 17 00:00:00 2001 From: Mial Lewis Date: Thu, 27 Nov 2025 00:52:28 +0000 Subject: [PATCH 74/84] correct return values --- models/cards.js | 5 +++-- 1 file changed, 3 insertions(+), 2 deletions(-) diff --git a/models/cards.js b/models/cards.js index 9c48546fc..eb55f4c4a 100644 --- a/models/cards.js +++ b/models/cards.js @@ -4468,12 +4468,13 @@ JsonRoutes.add('GET', '/api/boards/:boardId/cards_count', function( if (!card) { throw new Meteor.Error(404, 'Card not found'); } - const archive_res = card.archive(); + card.archive(); JsonRoutes.sendResult(res, { code: 200, data: { _id: paramCardId, - ...archive_res.$set, + archived: true, + archivedAt: new Date(), }, }); }, From 5ff9bf331f4f329ebc70dfa05cf53c4607f25861 Mon Sep 17 00:00:00 2001 From: Mial Lewis Date: Thu, 27 Nov 2025 08:23:56 +0000 Subject: [PATCH 75/84] add restore to api --- models/cards.js | 38 ++++++++++++++++++++++++++++++++++++++ 1 file changed, 38 insertions(+) diff --git a/models/cards.js b/models/cards.js index eb55f4c4a..948612bd6 100644 --- a/models/cards.js +++ b/models/cards.js @@ -4479,6 +4479,44 @@ JsonRoutes.add('GET', '/api/boards/:boardId/cards_count', function( }); }, ); + + /** + * @operation restore_card + * @summary Restore a card from the archive + * + * @description Restore a card from the archive + * @param {string} boardId the board ID of the card + * @param {string} listId the list ID of the card + * @param {string} cardId the ID of the card + * @return_type {_id: string, archived: boolean} + */ + JsonRoutes.add( + 'POST', + '/api/boards/:boardId/lists/:listId/cards/:cardId/restore', + function(req, res) { + const paramBoardId = req.params.boardId; + const paramCardId = req.params.cardId; + const paramListId = req.params.listId; + Authentication.checkBoardAccess(req.userId, paramBoardId); + const card = ReactiveCache.getCard({ + _id: paramCardId, + listId: paramListId, + boardId: paramBoardId, + archived: true, + }); + if (!card) { + throw new Meteor.Error(404, 'Card not found'); + } + card.restore(); + JsonRoutes.sendResult(res, { + code: 200, + data: { + _id: paramCardId, + archived: false, + }, + }); + }, + ); } // Position history tracking methods From a42915614a4f83175b91e2cae0d6aea854d60b58 Mon Sep 17 00:00:00 2001 From: Mial Lewis Date: Thu, 27 Nov 2025 08:25:59 +0000 Subject: [PATCH 76/84] add restore to wekan.yml --- public/api/wekan.yml | 45 ++++++++++++++++++++++++++++++++++++++++++++ 1 file changed, 45 insertions(+) diff --git a/public/api/wekan.yml b/public/api/wekan.yml index 5982969dc..c06c1c263 100644 --- a/public/api/wekan.yml +++ b/public/api/wekan.yml @@ -2705,6 +2705,51 @@ paths: type: string value: type: object + /api/boards/{board}/lists/{list}/cards/{card}/restore: + post: + operationId: restore_card + summary: Restore a card from the archive + description: | + Restore a card from the archive + tags: + - Cards + consumes: + - multipart/form-data + - application/json + parameters: + - name: board + in: path + description: | + the board ID of the card + type: string + required: true + - name: list + in: path + description: | + the list ID of the card + type: string + required: true + - name: card + in: path + description: | + the ID of the card + type: string + required: true + produces: + - application/json + security: + - UserSecurity: [] + responses: + '200': + description: |- + 200 response + schema: + type: object + properties: + _id: + type: string + archived: + type: boolean /api/boards/{board}/lists/{list}/cards_count: get: operationId: get_list_cards_count From bac0fa81fc1f0624e523e4754fb35d127f6e5ddd Mon Sep 17 00:00:00 2001 From: Mial Lewis Date: Thu, 27 Nov 2025 08:27:38 +0000 Subject: [PATCH 77/84] correce indent --- models/cards.js | 2 +- 1 file changed, 1 insertion(+), 1 deletion(-) diff --git a/models/cards.js b/models/cards.js index 948612bd6..1b1831d7d 100644 --- a/models/cards.js +++ b/models/cards.js @@ -4512,7 +4512,7 @@ JsonRoutes.add('GET', '/api/boards/:boardId/cards_count', function( code: 200, data: { _id: paramCardId, - archived: false, + archived: false, }, }); }, From d3c237bc664eb2f4083ff8249ab74c6858f6dc41 Mon Sep 17 00:00:00 2001 From: Mial Lewis Date: Thu, 27 Nov 2025 08:29:36 +0000 Subject: [PATCH 78/84] fix more indenting --- models/cards.js | 4 ++-- 1 file changed, 2 insertions(+), 2 deletions(-) diff --git a/models/cards.js b/models/cards.js index 1b1831d7d..72b353cb4 100644 --- a/models/cards.js +++ b/models/cards.js @@ -4473,8 +4473,8 @@ JsonRoutes.add('GET', '/api/boards/:boardId/cards_count', function( code: 200, data: { _id: paramCardId, - archived: true, - archivedAt: new Date(), + archived: true, + archivedAt: new Date(), }, }); }, From 003a07ebce9ec9b65c8a074f3c18712f8521d623 Mon Sep 17 00:00:00 2001 From: Mial Lewis Date: Thu, 27 Nov 2025 22:00:43 +0000 Subject: [PATCH 79/84] change restore to unarchive --- models/cards.js | 8 ++++---- public/api/wekan.yml | 8 ++++---- 2 files changed, 8 insertions(+), 8 deletions(-) diff --git a/models/cards.js b/models/cards.js index 72b353cb4..a0eaaa8ca 100644 --- a/models/cards.js +++ b/models/cards.js @@ -4481,10 +4481,10 @@ JsonRoutes.add('GET', '/api/boards/:boardId/cards_count', function( ); /** - * @operation restore_card - * @summary Restore a card from the archive + * @operation unarchive_card + * @summary Unarchive card * - * @description Restore a card from the archive + * @description Unarchive card * @param {string} boardId the board ID of the card * @param {string} listId the list ID of the card * @param {string} cardId the ID of the card @@ -4492,7 +4492,7 @@ JsonRoutes.add('GET', '/api/boards/:boardId/cards_count', function( */ JsonRoutes.add( 'POST', - '/api/boards/:boardId/lists/:listId/cards/:cardId/restore', + '/api/boards/:boardId/lists/:listId/cards/:cardId/unarchive', function(req, res) { const paramBoardId = req.params.boardId; const paramCardId = req.params.cardId; diff --git a/public/api/wekan.yml b/public/api/wekan.yml index c06c1c263..947aa3862 100644 --- a/public/api/wekan.yml +++ b/public/api/wekan.yml @@ -2705,12 +2705,12 @@ paths: type: string value: type: object - /api/boards/{board}/lists/{list}/cards/{card}/restore: + /api/boards/{board}/lists/{list}/cards/{card}/unarchive: post: - operationId: restore_card - summary: Restore a card from the archive + operationId: unarchive_card + summary: Unarchive card description: | - Restore a card from the archive + Unarchive card tags: - Cards consumes: From 88ea716d63319e5a0524ab9cdaaa2365bcf135a4 Mon Sep 17 00:00:00 2001 From: "dependabot[bot]" <49699333+dependabot[bot]@users.noreply.github.com> Date: Mon, 1 Dec 2025 23:35:03 +0000 Subject: [PATCH 80/84] Bump docker/metadata-action from 5.9.0 to 5.10.0 Bumps [docker/metadata-action](https://github.com/docker/metadata-action) from 5.9.0 to 5.10.0. - [Release notes](https://github.com/docker/metadata-action/releases) - [Commits](https://github.com/docker/metadata-action/compare/318604b99e75e41977312d83839a89be02ca4893...c299e40c65443455700f0fdfc63efafe5b349051) --- updated-dependencies: - dependency-name: docker/metadata-action dependency-version: 5.10.0 dependency-type: direct:production update-type: version-update:semver-minor ... Signed-off-by: dependabot[bot] --- .github/workflows/docker-publish.yml | 2 +- 1 file changed, 1 insertion(+), 1 deletion(-) diff --git a/.github/workflows/docker-publish.yml b/.github/workflows/docker-publish.yml index ecc6c5044..febfde53f 100644 --- a/.github/workflows/docker-publish.yml +++ b/.github/workflows/docker-publish.yml @@ -48,7 +48,7 @@ jobs: # https://github.com/docker/metadata-action - name: Extract Docker metadata id: meta - uses: docker/metadata-action@318604b99e75e41977312d83839a89be02ca4893 + uses: docker/metadata-action@c299e40c65443455700f0fdfc63efafe5b349051 with: images: ${{ env.REGISTRY }}/${{ env.IMAGE_NAME }} From 5b77ac1b44b6e7210de3c26c66161b0a1eba4d6b Mon Sep 17 00:00:00 2001 From: Lauri Ojansivu Date: Thu, 11 Dec 2025 03:33:17 +0200 Subject: [PATCH 81/84] Updated translations --- imports/i18n/data/fr.i18n.json | 12 +- imports/i18n/data/he.i18n.json | 42 ++--- imports/i18n/data/sv.i18n.json | 282 ++++++++++++++++----------------- 3 files changed, 168 insertions(+), 168 deletions(-) diff --git a/imports/i18n/data/fr.i18n.json b/imports/i18n/data/fr.i18n.json index dc83c8e89..d7f615886 100644 --- a/imports/i18n/data/fr.i18n.json +++ b/imports/i18n/data/fr.i18n.json @@ -1421,18 +1421,18 @@ "card-show-lists-on-minicard": "Afficher les listes sur la mini-carte", "comprehensive-board-migration": "Migration complète de tableau", "comprehensive-board-migration-description": "Performs comprehensive checks and fixes for board data integrity, including list ordering, card positions, and swimlane structure.", - "delete-duplicate-empty-lists-migration": "Delete Duplicate Empty Lists", + "delete-duplicate-empty-lists-migration": "Supprimer les listes vides en doublon ? ", "delete-duplicate-empty-lists-migration-description": "Safely deletes empty duplicate lists. Only removes lists that have no cards AND have another list with the same title that contains cards.", "lost-cards": "Cartes perdues", - "lost-cards-list": "Restored Items", + "lost-cards-list": "Éléments restaurés", "restore-lost-cards-migration": "Restaurer les cartes perdues", "restore-lost-cards-migration-description": "Finds and restores cards and lists with missing swimlaneId or listId. Creates a 'Lost Cards' swimlane to make all lost items visible again.", "restore-all-archived-migration": "Restore All Archived", "restore-all-archived-migration-description": "Restores all archived swimlanes, lists, and cards. Automatically fixes any missing swimlaneId or listId to make items visible.", "fix-missing-lists-migration": "Fix Missing Lists", "fix-missing-lists-migration-description": "Detects and repairs missing or corrupted lists in the board structure.", - "fix-avatar-urls-migration": "Fix Avatar URLs", - "fix-avatar-urls-migration-description": "Updates avatar URLs for board members to use the correct storage backend and fixes broken avatar references.", + "fix-avatar-urls-migration": "Corriger les URLs d'avatar", + "fix-avatar-urls-migration-description": "Mets à jour les URLs d'avatar des participants du tableau pour utiliser le bon gestionnaire de stockage et corriger les références défaillantes d'avatar ", "fix-all-file-urls-migration": "Fix All File URLs", "fix-all-file-urls-migration-description": "Updates all file attachment URLs on this board to use the correct storage backend and fixes broken file references.", "migration-needed": "Migration requise", @@ -1466,13 +1466,13 @@ "step-convert-shared-lists": "Convert Shared Lists", "step-ensure-per-swimlane-lists": "Ensure Per-Swimlane Lists", "step-validate-migration": "Valider la migration", - "step-fix-avatar-urls": "Fix Avatar URLs", + "step-fix-avatar-urls": "Corriger les URLs d'avatar", "step-fix-attachment-urls": "Fix Attachment URLs", "step-analyze-lists": "Analyze Lists", "step-create-missing-lists": "Create Missing Lists", "step-update-cards": "Update Cards", "step-finalize": "Finalize", - "step-delete-duplicate-empty-lists": "Delete Duplicate Empty Lists", + "step-delete-duplicate-empty-lists": "Supprimer les listes vides en doublon ? ", "step-ensure-lost-cards-swimlane": "Ensure Lost Cards Swimlane", "step-restore-lists": "Restore Lists", "step-restore-cards": "Restaurer les cartes", diff --git a/imports/i18n/data/he.i18n.json b/imports/i18n/data/he.i18n.json index 0592d208b..69afb7d25 100644 --- a/imports/i18n/data/he.i18n.json +++ b/imports/i18n/data/he.i18n.json @@ -78,18 +78,18 @@ "activity-deleteComment": "התגובה %s נמחקה", "activity-receivedDate": "תאריך הקבלה השתנה מ־%s ל־%s", "activity-startDate": "תאריך ההתחלה השתנה מ־%s ל־%s", - "allboards.starred": "Starred", + "allboards.starred": "סומן בכוכב", "allboards.templates": "תבניות", - "allboards.remaining": "Remaining", - "allboards.workspaces": "Workspaces", - "allboards.add-workspace": "Add Workspace", - "allboards.add-workspace-prompt": "Workspace name", - "allboards.add-subworkspace": "Add Subworkspace", - "allboards.add-subworkspace-prompt": "Subworkspace name", - "allboards.edit-workspace": "Edit workspace", - "allboards.edit-workspace-name": "Workspace name", - "allboards.edit-workspace-icon": "Workspace icon (markdown)", - "multi-selection-active": "Click checkboxes to select boards", + "allboards.remaining": "נותרו", + "allboards.workspaces": "מרחבי עבודה", + "allboards.add-workspace": "הוספת מרחב עבודה", + "allboards.add-workspace-prompt": "שם מרחב עבודה", + "allboards.add-subworkspace": "הוספת תת־מרחב עבודה", + "allboards.add-subworkspace-prompt": "שם תת־מרחב עבודה", + "allboards.edit-workspace": "עריכת מרחב עבודה", + "allboards.edit-workspace-name": "שם מרחב עבודה", + "allboards.edit-workspace-icon": "סמל מרחב עבודה (markdown)", + "multi-selection-active": "יש לסמן את התיבות כדי לבחור לוחות", "activity-dueDate": "תאריך היעד השתנה מ־%s ל־%s", "activity-endDate": "תאריך הסיום השתנה מ־%s ל־%s", "add-attachment": "הוספת קובץ מצורף", @@ -398,7 +398,7 @@ "editNotificationPopup-title": "שינוי דיווח", "editProfilePopup-title": "עריכת פרופיל", "email": "דוא״ל", - "email-address": "Email Address", + "email-address": "כתובת דוא״ל", "email-enrollAccount-subject": "נוצר עבורך חשבון באתר __siteName__", "email-enrollAccount-text": "__user__ שלום,\n\nכדי להתחיל להשתמש בשירות, יש ללחוץ על הקישור המופיע להלן.\n\n__url__\n\nתודה.", "email-fail": "שליחת ההודעה בדוא״ל נכשלה", @@ -768,8 +768,8 @@ "delete-board-confirm-popup": "כל הרשימות, הכרטיסים, התווית והפעולות יימחקו ולא תהיה לך דרך לשחזר את תכני הלוח. אין אפשרות לבטל.", "boardDeletePopup-title": "למחוק את הלוח?", "delete-board": "מחיקת לוח", - "delete-duplicate-lists": "Delete Duplicate Lists", - "delete-duplicate-lists-confirm": "Are you sure? This will delete all duplicate lists that have the same name and contain no cards.", + "delete-duplicate-lists": "מחיקת רשימות כפולות", + "delete-duplicate-lists-confirm": "להמשיך? הפעולה הזאת תמחק את כל הרשימות הכפולות שיש להן את אותו השם ואינן מכילות כרטיסים.", "default-subtasks-board": "תת־משימות עבור הלוח __board__", "default": "בררת מחדל", "defaultdefault": "בררת מחדל", @@ -1417,15 +1417,15 @@ "back-to-settings": "חזרה להגדרות", "board-id": "מזהה לוח", "board-migration": "הסבת לוחות", - "board-migrations": "Board Migrations", + "board-migrations": "הסבות לוחות", "card-show-lists-on-minicard": "הצגת רשימות בכרטיסון", - "comprehensive-board-migration": "Comprehensive Board Migration", + "comprehensive-board-migration": "הסבת לוחות נרחבת", "comprehensive-board-migration-description": "Performs comprehensive checks and fixes for board data integrity, including list ordering, card positions, and swimlane structure.", "delete-duplicate-empty-lists-migration": "Delete Duplicate Empty Lists", "delete-duplicate-empty-lists-migration-description": "Safely deletes empty duplicate lists. Only removes lists that have no cards AND have another list with the same title that contains cards.", - "lost-cards": "Lost Cards", - "lost-cards-list": "Restored Items", - "restore-lost-cards-migration": "Restore Lost Cards", + "lost-cards": "כרטיסים אבודים", + "lost-cards-list": "פריטים משוחזרים", + "restore-lost-cards-migration": "שחזור כרטיסים אבודים", "restore-lost-cards-migration-description": "Finds and restores cards and lists with missing swimlaneId or listId. Creates a 'Lost Cards' swimlane to make all lost items visible again.", "restore-all-archived-migration": "Restore All Archived", "restore-all-archived-migration-description": "Restores all archived swimlanes, lists, and cards. Automatically fixes any missing swimlaneId or listId to make items visible.", @@ -1439,8 +1439,8 @@ "migration-complete": "הושלם", "migration-running": "Running...", "migration-successful": "Migration completed successfully", - "migration-failed": "Migration failed", - "migrations": "Migrations", + "migration-failed": "ההסבה נכשלה", + "migrations": "הסבות", "migrations-admin-only": "Only board administrators can run migrations", "migrations-description": "Run data integrity checks and repairs for this board. Each migration can be executed individually.", "no-issues-found": "No issues found", diff --git a/imports/i18n/data/sv.i18n.json b/imports/i18n/data/sv.i18n.json index de8ea3332..fa54b9991 100644 --- a/imports/i18n/data/sv.i18n.json +++ b/imports/i18n/data/sv.i18n.json @@ -1417,85 +1417,85 @@ "back-to-settings": "Tillbaka till inställningar", "board-id": "Tavlans ID", "board-migration": "Tavelmigration", - "board-migrations": "Board Migrations", - "card-show-lists-on-minicard": "Show Lists on Minicard", - "comprehensive-board-migration": "Comprehensive Board Migration", - "comprehensive-board-migration-description": "Performs comprehensive checks and fixes for board data integrity, including list ordering, card positions, and swimlane structure.", - "delete-duplicate-empty-lists-migration": "Delete Duplicate Empty Lists", - "delete-duplicate-empty-lists-migration-description": "Safely deletes empty duplicate lists. Only removes lists that have no cards AND have another list with the same title that contains cards.", - "lost-cards": "Lost Cards", - "lost-cards-list": "Restored Items", - "restore-lost-cards-migration": "Restore Lost Cards", - "restore-lost-cards-migration-description": "Finds and restores cards and lists with missing swimlaneId or listId. Creates a 'Lost Cards' swimlane to make all lost items visible again.", - "restore-all-archived-migration": "Restore All Archived", - "restore-all-archived-migration-description": "Restores all archived swimlanes, lists, and cards. Automatically fixes any missing swimlaneId or listId to make items visible.", - "fix-missing-lists-migration": "Fix Missing Lists", - "fix-missing-lists-migration-description": "Detects and repairs missing or corrupted lists in the board structure.", - "fix-avatar-urls-migration": "Fix Avatar URLs", - "fix-avatar-urls-migration-description": "Updates avatar URLs for board members to use the correct storage backend and fixes broken avatar references.", - "fix-all-file-urls-migration": "Fix All File URLs", - "fix-all-file-urls-migration-description": "Updates all file attachment URLs on this board to use the correct storage backend and fixes broken file references.", - "migration-needed": "Migration Needed", - "migration-complete": "Complete", - "migration-running": "Running...", - "migration-successful": "Migration completed successfully", - "migration-failed": "Migration failed", - "migrations": "Migrations", - "migrations-admin-only": "Only board administrators can run migrations", - "migrations-description": "Run data integrity checks and repairs for this board. Each migration can be executed individually.", - "no-issues-found": "No issues found", - "run-migration": "Run Migration", - "run-comprehensive-migration-confirm": "This will perform a comprehensive migration to check and fix board data integrity. This may take a few moments. Continue?", - "run-delete-duplicate-empty-lists-migration-confirm": "This will first convert any shared lists to per-swimlane lists, then delete empty lists that have a duplicate list with the same title containing cards. Only truly redundant empty lists will be removed. Continue?", - "run-restore-lost-cards-migration-confirm": "This will create a 'Lost Cards' swimlane and restore all cards and lists with missing swimlaneId or listId. This only affects non-archived items. Continue?", - "run-restore-all-archived-migration-confirm": "This will restore ALL archived swimlanes, lists, and cards, making them visible again. Any items with missing IDs will be automatically fixed. This cannot be easily undone. Continue?", - "run-fix-missing-lists-migration-confirm": "This will detect and repair missing or corrupted lists in the board structure. Continue?", - "run-fix-avatar-urls-migration-confirm": "This will update avatar URLs for board members to use the correct storage backend. Continue?", - "run-fix-all-file-urls-migration-confirm": "This will update all file attachment URLs on this board to use the correct storage backend. Continue?", - "restore-lost-cards-nothing-to-restore": "No lost swimlanes, lists, or cards to restore", + "board-migrations": "Migrering av tavlor", + "card-show-lists-on-minicard": "Visa listor på minikort", + "comprehensive-board-migration": "Fullständig migration av alla tavlor ", + "comprehensive-board-migration-description": "Utför omfattande kontroller och korrigeringar för tavlans dataintegritet, inklusive listordning, kortpositioner och swimlane-struktur.", + "delete-duplicate-empty-lists-migration": "Ta bort duplicerade tomma listor", + "delete-duplicate-empty-lists-migration-description": "Tar säkert bort tomma duplicerade listor. Tar endast bort listor som inte har några kort OCH där det finns en annan lista med samma titel som innehåller kort.", + "lost-cards": "Förlorade kort ", + "lost-cards-list": "Återställda objekt", + "restore-lost-cards-migration": "Återställ förlorade kort", + "restore-lost-cards-migration-description": "Hittar och återställer kort och listor som saknar simbaneid eller listId. Skapar en 'Förlorade kort'-simbana för att göra alla förlorade objekt synliga igen.", + "restore-all-archived-migration": "Återställ från Arkiv", + "restore-all-archived-migration-description": "Återställer alla arkiverade swimlanes, listor och kort. Korrigerar automatiskt alla saknade simbaneId eller listId för att göra objekt synliga.", + "fix-missing-lists-migration": "Fixa saknade listor", + "fix-missing-lists-migration-description": "Upptäcker och reparerar saknade eller korrupta listor i tavlans struktur.", + "fix-avatar-urls-migration": "Fixa avatar-URL:er", + "fix-avatar-urls-migration-description": "Uppdaterar avatar-URL:er för tavlans medlemmar till att använda rätt lagrings-backend och fixar trasiga avatar-referenser.", + "fix-all-file-urls-migration": "Fixa alla fil-URL:er", + "fix-all-file-urls-migration-description": "Uppdaterar alla URL:er för filbilagor på denna tavla till att använda rätt lagrings-backend och fixar trasiga filreferenser.", + "migration-needed": "Migrering krävs", + "migration-complete": "Avslutad", + "migration-running": "Körs...", + "migration-successful": "Migrering slutförd", + "migration-failed": "Migrering misslyckades", + "migrations": "Migreringar", + "migrations-admin-only": "Endast tavlans administratörer kan köra migreringar", + "migrations-description": "Kör dataintegritetskontroller och reparationer för denna tavla. Varje migrering kan utföras individuellt.", + "no-issues-found": "Inga problem hittades", + "run-migration": "Kör migrering", + "run-comprehensive-migration-confirm": "Detta kommer att utföra en omfattande migrering för att kontrollera och fixa tavlans dataintegritet. Detta kan ta en liten stund. Fortsätt?", + "run-delete-duplicate-empty-lists-migration-confirm": "Detta kommer först att konvertera alla delade listor till listor per simbana, och sedan ta bort tomma listor som har en dubblettlista med samma titel som innehåller kort. Endast överflödiga tomma listor kommer att tas bort. Fortsätt?", + "run-restore-lost-cards-migration-confirm": "Detta kommer att skapa en 'Förlorade kort'-simbana och återställa alla kort och listor som saknar simbaneId eller listId. Detta påverkar endast icke-arkiverade objekt. Fortsätt?", + "run-restore-all-archived-migration-confirm": "Detta kommer att återställa ALLA arkiverade simbanor, listor och kort, vilket gör dem synliga igen. Alla objekt med saknade ID:n kommer att fixas automatiskt. Detta kan inte enkelt ångras. Fortsätt?", + "run-fix-missing-lists-migration-confirm": "Detta kommer att upptäcka och reparera saknade eller korrupta listor i tavlans struktur. Fortsätt?", + "run-fix-avatar-urls-migration-confirm": "Detta kommer att uppdatera avatar-URL:er för tavlans medlemmar till att använda rätt lagrings-backend. Fortsätt?", + "run-fix-all-file-urls-migration-confirm": "Detta kommer att uppdatera alla URL:er för filbilagor på denna tavla till att använda rätt lagrings-backend. Fortsätt?", + "restore-lost-cards-nothing-to-restore": "Inga förlorade simbanor, listor eller kort att återställa", - "migration-progress-title": "Board Migration in Progress", - "migration-progress-overall": "Overall Progress", - "migration-progress-current-step": "Current Step", + "migration-progress-title": "Tavlans migrering pågår", + "migration-progress-overall": "Övergripande förlopp", + "migration-progress-current-step": "Nuvarande steg", "migration-progress-status": "Status", "migration-progress-details": "Detaljer", - "migration-progress-note": "Please wait while we migrate your board to the latest structure...", + "migration-progress-note": "Vänta medan vi migrerar din tavla till den senaste strukturen...", - "step-analyze-board-structure": "Analyze Board Structure", - "step-fix-orphaned-cards": "Fix Orphaned Cards", - "step-convert-shared-lists": "Convert Shared Lists", - "step-ensure-per-swimlane-lists": "Ensure Per-Swimlane Lists", - "step-validate-migration": "Validate Migration", - "step-fix-avatar-urls": "Fix Avatar URLs", - "step-fix-attachment-urls": "Fix Attachment URLs", - "step-analyze-lists": "Analyze Lists", + "step-analyze-board-structure": "Analysera tavlans struktur", + "step-fix-orphaned-cards": "Fixa övergivna kort", + "step-convert-shared-lists": "Konvertera delade listor", + "step-ensure-per-swimlane-lists": "Säkerställ listor per simbana", + "step-validate-migration": "Validera migrering", + "step-fix-avatar-urls": "Fixa avatar-URL:er", + "step-fix-attachment-urls": "Fixa bilage-URL:er", + "step-analyze-lists": "Analysera listor", "step-create-missing-lists": "Create Missing Lists", - "step-update-cards": "Update Cards", - "step-finalize": "Finalize", - "step-delete-duplicate-empty-lists": "Delete Duplicate Empty Lists", - "step-ensure-lost-cards-swimlane": "Ensure Lost Cards Swimlane", - "step-restore-lists": "Restore Lists", - "step-restore-cards": "Restore Cards", - "step-restore-swimlanes": "Restore Swimlanes", - "step-fix-missing-ids": "Fix Missing IDs", - "step-scan-users": "Checking board member avatars", - "step-scan-files": "Checking board file attachments", - "step-fix-file-urls": "Fixing file URLs", - "cleanup": "Cleanup", - "cleanup-old-jobs": "Cleanup Old Jobs", + "step-update-cards": "Uppdatera kort", + "step-finalize": "Slutför", + "step-delete-duplicate-empty-lists": "Ta bort duplicerade tomma listor", + "step-ensure-lost-cards-swimlane": "Säkerställ simbana för förlorade kort", + "step-restore-lists": "Återställ listor", + "step-restore-cards": "Återställ kort", + "step-restore-swimlanes": "Återställ simbanor", + "step-fix-missing-ids": "Fixa saknade ID:n", + "step-scan-users": "Kontrollerar tavlans medlemsavatarer", + "step-scan-files": "Kontrollerar tavlans filbilagor", + "step-fix-file-urls": "Fixar fil-URL:er", + "cleanup": "Rensning", + "cleanup-old-jobs": "Rensa gamla jobb", "completed": "Avslutad", - "conversion-info-text": "This conversion is performed once per board and improves performance. You can continue using the board normally.", - "converting-board": "Converting Board", - "converting-board-description": "Converting board structure for improved functionality. This may take a few moments.", - "cpu-cores": "CPU Cores", - "cpu-usage": "CPU Usage", - "current-action": "Current Action", - "database-migration": "Database Migration", - "database-migration-description": "Updating database structure for improved functionality and performance. This process may take several minutes.", - "database-migrations": "Database Migrations", - "days-old": "Days Old", - "duration": "Duration", - "errors": "Errors", + "conversion-info-text": "Denna konvertering utförs en gång per tavla och förbättrar prestandan. Ni kan fortsätta använda tavlan som vanligt.", + "converting-board": "Konverterar tavla", + "converting-board-description": "Konverterar tavlans struktur för förbättrad funktionalitet. Detta kan ta en liten stund.", + "cpu-cores": "CPU-kärnor", + "cpu-usage": "CPU-användning", + "current-action": "Nuvarande åtgärd", + "database-migration": "Databas­migrering", + "database-migration-description": "Uppdaterar databasstrukturen för förbättrad funktionalitet och prestanda. Denna process kan ta flera minuter.", + "database-migrations": "Databas­migreringar", + "days-old": "Dagar gammal", + "duration": "Varaktighet", + "errors": "Fel", "estimated-time-remaining": "Beräknad återstående tid", "every-1-day": "Var 1 dag", "every-1-hour": "Var 1 timme", @@ -1504,76 +1504,76 @@ "every-30-minutes": "Var 30 minuter", "every-5-minutes": "Var 5 minuter", "every-6-hours": "Var 6 timmar", - "export-monitoring": "Export Monitoring", - "filesystem-attachments": "Filesystem Attachments", - "filesystem-size": "Filesystem Size", - "filesystem-storage": "Filesystem Storage", - "force-board-scan": "Force Board Scan", - "gridfs-attachments": "GridFS Attachments", - "gridfs-size": "GridFS Size", + "export-monitoring": "Exportövervakning", + "filesystem-attachments": "Filssystem-bilagor", + "filesystem-size": "Filssystemstorlek", + "filesystem-storage": "Filssystemlagring", + "force-board-scan": "Tvinga tavelskanning", + "gridfs-attachments": "GridFS-bilagor", + "gridfs-size": "GridFS-storlek", "gridfs-storage": "GridFS", - "hide-list-on-minicard": "Hide List on Minicard", - "idle-migration": "Idle Migration", - "job-description": "Job Description", - "job-details": "Job Details", - "job-name": "Job Name", - "job-queue": "Job Queue", - "last-run": "Last Run", - "max-concurrent": "Max Concurrent", - "memory-usage": "Memory Usage", - "migrate-all-to-filesystem": "Migrate All to Filesystem", - "migrate-all-to-gridfs": "Migrate All to GridFS", - "migrate-all-to-s3": "Migrate All to S3", - "migrated-attachments": "Migrated Attachments", - "migration-batch-size": "Batch Size", - "migration-batch-size-description": "Number of attachments to process in each batch (1-100)", - "migration-cpu-threshold": "CPU Threshold (%)", - "migration-cpu-threshold-description": "Pause migration when CPU usage exceeds this percentage (10-90)", - "migration-delay-ms": "Delay (ms)", - "migration-delay-ms-description": "Delay between batches in milliseconds (100-10000)", - "migration-detector": "Migration Detector", - "migration-info-text": "Database migrations are performed once and improve system performance. The process continues in the background even if you close your browser.", - "migration-log": "Migration Log", - "migration-markers": "Migration Markers", - "migration-resume-failed": "Failed to resume migration", - "migration-resumed": "Migration resumed", - "migration-steps": "Migration Steps", - "migration-warning-text": "Please do not close your browser during migration. The process will continue in the background but may take longer to complete.", - "monitoring-export-failed": "Failed to export monitoring data", - "monitoring-refresh-failed": "Failed to refresh monitoring data", - "next": "Next", - "next-run": "Next Run", + "hide-list-on-minicard": "Göm lista på minikort", + "idle-migration": "Inaktiv migrering", + "job-description": "Jobbeskrivning", + "job-details": "Jobbdetaljer", + "job-name": "Jobbnamn", + "job-queue": "Jobbkö", + "last-run": "Senaste körning", + "max-concurrent": "Max samtidigt", + "memory-usage": "Minnesanvändning", + "migrate-all-to-filesystem": "Migrera allt till filsystem", + "migrate-all-to-gridfs": "Migrera allt till GridFS", + "migrate-all-to-s3": "Migrera allt till S3", + "migrated-attachments": "Migrerade bilagor", + "migration-batch-size": "Batchstorlek", + "migration-batch-size-description": "Antal bilagor att bearbeta i varje batch (1-100)", + "migration-cpu-threshold": "CPU-tröskel (%)", + "migration-cpu-threshold-description": "Pausa migreringen när CPU-användningen överstiger denna procentsats (10-90)", + "migration-delay-ms": "Fördröjning (ms)", + "migration-delay-ms-description": "Fördröjning mellan batchar i millisekunder (100-10000)", + "migration-detector": "Migreringsdetektor", + "migration-info-text": "Databas­migreringar utförs en gång och förbättrar systemets prestanda. Processen fortsätter i bakgrunden även om du stänger din webbläsare.", + "migration-log": "Migreringslogg", + "migration-markers": "Migreringsmarkörer", + "migration-resume-failed": "Misslyckades med att återuppta migrering", + "migration-resumed": "Migrering återupptogs", + "migration-steps": "Migreringssteg", + "migration-warning-text": "Vänligen stäng inte din webbläsare under migreringen. Processen fortsätter i bakgrunden men kan ta längre tid att slutföra.", + "monitoring-export-failed": "Misslyckades med att exportera övervakningsdata", + "monitoring-refresh-failed": "Misslyckades med att uppdatera övervakningsdata", + "next": "Nästa", + "next-run": "Nästa körning", "of": "av", - "operation-type": "Operation Type", - "overall-progress": "Overall Progress", - "page": "Page", - "pause-migration": "Pause Migration", - "previous": "Previous", - "refresh": "Refresh", - "refresh-monitoring": "Refresh Monitoring", - "remaining-attachments": "Remaining Attachments", - "resume-migration": "Resume Migration", - "run-once": "Run once", - "s3-attachments": "S3 Attachments", - "s3-size": "S3 Size", + "operation-type": "Operationstyp", + "overall-progress": "Övergripande förlopp", + "page": "Sida", + "pause-migration": "Pausa migrering", + "previous": "Föregående", + "refresh": "Uppdatera", + "refresh-monitoring": "Uppdatera övervakning", + "remaining-attachments": "Återstående bilagor", + "resume-migration": "Återuppta migrering", + "run-once": "Kör en gång", + "s3-attachments": "S3-bilagor", + "s3-size": "S3-storlek", "s3-storage": "S3", - "scanning-status": "Scanning Status", - "schedule": "Schedule", - "search-boards-or-operations": "Search boards or operations...", - "show-list-on-minicard": "Show List on Minicard", - "showing": "Showing", - "start-test-operation": "Start Test Operation", - "start-time": "Start Time", - "step-progress": "Step Progress", - "stop-migration": "Stop Migration", - "storage-distribution": "Storage Distribution", - "system-resources": "System Resources", - "total-attachments": "Total Attachments", - "total-operations": "Total Operations", - "total-size": "Total Size", - "unmigrated-boards": "Unmigrated Boards", - "weight": "Weight", - "idle": "Idle", - "complete": "Complete", + "scanning-status": "Skanningsstatus", + "schedule": "Schema", + "search-boards-or-operations": "Sök tavlor eller operationer...", + "show-list-on-minicard": "Visa lista på minikort", + "showing": "Visar", + "start-test-operation": "Starta testoperation", + "start-time": "Starttid", + "step-progress": "Stegförlopp", + "stop-migration": "Stoppa migrering", + "storage-distribution": "Lagringsdistribution", + "system-resources": "Systemresurser", + "total-attachments": "Totala bilagor", + "total-operations": "Totala operationer", + "total-size": "Total storlek", + "unmigrated-boards": "Okonverterade tavlor", + "weight": "Vikt", + "idle": "Inaktiv", + "complete": "Avslutad", "cron": "Cron" } From a290c7b34beaaa21f1379116e042efb120336e0d Mon Sep 17 00:00:00 2001 From: "dependabot[bot]" <49699333+dependabot[bot]@users.noreply.github.com> Date: Mon, 15 Dec 2025 20:12:55 +0000 Subject: [PATCH 82/84] Bump actions/upload-artifact from 5 to 6 Bumps [actions/upload-artifact](https://github.com/actions/upload-artifact) from 5 to 6. - [Release notes](https://github.com/actions/upload-artifact/releases) - [Commits](https://github.com/actions/upload-artifact/compare/v5...v6) --- updated-dependencies: - dependency-name: actions/upload-artifact dependency-version: '6' dependency-type: direct:production update-type: version-update:semver-major ... Signed-off-by: dependabot[bot] --- .github/workflows/test_suite.yml | 2 +- 1 file changed, 1 insertion(+), 1 deletion(-) diff --git a/.github/workflows/test_suite.yml b/.github/workflows/test_suite.yml index 6de010b61..afee5ce08 100644 --- a/.github/workflows/test_suite.yml +++ b/.github/workflows/test_suite.yml @@ -136,7 +136,7 @@ jobs: run: sh ./test-wekan.sh -cv - name: Upload coverage - uses: actions/upload-artifact@v5 + uses: actions/upload-artifact@v6 with: name: coverage-folder path: .coverage/ From cec625607d2632eddcf156281682b53aaaab4ea1 Mon Sep 17 00:00:00 2001 From: "dependabot[bot]" <49699333+dependabot[bot]@users.noreply.github.com> Date: Mon, 15 Dec 2025 20:13:00 +0000 Subject: [PATCH 83/84] Bump actions/download-artifact from 6 to 7 Bumps [actions/download-artifact](https://github.com/actions/download-artifact) from 6 to 7. - [Release notes](https://github.com/actions/download-artifact/releases) - [Commits](https://github.com/actions/download-artifact/compare/v6...v7) --- updated-dependencies: - dependency-name: actions/download-artifact dependency-version: '7' dependency-type: direct:production update-type: version-update:semver-major ... Signed-off-by: dependabot[bot] --- .github/workflows/test_suite.yml | 2 +- 1 file changed, 1 insertion(+), 1 deletion(-) diff --git a/.github/workflows/test_suite.yml b/.github/workflows/test_suite.yml index 6de010b61..be2be4533 100644 --- a/.github/workflows/test_suite.yml +++ b/.github/workflows/test_suite.yml @@ -150,7 +150,7 @@ jobs: uses: actions/checkout@v6 - name: Download coverage - uses: actions/download-artifact@v6 + uses: actions/download-artifact@v7 with: name: coverage-folder path: .coverage/ From 07f69950a764395a413f0630666484044367e6fb Mon Sep 17 00:00:00 2001 From: "dependabot[bot]" <49699333+dependabot[bot]@users.noreply.github.com> Date: Mon, 15 Dec 2025 20:13:04 +0000 Subject: [PATCH 84/84] Bump actions/cache from 4 to 5 Bumps [actions/cache](https://github.com/actions/cache) from 4 to 5. - [Release notes](https://github.com/actions/cache/releases) - [Changelog](https://github.com/actions/cache/blob/main/RELEASES.md) - [Commits](https://github.com/actions/cache/compare/v4...v5) --- updated-dependencies: - dependency-name: actions/cache dependency-version: '5' dependency-type: direct:production update-type: version-update:semver-major ... Signed-off-by: dependabot[bot] --- .github/workflows/test_suite.yml | 6 +++--- 1 file changed, 3 insertions(+), 3 deletions(-) diff --git a/.github/workflows/test_suite.yml b/.github/workflows/test_suite.yml index 6de010b61..850500584 100644 --- a/.github/workflows/test_suite.yml +++ b/.github/workflows/test_suite.yml @@ -95,7 +95,7 @@ jobs: # CACHING - name: Install Meteor id: cache-meteor-install - uses: actions/cache@v4 + uses: actions/cache@v5 with: path: ~/.meteor key: v1-meteor-${{ hashFiles('.meteor/versions') }} @@ -104,7 +104,7 @@ jobs: - name: Cache NPM dependencies id: cache-meteor-npm - uses: actions/cache@v4 + uses: actions/cache@v5 with: path: ~/.npm key: v1-npm-${{ hashFiles('package-lock.json') }} @@ -113,7 +113,7 @@ jobs: - name: Cache Meteor build id: cache-meteor-build - uses: actions/cache@v4 + uses: actions/cache@v5 with: path: | .meteor/local/resolver-result-cache.json