Add standalone desktop workflow
This commit is contained in:
@@ -4,16 +4,17 @@
|
||||
<section class="panel">
|
||||
<h2>Copy Sources</h2>
|
||||
<div class="panel-body">
|
||||
<div class="form-hint">Select top-level folders or expand branches and choose individual nested directories.</div>
|
||||
<div class="form-hint">Add one or more root folders with source files. After that, expand each root and enable or disable individual nested folders with checkboxes.</div>
|
||||
</div>
|
||||
<div class="btn-row">
|
||||
<button type="button" class="button-primary" onclick="addSourceRoot()">Add source folder</button>
|
||||
<button type="button" class="button-secondary button-sm" onclick="reloadAllSourceTrees()">Refresh trees</button>
|
||||
</div>
|
||||
<div class="source-list">
|
||||
<div class="source-tree" id="sourceTree">
|
||||
<div class="text-muted source-tree-empty">Loading...</div>
|
||||
<div class="text-muted source-tree-empty">No source folders added yet.</div>
|
||||
</div>
|
||||
</div>
|
||||
<div class="btn-row">
|
||||
<button type="button" class="button-secondary button-sm" onclick="reloadSourceTree()">Refresh list</button>
|
||||
</div>
|
||||
</section>
|
||||
|
||||
<section class="panel">
|
||||
@@ -77,6 +78,7 @@
|
||||
const sourceTree = new Map();
|
||||
const expandedNodes = new Set();
|
||||
const loadingNodes = new Set();
|
||||
let sourceRoots = [];
|
||||
let sourceConfig = {};
|
||||
|
||||
function escapeHTML(value) {
|
||||
@@ -89,13 +91,43 @@ function escapeHTML(value) {
|
||||
}[char]));
|
||||
}
|
||||
|
||||
function pathDepth(path) {
|
||||
return path ? path.split('/').length : 0;
|
||||
function pathSegments(path) {
|
||||
return String(path || '').split(/[\\/]+/).filter(Boolean);
|
||||
}
|
||||
|
||||
function nodeName(path) {
|
||||
const parts = pathSegments(path);
|
||||
return parts.length ? parts[parts.length - 1] : path;
|
||||
}
|
||||
|
||||
function normalizeComparePath(path) {
|
||||
return String(path || '').replace(/\\/g, '/').replace(/\/+$/, '').toLowerCase();
|
||||
}
|
||||
|
||||
function isSamePath(a, b) {
|
||||
return normalizeComparePath(a) === normalizeComparePath(b);
|
||||
}
|
||||
|
||||
function isPathWithin(base, candidate) {
|
||||
const baseNorm = normalizeComparePath(base);
|
||||
const candidateNorm = normalizeComparePath(candidate);
|
||||
return candidateNorm === baseNorm || candidateNorm.startsWith(baseNorm + '/');
|
||||
}
|
||||
|
||||
function parentPath(path) {
|
||||
if (!path || !path.includes('/')) return '';
|
||||
return path.slice(0, path.lastIndexOf('/'));
|
||||
const value = String(path || '').replace(/[\\/]+$/, '');
|
||||
const slash = Math.max(value.lastIndexOf('/'), value.lastIndexOf('\\'));
|
||||
if (slash < 0) return '';
|
||||
if (slash === 2 && /^[A-Za-z]:/.test(value)) return value.slice(0, slash + 1);
|
||||
if (slash === 0) return value.slice(0, 1);
|
||||
return value.slice(0, slash);
|
||||
}
|
||||
|
||||
function relativeDepth(root, path) {
|
||||
if (isSamePath(root, path)) return 0;
|
||||
const rootParts = pathSegments(root);
|
||||
const pathParts = pathSegments(path);
|
||||
return Math.max(0, pathParts.length - rootParts.length);
|
||||
}
|
||||
|
||||
function effectiveSourceState(path) {
|
||||
@@ -104,37 +136,45 @@ function effectiveSourceState(path) {
|
||||
if (Object.prototype.hasOwnProperty.call(sourceConfig, current)) {
|
||||
return sourceConfig[current];
|
||||
}
|
||||
if (!current) return true;
|
||||
current = parentPath(current);
|
||||
if (!current) return true;
|
||||
}
|
||||
}
|
||||
|
||||
function collectSourcesForSave() {
|
||||
const items = [];
|
||||
const seen = new Set();
|
||||
const roots = sourceTree.get('') || [];
|
||||
|
||||
for (const item of roots) {
|
||||
items.push({ path: item.path, enabled: effectiveSourceState(item.path) });
|
||||
seen.add(item.path);
|
||||
}
|
||||
sourceRoots.forEach((root) => {
|
||||
items.push({ path: root, enabled: effectiveSourceState(root), root: true });
|
||||
seen.add(normalizeComparePath(root));
|
||||
});
|
||||
|
||||
Object.entries(sourceConfig).forEach(([path, enabled]) => {
|
||||
if (seen.has(path)) return;
|
||||
items.push({ path, enabled });
|
||||
const key = normalizeComparePath(path);
|
||||
if (seen.has(key)) return;
|
||||
items.push({ path, enabled, root: false });
|
||||
});
|
||||
|
||||
return items.sort((a, b) => a.path.localeCompare(b.path));
|
||||
}
|
||||
|
||||
async function loadSourceChildren(path = '') {
|
||||
if (loadingNodes.has(path)) return;
|
||||
async function pickFolder() {
|
||||
const response = await fetch('/api/system/pick-folder', { method: 'POST' });
|
||||
const payload = await response.json();
|
||||
if (!response.ok) {
|
||||
throw new Error(payload.error || 'Failed to choose folder');
|
||||
}
|
||||
return payload.path || '';
|
||||
}
|
||||
|
||||
async function loadSourceChildren(path) {
|
||||
if (!path || loadingNodes.has(path)) return;
|
||||
loadingNodes.add(path);
|
||||
renderSources();
|
||||
|
||||
try {
|
||||
const query = path ? '?path=' + encodeURIComponent(path) : '';
|
||||
const response = await fetch('/api/sources' + query);
|
||||
const response = await fetch('/api/sources?path=' + encodeURIComponent(path));
|
||||
if (!response.ok) return;
|
||||
const payload = await response.json();
|
||||
sourceTree.set(path, payload.items || []);
|
||||
@@ -159,15 +199,59 @@ function toggleSource(path, checked) {
|
||||
renderSources();
|
||||
}
|
||||
|
||||
function renderSourceNodes(parent = '') {
|
||||
const items = sourceTree.get(parent) || [];
|
||||
function removeRoot(path) {
|
||||
sourceRoots = sourceRoots.filter((root) => !isSamePath(root, path));
|
||||
sourceTree.delete(path);
|
||||
expandedNodes.delete(path);
|
||||
loadingNodes.delete(path);
|
||||
|
||||
Object.keys(sourceConfig).forEach((key) => {
|
||||
if (isPathWithin(path, key)) {
|
||||
delete sourceConfig[key];
|
||||
}
|
||||
});
|
||||
|
||||
renderSources();
|
||||
}
|
||||
|
||||
async function addSourceRoot() {
|
||||
try {
|
||||
const path = await pickFolder();
|
||||
if (!path) return;
|
||||
if (sourceRoots.some((root) => isSamePath(root, path))) {
|
||||
toast('This source folder is already added', 'error');
|
||||
return;
|
||||
}
|
||||
sourceRoots.push(path);
|
||||
sourceRoots.sort((a, b) => a.localeCompare(b));
|
||||
sourceConfig[path] = true;
|
||||
expandedNodes.add(path);
|
||||
await loadSourceChildren(path);
|
||||
} catch (error) {
|
||||
toast(error.message || 'Failed to choose folder', 'error');
|
||||
}
|
||||
}
|
||||
|
||||
async function reloadAllSourceTrees() {
|
||||
const roots = [...sourceRoots];
|
||||
sourceTree.clear();
|
||||
for (const root of roots) {
|
||||
if (expandedNodes.has(root)) {
|
||||
await loadSourceChildren(root);
|
||||
}
|
||||
}
|
||||
renderSources();
|
||||
}
|
||||
|
||||
function renderSourceNodes(root, parentPathValue) {
|
||||
const items = sourceTree.get(parentPathValue) || [];
|
||||
return items.map((item) => {
|
||||
const checked = effectiveSourceState(item.path);
|
||||
const expanded = expandedNodes.has(item.path);
|
||||
const childrenKnown = sourceTree.has(item.path);
|
||||
const children = childrenKnown ? sourceTree.get(item.path) : [];
|
||||
const hasChildren = !childrenKnown || children.length > 0;
|
||||
const pad = 16 + pathDepth(item.path) * 20;
|
||||
const pad = 16 + (relativeDepth(root, item.path) + 1) * 20;
|
||||
|
||||
return `
|
||||
<div class="source-node">
|
||||
@@ -182,11 +266,11 @@ function renderSourceNodes(parent = '') {
|
||||
<input class="source-check" type="checkbox" data-action="toggle-check" data-path="${escapeHTML(item.path)}" ${checked ? 'checked' : ''}>
|
||||
<div class="source-label">
|
||||
<span class="source-item-name">${escapeHTML(item.name)}</span>
|
||||
<span class="source-item-path">/media/${escapeHTML(item.path)}</span>
|
||||
<span class="source-item-path">${escapeHTML(item.path)}</span>
|
||||
</div>
|
||||
</div>
|
||||
${expanded && loadingNodes.has(item.path) ? '<div class="source-loading">Loading...</div>' : ''}
|
||||
${expanded && childrenKnown && children.length ? `<div class="source-children">${renderSourceNodes(item.path)}</div>` : ''}
|
||||
${expanded && childrenKnown && children.length ? `<div class="source-children">${renderSourceNodes(root, item.path)}</div>` : ''}
|
||||
</div>
|
||||
`;
|
||||
}).join('');
|
||||
@@ -194,24 +278,54 @@ function renderSourceNodes(parent = '') {
|
||||
|
||||
function renderSources() {
|
||||
const el = document.getElementById('sourceTree');
|
||||
const roots = sourceTree.get('');
|
||||
|
||||
if (loadingNodes.has('') && !roots) {
|
||||
el.innerHTML = '<div class="text-muted source-tree-empty">Loading...</div>';
|
||||
return;
|
||||
}
|
||||
if (!roots || !roots.length) {
|
||||
el.innerHTML = '<div class="text-muted source-tree-empty">No folders found in /media.</div>';
|
||||
if (!sourceRoots.length) {
|
||||
el.innerHTML = '<div class="text-muted source-tree-empty">No source folders added yet.</div>';
|
||||
return;
|
||||
}
|
||||
|
||||
el.innerHTML = renderSourceNodes('');
|
||||
el.innerHTML = sourceRoots.map((root) => {
|
||||
const checked = effectiveSourceState(root);
|
||||
const expanded = expandedNodes.has(root);
|
||||
const childrenKnown = sourceTree.has(root);
|
||||
const children = childrenKnown ? sourceTree.get(root) : [];
|
||||
const hasChildren = !childrenKnown || children.length > 0;
|
||||
|
||||
return `
|
||||
<div class="source-root-card">
|
||||
<div class="source-row source-root-row">
|
||||
<button
|
||||
type="button"
|
||||
class="source-toggle ${hasChildren ? '' : 'source-toggle-empty'}"
|
||||
data-action="toggle-expand"
|
||||
data-path="${escapeHTML(root)}"
|
||||
${hasChildren ? '' : 'tabindex="-1" aria-hidden="true"'}
|
||||
>${expanded ? '▾' : '▸'}</button>
|
||||
<input class="source-check" type="checkbox" data-action="toggle-check" data-path="${escapeHTML(root)}" ${checked ? 'checked' : ''}>
|
||||
<div class="source-label">
|
||||
<div class="source-root-title">
|
||||
<span class="source-item-name">${escapeHTML(nodeName(root))}</span>
|
||||
<span class="source-root-badge">Root</span>
|
||||
</div>
|
||||
<span class="source-item-path">${escapeHTML(root)}</span>
|
||||
</div>
|
||||
<button type="button" class="button-secondary button-sm" data-action="remove-root" data-path="${escapeHTML(root)}">Remove</button>
|
||||
</div>
|
||||
${expanded && loadingNodes.has(root) ? '<div class="source-loading">Loading...</div>' : ''}
|
||||
${expanded && childrenKnown && children.length ? `<div class="source-children">${renderSourceNodes(root, root)}</div>` : ''}
|
||||
</div>
|
||||
`;
|
||||
}).join('');
|
||||
}
|
||||
|
||||
async function reloadSourceTree() {
|
||||
sourceTree.clear();
|
||||
expandedNodes.clear();
|
||||
await loadSourceChildren('');
|
||||
function deriveRootsFromSources(sources) {
|
||||
const explicitRoots = sources.filter((source) => source.root).map((source) => source.path);
|
||||
if (explicitRoots.length) {
|
||||
return explicitRoots;
|
||||
}
|
||||
|
||||
return sources
|
||||
.map((source) => source.path)
|
||||
.filter((path, index, all) => !all.some((other, otherIndex) => otherIndex !== index && isPathWithin(other, path) && !isSamePath(other, path)));
|
||||
}
|
||||
|
||||
async function loadSettings() {
|
||||
@@ -229,7 +343,10 @@ async function loadSettings() {
|
||||
(cfg.sources || []).forEach((source) => {
|
||||
sourceConfig[source.path] = !!source.enabled;
|
||||
});
|
||||
renderSources();
|
||||
sourceRoots = deriveRootsFromSources(cfg.sources || []).sort((a, b) => a.localeCompare(b));
|
||||
expandedNodes.clear();
|
||||
sourceTree.clear();
|
||||
await reloadAllSourceTrees();
|
||||
} catch (error) {}
|
||||
}
|
||||
|
||||
@@ -264,16 +381,22 @@ async function saveSettings(event) {
|
||||
}
|
||||
|
||||
document.getElementById('sourceTree').addEventListener('click', async (event) => {
|
||||
const button = event.target.closest('[data-action="toggle-expand"]');
|
||||
const button = event.target.closest('button[data-action]');
|
||||
if (!button) return;
|
||||
|
||||
const action = button.dataset.action;
|
||||
const path = button.dataset.path;
|
||||
if (expandedNodes.has(path)) {
|
||||
expandedNodes.delete(path);
|
||||
renderSources();
|
||||
return;
|
||||
if (action === 'toggle-expand') {
|
||||
if (expandedNodes.has(path)) {
|
||||
expandedNodes.delete(path);
|
||||
renderSources();
|
||||
return;
|
||||
}
|
||||
await ensureExpanded(path);
|
||||
}
|
||||
if (action === 'remove-root') {
|
||||
removeRoot(path);
|
||||
}
|
||||
await ensureExpanded(path);
|
||||
});
|
||||
|
||||
document.getElementById('sourceTree').addEventListener('change', (event) => {
|
||||
@@ -283,6 +406,5 @@ document.getElementById('sourceTree').addEventListener('change', (event) => {
|
||||
});
|
||||
|
||||
loadSettings();
|
||||
loadSourceChildren('');
|
||||
</script>
|
||||
{{end}}
|
||||
|
||||
Reference in New Issue
Block a user