v1.2 - collections, etc
This commit is contained in:
File diff suppressed because it is too large
Load Diff
Binary file not shown.
@@ -4,6 +4,10 @@
|
|||||||
"private": true,
|
"private": true,
|
||||||
"type": "module",
|
"type": "module",
|
||||||
"scripts": {
|
"scripts": {
|
||||||
|
"test": "node tests/run-tests.js",
|
||||||
|
"predev": "npm run test",
|
||||||
|
"prebuild": "npm run test",
|
||||||
|
"prepreview": "npm run test",
|
||||||
"dev": "vite dev",
|
"dev": "vite dev",
|
||||||
"build": "vite build",
|
"build": "vite build",
|
||||||
"preview": "vite preview"
|
"preview": "vite preview"
|
||||||
|
|||||||
+7
-1
@@ -3,7 +3,13 @@
|
|||||||
<head>
|
<head>
|
||||||
<meta charset="utf-8" />
|
<meta charset="utf-8" />
|
||||||
<meta name="viewport" content="width=device-width, initial-scale=1" />
|
<meta name="viewport" content="width=device-width, initial-scale=1" />
|
||||||
<title>Emby Home Screen Editor</title>
|
<link rel="preconnect" href="https://fonts.googleapis.com" />
|
||||||
|
<link rel="preconnect" href="https://fonts.gstatic.com" crossorigin />
|
||||||
|
<link
|
||||||
|
href="https://fonts.googleapis.com/css2?family=Inter:wght@400;500;600;700;800&display=swap"
|
||||||
|
rel="stylesheet"
|
||||||
|
/>
|
||||||
|
<title>HomeScreenPal</title>
|
||||||
%sveltekit.head%
|
%sveltekit.head%
|
||||||
</head>
|
</head>
|
||||||
<body data-sveltekit-preload-data="hover">
|
<body data-sveltekit-preload-data="hover">
|
||||||
|
|||||||
@@ -0,0 +1,635 @@
|
|||||||
|
<script>
|
||||||
|
import { createEventDispatcher } from 'svelte';
|
||||||
|
|
||||||
|
export let open = false;
|
||||||
|
export let users = [];
|
||||||
|
export let selectedUserId = null;
|
||||||
|
|
||||||
|
const dispatch = createEventDispatcher();
|
||||||
|
|
||||||
|
let sourceUserId = null;
|
||||||
|
let targetUserIds = [];
|
||||||
|
let seedType = 'Movie';
|
||||||
|
let collectionName = '';
|
||||||
|
let seedSearchTerm = '';
|
||||||
|
let seedResults = [];
|
||||||
|
let selectedSeeds = [];
|
||||||
|
let recommendations = [];
|
||||||
|
let includeSeeds = true;
|
||||||
|
let recommendationLimit = 18;
|
||||||
|
let searchBusy = false;
|
||||||
|
let previewBusy = false;
|
||||||
|
let createBusy = false;
|
||||||
|
let searchError = '';
|
||||||
|
let actionError = '';
|
||||||
|
let actionMessage = '';
|
||||||
|
let existingCollection = null;
|
||||||
|
let wasOpen = false;
|
||||||
|
|
||||||
|
$: sourceUsers = users.filter((user) => user.embyGuid);
|
||||||
|
$: if (open && !wasOpen) {
|
||||||
|
sourceUserId = sourceUsers.some((user) => user.id === selectedUserId)
|
||||||
|
? selectedUserId
|
||||||
|
: (sourceUsers[0]?.id || null);
|
||||||
|
targetUserIds = users.some((user) => user.id === selectedUserId) ? [selectedUserId] : [];
|
||||||
|
wasOpen = true;
|
||||||
|
} else if (!open && wasOpen) {
|
||||||
|
resetState();
|
||||||
|
wasOpen = false;
|
||||||
|
}
|
||||||
|
$: if (open && !sourceUsers.some((user) => user.id === sourceUserId)) {
|
||||||
|
sourceUserId = sourceUsers.some((user) => user.id === selectedUserId)
|
||||||
|
? selectedUserId
|
||||||
|
: (sourceUsers[0]?.id || null);
|
||||||
|
}
|
||||||
|
$: if (open && targetUserIds.length === 0 && selectedUserId != null) {
|
||||||
|
targetUserIds = users.some((user) => user.id === selectedUserId) ? [selectedUserId] : [];
|
||||||
|
}
|
||||||
|
$: sourceUser = users.find((user) => user.id === sourceUserId);
|
||||||
|
$: lookupType = seedType === 'Movie' ? 'Movie' : 'Series';
|
||||||
|
$: canPreview = !!sourceUser?.embyGuid && selectedSeeds.length > 0;
|
||||||
|
$: canCreate = canPreview && !!collectionName.trim() && !createBusy && recommendations.length > 0;
|
||||||
|
|
||||||
|
function close() {
|
||||||
|
dispatch('close');
|
||||||
|
}
|
||||||
|
|
||||||
|
function resetState() {
|
||||||
|
sourceUserId = null;
|
||||||
|
targetUserIds = [];
|
||||||
|
seedType = 'Movie';
|
||||||
|
collectionName = '';
|
||||||
|
seedSearchTerm = '';
|
||||||
|
seedResults = [];
|
||||||
|
selectedSeeds = [];
|
||||||
|
recommendations = [];
|
||||||
|
existingCollection = null;
|
||||||
|
includeSeeds = true;
|
||||||
|
recommendationLimit = 18;
|
||||||
|
searchBusy = false;
|
||||||
|
previewBusy = false;
|
||||||
|
createBusy = false;
|
||||||
|
searchError = '';
|
||||||
|
actionError = '';
|
||||||
|
actionMessage = '';
|
||||||
|
}
|
||||||
|
|
||||||
|
function toggleTargetUser(userId) {
|
||||||
|
if (targetUserIds.includes(userId)) {
|
||||||
|
targetUserIds = targetUserIds.filter((id) => id !== userId);
|
||||||
|
} else {
|
||||||
|
targetUserIds = [...targetUserIds, userId];
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
function addSeed(item) {
|
||||||
|
if (selectedSeeds.some((seed) => seed.id === item.id)) return;
|
||||||
|
selectedSeeds = [...selectedSeeds, item];
|
||||||
|
recommendations = [];
|
||||||
|
existingCollection = null;
|
||||||
|
actionError = '';
|
||||||
|
actionMessage = '';
|
||||||
|
}
|
||||||
|
|
||||||
|
function removeSeed(seedId) {
|
||||||
|
selectedSeeds = selectedSeeds.filter((seed) => seed.id !== seedId);
|
||||||
|
recommendations = [];
|
||||||
|
existingCollection = null;
|
||||||
|
}
|
||||||
|
|
||||||
|
async function searchSeeds() {
|
||||||
|
if (!sourceUser?.embyGuid) {
|
||||||
|
searchError = 'Select a source user with a linked Emby account.';
|
||||||
|
return;
|
||||||
|
}
|
||||||
|
|
||||||
|
if (!seedSearchTerm.trim()) {
|
||||||
|
seedResults = [];
|
||||||
|
searchError = '';
|
||||||
|
return;
|
||||||
|
}
|
||||||
|
|
||||||
|
searchBusy = true;
|
||||||
|
searchError = '';
|
||||||
|
|
||||||
|
try {
|
||||||
|
const params = new URLSearchParams({
|
||||||
|
userId: sourceUser.embyGuid,
|
||||||
|
term: seedSearchTerm.trim(),
|
||||||
|
types: lookupType,
|
||||||
|
limit: '10'
|
||||||
|
});
|
||||||
|
const response = await fetch(`/api/emby-item-search?${params.toString()}`);
|
||||||
|
const body = await response.json().catch(() => ({ items: [] }));
|
||||||
|
if (!response.ok) throw new Error(body.message || body.error || response.statusText);
|
||||||
|
seedResults = body.items || [];
|
||||||
|
} catch (err) {
|
||||||
|
searchError = err.message;
|
||||||
|
seedResults = [];
|
||||||
|
} finally {
|
||||||
|
searchBusy = false;
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
async function previewRecommendations() {
|
||||||
|
if (!canPreview) return;
|
||||||
|
|
||||||
|
previewBusy = true;
|
||||||
|
actionError = '';
|
||||||
|
actionMessage = '';
|
||||||
|
|
||||||
|
try {
|
||||||
|
const response = await fetch('/api/emby-collections', {
|
||||||
|
method: 'POST',
|
||||||
|
headers: { 'Content-Type': 'application/json' },
|
||||||
|
body: JSON.stringify({
|
||||||
|
mode: 'preview',
|
||||||
|
userId: sourceUser.embyGuid,
|
||||||
|
seedIds: selectedSeeds.map((seed) => seed.id),
|
||||||
|
name: collectionName.trim(),
|
||||||
|
limit: recommendationLimit
|
||||||
|
})
|
||||||
|
});
|
||||||
|
const body = await response.json().catch(() => ({}));
|
||||||
|
if (!response.ok) throw new Error(body.message || body.error || response.statusText);
|
||||||
|
recommendations = body.recommendations || [];
|
||||||
|
existingCollection = body.collection || null;
|
||||||
|
actionMessage = recommendations.length
|
||||||
|
? existingCollection?.updated
|
||||||
|
? `Generated ${recommendations.length} recommendations. Re-running create will update "${existingCollection.name}".`
|
||||||
|
: `Generated ${recommendations.length} recommendations.`
|
||||||
|
: 'No recommendations were returned for these seeds.';
|
||||||
|
} catch (err) {
|
||||||
|
actionError = err.message;
|
||||||
|
recommendations = [];
|
||||||
|
existingCollection = null;
|
||||||
|
} finally {
|
||||||
|
previewBusy = false;
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
async function createCollection() {
|
||||||
|
if (!canCreate) return;
|
||||||
|
|
||||||
|
createBusy = true;
|
||||||
|
actionError = '';
|
||||||
|
actionMessage = '';
|
||||||
|
|
||||||
|
try {
|
||||||
|
const response = await fetch('/api/emby-collections', {
|
||||||
|
method: 'POST',
|
||||||
|
headers: { 'Content-Type': 'application/json' },
|
||||||
|
body: JSON.stringify({
|
||||||
|
mode: 'create',
|
||||||
|
userId: sourceUser.embyGuid,
|
||||||
|
name: collectionName.trim(),
|
||||||
|
seedIds: selectedSeeds.map((seed) => seed.id),
|
||||||
|
limit: recommendationLimit,
|
||||||
|
includeSeeds
|
||||||
|
})
|
||||||
|
});
|
||||||
|
const body = await response.json().catch(() => ({}));
|
||||||
|
if (!response.ok) throw new Error(body.message || body.error || response.statusText);
|
||||||
|
|
||||||
|
dispatch('created', {
|
||||||
|
collection: body.collection,
|
||||||
|
targetUserIds,
|
||||||
|
seeds: selectedSeeds,
|
||||||
|
recommendations: body.recommendations || []
|
||||||
|
});
|
||||||
|
existingCollection = body.collection || null;
|
||||||
|
actionMessage = body.collection?.updated
|
||||||
|
? `Updated collection "${body.collection.name}".`
|
||||||
|
: `Created collection "${body.collection?.name || collectionName.trim()}".`;
|
||||||
|
} catch (err) {
|
||||||
|
actionError = err.message;
|
||||||
|
} finally {
|
||||||
|
createBusy = false;
|
||||||
|
}
|
||||||
|
}
|
||||||
|
</script>
|
||||||
|
|
||||||
|
{#if open}
|
||||||
|
<div
|
||||||
|
class="overlay"
|
||||||
|
role="button"
|
||||||
|
tabindex="0"
|
||||||
|
aria-label="Close collection builder"
|
||||||
|
on:click|self={close}
|
||||||
|
on:keydown={(event) => (event.key === 'Escape' || event.key === 'Enter') && close()}
|
||||||
|
>
|
||||||
|
<div class="modal" role="dialog" aria-modal="true" aria-label="Recommendation collection builder">
|
||||||
|
<div class="modal-header">
|
||||||
|
<div>
|
||||||
|
<h3>Recommendation Collection Builder</h3>
|
||||||
|
<p>Pick seed movies or shows, generate similar items, then create a reusable Emby collection.</p>
|
||||||
|
</div>
|
||||||
|
<button class="close-btn" on:click={close}>×</button>
|
||||||
|
</div>
|
||||||
|
|
||||||
|
<div class="modal-body">
|
||||||
|
<div class="field-grid">
|
||||||
|
<label class="field">
|
||||||
|
<span class="field-label">Source user</span>
|
||||||
|
<select bind:value={sourceUserId}>
|
||||||
|
{#each sourceUsers as user}
|
||||||
|
<option value={user.id}>{user.name}</option>
|
||||||
|
{/each}
|
||||||
|
</select>
|
||||||
|
</label>
|
||||||
|
|
||||||
|
<label class="field">
|
||||||
|
<span class="field-label">Seed type</span>
|
||||||
|
<select bind:value={seedType}>
|
||||||
|
<option value="Movie">Movies</option>
|
||||||
|
<option value="Series">Shows</option>
|
||||||
|
</select>
|
||||||
|
</label>
|
||||||
|
|
||||||
|
<label class="field span2">
|
||||||
|
<span class="field-label">Collection name</span>
|
||||||
|
<input
|
||||||
|
type="text"
|
||||||
|
bind:value={collectionName}
|
||||||
|
placeholder="Recommended for Family Movie Night"
|
||||||
|
/>
|
||||||
|
</label>
|
||||||
|
</div>
|
||||||
|
|
||||||
|
<div class="field-group">
|
||||||
|
<div class="field-label-row">
|
||||||
|
<span class="field-label">Seed lookup</span>
|
||||||
|
<label class="field-inline compact">
|
||||||
|
<input type="checkbox" bind:checked={includeSeeds} />
|
||||||
|
<span>Include seeds in final collection</span>
|
||||||
|
</label>
|
||||||
|
</div>
|
||||||
|
<div class="lookup-row">
|
||||||
|
<input
|
||||||
|
type="text"
|
||||||
|
bind:value={seedSearchTerm}
|
||||||
|
on:keydown={(event) => event.key === 'Enter' && searchSeeds()}
|
||||||
|
placeholder={`Search ${seedType === 'Movie' ? 'movies' : 'shows'}...`}
|
||||||
|
/>
|
||||||
|
<button class="btn ghost" on:click={searchSeeds} disabled={searchBusy || !sourceUser?.embyGuid}>
|
||||||
|
{searchBusy ? 'Searching…' : 'Search'}
|
||||||
|
</button>
|
||||||
|
</div>
|
||||||
|
{#if searchError}
|
||||||
|
<div class="status error">{searchError}</div>
|
||||||
|
{/if}
|
||||||
|
{#if seedResults.length}
|
||||||
|
<div class="result-list">
|
||||||
|
{#each seedResults as item}
|
||||||
|
<button class="result-card" on:click={() => addSeed(item)}>
|
||||||
|
<div class="result-name">{item.name}</div>
|
||||||
|
<div class="result-meta">{item.type}{item.year ? ` · ${item.year}` : ''}</div>
|
||||||
|
{#if item.overview}
|
||||||
|
<div class="result-overview">{item.overview}</div>
|
||||||
|
{/if}
|
||||||
|
</button>
|
||||||
|
{/each}
|
||||||
|
</div>
|
||||||
|
{/if}
|
||||||
|
</div>
|
||||||
|
|
||||||
|
<div class="field-group">
|
||||||
|
<div class="field-label-row">
|
||||||
|
<span class="field-label">Selected seeds</span>
|
||||||
|
<label class="field-inline compact">
|
||||||
|
<span>Recommendation count</span>
|
||||||
|
<select bind:value={recommendationLimit}>
|
||||||
|
<option value={12}>12</option>
|
||||||
|
<option value={18}>18</option>
|
||||||
|
<option value={24}>24</option>
|
||||||
|
<option value={30}>30</option>
|
||||||
|
</select>
|
||||||
|
</label>
|
||||||
|
</div>
|
||||||
|
{#if selectedSeeds.length}
|
||||||
|
<div class="seed-list">
|
||||||
|
{#each selectedSeeds as item}
|
||||||
|
<div class="seed-chip">
|
||||||
|
<div>
|
||||||
|
<div class="seed-name">{item.name}</div>
|
||||||
|
<div class="seed-meta">{item.type}{item.year ? ` · ${item.year}` : ''}</div>
|
||||||
|
</div>
|
||||||
|
<button class="remove-btn" on:click={() => removeSeed(item.id)}>×</button>
|
||||||
|
</div>
|
||||||
|
{/each}
|
||||||
|
</div>
|
||||||
|
{:else}
|
||||||
|
<div class="empty-note">Add at least one seed item to generate recommendations.</div>
|
||||||
|
{/if}
|
||||||
|
</div>
|
||||||
|
|
||||||
|
<div class="field-group">
|
||||||
|
<span class="field-label">Apply linked section to users</span>
|
||||||
|
<div class="target-list">
|
||||||
|
{#each users.filter((user) => user.embyGuid) as user}
|
||||||
|
<label class="target-option" class:selected={targetUserIds.includes(user.id)}>
|
||||||
|
<input
|
||||||
|
type="checkbox"
|
||||||
|
checked={targetUserIds.includes(user.id)}
|
||||||
|
on:change={() => toggleTargetUser(user.id)}
|
||||||
|
/>
|
||||||
|
<span>{user.name}</span>
|
||||||
|
<span class="target-count">{user.sections?.length || 0} sections</span>
|
||||||
|
</label>
|
||||||
|
{/each}
|
||||||
|
</div>
|
||||||
|
</div>
|
||||||
|
|
||||||
|
{#if recommendations.length}
|
||||||
|
<div class="field-group">
|
||||||
|
<span class="field-label">Recommendation preview</span>
|
||||||
|
<div class="preview-list">
|
||||||
|
{#each recommendations as item}
|
||||||
|
<div class="preview-item">
|
||||||
|
<div>
|
||||||
|
<div class="result-name">{item.name}</div>
|
||||||
|
<div class="result-meta">
|
||||||
|
{item.type}{item.year ? ` · ${item.year}` : ''} · matched {item.matchCount} seed{item.matchCount === 1 ? '' : 's'}
|
||||||
|
</div>
|
||||||
|
</div>
|
||||||
|
</div>
|
||||||
|
{/each}
|
||||||
|
</div>
|
||||||
|
</div>
|
||||||
|
{/if}
|
||||||
|
|
||||||
|
{#if existingCollection?.id}
|
||||||
|
<div class="status">Collection target: {existingCollection.name} · existing box set will be updated</div>
|
||||||
|
{/if}
|
||||||
|
|
||||||
|
{#if actionError}
|
||||||
|
<div class="status error">{actionError}</div>
|
||||||
|
{:else if actionMessage}
|
||||||
|
<div class="status ok">{actionMessage}</div>
|
||||||
|
{/if}
|
||||||
|
</div>
|
||||||
|
|
||||||
|
<div class="modal-actions">
|
||||||
|
<button class="btn ghost" on:click={previewRecommendations} disabled={!canPreview || previewBusy || createBusy}>
|
||||||
|
{previewBusy ? 'Generating…' : 'Preview Recommendations'}
|
||||||
|
</button>
|
||||||
|
<button class="btn accent" on:click={createCollection} disabled={!canCreate || previewBusy || createBusy}>
|
||||||
|
{createBusy ? (existingCollection?.updated ? 'Updating…' : 'Creating…') : (existingCollection?.updated ? 'Update Collection' : 'Create Collection')}
|
||||||
|
</button>
|
||||||
|
</div>
|
||||||
|
</div>
|
||||||
|
</div>
|
||||||
|
{/if}
|
||||||
|
|
||||||
|
<style>
|
||||||
|
.overlay {
|
||||||
|
position: fixed;
|
||||||
|
inset: 0;
|
||||||
|
background: rgba(0, 0, 0, 0.78);
|
||||||
|
display: flex;
|
||||||
|
align-items: center;
|
||||||
|
justify-content: center;
|
||||||
|
padding: 20px;
|
||||||
|
z-index: 120;
|
||||||
|
}
|
||||||
|
.modal {
|
||||||
|
width: min(980px, 100%);
|
||||||
|
max-height: 90vh;
|
||||||
|
display: flex;
|
||||||
|
flex-direction: column;
|
||||||
|
border-radius: 20px;
|
||||||
|
background: var(--surface-strong);
|
||||||
|
border: 1px solid var(--border);
|
||||||
|
box-shadow: 0 24px 60px rgba(0, 0, 0, 0.35);
|
||||||
|
}
|
||||||
|
.modal-header,
|
||||||
|
.modal-actions {
|
||||||
|
padding: 18px 20px;
|
||||||
|
border-bottom: 1px solid var(--border);
|
||||||
|
}
|
||||||
|
.modal-actions {
|
||||||
|
border-bottom: none;
|
||||||
|
border-top: 1px solid var(--border);
|
||||||
|
display: flex;
|
||||||
|
justify-content: flex-end;
|
||||||
|
gap: 10px;
|
||||||
|
}
|
||||||
|
.modal-header {
|
||||||
|
display: flex;
|
||||||
|
justify-content: space-between;
|
||||||
|
gap: 16px;
|
||||||
|
}
|
||||||
|
.modal-header h3 {
|
||||||
|
margin: 0 0 6px;
|
||||||
|
font-size: 18px;
|
||||||
|
color: var(--text);
|
||||||
|
}
|
||||||
|
.modal-header p {
|
||||||
|
margin: 0;
|
||||||
|
font-size: 13px;
|
||||||
|
line-height: 1.5;
|
||||||
|
color: var(--text-muted);
|
||||||
|
}
|
||||||
|
.close-btn,
|
||||||
|
.remove-btn {
|
||||||
|
all: unset;
|
||||||
|
cursor: pointer;
|
||||||
|
color: var(--text-muted);
|
||||||
|
font-size: 22px;
|
||||||
|
line-height: 1;
|
||||||
|
}
|
||||||
|
.modal-body {
|
||||||
|
padding: 20px;
|
||||||
|
overflow-y: auto;
|
||||||
|
display: flex;
|
||||||
|
flex-direction: column;
|
||||||
|
gap: 18px;
|
||||||
|
}
|
||||||
|
.field-grid {
|
||||||
|
display: grid;
|
||||||
|
grid-template-columns: repeat(2, minmax(0, 1fr));
|
||||||
|
gap: 12px;
|
||||||
|
}
|
||||||
|
.field,
|
||||||
|
.field-group {
|
||||||
|
display: flex;
|
||||||
|
flex-direction: column;
|
||||||
|
gap: 8px;
|
||||||
|
}
|
||||||
|
.field.span2 {
|
||||||
|
grid-column: span 2;
|
||||||
|
}
|
||||||
|
.field-label {
|
||||||
|
font-size: 11px;
|
||||||
|
font-weight: 700;
|
||||||
|
text-transform: uppercase;
|
||||||
|
letter-spacing: 0.08em;
|
||||||
|
color: var(--text-muted);
|
||||||
|
}
|
||||||
|
.field-label-row {
|
||||||
|
display: flex;
|
||||||
|
justify-content: space-between;
|
||||||
|
align-items: center;
|
||||||
|
gap: 12px;
|
||||||
|
}
|
||||||
|
.field-inline {
|
||||||
|
display: flex;
|
||||||
|
align-items: center;
|
||||||
|
gap: 8px;
|
||||||
|
color: var(--text);
|
||||||
|
font-size: 13px;
|
||||||
|
}
|
||||||
|
.field-inline.compact {
|
||||||
|
font-size: 12px;
|
||||||
|
}
|
||||||
|
input,
|
||||||
|
select {
|
||||||
|
width: 100%;
|
||||||
|
box-sizing: border-box;
|
||||||
|
padding: 10px 12px;
|
||||||
|
border-radius: 12px;
|
||||||
|
border: 1px solid var(--border);
|
||||||
|
background: #0b0f14;
|
||||||
|
color: var(--text);
|
||||||
|
font-size: 13px;
|
||||||
|
font-family: inherit;
|
||||||
|
}
|
||||||
|
input:focus,
|
||||||
|
select:focus {
|
||||||
|
outline: none;
|
||||||
|
border-color: var(--border-strong);
|
||||||
|
}
|
||||||
|
.lookup-row {
|
||||||
|
display: grid;
|
||||||
|
grid-template-columns: minmax(0, 1fr) auto;
|
||||||
|
gap: 10px;
|
||||||
|
}
|
||||||
|
.result-list,
|
||||||
|
.preview-list,
|
||||||
|
.target-list,
|
||||||
|
.seed-list {
|
||||||
|
display: flex;
|
||||||
|
flex-direction: column;
|
||||||
|
gap: 8px;
|
||||||
|
}
|
||||||
|
.result-card,
|
||||||
|
.preview-item,
|
||||||
|
.seed-chip,
|
||||||
|
.target-option {
|
||||||
|
border: 1px solid var(--border);
|
||||||
|
border-radius: 14px;
|
||||||
|
background: var(--surface);
|
||||||
|
padding: 12px 14px;
|
||||||
|
}
|
||||||
|
.result-card {
|
||||||
|
all: unset;
|
||||||
|
cursor: pointer;
|
||||||
|
border: 1px solid var(--border);
|
||||||
|
border-radius: 14px;
|
||||||
|
background: var(--surface);
|
||||||
|
padding: 12px 14px;
|
||||||
|
transition: border-color 0.12s ease, background 0.12s ease;
|
||||||
|
}
|
||||||
|
.result-card:hover,
|
||||||
|
.target-option:hover {
|
||||||
|
border-color: var(--border-strong);
|
||||||
|
background: var(--surface-hover);
|
||||||
|
}
|
||||||
|
.result-name,
|
||||||
|
.seed-name {
|
||||||
|
font-size: 14px;
|
||||||
|
font-weight: 600;
|
||||||
|
color: var(--text);
|
||||||
|
}
|
||||||
|
.result-meta,
|
||||||
|
.seed-meta,
|
||||||
|
.target-count {
|
||||||
|
font-size: 12px;
|
||||||
|
color: var(--text-muted);
|
||||||
|
}
|
||||||
|
.result-overview {
|
||||||
|
margin-top: 6px;
|
||||||
|
font-size: 12px;
|
||||||
|
color: var(--text-muted);
|
||||||
|
line-height: 1.45;
|
||||||
|
}
|
||||||
|
.seed-chip,
|
||||||
|
.target-option {
|
||||||
|
display: flex;
|
||||||
|
justify-content: space-between;
|
||||||
|
align-items: center;
|
||||||
|
gap: 12px;
|
||||||
|
}
|
||||||
|
.target-option {
|
||||||
|
cursor: pointer;
|
||||||
|
}
|
||||||
|
.target-option.selected {
|
||||||
|
background: var(--surface-active);
|
||||||
|
border-color: var(--border-strong);
|
||||||
|
}
|
||||||
|
.empty-note {
|
||||||
|
padding: 14px;
|
||||||
|
border-radius: 14px;
|
||||||
|
background: var(--surface);
|
||||||
|
border: 1px dashed var(--border);
|
||||||
|
color: var(--text-muted);
|
||||||
|
font-size: 13px;
|
||||||
|
}
|
||||||
|
.status {
|
||||||
|
padding: 10px 12px;
|
||||||
|
border-radius: 14px;
|
||||||
|
font-size: 12px;
|
||||||
|
background: var(--surface);
|
||||||
|
border: 1px solid var(--border);
|
||||||
|
color: var(--text-muted);
|
||||||
|
}
|
||||||
|
.status.ok {
|
||||||
|
border-color: rgba(34, 197, 94, 0.3);
|
||||||
|
color: #86efac;
|
||||||
|
}
|
||||||
|
.status.error {
|
||||||
|
border-color: rgba(239, 68, 68, 0.3);
|
||||||
|
color: #fca5a5;
|
||||||
|
}
|
||||||
|
.btn {
|
||||||
|
all: unset;
|
||||||
|
cursor: pointer;
|
||||||
|
padding: 9px 16px;
|
||||||
|
border-radius: 12px;
|
||||||
|
font-size: 13px;
|
||||||
|
font-weight: 600;
|
||||||
|
font-family: inherit;
|
||||||
|
border: 1px solid transparent;
|
||||||
|
}
|
||||||
|
.btn.ghost {
|
||||||
|
color: #d9e7f8;
|
||||||
|
border-color: var(--border);
|
||||||
|
background: var(--surface);
|
||||||
|
}
|
||||||
|
.btn.accent {
|
||||||
|
background: var(--accent);
|
||||||
|
color: #fff;
|
||||||
|
}
|
||||||
|
.btn:disabled {
|
||||||
|
opacity: 0.45;
|
||||||
|
cursor: default;
|
||||||
|
}
|
||||||
|
|
||||||
|
@media (max-width: 720px) {
|
||||||
|
.field-grid,
|
||||||
|
.lookup-row {
|
||||||
|
grid-template-columns: 1fr;
|
||||||
|
}
|
||||||
|
.modal-header,
|
||||||
|
.modal-body,
|
||||||
|
.modal-actions {
|
||||||
|
padding: 16px;
|
||||||
|
}
|
||||||
|
.modal-actions {
|
||||||
|
flex-direction: column;
|
||||||
|
}
|
||||||
|
.field.span2 {
|
||||||
|
grid-column: span 1;
|
||||||
|
}
|
||||||
|
}
|
||||||
|
</style>
|
||||||
@@ -0,0 +1,857 @@
|
|||||||
|
<script>
|
||||||
|
import { createEventDispatcher } from 'svelte';
|
||||||
|
import Icon from '$lib/Icon.svelte';
|
||||||
|
import { RECOMMENDATION_PROFILES } from '$lib/collection-tools.js';
|
||||||
|
|
||||||
|
export let users = [];
|
||||||
|
export let selectedUserId = null;
|
||||||
|
|
||||||
|
const dispatch = createEventDispatcher();
|
||||||
|
|
||||||
|
let sourceUserId = null;
|
||||||
|
let targetUserIds = [];
|
||||||
|
let collectionName = '';
|
||||||
|
let selectedSeeds = [];
|
||||||
|
let recommendations = [];
|
||||||
|
let includeSeeds = true;
|
||||||
|
let recommendationLimit = 18;
|
||||||
|
let recommendationProfile = 'balanced';
|
||||||
|
let recentItems = [];
|
||||||
|
let recentBusy = false;
|
||||||
|
let recentError = '';
|
||||||
|
let movieSearchTerm = '';
|
||||||
|
let showSearchTerm = '';
|
||||||
|
let movieResults = [];
|
||||||
|
let showResults = [];
|
||||||
|
let movieSearchBusy = false;
|
||||||
|
let showSearchBusy = false;
|
||||||
|
let movieSearchError = '';
|
||||||
|
let showSearchError = '';
|
||||||
|
let previewBusy = false;
|
||||||
|
let createBusy = false;
|
||||||
|
let actionError = '';
|
||||||
|
let actionMessage = '';
|
||||||
|
let existingCollection = null;
|
||||||
|
|
||||||
|
$: sourceUsers = users.filter((user) => user.embyGuid);
|
||||||
|
$: profileOptions = Object.values(RECOMMENDATION_PROFILES);
|
||||||
|
$: selectedProfile = RECOMMENDATION_PROFILES[recommendationProfile] || RECOMMENDATION_PROFILES.balanced;
|
||||||
|
$: if (!sourceUsers.some((user) => user.id === sourceUserId)) {
|
||||||
|
sourceUserId = sourceUsers.some((user) => user.id === selectedUserId)
|
||||||
|
? selectedUserId
|
||||||
|
: (sourceUsers[0]?.id || null);
|
||||||
|
}
|
||||||
|
$: sourceUser = users.find((user) => user.id === sourceUserId);
|
||||||
|
$: canPreview = !!sourceUser?.embyGuid && selectedSeeds.length > 0;
|
||||||
|
$: canCreate = canPreview && !!collectionName.trim() && recommendations.length > 0 && !createBusy;
|
||||||
|
|
||||||
|
$: if (sourceUser?.embyGuid) {
|
||||||
|
loadRecentActivity(sourceUser.embyGuid);
|
||||||
|
}
|
||||||
|
|
||||||
|
let lastRecentUserId = '';
|
||||||
|
|
||||||
|
async function loadRecentActivity(embyGuid) {
|
||||||
|
if (!embyGuid || lastRecentUserId === embyGuid) return;
|
||||||
|
lastRecentUserId = embyGuid;
|
||||||
|
recentBusy = true;
|
||||||
|
recentError = '';
|
||||||
|
recentItems = [];
|
||||||
|
|
||||||
|
try {
|
||||||
|
const params = new URLSearchParams({ embyGuid });
|
||||||
|
const response = await fetch(`/api/emby-user-context?${params.toString()}`);
|
||||||
|
const body = await response.json().catch(() => ({}));
|
||||||
|
if (!response.ok) throw new Error(body.message || body.error || response.statusText);
|
||||||
|
const activity = Array.isArray(body.recentlyPlayed) ? body.recentlyPlayed : [];
|
||||||
|
recentItems = activity.filter((item) => item.type === 'Movie' || item.type === 'Series');
|
||||||
|
} catch (error) {
|
||||||
|
recentError = error.message;
|
||||||
|
} finally {
|
||||||
|
recentBusy = false;
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
function toggleTargetUser(userId) {
|
||||||
|
if (targetUserIds.includes(userId)) {
|
||||||
|
targetUserIds = targetUserIds.filter((id) => id !== userId);
|
||||||
|
} else {
|
||||||
|
targetUserIds = [...targetUserIds, userId];
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
function addSeed(item) {
|
||||||
|
if (!item?.id || selectedSeeds.some((seed) => seed.id === item.id)) return;
|
||||||
|
selectedSeeds = [...selectedSeeds, item];
|
||||||
|
recommendations = [];
|
||||||
|
existingCollection = null;
|
||||||
|
actionError = '';
|
||||||
|
actionMessage = '';
|
||||||
|
}
|
||||||
|
|
||||||
|
function removeSeed(seedId) {
|
||||||
|
selectedSeeds = selectedSeeds.filter((seed) => seed.id !== seedId);
|
||||||
|
recommendations = [];
|
||||||
|
existingCollection = null;
|
||||||
|
}
|
||||||
|
|
||||||
|
function handleProfileChange() {
|
||||||
|
recommendations = [];
|
||||||
|
existingCollection = null;
|
||||||
|
actionError = '';
|
||||||
|
actionMessage = '';
|
||||||
|
}
|
||||||
|
|
||||||
|
function logPreviewDiagnostics(requestPayload, responseBody) {
|
||||||
|
if (typeof window === 'undefined') return;
|
||||||
|
|
||||||
|
const diagnostics = responseBody?.diagnostics || {};
|
||||||
|
const perSeed = Array.isArray(diagnostics.perSeed) ? diagnostics.perSeed : [];
|
||||||
|
|
||||||
|
console.groupCollapsed('[Collections] Recommendation preview');
|
||||||
|
console.info('Request', requestPayload);
|
||||||
|
console.info('Response summary', {
|
||||||
|
profile: responseBody?.profile,
|
||||||
|
recommendationCount: (responseBody?.recommendations || []).length,
|
||||||
|
libraryItemCount: diagnostics.libraryItemCount ?? 0,
|
||||||
|
watchedItemCount: diagnostics.watchedItemCount ?? 0,
|
||||||
|
seedCount: diagnostics.seedCount ?? 0,
|
||||||
|
embyCandidates: diagnostics.embyCandidates ?? 0,
|
||||||
|
tmdbEnabled: diagnostics.tmdbEnabled ?? false,
|
||||||
|
tmdbCandidates: diagnostics.tmdbCandidates ?? 0,
|
||||||
|
tmdbResolved: diagnostics.tmdbResolved ?? 0,
|
||||||
|
uniqueCandidateCount: diagnostics.uniqueCandidateCount ?? 0,
|
||||||
|
excludedSeedCandidateCount: diagnostics.excludedSeedCandidateCount ?? 0
|
||||||
|
});
|
||||||
|
console.info('Seeds returned by API', responseBody?.seeds || []);
|
||||||
|
if ((diagnostics.sampleExcludedSeedCandidates || []).length) {
|
||||||
|
console.warn('Candidates dropped because they resolved back onto the seed items', diagnostics.sampleExcludedSeedCandidates);
|
||||||
|
}
|
||||||
|
if (perSeed.length) {
|
||||||
|
console.table(
|
||||||
|
perSeed.map((seed) => ({
|
||||||
|
seed: seed.seedName,
|
||||||
|
type: seed.seedType,
|
||||||
|
year: seed.seedYear,
|
||||||
|
tmdbProviderId: seed.providerTmdbId || '',
|
||||||
|
tmdbMatchId: seed.tmdbMatch?.tmdbId || '',
|
||||||
|
tmdbMediaType: seed.tmdbMatch?.mediaType || '',
|
||||||
|
similar: seed.similarCount ?? 0,
|
||||||
|
recommendations: seed.recommendationCount ?? 0,
|
||||||
|
localSimilar: seed.localSimilarCount ?? 0,
|
||||||
|
localRecommendations: seed.localRecommendationCount ?? 0,
|
||||||
|
error: seed.error || ''
|
||||||
|
}))
|
||||||
|
);
|
||||||
|
console.info('Per-seed detail', perSeed);
|
||||||
|
}
|
||||||
|
if ((diagnostics.discoverQueries || []).length) {
|
||||||
|
console.table(
|
||||||
|
diagnostics.discoverQueries.map((query) => ({
|
||||||
|
query: query.label,
|
||||||
|
tmdbCount: query.tmdbCount,
|
||||||
|
localCount: query.localCount,
|
||||||
|
error: query.error || ''
|
||||||
|
}))
|
||||||
|
);
|
||||||
|
console.info('Discover query detail', diagnostics.discoverQueries);
|
||||||
|
}
|
||||||
|
if ((responseBody?.recommendations || []).length) {
|
||||||
|
console.info('Recommendations', responseBody.recommendations);
|
||||||
|
}
|
||||||
|
console.groupEnd();
|
||||||
|
}
|
||||||
|
|
||||||
|
async function searchItems(type) {
|
||||||
|
const searchTerm = type === 'Movie' ? movieSearchTerm.trim() : showSearchTerm.trim();
|
||||||
|
if (!sourceUser?.embyGuid) {
|
||||||
|
if (type === 'Movie') movieSearchError = 'Select a source user first.';
|
||||||
|
else showSearchError = 'Select a source user first.';
|
||||||
|
return;
|
||||||
|
}
|
||||||
|
if (!searchTerm) {
|
||||||
|
if (type === 'Movie') movieResults = [];
|
||||||
|
else showResults = [];
|
||||||
|
return;
|
||||||
|
}
|
||||||
|
|
||||||
|
if (type === 'Movie') {
|
||||||
|
movieSearchBusy = true;
|
||||||
|
movieSearchError = '';
|
||||||
|
} else {
|
||||||
|
showSearchBusy = true;
|
||||||
|
showSearchError = '';
|
||||||
|
}
|
||||||
|
|
||||||
|
try {
|
||||||
|
const params = new URLSearchParams({
|
||||||
|
userId: sourceUser.embyGuid,
|
||||||
|
term: searchTerm,
|
||||||
|
types: type,
|
||||||
|
limit: '10'
|
||||||
|
});
|
||||||
|
const response = await fetch(`/api/emby-item-search?${params.toString()}`);
|
||||||
|
const body = await response.json().catch(() => ({ items: [] }));
|
||||||
|
if (!response.ok) throw new Error(body.message || body.error || response.statusText);
|
||||||
|
if (type === 'Movie') movieResults = body.items || [];
|
||||||
|
else showResults = body.items || [];
|
||||||
|
} catch (error) {
|
||||||
|
if (type === 'Movie') {
|
||||||
|
movieSearchError = error.message;
|
||||||
|
movieResults = [];
|
||||||
|
} else {
|
||||||
|
showSearchError = error.message;
|
||||||
|
showResults = [];
|
||||||
|
}
|
||||||
|
} finally {
|
||||||
|
if (type === 'Movie') movieSearchBusy = false;
|
||||||
|
else showSearchBusy = false;
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
async function previewRecommendations() {
|
||||||
|
if (!canPreview) return;
|
||||||
|
previewBusy = true;
|
||||||
|
actionError = '';
|
||||||
|
actionMessage = '';
|
||||||
|
const requestPayload = {
|
||||||
|
mode: 'preview',
|
||||||
|
userId: sourceUser.embyGuid,
|
||||||
|
seedIds: selectedSeeds.map((seed) => seed.id),
|
||||||
|
seeds: selectedSeeds.map((seed) => ({
|
||||||
|
id: seed.id,
|
||||||
|
name: seed.name,
|
||||||
|
type: seed.type,
|
||||||
|
year: seed.year
|
||||||
|
})),
|
||||||
|
name: collectionName.trim(),
|
||||||
|
limit: recommendationLimit,
|
||||||
|
profile: recommendationProfile
|
||||||
|
};
|
||||||
|
|
||||||
|
try {
|
||||||
|
const response = await fetch('/api/emby-collections', {
|
||||||
|
method: 'POST',
|
||||||
|
headers: { 'Content-Type': 'application/json' },
|
||||||
|
body: JSON.stringify(requestPayload)
|
||||||
|
});
|
||||||
|
const body = await response.json().catch(() => ({}));
|
||||||
|
if (!response.ok) throw new Error(body.message || body.error || response.statusText);
|
||||||
|
logPreviewDiagnostics(requestPayload, body);
|
||||||
|
recommendations = body.recommendations || [];
|
||||||
|
existingCollection = body.collection || null;
|
||||||
|
if (recommendations.length) {
|
||||||
|
actionMessage = existingCollection?.updated
|
||||||
|
? `Generated ${recommendations.length} recommendations. Re-running create will update "${existingCollection.name}".`
|
||||||
|
: `Generated ${recommendations.length} recommendations.`;
|
||||||
|
} else if (body?.diagnostics?.tmdbEnabled && body?.diagnostics?.tmdbCandidates > 0 && body?.diagnostics?.tmdbResolved === 0) {
|
||||||
|
actionMessage = 'TMDB found similar titles, but none of them matched items currently in your Emby library.';
|
||||||
|
} else {
|
||||||
|
actionMessage = 'No recommendations were returned for these seeds.';
|
||||||
|
}
|
||||||
|
} catch (error) {
|
||||||
|
console.error('[Collections] Recommendation preview failed', {
|
||||||
|
request: requestPayload,
|
||||||
|
error
|
||||||
|
});
|
||||||
|
actionError = error.message;
|
||||||
|
recommendations = [];
|
||||||
|
existingCollection = null;
|
||||||
|
} finally {
|
||||||
|
previewBusy = false;
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
async function createCollection() {
|
||||||
|
if (!canCreate) return;
|
||||||
|
createBusy = true;
|
||||||
|
actionError = '';
|
||||||
|
actionMessage = '';
|
||||||
|
|
||||||
|
try {
|
||||||
|
const response = await fetch('/api/emby-collections', {
|
||||||
|
method: 'POST',
|
||||||
|
headers: { 'Content-Type': 'application/json' },
|
||||||
|
body: JSON.stringify({
|
||||||
|
mode: 'create',
|
||||||
|
userId: sourceUser.embyGuid,
|
||||||
|
name: collectionName.trim(),
|
||||||
|
seedIds: selectedSeeds.map((seed) => seed.id),
|
||||||
|
limit: recommendationLimit,
|
||||||
|
includeSeeds,
|
||||||
|
profile: recommendationProfile
|
||||||
|
})
|
||||||
|
});
|
||||||
|
const body = await response.json().catch(() => ({}));
|
||||||
|
if (!response.ok) throw new Error(body.message || body.error || response.statusText);
|
||||||
|
|
||||||
|
dispatch('created', {
|
||||||
|
collection: body.collection,
|
||||||
|
targetUserIds,
|
||||||
|
seeds: selectedSeeds,
|
||||||
|
recommendations: body.recommendations || []
|
||||||
|
});
|
||||||
|
existingCollection = body.collection || null;
|
||||||
|
actionMessage = body.collection?.updated
|
||||||
|
? `Updated collection "${body.collection.name}".`
|
||||||
|
: `Created collection "${body.collection?.name || collectionName.trim()}".`;
|
||||||
|
} catch (error) {
|
||||||
|
actionError = error.message;
|
||||||
|
} finally {
|
||||||
|
createBusy = false;
|
||||||
|
}
|
||||||
|
}
|
||||||
|
</script>
|
||||||
|
|
||||||
|
<div class="collection-page">
|
||||||
|
<div class="page-header">
|
||||||
|
<div>
|
||||||
|
<div class="page-title-row">
|
||||||
|
<span class="page-icon"><Icon name="collections" size={18} /></span>
|
||||||
|
<h2>Collections</h2>
|
||||||
|
</div>
|
||||||
|
<p class="page-copy">Build recommendation collections from recent activity, seed movies, and seed shows.</p>
|
||||||
|
</div>
|
||||||
|
<div class="page-actions">
|
||||||
|
<label class="field-inline compact field-select">
|
||||||
|
<span>Profile</span>
|
||||||
|
<select bind:value={recommendationProfile} on:change={handleProfileChange}>
|
||||||
|
{#each profileOptions as profile}
|
||||||
|
<option value={profile.id}>{profile.label}</option>
|
||||||
|
{/each}
|
||||||
|
</select>
|
||||||
|
</label>
|
||||||
|
<label class="field-inline compact">
|
||||||
|
<input type="checkbox" bind:checked={includeSeeds} />
|
||||||
|
<span>Include seeds in collection</span>
|
||||||
|
</label>
|
||||||
|
<label class="field-inline compact">
|
||||||
|
<span>Recommendation count</span>
|
||||||
|
<select bind:value={recommendationLimit}>
|
||||||
|
<option value={12}>12</option>
|
||||||
|
<option value={18}>18</option>
|
||||||
|
<option value={24}>24</option>
|
||||||
|
<option value={30}>30</option>
|
||||||
|
</select>
|
||||||
|
</label>
|
||||||
|
</div>
|
||||||
|
</div>
|
||||||
|
|
||||||
|
<div class="builder-grid">
|
||||||
|
<div class="builder-main">
|
||||||
|
<div class="builder-card hero-card">
|
||||||
|
<div class="field-grid">
|
||||||
|
<label class="field">
|
||||||
|
<span class="field-label">Source user</span>
|
||||||
|
<select bind:value={sourceUserId}>
|
||||||
|
{#each sourceUsers as user}
|
||||||
|
<option value={user.id}>{user.name}</option>
|
||||||
|
{/each}
|
||||||
|
</select>
|
||||||
|
</label>
|
||||||
|
<label class="field">
|
||||||
|
<span class="field-label">Collection name</span>
|
||||||
|
<input type="text" bind:value={collectionName} placeholder="Recommended for Family Night" />
|
||||||
|
</label>
|
||||||
|
</div>
|
||||||
|
<p class="profile-note">{selectedProfile.description}</p>
|
||||||
|
</div>
|
||||||
|
|
||||||
|
<div class="seed-grid">
|
||||||
|
<section class="builder-card">
|
||||||
|
<div class="card-header">
|
||||||
|
<div class="card-title-row">
|
||||||
|
<Icon name="activity" size={16} />
|
||||||
|
<h3>Watched Activity</h3>
|
||||||
|
</div>
|
||||||
|
<span class="card-copy">Quick-add from recent movies and shows.</span>
|
||||||
|
</div>
|
||||||
|
{#if recentError}
|
||||||
|
<div class="status error">{recentError}</div>
|
||||||
|
{:else if recentBusy}
|
||||||
|
<div class="status">Loading recent activity…</div>
|
||||||
|
{:else if recentItems.length}
|
||||||
|
<div class="result-list">
|
||||||
|
{#each recentItems as item}
|
||||||
|
<button class="result-card" on:click={() => addSeed(item)}>
|
||||||
|
<div class="result-name">{item.name}</div>
|
||||||
|
<div class="result-meta">{item.type}{item.datePlayed ? ` · ${new Date(item.datePlayed).toLocaleDateString()}` : ''}</div>
|
||||||
|
</button>
|
||||||
|
{/each}
|
||||||
|
</div>
|
||||||
|
{:else}
|
||||||
|
<div class="empty-note">No recent movie or show activity was available for this user.</div>
|
||||||
|
{/if}
|
||||||
|
</section>
|
||||||
|
|
||||||
|
<section class="builder-card">
|
||||||
|
<div class="card-header">
|
||||||
|
<div class="card-title-row">
|
||||||
|
<Icon name="search" size={16} />
|
||||||
|
<h3>Seed Movies</h3>
|
||||||
|
</div>
|
||||||
|
<span class="card-copy">Search Emby and add movie seeds.</span>
|
||||||
|
</div>
|
||||||
|
<div class="lookup-row">
|
||||||
|
<input
|
||||||
|
type="text"
|
||||||
|
bind:value={movieSearchTerm}
|
||||||
|
on:keydown={(event) => event.key === 'Enter' && searchItems('Movie')}
|
||||||
|
placeholder="Search movies..."
|
||||||
|
/>
|
||||||
|
<button class="btn ghost" on:click={() => searchItems('Movie')} disabled={movieSearchBusy}>
|
||||||
|
{movieSearchBusy ? 'Searching…' : 'Search'}
|
||||||
|
</button>
|
||||||
|
</div>
|
||||||
|
{#if movieSearchError}
|
||||||
|
<div class="status error">{movieSearchError}</div>
|
||||||
|
{/if}
|
||||||
|
{#if movieResults.length}
|
||||||
|
<div class="result-list">
|
||||||
|
{#each movieResults as item}
|
||||||
|
<button class="result-card" on:click={() => addSeed(item)}>
|
||||||
|
<div class="result-name">{item.name}</div>
|
||||||
|
<div class="result-meta">{item.year ? `${item.year} · ` : ''}{item.type}</div>
|
||||||
|
</button>
|
||||||
|
{/each}
|
||||||
|
</div>
|
||||||
|
{/if}
|
||||||
|
</section>
|
||||||
|
|
||||||
|
<section class="builder-card">
|
||||||
|
<div class="card-header">
|
||||||
|
<div class="card-title-row">
|
||||||
|
<Icon name="search" size={16} />
|
||||||
|
<h3>Seed Shows</h3>
|
||||||
|
</div>
|
||||||
|
<span class="card-copy">Search Emby and add TV seeds.</span>
|
||||||
|
</div>
|
||||||
|
<div class="lookup-row">
|
||||||
|
<input
|
||||||
|
type="text"
|
||||||
|
bind:value={showSearchTerm}
|
||||||
|
on:keydown={(event) => event.key === 'Enter' && searchItems('Series')}
|
||||||
|
placeholder="Search shows..."
|
||||||
|
/>
|
||||||
|
<button class="btn ghost" on:click={() => searchItems('Series')} disabled={showSearchBusy}>
|
||||||
|
{showSearchBusy ? 'Searching…' : 'Search'}
|
||||||
|
</button>
|
||||||
|
</div>
|
||||||
|
{#if showSearchError}
|
||||||
|
<div class="status error">{showSearchError}</div>
|
||||||
|
{/if}
|
||||||
|
{#if showResults.length}
|
||||||
|
<div class="result-list">
|
||||||
|
{#each showResults as item}
|
||||||
|
<button class="result-card" on:click={() => addSeed(item)}>
|
||||||
|
<div class="result-name">{item.name}</div>
|
||||||
|
<div class="result-meta">{item.year ? `${item.year} · ` : ''}{item.type}</div>
|
||||||
|
</button>
|
||||||
|
{/each}
|
||||||
|
</div>
|
||||||
|
{/if}
|
||||||
|
</section>
|
||||||
|
</div>
|
||||||
|
</div>
|
||||||
|
|
||||||
|
<div class="builder-side">
|
||||||
|
<section class="builder-card sticky-card">
|
||||||
|
<div class="card-header">
|
||||||
|
<div class="card-title-row">
|
||||||
|
<Icon name="spark" size={16} />
|
||||||
|
<h3>Selected Seeds</h3>
|
||||||
|
</div>
|
||||||
|
<span class="card-copy">Mix watched activity with manual seeds.</span>
|
||||||
|
</div>
|
||||||
|
{#if selectedSeeds.length}
|
||||||
|
<div class="seed-list">
|
||||||
|
{#each selectedSeeds as item}
|
||||||
|
<div class="seed-chip">
|
||||||
|
<div>
|
||||||
|
<div class="result-name">{item.name}</div>
|
||||||
|
<div class="result-meta">{item.type}{item.year ? ` · ${item.year}` : ''}</div>
|
||||||
|
</div>
|
||||||
|
<button class="remove-btn" on:click={() => removeSeed(item.id)}>×</button>
|
||||||
|
</div>
|
||||||
|
{/each}
|
||||||
|
</div>
|
||||||
|
{:else}
|
||||||
|
<div class="empty-note">Add at least one seed to generate recommendations.</div>
|
||||||
|
{/if}
|
||||||
|
|
||||||
|
<div class="divider"></div>
|
||||||
|
|
||||||
|
<div class="card-header compact">
|
||||||
|
<div class="card-title-row">
|
||||||
|
<Icon name="boxset" size={16} />
|
||||||
|
<h3>Apply Section To</h3>
|
||||||
|
</div>
|
||||||
|
</div>
|
||||||
|
<div class="target-list">
|
||||||
|
{#each users.filter((user) => user.embyGuid) as user}
|
||||||
|
<label class="target-option" class:selected={targetUserIds.includes(user.id)}>
|
||||||
|
<input
|
||||||
|
type="checkbox"
|
||||||
|
checked={targetUserIds.includes(user.id)}
|
||||||
|
on:change={() => toggleTargetUser(user.id)}
|
||||||
|
/>
|
||||||
|
<span>{user.name}</span>
|
||||||
|
<span class="target-count">{user.sections?.length || 0} sections</span>
|
||||||
|
</label>
|
||||||
|
{/each}
|
||||||
|
</div>
|
||||||
|
|
||||||
|
{#if recommendations.length}
|
||||||
|
<div class="divider"></div>
|
||||||
|
<div class="card-header compact">
|
||||||
|
<div class="card-title-row">
|
||||||
|
<Icon name="collections" size={16} />
|
||||||
|
<h3>Recommendation Preview</h3>
|
||||||
|
</div>
|
||||||
|
</div>
|
||||||
|
<div class="preview-list">
|
||||||
|
{#each recommendations as item}
|
||||||
|
<div class="preview-item">
|
||||||
|
<div class="result-name">{item.name}</div>
|
||||||
|
<div class="result-meta">
|
||||||
|
{item.type}{item.year ? ` · ${item.year}` : ''} · matched {item.matchCount}
|
||||||
|
{#if item.styleScore !== null && item.styleScore !== undefined}
|
||||||
|
· style {item.styleScore}
|
||||||
|
{/if}
|
||||||
|
{#if item.qualityScore !== null && item.qualityScore !== undefined}
|
||||||
|
· quality {item.qualityScore}
|
||||||
|
{/if}
|
||||||
|
</div>
|
||||||
|
</div>
|
||||||
|
{/each}
|
||||||
|
</div>
|
||||||
|
{/if}
|
||||||
|
|
||||||
|
{#if existingCollection?.id}
|
||||||
|
<div class="divider"></div>
|
||||||
|
<div class="status">
|
||||||
|
Collection target: {existingCollection.name} · existing box set will be updated
|
||||||
|
</div>
|
||||||
|
{/if}
|
||||||
|
|
||||||
|
{#if actionError}
|
||||||
|
<div class="status error">{actionError}</div>
|
||||||
|
{:else if actionMessage}
|
||||||
|
<div class="status ok">{actionMessage}</div>
|
||||||
|
{/if}
|
||||||
|
|
||||||
|
<div class="action-row">
|
||||||
|
<button class="btn ghost" on:click={previewRecommendations} disabled={!canPreview || previewBusy || createBusy}>
|
||||||
|
{previewBusy ? 'Generating…' : 'Preview Recommendations'}
|
||||||
|
</button>
|
||||||
|
<button class="btn accent" on:click={createCollection} disabled={!canCreate || previewBusy || createBusy}>
|
||||||
|
{createBusy ? (existingCollection?.updated ? 'Updating…' : 'Creating…') : (existingCollection?.updated ? 'Update Collection' : 'Create Collection')}
|
||||||
|
</button>
|
||||||
|
</div>
|
||||||
|
</section>
|
||||||
|
</div>
|
||||||
|
</div>
|
||||||
|
</div>
|
||||||
|
|
||||||
|
<style>
|
||||||
|
.collection-page {
|
||||||
|
display: flex;
|
||||||
|
flex-direction: column;
|
||||||
|
gap: 18px;
|
||||||
|
}
|
||||||
|
.page-header {
|
||||||
|
display: flex;
|
||||||
|
justify-content: space-between;
|
||||||
|
gap: 16px;
|
||||||
|
align-items: flex-start;
|
||||||
|
}
|
||||||
|
.page-title-row,
|
||||||
|
.card-title-row {
|
||||||
|
display: flex;
|
||||||
|
align-items: center;
|
||||||
|
gap: 10px;
|
||||||
|
}
|
||||||
|
h2,
|
||||||
|
h3 {
|
||||||
|
margin: 0;
|
||||||
|
color: var(--text);
|
||||||
|
}
|
||||||
|
h2 {
|
||||||
|
font-size: 26px;
|
||||||
|
font-weight: 800;
|
||||||
|
}
|
||||||
|
h3 {
|
||||||
|
font-size: 15px;
|
||||||
|
font-weight: 700;
|
||||||
|
}
|
||||||
|
.page-copy,
|
||||||
|
.card-copy {
|
||||||
|
margin: 6px 0 0;
|
||||||
|
font-size: 13px;
|
||||||
|
line-height: 1.5;
|
||||||
|
color: var(--text-muted);
|
||||||
|
}
|
||||||
|
.page-actions {
|
||||||
|
display: flex;
|
||||||
|
gap: 12px;
|
||||||
|
flex-wrap: wrap;
|
||||||
|
}
|
||||||
|
.field-select {
|
||||||
|
min-width: 220px;
|
||||||
|
}
|
||||||
|
.builder-grid {
|
||||||
|
display: grid;
|
||||||
|
grid-template-columns: minmax(0, 1.7fr) minmax(320px, 0.9fr);
|
||||||
|
gap: 18px;
|
||||||
|
align-items: start;
|
||||||
|
}
|
||||||
|
.builder-main,
|
||||||
|
.builder-side,
|
||||||
|
.seed-grid {
|
||||||
|
display: flex;
|
||||||
|
flex-direction: column;
|
||||||
|
gap: 18px;
|
||||||
|
}
|
||||||
|
.seed-grid {
|
||||||
|
display: grid;
|
||||||
|
grid-template-columns: repeat(3, minmax(0, 1fr));
|
||||||
|
}
|
||||||
|
.builder-card {
|
||||||
|
background: var(--surface);
|
||||||
|
border: 1px solid var(--border);
|
||||||
|
border-radius: 18px;
|
||||||
|
padding: 18px;
|
||||||
|
box-shadow: 0 10px 28px rgba(0, 0, 0, 0.16);
|
||||||
|
}
|
||||||
|
.hero-card {
|
||||||
|
padding-bottom: 14px;
|
||||||
|
}
|
||||||
|
.profile-note {
|
||||||
|
margin: 12px 0 0;
|
||||||
|
font-size: 12px;
|
||||||
|
line-height: 1.5;
|
||||||
|
color: var(--text-muted);
|
||||||
|
}
|
||||||
|
.sticky-card {
|
||||||
|
position: sticky;
|
||||||
|
top: 24px;
|
||||||
|
}
|
||||||
|
.card-header.compact {
|
||||||
|
margin-bottom: 8px;
|
||||||
|
}
|
||||||
|
.field-grid {
|
||||||
|
display: grid;
|
||||||
|
grid-template-columns: 1fr 1fr;
|
||||||
|
gap: 12px;
|
||||||
|
}
|
||||||
|
.field {
|
||||||
|
display: flex;
|
||||||
|
flex-direction: column;
|
||||||
|
gap: 6px;
|
||||||
|
}
|
||||||
|
.field-label {
|
||||||
|
font-size: 11px;
|
||||||
|
font-weight: 700;
|
||||||
|
text-transform: uppercase;
|
||||||
|
letter-spacing: 0.08em;
|
||||||
|
color: var(--text-muted);
|
||||||
|
}
|
||||||
|
.field-inline {
|
||||||
|
display: flex;
|
||||||
|
align-items: center;
|
||||||
|
gap: 8px;
|
||||||
|
color: var(--text);
|
||||||
|
font-size: 13px;
|
||||||
|
}
|
||||||
|
.field-inline.compact {
|
||||||
|
font-size: 12px;
|
||||||
|
}
|
||||||
|
input,
|
||||||
|
select {
|
||||||
|
width: 100%;
|
||||||
|
box-sizing: border-box;
|
||||||
|
padding: 10px 12px;
|
||||||
|
border-radius: 12px;
|
||||||
|
border: 1px solid var(--border);
|
||||||
|
background: #0b0f14;
|
||||||
|
color: var(--text);
|
||||||
|
font-size: 13px;
|
||||||
|
font-family: inherit;
|
||||||
|
}
|
||||||
|
input:focus,
|
||||||
|
select:focus {
|
||||||
|
outline: none;
|
||||||
|
border-color: var(--border-strong);
|
||||||
|
}
|
||||||
|
.lookup-row {
|
||||||
|
display: grid;
|
||||||
|
grid-template-columns: minmax(0, 1fr) auto;
|
||||||
|
gap: 10px;
|
||||||
|
margin-top: 12px;
|
||||||
|
}
|
||||||
|
.result-list,
|
||||||
|
.seed-list,
|
||||||
|
.target-list,
|
||||||
|
.preview-list {
|
||||||
|
display: flex;
|
||||||
|
flex-direction: column;
|
||||||
|
gap: 8px;
|
||||||
|
margin-top: 12px;
|
||||||
|
}
|
||||||
|
.result-card,
|
||||||
|
.seed-chip,
|
||||||
|
.target-option,
|
||||||
|
.preview-item {
|
||||||
|
border: 1px solid var(--border);
|
||||||
|
border-radius: 14px;
|
||||||
|
background: var(--bg-secondary);
|
||||||
|
padding: 12px 14px;
|
||||||
|
}
|
||||||
|
.result-card {
|
||||||
|
all: unset;
|
||||||
|
cursor: pointer;
|
||||||
|
border: 1px solid var(--border);
|
||||||
|
border-radius: 14px;
|
||||||
|
background: var(--bg-secondary);
|
||||||
|
padding: 12px 14px;
|
||||||
|
transition: background 0.12s ease, border-color 0.12s ease;
|
||||||
|
}
|
||||||
|
.result-card:hover,
|
||||||
|
.target-option:hover {
|
||||||
|
background: var(--surface-hover);
|
||||||
|
border-color: var(--border-strong);
|
||||||
|
}
|
||||||
|
.result-name {
|
||||||
|
font-size: 14px;
|
||||||
|
font-weight: 600;
|
||||||
|
color: var(--text);
|
||||||
|
}
|
||||||
|
.result-meta,
|
||||||
|
.target-count {
|
||||||
|
font-size: 12px;
|
||||||
|
color: var(--text-muted);
|
||||||
|
}
|
||||||
|
.seed-chip,
|
||||||
|
.target-option {
|
||||||
|
display: flex;
|
||||||
|
align-items: center;
|
||||||
|
justify-content: space-between;
|
||||||
|
gap: 12px;
|
||||||
|
}
|
||||||
|
.target-option {
|
||||||
|
cursor: pointer;
|
||||||
|
}
|
||||||
|
.target-option.selected {
|
||||||
|
background: var(--surface-active);
|
||||||
|
border-color: var(--border-strong);
|
||||||
|
}
|
||||||
|
.preview-item {
|
||||||
|
display: flex;
|
||||||
|
flex-direction: column;
|
||||||
|
gap: 4px;
|
||||||
|
}
|
||||||
|
.remove-btn {
|
||||||
|
all: unset;
|
||||||
|
cursor: pointer;
|
||||||
|
color: var(--text-muted);
|
||||||
|
font-size: 20px;
|
||||||
|
line-height: 1;
|
||||||
|
}
|
||||||
|
.remove-btn:hover {
|
||||||
|
color: var(--danger);
|
||||||
|
}
|
||||||
|
.empty-note {
|
||||||
|
padding: 14px;
|
||||||
|
border-radius: 14px;
|
||||||
|
background: var(--bg-secondary);
|
||||||
|
border: 1px dashed var(--border);
|
||||||
|
color: var(--text-muted);
|
||||||
|
font-size: 13px;
|
||||||
|
margin-top: 12px;
|
||||||
|
}
|
||||||
|
.divider {
|
||||||
|
height: 1px;
|
||||||
|
background: rgba(148, 163, 184, 0.1);
|
||||||
|
margin: 16px 0;
|
||||||
|
}
|
||||||
|
.action-row {
|
||||||
|
display: flex;
|
||||||
|
flex-direction: column;
|
||||||
|
gap: 10px;
|
||||||
|
margin-top: 16px;
|
||||||
|
}
|
||||||
|
.status {
|
||||||
|
padding: 10px 12px;
|
||||||
|
border-radius: 14px;
|
||||||
|
font-size: 12px;
|
||||||
|
background: var(--bg-secondary);
|
||||||
|
border: 1px solid var(--border);
|
||||||
|
color: var(--text-muted);
|
||||||
|
margin-top: 12px;
|
||||||
|
}
|
||||||
|
.status.ok {
|
||||||
|
border-color: rgba(34, 197, 94, 0.3);
|
||||||
|
color: #86efac;
|
||||||
|
}
|
||||||
|
.status.error {
|
||||||
|
border-color: rgba(239, 68, 68, 0.3);
|
||||||
|
color: #fca5a5;
|
||||||
|
}
|
||||||
|
.btn {
|
||||||
|
all: unset;
|
||||||
|
cursor: pointer;
|
||||||
|
padding: 9px 16px;
|
||||||
|
border-radius: 12px;
|
||||||
|
font-size: 13px;
|
||||||
|
font-weight: 600;
|
||||||
|
font-family: inherit;
|
||||||
|
border: 1px solid transparent;
|
||||||
|
text-align: center;
|
||||||
|
}
|
||||||
|
.btn.ghost {
|
||||||
|
color: #d9e7f8;
|
||||||
|
border-color: var(--border);
|
||||||
|
background: var(--surface);
|
||||||
|
}
|
||||||
|
.btn.accent {
|
||||||
|
background: var(--accent);
|
||||||
|
color: #fff;
|
||||||
|
}
|
||||||
|
.btn:disabled {
|
||||||
|
opacity: 0.45;
|
||||||
|
cursor: default;
|
||||||
|
}
|
||||||
|
.page-icon {
|
||||||
|
display: inline-flex;
|
||||||
|
align-items: center;
|
||||||
|
justify-content: center;
|
||||||
|
width: 34px;
|
||||||
|
height: 34px;
|
||||||
|
border-radius: 10px;
|
||||||
|
background: var(--surface-active);
|
||||||
|
color: #a9d6ff;
|
||||||
|
}
|
||||||
|
|
||||||
|
@media (max-width: 1100px) {
|
||||||
|
.builder-grid {
|
||||||
|
grid-template-columns: 1fr;
|
||||||
|
}
|
||||||
|
.sticky-card {
|
||||||
|
position: static;
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
@media (max-width: 860px) {
|
||||||
|
.seed-grid,
|
||||||
|
.field-grid,
|
||||||
|
.lookup-row {
|
||||||
|
grid-template-columns: 1fr;
|
||||||
|
}
|
||||||
|
.page-header {
|
||||||
|
flex-direction: column;
|
||||||
|
}
|
||||||
|
.page-actions {
|
||||||
|
width: 100%;
|
||||||
|
}
|
||||||
|
}
|
||||||
|
</style>
|
||||||
@@ -0,0 +1,107 @@
|
|||||||
|
<script>
|
||||||
|
export let name = 'spark';
|
||||||
|
export let size = 16;
|
||||||
|
export let stroke = 1.8;
|
||||||
|
|
||||||
|
const paths = {
|
||||||
|
edit: [
|
||||||
|
'M3 21h3.8L19.4 8.4a2.2 2.2 0 0 0 0-3.1l-.7-.7a2.2 2.2 0 0 0-3.1 0L3 17.2V21Z',
|
||||||
|
'M13.5 6.5l4 4'
|
||||||
|
],
|
||||||
|
sync: [
|
||||||
|
'M17 3l4 4-4 4',
|
||||||
|
'M3 7h18',
|
||||||
|
'M7 21l-4-4 4-4',
|
||||||
|
'M21 17H3'
|
||||||
|
],
|
||||||
|
settings: [
|
||||||
|
'M12 8.5a3.5 3.5 0 1 1 0 7a3.5 3.5 0 0 1 0-7Z',
|
||||||
|
'M19.4 15a1 1 0 0 0 .2 1.1l.1.1a1.2 1.2 0 0 1 0 1.7l-1.4 1.4a1.2 1.2 0 0 1-1.7 0l-.1-.1a1 1 0 0 0-1.1-.2a1 1 0 0 0-.6.9V21a1.2 1.2 0 0 1-1.2 1.2h-2a1.2 1.2 0 0 1-1.2-1.2v-.2a1 1 0 0 0-.6-.9a1 1 0 0 0-1.1.2l-.1.1a1.2 1.2 0 0 1-1.7 0L4.3 18a1.2 1.2 0 0 1 0-1.7l.1-.1a1 1 0 0 0 .2-1.1a1 1 0 0 0-.9-.6H3.5A1.2 1.2 0 0 1 2.3 13v-2a1.2 1.2 0 0 1 1.2-1.2h.2a1 1 0 0 0 .9-.6a1 1 0 0 0-.2-1.1l-.1-.1a1.2 1.2 0 0 1 0-1.7l1.4-1.4a1.2 1.2 0 0 1 1.7 0l.1.1a1 1 0 0 0 1.1.2a1 1 0 0 0 .6-.9V3A1.2 1.2 0 0 1 10.5 1.8h2A1.2 1.2 0 0 1 13.7 3v.2a1 1 0 0 0 .6.9a1 1 0 0 0 1.1-.2l.1-.1a1.2 1.2 0 0 1 1.7 0L18.6 5a1.2 1.2 0 0 1 0 1.7l-.1.1a1 1 0 0 0-.2 1.1a1 1 0 0 0 .9.6h.2A1.2 1.2 0 0 1 20.6 11v2a1.2 1.2 0 0 1-1.2 1.2h-.2a1 1 0 0 0-.9.8Z'
|
||||||
|
],
|
||||||
|
collections: [
|
||||||
|
'M4 7.5h16',
|
||||||
|
'M4 12h16',
|
||||||
|
'M4 16.5h10',
|
||||||
|
'M17.5 14.5v6',
|
||||||
|
'M14.5 17.5h6'
|
||||||
|
],
|
||||||
|
user: [
|
||||||
|
'M12 12a4 4 0 1 0 0-8a4 4 0 0 0 0 8Z',
|
||||||
|
'M4 20a8 8 0 0 1 16 0'
|
||||||
|
],
|
||||||
|
resume: [
|
||||||
|
'M7 5v14l11-7Z'
|
||||||
|
],
|
||||||
|
items: [
|
||||||
|
'M4 6.5h16',
|
||||||
|
'M4 12h16',
|
||||||
|
'M4 17.5h16'
|
||||||
|
],
|
||||||
|
userviews: [
|
||||||
|
'M4 5h7v6H4Z',
|
||||||
|
'M13 5h7v6h-7Z',
|
||||||
|
'M4 13h7v6H4Z',
|
||||||
|
'M13 13h7v6h-7Z'
|
||||||
|
],
|
||||||
|
boxset: [
|
||||||
|
'M4 8l8-4l8 4-8 4-8-4Z',
|
||||||
|
'M4 8v8l8 4l8-4V8'
|
||||||
|
],
|
||||||
|
latestepisodereleases: [
|
||||||
|
'M4 18h16',
|
||||||
|
'M7 18V9l5-4l5 4v9'
|
||||||
|
],
|
||||||
|
latestmoviereleases: [
|
||||||
|
'M4 7h16v10H4Z',
|
||||||
|
'M8 7V5',
|
||||||
|
'M16 7V5',
|
||||||
|
'M8 17v2',
|
||||||
|
'M16 17v2'
|
||||||
|
],
|
||||||
|
latestmediablock: [
|
||||||
|
'M4 6h16',
|
||||||
|
'M4 12h10',
|
||||||
|
'M4 18h16'
|
||||||
|
],
|
||||||
|
spark: [
|
||||||
|
'M12 3l1.6 4.4L18 9l-4.4 1.6L12 15l-1.6-4.4L6 9l4.4-1.6L12 3Z'
|
||||||
|
],
|
||||||
|
search: [
|
||||||
|
'M11 18a7 7 0 1 0 0-14a7 7 0 0 0 0 14Z',
|
||||||
|
'M20 20l-3.5-3.5'
|
||||||
|
],
|
||||||
|
activity: [
|
||||||
|
'M5 12h3l2-5l4 10l2-5h3'
|
||||||
|
],
|
||||||
|
plus: [
|
||||||
|
'M12 5v14',
|
||||||
|
'M5 12h14'
|
||||||
|
]
|
||||||
|
};
|
||||||
|
|
||||||
|
$: selectedPaths = paths[name] || paths.spark;
|
||||||
|
</script>
|
||||||
|
|
||||||
|
<svg
|
||||||
|
class="icon"
|
||||||
|
width={size}
|
||||||
|
height={size}
|
||||||
|
viewBox="0 0 24 24"
|
||||||
|
fill="none"
|
||||||
|
stroke="currentColor"
|
||||||
|
stroke-width={stroke}
|
||||||
|
stroke-linecap="round"
|
||||||
|
stroke-linejoin="round"
|
||||||
|
aria-hidden="true"
|
||||||
|
>
|
||||||
|
{#each selectedPaths as path}
|
||||||
|
<path d={path}></path>
|
||||||
|
{/each}
|
||||||
|
</svg>
|
||||||
|
|
||||||
|
<style>
|
||||||
|
.icon {
|
||||||
|
display: block;
|
||||||
|
flex: none;
|
||||||
|
}
|
||||||
|
</style>
|
||||||
+266
-66
@@ -6,20 +6,25 @@
|
|||||||
SORT_OPTIONS,
|
SORT_OPTIONS,
|
||||||
IMAGE_TYPES,
|
IMAGE_TYPES,
|
||||||
GENRES,
|
GENRES,
|
||||||
|
getSectionIconName,
|
||||||
getSectionTypeLabel,
|
getSectionTypeLabel,
|
||||||
getGenreNames
|
getGenreNames
|
||||||
} from '$lib/constants.js';
|
} from '$lib/constants.js';
|
||||||
|
import Icon from '$lib/Icon.svelte';
|
||||||
|
|
||||||
export let section;
|
export let section;
|
||||||
export let index;
|
export let index;
|
||||||
export let total;
|
export let total;
|
||||||
export let expanded = false;
|
export let expanded = false;
|
||||||
|
export let excludedFolderLookup = {};
|
||||||
|
export let lookupUserId = '';
|
||||||
|
|
||||||
import { createEventDispatcher } from 'svelte';
|
import { createEventDispatcher } from 'svelte';
|
||||||
const dispatch = createEventDispatcher();
|
const dispatch = createEventDispatcher();
|
||||||
|
|
||||||
$: genreNames = getGenreNames(section.Query?.GenreIds);
|
$: genreNames = getGenreNames(section.Query?.GenreIds);
|
||||||
$: typeLabel = getSectionTypeLabel(section.SectionType);
|
$: typeLabel = getSectionTypeLabel(section.SectionType);
|
||||||
|
$: sectionIcon = getSectionIconName(section.SectionType);
|
||||||
$: showFilters =
|
$: showFilters =
|
||||||
section.SectionType === 'items' ||
|
section.SectionType === 'items' ||
|
||||||
section.SectionType === 'collections' ||
|
section.SectionType === 'collections' ||
|
||||||
@@ -28,10 +33,20 @@
|
|||||||
|
|
||||||
// ExcludedFolders as a comma-separated string for editing
|
// ExcludedFolders as a comma-separated string for editing
|
||||||
$: excludedFoldersStr = (section.ExcludedFolders || []).join(', ');
|
$: excludedFoldersStr = (section.ExcludedFolders || []).join(', ');
|
||||||
|
$: excludedFolderDetails = (section.ExcludedFolders || []).map((id) => ({
|
||||||
|
id,
|
||||||
|
label: excludedFolderLookup?.[id]?.name || null,
|
||||||
|
type: excludedFolderLookup?.[id]?.type || null
|
||||||
|
}));
|
||||||
|
|
||||||
// TagIds as a comma-separated string for editing
|
// TagIds as a comma-separated string for editing
|
||||||
$: tagIdsStr = (section.Query?.TagIds || []).join(', ');
|
$: tagIdsStr = (section.Query?.TagIds || []).join(', ');
|
||||||
|
|
||||||
|
let collectionSearchTerm = '';
|
||||||
|
let collectionResults = [];
|
||||||
|
let collectionLookupBusy = false;
|
||||||
|
let collectionLookupError = '';
|
||||||
|
|
||||||
function toggleGenre(id) {
|
function toggleGenre(id) {
|
||||||
if (!section.Query) section.Query = { StudioIds: [], TagIds: [], GenreIds: [], CollectionTypes: [] };
|
if (!section.Query) section.Query = { StudioIds: [], TagIds: [], GenreIds: [], CollectionTypes: [] };
|
||||||
const idx = section.Query.GenreIds.indexOf(id);
|
const idx = section.Query.GenreIds.indexOf(id);
|
||||||
@@ -75,6 +90,59 @@
|
|||||||
.filter((s) => s !== '');
|
.filter((s) => s !== '');
|
||||||
dispatch('change');
|
dispatch('change');
|
||||||
}
|
}
|
||||||
|
|
||||||
|
async function searchCollections() {
|
||||||
|
if (!lookupUserId) {
|
||||||
|
collectionLookupError = 'Select a user with a linked Emby account to search collections.';
|
||||||
|
return;
|
||||||
|
}
|
||||||
|
|
||||||
|
if (!collectionSearchTerm.trim()) {
|
||||||
|
collectionResults = [];
|
||||||
|
collectionLookupError = '';
|
||||||
|
return;
|
||||||
|
}
|
||||||
|
|
||||||
|
collectionLookupBusy = true;
|
||||||
|
collectionLookupError = '';
|
||||||
|
|
||||||
|
try {
|
||||||
|
const params = new URLSearchParams({
|
||||||
|
userId: lookupUserId,
|
||||||
|
term: collectionSearchTerm.trim(),
|
||||||
|
types: 'BoxSet',
|
||||||
|
limit: '8'
|
||||||
|
});
|
||||||
|
const response = await fetch(`/api/emby-item-search?${params.toString()}`);
|
||||||
|
const body = await response.json().catch(() => ({ items: [] }));
|
||||||
|
if (!response.ok) throw new Error(body.message || body.error || response.statusText);
|
||||||
|
collectionResults = body.items || [];
|
||||||
|
} catch (err) {
|
||||||
|
collectionLookupError = err.message;
|
||||||
|
collectionResults = [];
|
||||||
|
} finally {
|
||||||
|
collectionLookupBusy = false;
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
function applyCollection(item) {
|
||||||
|
const previousName = section.ParentItem?.Name || section.Name || section.CustomName || '';
|
||||||
|
if (!section.ParentItem) section.ParentItem = { Name: '', Id: '' };
|
||||||
|
section.ParentItem.Name = item.name;
|
||||||
|
section.ParentItem.Id = item.id;
|
||||||
|
section.ParentId = item.id;
|
||||||
|
|
||||||
|
if (!section.CustomName || section.CustomName === previousName) {
|
||||||
|
section.CustomName = item.name;
|
||||||
|
}
|
||||||
|
if (!section.Name || section.Name === previousName) {
|
||||||
|
section.Name = item.name;
|
||||||
|
}
|
||||||
|
|
||||||
|
collectionSearchTerm = item.name;
|
||||||
|
collectionResults = [];
|
||||||
|
dispatch('change');
|
||||||
|
}
|
||||||
</script>
|
</script>
|
||||||
|
|
||||||
<div class="section-card" class:expanded>
|
<div class="section-card" class:expanded>
|
||||||
@@ -97,6 +165,10 @@
|
|||||||
|
|
||||||
<span class="section-index">{index + 1}</span>
|
<span class="section-index">{index + 1}</span>
|
||||||
|
|
||||||
|
<span class="section-icon">
|
||||||
|
<Icon name={sectionIcon} size={16} />
|
||||||
|
</span>
|
||||||
|
|
||||||
<div class="section-info">
|
<div class="section-info">
|
||||||
<div class="section-name">{section.CustomName || section.Name || '(unnamed)'}</div>
|
<div class="section-name">{section.CustomName || section.Name || '(unnamed)'}</div>
|
||||||
<div class="section-meta">
|
<div class="section-meta">
|
||||||
@@ -312,42 +384,66 @@
|
|||||||
{#if section.SectionType === 'boxset'}
|
{#if section.SectionType === 'boxset'}
|
||||||
<div class="field-group">
|
<div class="field-group">
|
||||||
<span class="field-label">Linked box set</span>
|
<span class="field-label">Linked box set</span>
|
||||||
{#if section.ParentItem}
|
<div class="lookup-row">
|
||||||
|
<input
|
||||||
|
type="text"
|
||||||
|
class="ids-input"
|
||||||
|
bind:value={collectionSearchTerm}
|
||||||
|
on:keydown={(event) => event.key === 'Enter' && searchCollections()}
|
||||||
|
placeholder="Search existing collections..."
|
||||||
|
/>
|
||||||
|
<button class="chip lookup-btn" on:click={searchCollections} disabled={collectionLookupBusy}>
|
||||||
|
{collectionLookupBusy ? 'Searching…' : 'Lookup'}
|
||||||
|
</button>
|
||||||
|
</div>
|
||||||
|
{#if collectionLookupError}
|
||||||
|
<div class="inline-status error">{collectionLookupError}</div>
|
||||||
|
{/if}
|
||||||
|
{#if collectionResults.length}
|
||||||
|
<div class="lookup-results">
|
||||||
|
{#each collectionResults as item}
|
||||||
|
<button class="resolved-item lookup-result" on:click={() => applyCollection(item)}>
|
||||||
|
<span class="resolved-name">{item.name}</span>
|
||||||
|
<span class="resolved-meta">{item.id}</span>
|
||||||
|
</button>
|
||||||
|
{/each}
|
||||||
|
</div>
|
||||||
|
{/if}
|
||||||
|
<div class="field-grid">
|
||||||
|
<label class="field">
|
||||||
|
<span class="field-label">Box set name</span>
|
||||||
|
<input
|
||||||
|
type="text"
|
||||||
|
value={section.ParentItem?.Name || ''}
|
||||||
|
on:input={(e) => {
|
||||||
|
if (!section.ParentItem) section.ParentItem = { Name: '', Id: '' };
|
||||||
|
section.ParentItem.Name = e.target.value;
|
||||||
|
section.Name = e.target.value;
|
||||||
|
handleChange();
|
||||||
|
}}
|
||||||
|
placeholder="Display name"
|
||||||
|
/>
|
||||||
|
</label>
|
||||||
|
<label class="field">
|
||||||
|
<span class="field-label">Box set ID</span>
|
||||||
|
<input
|
||||||
|
type="text"
|
||||||
|
value={section.ParentItem?.Id || section.ParentId || ''}
|
||||||
|
on:input={(e) => {
|
||||||
|
if (!section.ParentItem) section.ParentItem = { Name: '', Id: '' };
|
||||||
|
section.ParentItem.Id = e.target.value;
|
||||||
|
section.ParentId = e.target.value;
|
||||||
|
handleChange();
|
||||||
|
}}
|
||||||
|
placeholder="Emby item ID"
|
||||||
|
/>
|
||||||
|
</label>
|
||||||
|
</div>
|
||||||
|
{#if section.ParentItem?.Id}
|
||||||
<div class="boxset-info">
|
<div class="boxset-info">
|
||||||
<span>{section.ParentItem.Name}</span>
|
<span>{section.ParentItem.Name}</span>
|
||||||
<span class="text-muted">ID: {section.ParentItem.Id}</span>
|
<span class="text-muted">ID: {section.ParentItem.Id}</span>
|
||||||
</div>
|
</div>
|
||||||
{:else}
|
|
||||||
<div class="field-grid">
|
|
||||||
<label class="field">
|
|
||||||
<span class="field-label">Box set name</span>
|
|
||||||
<input
|
|
||||||
type="text"
|
|
||||||
value={section.ParentItem?.Name || ''}
|
|
||||||
on:input={(e) => {
|
|
||||||
if (!section.ParentItem) section.ParentItem = { Name: '', Id: '' };
|
|
||||||
section.ParentItem.Name = e.target.value;
|
|
||||||
section.Name = e.target.value;
|
|
||||||
handleChange();
|
|
||||||
}}
|
|
||||||
placeholder="Display name"
|
|
||||||
/>
|
|
||||||
</label>
|
|
||||||
<label class="field">
|
|
||||||
<span class="field-label">Box set ID</span>
|
|
||||||
<input
|
|
||||||
type="text"
|
|
||||||
value={section.ParentItem?.Id || section.ParentId || ''}
|
|
||||||
on:input={(e) => {
|
|
||||||
if (!section.ParentItem) section.ParentItem = { Name: '', Id: '' };
|
|
||||||
section.ParentItem.Id = e.target.value;
|
|
||||||
section.ParentId = e.target.value;
|
|
||||||
handleChange();
|
|
||||||
}}
|
|
||||||
placeholder="Emby item ID"
|
|
||||||
/>
|
|
||||||
</label>
|
|
||||||
</div>
|
|
||||||
{/if}
|
{/if}
|
||||||
</div>
|
</div>
|
||||||
{/if}
|
{/if}
|
||||||
@@ -381,6 +477,16 @@
|
|||||||
{/each}
|
{/each}
|
||||||
</div>
|
</div>
|
||||||
{/if}
|
{/if}
|
||||||
|
{#if excludedFolderDetails.some((folder) => folder.label)}
|
||||||
|
<div class="resolved-list">
|
||||||
|
{#each excludedFolderDetails.filter((folder) => folder.label) as folder}
|
||||||
|
<div class="resolved-item">
|
||||||
|
<span class="resolved-name">{folder.label}</span>
|
||||||
|
<span class="resolved-meta">{folder.id}{folder.type ? ` · ${folder.type}` : ''}</span>
|
||||||
|
</div>
|
||||||
|
{/each}
|
||||||
|
</div>
|
||||||
|
{/if}
|
||||||
</div>
|
</div>
|
||||||
|
|
||||||
</div>
|
</div>
|
||||||
@@ -390,20 +496,26 @@
|
|||||||
<style>
|
<style>
|
||||||
.section-card {
|
.section-card {
|
||||||
background: var(--surface);
|
background: var(--surface);
|
||||||
border-radius: 10px;
|
border-radius: 20px;
|
||||||
border: 1px solid var(--border);
|
border: 1px solid var(--border);
|
||||||
margin-bottom: 6px;
|
margin-bottom: 12px;
|
||||||
overflow: hidden;
|
overflow: hidden;
|
||||||
transition: border-color 0.15s;
|
box-shadow: 0 18px 36px rgba(0, 0, 0, 0.18);
|
||||||
|
transition: border-color 0.15s, transform 0.15s ease, box-shadow 0.15s ease;
|
||||||
|
}
|
||||||
|
.section-card:hover {
|
||||||
|
transform: translateY(-1px);
|
||||||
|
box-shadow: 0 22px 42px rgba(0, 0, 0, 0.22);
|
||||||
}
|
}
|
||||||
.section-card.expanded {
|
.section-card.expanded {
|
||||||
border-color: var(--accent);
|
border-color: var(--border-strong);
|
||||||
|
box-shadow: 0 24px 48px rgba(0, 0, 0, 0.26);
|
||||||
}
|
}
|
||||||
.section-header {
|
.section-header {
|
||||||
all: unset;
|
all: unset;
|
||||||
display: flex;
|
display: flex;
|
||||||
align-items: center;
|
align-items: center;
|
||||||
padding: 10px 14px;
|
padding: 14px 16px;
|
||||||
gap: 10px;
|
gap: 10px;
|
||||||
cursor: pointer;
|
cursor: pointer;
|
||||||
width: 100%;
|
width: 100%;
|
||||||
@@ -422,8 +534,8 @@
|
|||||||
cursor: pointer;
|
cursor: pointer;
|
||||||
font-size: 10px;
|
font-size: 10px;
|
||||||
line-height: 1;
|
line-height: 1;
|
||||||
opacity: 0.5;
|
opacity: 0.6;
|
||||||
padding: 1px 2px;
|
padding: 3px 4px;
|
||||||
color: var(--text);
|
color: var(--text);
|
||||||
}
|
}
|
||||||
.move-btn:disabled {
|
.move-btn:disabled {
|
||||||
@@ -436,17 +548,29 @@
|
|||||||
.section-index {
|
.section-index {
|
||||||
font-size: 11px;
|
font-size: 11px;
|
||||||
color: var(--text-muted);
|
color: var(--text-muted);
|
||||||
min-width: 18px;
|
min-width: 24px;
|
||||||
text-align: center;
|
text-align: center;
|
||||||
font-variant-numeric: tabular-nums;
|
font-variant-numeric: tabular-nums;
|
||||||
}
|
}
|
||||||
|
.section-icon {
|
||||||
|
display: inline-flex;
|
||||||
|
align-items: center;
|
||||||
|
justify-content: center;
|
||||||
|
width: 34px;
|
||||||
|
height: 34px;
|
||||||
|
border-radius: 12px;
|
||||||
|
background: var(--bg-secondary);
|
||||||
|
border: 1px solid var(--border);
|
||||||
|
color: #a9d6ff;
|
||||||
|
flex-shrink: 0;
|
||||||
|
}
|
||||||
.section-info {
|
.section-info {
|
||||||
flex: 1;
|
flex: 1;
|
||||||
min-width: 0;
|
min-width: 0;
|
||||||
}
|
}
|
||||||
.section-name {
|
.section-name {
|
||||||
font-weight: 600;
|
font-weight: 600;
|
||||||
font-size: 14px;
|
font-size: 15px;
|
||||||
color: var(--text);
|
color: var(--text);
|
||||||
white-space: nowrap;
|
white-space: nowrap;
|
||||||
overflow: hidden;
|
overflow: hidden;
|
||||||
@@ -455,7 +579,7 @@
|
|||||||
.section-meta {
|
.section-meta {
|
||||||
font-size: 12px;
|
font-size: 12px;
|
||||||
color: var(--text-muted);
|
color: var(--text-muted);
|
||||||
margin-top: 2px;
|
margin-top: 4px;
|
||||||
white-space: nowrap;
|
white-space: nowrap;
|
||||||
overflow: hidden;
|
overflow: hidden;
|
||||||
text-overflow: ellipsis;
|
text-overflow: ellipsis;
|
||||||
@@ -464,7 +588,7 @@
|
|||||||
all: unset;
|
all: unset;
|
||||||
cursor: pointer;
|
cursor: pointer;
|
||||||
color: var(--danger);
|
color: var(--danger);
|
||||||
font-size: 20px;
|
font-size: 22px;
|
||||||
line-height: 1;
|
line-height: 1;
|
||||||
padding: 0 4px;
|
padding: 0 4px;
|
||||||
opacity: 0.5;
|
opacity: 0.5;
|
||||||
@@ -480,14 +604,14 @@
|
|||||||
|
|
||||||
/* Editor panel */
|
/* Editor panel */
|
||||||
.section-editor {
|
.section-editor {
|
||||||
padding: 4px 16px 16px;
|
padding: 8px 16px 18px;
|
||||||
border-top: 1px solid var(--border);
|
border-top: 1px solid var(--border);
|
||||||
}
|
}
|
||||||
.field-grid {
|
.field-grid {
|
||||||
display: grid;
|
display: grid;
|
||||||
grid-template-columns: 1fr 1fr;
|
grid-template-columns: 1fr 1fr;
|
||||||
gap: 10px;
|
gap: 12px;
|
||||||
margin-bottom: 12px;
|
margin-bottom: 14px;
|
||||||
}
|
}
|
||||||
.field-grid.checkboxes {
|
.field-grid.checkboxes {
|
||||||
grid-template-columns: auto auto auto;
|
grid-template-columns: auto auto auto;
|
||||||
@@ -498,7 +622,7 @@
|
|||||||
.field {
|
.field {
|
||||||
display: flex;
|
display: flex;
|
||||||
flex-direction: column;
|
flex-direction: column;
|
||||||
gap: 4px;
|
gap: 6px;
|
||||||
}
|
}
|
||||||
.field.span2 {
|
.field.span2 {
|
||||||
grid-column: span 2;
|
grid-column: span 2;
|
||||||
@@ -519,10 +643,10 @@
|
|||||||
}
|
}
|
||||||
.field input[type='text'],
|
.field input[type='text'],
|
||||||
.field select {
|
.field select {
|
||||||
padding: 7px 10px;
|
padding: 10px 12px;
|
||||||
border-radius: 6px;
|
border-radius: 12px;
|
||||||
border: 1px solid var(--border);
|
border: 1px solid var(--border);
|
||||||
background: var(--bg);
|
background: #0b0f14;
|
||||||
color: var(--text);
|
color: var(--text);
|
||||||
font-size: 13px;
|
font-size: 13px;
|
||||||
font-family: inherit;
|
font-family: inherit;
|
||||||
@@ -530,12 +654,52 @@
|
|||||||
.field input:focus,
|
.field input:focus,
|
||||||
.field select:focus {
|
.field select:focus {
|
||||||
outline: none;
|
outline: none;
|
||||||
border-color: var(--accent);
|
border-color: var(--border-strong);
|
||||||
}
|
}
|
||||||
|
|
||||||
.field-group {
|
.field-group {
|
||||||
margin-bottom: 12px;
|
margin-bottom: 12px;
|
||||||
}
|
}
|
||||||
|
.lookup-row {
|
||||||
|
display: grid;
|
||||||
|
grid-template-columns: minmax(0, 1fr) auto;
|
||||||
|
gap: 8px;
|
||||||
|
align-items: start;
|
||||||
|
}
|
||||||
|
.lookup-btn {
|
||||||
|
padding: 10px 12px;
|
||||||
|
border-radius: 12px;
|
||||||
|
text-align: center;
|
||||||
|
color: var(--text);
|
||||||
|
background: var(--surface);
|
||||||
|
}
|
||||||
|
.lookup-btn:disabled {
|
||||||
|
opacity: 0.45;
|
||||||
|
cursor: default;
|
||||||
|
}
|
||||||
|
.lookup-results {
|
||||||
|
margin-top: 8px;
|
||||||
|
display: flex;
|
||||||
|
flex-direction: column;
|
||||||
|
gap: 6px;
|
||||||
|
}
|
||||||
|
.lookup-result {
|
||||||
|
all: unset;
|
||||||
|
cursor: pointer;
|
||||||
|
}
|
||||||
|
.inline-status {
|
||||||
|
margin-top: 8px;
|
||||||
|
padding: 10px 12px;
|
||||||
|
border-radius: 12px;
|
||||||
|
border: 1px solid var(--border);
|
||||||
|
background: var(--surface);
|
||||||
|
font-size: 12px;
|
||||||
|
color: var(--text-muted);
|
||||||
|
}
|
||||||
|
.inline-status.error {
|
||||||
|
color: #fca5a5;
|
||||||
|
border-color: rgba(239, 68, 68, 0.24);
|
||||||
|
}
|
||||||
.chip-group {
|
.chip-group {
|
||||||
display: flex;
|
display: flex;
|
||||||
flex-wrap: wrap;
|
flex-wrap: wrap;
|
||||||
@@ -545,8 +709,8 @@
|
|||||||
.chip {
|
.chip {
|
||||||
all: unset;
|
all: unset;
|
||||||
cursor: pointer;
|
cursor: pointer;
|
||||||
padding: 4px 10px;
|
padding: 6px 11px;
|
||||||
border-radius: 14px;
|
border-radius: 999px;
|
||||||
font-size: 12px;
|
font-size: 12px;
|
||||||
border: 1px solid var(--border);
|
border: 1px solid var(--border);
|
||||||
color: var(--text-muted);
|
color: var(--text-muted);
|
||||||
@@ -558,7 +722,7 @@
|
|||||||
}
|
}
|
||||||
.chip.active {
|
.chip.active {
|
||||||
background: var(--accent);
|
background: var(--accent);
|
||||||
border-color: var(--accent);
|
border-color: transparent;
|
||||||
color: #fff;
|
color: #fff;
|
||||||
}
|
}
|
||||||
|
|
||||||
@@ -575,10 +739,10 @@
|
|||||||
.ids-input {
|
.ids-input {
|
||||||
width: 100%;
|
width: 100%;
|
||||||
margin-top: 6px;
|
margin-top: 6px;
|
||||||
padding: 7px 10px;
|
padding: 10px 12px;
|
||||||
border-radius: 6px;
|
border-radius: 12px;
|
||||||
border: 1px solid var(--border);
|
border: 1px solid var(--border);
|
||||||
background: var(--bg);
|
background: #0b0f14;
|
||||||
color: var(--text);
|
color: var(--text);
|
||||||
font-size: 12px;
|
font-size: 12px;
|
||||||
font-family: monospace;
|
font-family: monospace;
|
||||||
@@ -586,7 +750,7 @@
|
|||||||
}
|
}
|
||||||
.ids-input:focus {
|
.ids-input:focus {
|
||||||
outline: none;
|
outline: none;
|
||||||
border-color: var(--accent);
|
border-color: var(--border-strong);
|
||||||
}
|
}
|
||||||
.ids-preview {
|
.ids-preview {
|
||||||
display: flex;
|
display: flex;
|
||||||
@@ -598,10 +762,10 @@
|
|||||||
display: inline-flex;
|
display: inline-flex;
|
||||||
align-items: center;
|
align-items: center;
|
||||||
gap: 3px;
|
gap: 3px;
|
||||||
background: var(--bg);
|
background: #0b0f14;
|
||||||
border: 1px solid var(--border);
|
border: 1px solid var(--border);
|
||||||
border-radius: 4px;
|
border-radius: 999px;
|
||||||
padding: 2px 6px 2px 8px;
|
padding: 4px 8px 4px 10px;
|
||||||
font-size: 11px;
|
font-size: 11px;
|
||||||
font-family: monospace;
|
font-family: monospace;
|
||||||
color: var(--text-muted);
|
color: var(--text-muted);
|
||||||
@@ -621,8 +785,8 @@
|
|||||||
display: inline-block;
|
display: inline-block;
|
||||||
background: var(--accent);
|
background: var(--accent);
|
||||||
color: #fff;
|
color: #fff;
|
||||||
border-radius: 8px;
|
border-radius: 999px;
|
||||||
padding: 0 5px;
|
padding: 2px 6px;
|
||||||
font-size: 10px;
|
font-size: 10px;
|
||||||
font-weight: 700;
|
font-weight: 700;
|
||||||
margin-left: 4px;
|
margin-left: 4px;
|
||||||
@@ -638,12 +802,48 @@
|
|||||||
font-size: 13px;
|
font-size: 13px;
|
||||||
color: var(--text);
|
color: var(--text);
|
||||||
margin-top: 6px;
|
margin-top: 6px;
|
||||||
padding: 6px 10px;
|
padding: 10px 12px;
|
||||||
background: var(--bg);
|
background: #0b0f14;
|
||||||
border-radius: 6px;
|
border-radius: 12px;
|
||||||
}
|
}
|
||||||
.text-muted {
|
.text-muted {
|
||||||
color: var(--text-muted);
|
color: var(--text-muted);
|
||||||
font-size: 12px;
|
font-size: 12px;
|
||||||
}
|
}
|
||||||
|
.resolved-list {
|
||||||
|
margin-top: 10px;
|
||||||
|
display: flex;
|
||||||
|
flex-direction: column;
|
||||||
|
gap: 6px;
|
||||||
|
}
|
||||||
|
.resolved-item {
|
||||||
|
display: flex;
|
||||||
|
justify-content: space-between;
|
||||||
|
gap: 12px;
|
||||||
|
padding: 8px 10px;
|
||||||
|
border-radius: 12px;
|
||||||
|
background: var(--bg-secondary);
|
||||||
|
border: 1px solid var(--border);
|
||||||
|
}
|
||||||
|
.lookup-result:hover {
|
||||||
|
border-color: var(--border-strong);
|
||||||
|
background: var(--surface-hover);
|
||||||
|
}
|
||||||
|
.resolved-name {
|
||||||
|
color: var(--text);
|
||||||
|
font-size: 12px;
|
||||||
|
font-weight: 600;
|
||||||
|
}
|
||||||
|
.resolved-meta {
|
||||||
|
color: var(--text-muted);
|
||||||
|
font-size: 11px;
|
||||||
|
font-family: monospace;
|
||||||
|
text-align: right;
|
||||||
|
}
|
||||||
|
|
||||||
|
@media (max-width: 640px) {
|
||||||
|
.lookup-row {
|
||||||
|
grid-template-columns: 1fr;
|
||||||
|
}
|
||||||
|
}
|
||||||
</style>
|
</style>
|
||||||
|
|||||||
@@ -2,7 +2,7 @@
|
|||||||
import { createEventDispatcher } from 'svelte';
|
import { createEventDispatcher } from 'svelte';
|
||||||
const dispatch = createEventDispatcher();
|
const dispatch = createEventDispatcher();
|
||||||
|
|
||||||
export let config = { embyUrl: '', apiKey: '', dbPath: '' };
|
export let config = { embyUrl: '', apiKey: '', tmdbApiKey: '', dbPath: '' };
|
||||||
|
|
||||||
let localConfig = { ...config };
|
let localConfig = { ...config };
|
||||||
let status = '';
|
let status = '';
|
||||||
@@ -14,6 +14,19 @@
|
|||||||
statusType = type;
|
statusType = type;
|
||||||
}
|
}
|
||||||
|
|
||||||
|
async function fetchEmbyUsers() {
|
||||||
|
const res = await fetch('/api/emby-users');
|
||||||
|
if (!res.ok) {
|
||||||
|
const err = await res.json().catch(() => ({ message: res.statusText }));
|
||||||
|
throw new Error(err.message || res.statusText);
|
||||||
|
}
|
||||||
|
|
||||||
|
const payload = await res.json();
|
||||||
|
return Array.isArray(payload)
|
||||||
|
? { users: payload, source: 'live', lastSyncedAt: null, message: '' }
|
||||||
|
: payload;
|
||||||
|
}
|
||||||
|
|
||||||
async function saveConfig() {
|
async function saveConfig() {
|
||||||
busy = true;
|
busy = true;
|
||||||
setStatus('Saving…');
|
setStatus('Saving…');
|
||||||
@@ -48,13 +61,15 @@
|
|||||||
headers: { 'Content-Type': 'application/json' },
|
headers: { 'Content-Type': 'application/json' },
|
||||||
body: JSON.stringify(localConfig)
|
body: JSON.stringify(localConfig)
|
||||||
});
|
});
|
||||||
const res = await fetch('/api/emby-users');
|
const payload = await fetchEmbyUsers();
|
||||||
if (!res.ok) {
|
if (payload.source === 'cache') {
|
||||||
const err = await res.json().catch(() => ({ message: res.statusText }));
|
setStatus(
|
||||||
throw new Error(err.message || res.statusText);
|
`${payload.message} Loaded ${payload.users.length} cached users from ${payload.lastSyncedAt || 'the last successful sync'}.`,
|
||||||
|
'ok'
|
||||||
|
);
|
||||||
|
} else {
|
||||||
|
setStatus(`Connected — ${payload.users.length} users found and cached locally.`, 'ok');
|
||||||
}
|
}
|
||||||
const users = await res.json();
|
|
||||||
setStatus(`Connected — ${users.length} users found.`, 'ok');
|
|
||||||
} catch (e) {
|
} catch (e) {
|
||||||
setStatus(`Connection failed: ${e.message}`, 'error');
|
setStatus(`Connection failed: ${e.message}`, 'error');
|
||||||
} finally {
|
} finally {
|
||||||
@@ -66,14 +81,16 @@
|
|||||||
busy = true;
|
busy = true;
|
||||||
setStatus('Fetching user names from Emby…');
|
setStatus('Fetching user names from Emby…');
|
||||||
try {
|
try {
|
||||||
const res = await fetch('/api/emby-users');
|
const payload = await fetchEmbyUsers();
|
||||||
if (!res.ok) {
|
dispatch('namesRefreshed', payload);
|
||||||
const err = await res.json().catch(() => ({ message: res.statusText }));
|
if (payload.source === 'cache') {
|
||||||
throw new Error(err.message || res.statusText);
|
setStatus(
|
||||||
|
`${payload.message} Refreshed ${payload.users.length} users from the local cache.`,
|
||||||
|
'ok'
|
||||||
|
);
|
||||||
|
} else {
|
||||||
|
setStatus(`Names refreshed — ${payload.users.length} users from Emby and saved locally.`, 'ok');
|
||||||
}
|
}
|
||||||
const embyUsers = await res.json();
|
|
||||||
dispatch('namesRefreshed', embyUsers);
|
|
||||||
setStatus(`Names refreshed — ${embyUsers.length} users from Emby.`, 'ok');
|
|
||||||
} catch (e) {
|
} catch (e) {
|
||||||
setStatus(`Failed: ${e.message}`, 'error');
|
setStatus(`Failed: ${e.message}`, 'error');
|
||||||
} finally {
|
} finally {
|
||||||
@@ -148,6 +165,25 @@
|
|||||||
</div>
|
</div>
|
||||||
</section>
|
</section>
|
||||||
|
|
||||||
|
<section>
|
||||||
|
<h4>Recommendations</h4>
|
||||||
|
<label>
|
||||||
|
<span>TMDB API key</span>
|
||||||
|
<input
|
||||||
|
type="password"
|
||||||
|
bind:value={localConfig.tmdbApiKey}
|
||||||
|
placeholder="Paste your TMDB v3 API key"
|
||||||
|
/>
|
||||||
|
</label>
|
||||||
|
<p class="hint">
|
||||||
|
Stored locally in the app config so recommendation features can reuse it without re-entering
|
||||||
|
the key each time.
|
||||||
|
</p>
|
||||||
|
<div class="row">
|
||||||
|
<button class="btn ghost" on:click={saveConfig} disabled={busy}>Save TMDB key</button>
|
||||||
|
</div>
|
||||||
|
</section>
|
||||||
|
|
||||||
<section>
|
<section>
|
||||||
<h4>Database file</h4>
|
<h4>Database file</h4>
|
||||||
<label>
|
<label>
|
||||||
@@ -182,9 +218,9 @@
|
|||||||
padding: 0 4px;
|
padding: 0 4px;
|
||||||
}
|
}
|
||||||
h3 {
|
h3 {
|
||||||
font-size: 15px;
|
font-size: 18px;
|
||||||
font-weight: 700;
|
font-weight: 700;
|
||||||
margin: 0 0 16px;
|
margin: 0 0 18px;
|
||||||
color: var(--text);
|
color: var(--text);
|
||||||
}
|
}
|
||||||
h4 {
|
h4 {
|
||||||
@@ -196,12 +232,15 @@
|
|||||||
margin: 0 0 10px;
|
margin: 0 0 10px;
|
||||||
}
|
}
|
||||||
section {
|
section {
|
||||||
margin-bottom: 24px;
|
margin-bottom: 18px;
|
||||||
padding-bottom: 20px;
|
padding: 16px;
|
||||||
border-bottom: 1px solid var(--border);
|
border: 1px solid var(--border);
|
||||||
|
border-radius: 18px;
|
||||||
|
background: var(--surface);
|
||||||
|
box-shadow: 0 16px 32px rgba(0, 0, 0, 0.16);
|
||||||
}
|
}
|
||||||
section:last-of-type {
|
section:last-of-type {
|
||||||
border-bottom: none;
|
margin-bottom: 12px;
|
||||||
}
|
}
|
||||||
label {
|
label {
|
||||||
display: flex;
|
display: flex;
|
||||||
@@ -215,10 +254,10 @@
|
|||||||
font-weight: 500;
|
font-weight: 500;
|
||||||
}
|
}
|
||||||
input {
|
input {
|
||||||
background: var(--bg);
|
background: #0b0f14;
|
||||||
border: 1px solid var(--border);
|
border: 1px solid var(--border);
|
||||||
border-radius: 6px;
|
border-radius: 12px;
|
||||||
padding: 7px 10px;
|
padding: 10px 12px;
|
||||||
font-size: 13px;
|
font-size: 13px;
|
||||||
color: var(--text);
|
color: var(--text);
|
||||||
font-family: inherit;
|
font-family: inherit;
|
||||||
@@ -227,7 +266,7 @@
|
|||||||
}
|
}
|
||||||
input:focus {
|
input:focus {
|
||||||
outline: none;
|
outline: none;
|
||||||
border-color: var(--accent);
|
border-color: var(--border-strong);
|
||||||
}
|
}
|
||||||
.hint {
|
.hint {
|
||||||
font-size: 11px;
|
font-size: 11px;
|
||||||
@@ -237,9 +276,9 @@
|
|||||||
}
|
}
|
||||||
code {
|
code {
|
||||||
font-size: 11px;
|
font-size: 11px;
|
||||||
background: var(--bg);
|
background: #0b0f14;
|
||||||
padding: 1px 4px;
|
padding: 2px 5px;
|
||||||
border-radius: 3px;
|
border-radius: 6px;
|
||||||
}
|
}
|
||||||
.row {
|
.row {
|
||||||
display: flex;
|
display: flex;
|
||||||
@@ -249,16 +288,17 @@
|
|||||||
.btn {
|
.btn {
|
||||||
all: unset;
|
all: unset;
|
||||||
cursor: pointer;
|
cursor: pointer;
|
||||||
padding: 6px 14px;
|
padding: 9px 14px;
|
||||||
border-radius: 6px;
|
border-radius: 12px;
|
||||||
font-size: 12px;
|
font-size: 12px;
|
||||||
font-weight: 600;
|
font-weight: 600;
|
||||||
font-family: inherit;
|
font-family: inherit;
|
||||||
transition: all 0.12s;
|
transition: all 0.12s;
|
||||||
}
|
}
|
||||||
.btn.ghost {
|
.btn.ghost {
|
||||||
color: var(--text-muted);
|
color: #d9e7f8;
|
||||||
border: 1px solid var(--border);
|
border: 1px solid var(--border);
|
||||||
|
background: var(--bg-secondary);
|
||||||
}
|
}
|
||||||
.btn.ghost:hover:not(:disabled) {
|
.btn.ghost:hover:not(:disabled) {
|
||||||
background: var(--surface-hover);
|
background: var(--surface-hover);
|
||||||
@@ -277,10 +317,10 @@
|
|||||||
}
|
}
|
||||||
.status {
|
.status {
|
||||||
margin-top: 8px;
|
margin-top: 8px;
|
||||||
padding: 8px 12px;
|
padding: 10px 12px;
|
||||||
border-radius: 6px;
|
border-radius: 14px;
|
||||||
font-size: 12px;
|
font-size: 12px;
|
||||||
background: var(--surface-hover);
|
background: var(--surface);
|
||||||
color: var(--text-muted);
|
color: var(--text-muted);
|
||||||
border-left: 3px solid var(--border);
|
border-left: 3px solid var(--border);
|
||||||
}
|
}
|
||||||
|
|||||||
+11
-4
@@ -23,8 +23,15 @@
|
|||||||
}
|
}
|
||||||
</script>
|
</script>
|
||||||
|
|
||||||
<div class="overlay" on:click|self={() => dispatch('close')} on:keydown={() => {}}>
|
<div
|
||||||
<div class="modal">
|
class="overlay"
|
||||||
|
role="button"
|
||||||
|
tabindex="0"
|
||||||
|
aria-label="Close generated SQL modal"
|
||||||
|
on:click|self={() => dispatch('close')}
|
||||||
|
on:keydown={(event) => (event.key === 'Escape' || event.key === 'Enter') && dispatch('close')}
|
||||||
|
>
|
||||||
|
<div class="modal" role="dialog" aria-modal="true" aria-label="Generated SQL">
|
||||||
<div class="modal-header">
|
<div class="modal-header">
|
||||||
<h3>Generated SQL</h3>
|
<h3>Generated SQL</h3>
|
||||||
<button class="close-btn" on:click={() => dispatch('close')}>×</button>
|
<button class="close-btn" on:click={() => dispatch('close')}>×</button>
|
||||||
@@ -47,7 +54,7 @@
|
|||||||
.overlay {
|
.overlay {
|
||||||
position: fixed;
|
position: fixed;
|
||||||
inset: 0;
|
inset: 0;
|
||||||
background: rgba(0, 0, 0, 0.5);
|
background: rgba(0, 0, 0, 0.72);
|
||||||
display: flex;
|
display: flex;
|
||||||
align-items: center;
|
align-items: center;
|
||||||
justify-content: center;
|
justify-content: center;
|
||||||
@@ -55,7 +62,7 @@
|
|||||||
padding: 20px;
|
padding: 20px;
|
||||||
}
|
}
|
||||||
.modal {
|
.modal {
|
||||||
background: var(--bg);
|
background: var(--surface-strong);
|
||||||
border-radius: 14px;
|
border-radius: 14px;
|
||||||
width: 100%;
|
width: 100%;
|
||||||
max-width: 720px;
|
max-width: 720px;
|
||||||
|
|||||||
+23
-20
@@ -158,9 +158,9 @@
|
|||||||
padding: 0;
|
padding: 0;
|
||||||
}
|
}
|
||||||
h3 {
|
h3 {
|
||||||
font-size: 15px;
|
font-size: 18px;
|
||||||
font-weight: 600;
|
font-weight: 700;
|
||||||
margin: 0 0 16px;
|
margin: 0 0 18px;
|
||||||
color: var(--text);
|
color: var(--text);
|
||||||
}
|
}
|
||||||
.sync-step {
|
.sync-step {
|
||||||
@@ -183,10 +183,10 @@
|
|||||||
}
|
}
|
||||||
.select-input {
|
.select-input {
|
||||||
width: 100%;
|
width: 100%;
|
||||||
padding: 8px 10px;
|
padding: 10px 12px;
|
||||||
border-radius: 6px;
|
border-radius: 12px;
|
||||||
border: 1px solid var(--border);
|
border: 1px solid var(--border);
|
||||||
background: var(--bg);
|
background: #0b0f14;
|
||||||
color: var(--text);
|
color: var(--text);
|
||||||
font-size: 13px;
|
font-size: 13px;
|
||||||
font-family: inherit;
|
font-family: inherit;
|
||||||
@@ -201,15 +201,16 @@
|
|||||||
display: flex;
|
display: flex;
|
||||||
align-items: flex-start;
|
align-items: flex-start;
|
||||||
gap: 8px;
|
gap: 8px;
|
||||||
padding: 8px 12px;
|
padding: 10px 12px;
|
||||||
border-radius: 8px;
|
border-radius: 16px;
|
||||||
border: 1px solid var(--border);
|
border: 1px solid var(--border);
|
||||||
|
background: var(--surface);
|
||||||
cursor: pointer;
|
cursor: pointer;
|
||||||
transition: all 0.12s;
|
transition: all 0.12s;
|
||||||
}
|
}
|
||||||
.mode-option.active {
|
.mode-option.active {
|
||||||
border-color: var(--accent);
|
border-color: var(--border-strong);
|
||||||
background: color-mix(in srgb, var(--accent) 8%, transparent);
|
background: var(--surface-active);
|
||||||
}
|
}
|
||||||
.mode-option input {
|
.mode-option input {
|
||||||
margin-top: 2px;
|
margin-top: 2px;
|
||||||
@@ -235,8 +236,8 @@
|
|||||||
display: flex;
|
display: flex;
|
||||||
align-items: center;
|
align-items: center;
|
||||||
gap: 8px;
|
gap: 8px;
|
||||||
padding: 6px 10px;
|
padding: 8px 10px;
|
||||||
border-radius: 6px;
|
border-radius: 12px;
|
||||||
cursor: pointer;
|
cursor: pointer;
|
||||||
font-size: 13px;
|
font-size: 13px;
|
||||||
}
|
}
|
||||||
@@ -244,7 +245,7 @@
|
|||||||
background: var(--surface-hover);
|
background: var(--surface-hover);
|
||||||
}
|
}
|
||||||
.section-pick.selected {
|
.section-pick.selected {
|
||||||
background: color-mix(in srgb, var(--accent) 10%, transparent);
|
background: var(--surface-active);
|
||||||
}
|
}
|
||||||
.pick-name {
|
.pick-name {
|
||||||
flex: 1;
|
flex: 1;
|
||||||
@@ -264,8 +265,8 @@
|
|||||||
display: flex;
|
display: flex;
|
||||||
align-items: center;
|
align-items: center;
|
||||||
gap: 8px;
|
gap: 8px;
|
||||||
padding: 6px 10px;
|
padding: 8px 10px;
|
||||||
border-radius: 6px;
|
border-radius: 12px;
|
||||||
cursor: pointer;
|
cursor: pointer;
|
||||||
font-size: 13px;
|
font-size: 13px;
|
||||||
}
|
}
|
||||||
@@ -273,7 +274,7 @@
|
|||||||
background: var(--surface-hover);
|
background: var(--surface-hover);
|
||||||
}
|
}
|
||||||
.target-option.selected {
|
.target-option.selected {
|
||||||
background: color-mix(in srgb, var(--accent) 10%, transparent);
|
background: var(--surface-active);
|
||||||
}
|
}
|
||||||
.target-name {
|
.target-name {
|
||||||
flex: 1;
|
flex: 1;
|
||||||
@@ -300,18 +301,20 @@
|
|||||||
display: block;
|
display: block;
|
||||||
width: 100%;
|
width: 100%;
|
||||||
text-align: center;
|
text-align: center;
|
||||||
padding: 10px;
|
padding: 12px;
|
||||||
border-radius: 8px;
|
border-radius: 16px;
|
||||||
background: var(--accent);
|
background: var(--accent);
|
||||||
color: #fff;
|
color: #fff;
|
||||||
font-size: 14px;
|
font-size: 14px;
|
||||||
font-weight: 600;
|
font-weight: 600;
|
||||||
font-family: inherit;
|
font-family: inherit;
|
||||||
box-sizing: border-box;
|
box-sizing: border-box;
|
||||||
transition: opacity 0.12s;
|
box-shadow: 0 10px 24px rgba(0, 0, 0, 0.2);
|
||||||
|
transition: opacity 0.12s, transform 0.12s ease;
|
||||||
}
|
}
|
||||||
.sync-btn:hover:not(:disabled) {
|
.sync-btn:hover:not(:disabled) {
|
||||||
opacity: 0.9;
|
opacity: 0.95;
|
||||||
|
transform: translateY(-1px);
|
||||||
}
|
}
|
||||||
.sync-btn:disabled {
|
.sync-btn:disabled {
|
||||||
opacity: 0.4;
|
opacity: 0.4;
|
||||||
|
|||||||
@@ -0,0 +1,386 @@
|
|||||||
|
export function normalizeLookupItem(item) {
|
||||||
|
const derivedYear = Number(
|
||||||
|
String(
|
||||||
|
item?.year ??
|
||||||
|
item?.ProductionYear ??
|
||||||
|
item?.release_date ??
|
||||||
|
item?.first_air_date ??
|
||||||
|
''
|
||||||
|
).slice(0, 4)
|
||||||
|
) || null;
|
||||||
|
const providerIds =
|
||||||
|
item?.providerIds ||
|
||||||
|
item?.ProviderIds ||
|
||||||
|
(item?.tmdbId ? { Tmdb: String(item.tmdbId) } : {});
|
||||||
|
|
||||||
|
return {
|
||||||
|
id: String(item?.id ?? item?.Id ?? ''),
|
||||||
|
name: item?.name || item?.Name || item?.SeriesName || item?.title || 'Unnamed item',
|
||||||
|
type:
|
||||||
|
item?.type ||
|
||||||
|
item?.Type ||
|
||||||
|
item?.CollectionType ||
|
||||||
|
(item?.mediaType === 'tv' ? 'Series' : item?.mediaType === 'movie' ? 'Movie' : 'Item'),
|
||||||
|
overview: item?.overview || item?.Overview || '',
|
||||||
|
year: derivedYear,
|
||||||
|
communityRating: item?.communityRating ?? item?.CommunityRating ?? item?.voteAverage ?? item?.vote_average ?? null,
|
||||||
|
providerIds,
|
||||||
|
genres: Array.isArray(item?.genres)
|
||||||
|
? item.genres.filter(Boolean)
|
||||||
|
: Array.isArray(item?.Genres)
|
||||||
|
? item.Genres.filter(Boolean)
|
||||||
|
: Array.isArray(item?.GenreItems)
|
||||||
|
? item.GenreItems.map((genre) => genre?.Name).filter(Boolean)
|
||||||
|
: [],
|
||||||
|
parentId: item?.parentId ?? item?.ParentId ?? null
|
||||||
|
};
|
||||||
|
}
|
||||||
|
|
||||||
|
const POSITIVE_GENRES = {
|
||||||
|
Romance: 18,
|
||||||
|
Comedy: 16,
|
||||||
|
Drama: 10,
|
||||||
|
Music: 4
|
||||||
|
};
|
||||||
|
|
||||||
|
const NEGATIVE_GENRES = {
|
||||||
|
Animation: -30,
|
||||||
|
Horror: -25,
|
||||||
|
'Science Fiction': -20,
|
||||||
|
Action: -16,
|
||||||
|
Thriller: -14,
|
||||||
|
Crime: -10,
|
||||||
|
War: -18,
|
||||||
|
Western: -14,
|
||||||
|
Documentary: -18,
|
||||||
|
Fantasy: -8,
|
||||||
|
Family: -6
|
||||||
|
};
|
||||||
|
|
||||||
|
const POSITIVE_TERMS = [
|
||||||
|
'wedding',
|
||||||
|
'love',
|
||||||
|
'romance',
|
||||||
|
'relationship',
|
||||||
|
'bride',
|
||||||
|
'best friend',
|
||||||
|
'friendship',
|
||||||
|
'family',
|
||||||
|
'holiday',
|
||||||
|
'food',
|
||||||
|
'small town',
|
||||||
|
'bookstore',
|
||||||
|
'restaurant',
|
||||||
|
'writer',
|
||||||
|
'divorce',
|
||||||
|
'second chance',
|
||||||
|
'starting over',
|
||||||
|
'feel-good',
|
||||||
|
'mother',
|
||||||
|
'daughter',
|
||||||
|
'sisters',
|
||||||
|
'chosen family',
|
||||||
|
'comedy of manners'
|
||||||
|
];
|
||||||
|
|
||||||
|
const NEGATIVE_TERMS = [
|
||||||
|
'war',
|
||||||
|
'serial killer',
|
||||||
|
'murder spree',
|
||||||
|
'mercenary',
|
||||||
|
'zombie',
|
||||||
|
'post-apocalyptic',
|
||||||
|
'gang',
|
||||||
|
'assassin',
|
||||||
|
'combat',
|
||||||
|
'superhero',
|
||||||
|
'multiverse',
|
||||||
|
'alien invasion',
|
||||||
|
'dystopian',
|
||||||
|
'dragon',
|
||||||
|
'animated adventure'
|
||||||
|
];
|
||||||
|
|
||||||
|
const SEED_BONUS_TERMS = [
|
||||||
|
'ensemble',
|
||||||
|
'relationship',
|
||||||
|
'family',
|
||||||
|
'comedy',
|
||||||
|
'romantic',
|
||||||
|
'wedding',
|
||||||
|
'friendship',
|
||||||
|
'identity',
|
||||||
|
'midlife',
|
||||||
|
'second chance'
|
||||||
|
];
|
||||||
|
|
||||||
|
export const RECOMMENDATION_PROFILES = {
|
||||||
|
balanced: {
|
||||||
|
id: 'balanced',
|
||||||
|
label: 'Balanced',
|
||||||
|
description: 'Default Emby-style recommendation overlap ranking.'
|
||||||
|
},
|
||||||
|
classicComfort: {
|
||||||
|
id: 'classicComfort',
|
||||||
|
label: 'Classic Comfort',
|
||||||
|
description: 'Bias toward older, warm, highly-rated movies and shows inspired by test.py.'
|
||||||
|
}
|
||||||
|
};
|
||||||
|
|
||||||
|
function normalizeLimit(limit) {
|
||||||
|
return Math.min(Math.max(Number(limit || 24), 1), 48);
|
||||||
|
}
|
||||||
|
|
||||||
|
function resolveRankingOptions(limitOrOptions) {
|
||||||
|
if (typeof limitOrOptions === 'number' || limitOrOptions === undefined) {
|
||||||
|
return {
|
||||||
|
limit: normalizeLimit(limitOrOptions),
|
||||||
|
profile: 'balanced',
|
||||||
|
seeds: [],
|
||||||
|
excludeIds: [],
|
||||||
|
allowedTypes: []
|
||||||
|
};
|
||||||
|
}
|
||||||
|
|
||||||
|
return {
|
||||||
|
limit: normalizeLimit(limitOrOptions?.limit),
|
||||||
|
profile: RECOMMENDATION_PROFILES[limitOrOptions?.profile] ? limitOrOptions.profile : 'balanced',
|
||||||
|
seeds: Array.isArray(limitOrOptions?.seeds) ? limitOrOptions.seeds.map(normalizeLookupItem) : [],
|
||||||
|
excludeIds: Array.isArray(limitOrOptions?.excludeIds)
|
||||||
|
? limitOrOptions.excludeIds.map((id) => String(id).trim()).filter(Boolean)
|
||||||
|
: [],
|
||||||
|
allowedTypes: Array.isArray(limitOrOptions?.allowedTypes)
|
||||||
|
? limitOrOptions.allowedTypes.map((type) => String(type).trim()).filter(Boolean)
|
||||||
|
: []
|
||||||
|
};
|
||||||
|
}
|
||||||
|
|
||||||
|
function genreBiasScore(genreNames, overview) {
|
||||||
|
let score = 0;
|
||||||
|
|
||||||
|
for (const genre of genreNames || []) {
|
||||||
|
score += POSITIVE_GENRES[genre] || 0;
|
||||||
|
score += NEGATIVE_GENRES[genre] || 0;
|
||||||
|
}
|
||||||
|
|
||||||
|
const text = String(overview || '').toLowerCase();
|
||||||
|
for (const term of POSITIVE_TERMS) {
|
||||||
|
if (text.includes(term)) score += 2;
|
||||||
|
}
|
||||||
|
for (const term of NEGATIVE_TERMS) {
|
||||||
|
if (text.includes(term)) score -= 2;
|
||||||
|
}
|
||||||
|
|
||||||
|
return score;
|
||||||
|
}
|
||||||
|
|
||||||
|
function classicBonus(year, seedHits, overview, seedGenres) {
|
||||||
|
let bonus = 0;
|
||||||
|
|
||||||
|
if (year !== null && year !== undefined) {
|
||||||
|
if (year >= 1985 && year <= 2008) bonus += 10;
|
||||||
|
else if (year >= 2009 && year <= 2012) bonus += 4;
|
||||||
|
else if (year < 1985) bonus += 2;
|
||||||
|
else bonus -= 8;
|
||||||
|
}
|
||||||
|
|
||||||
|
bonus += Math.min(seedHits * 5, 20);
|
||||||
|
|
||||||
|
const text = String(overview || '').toLowerCase();
|
||||||
|
for (const term of SEED_BONUS_TERMS) {
|
||||||
|
if (text.includes(term)) bonus += 1;
|
||||||
|
}
|
||||||
|
|
||||||
|
for (const genre of seedGenres) {
|
||||||
|
if ((overview || '').toLowerCase().includes(genre.toLowerCase())) {
|
||||||
|
bonus += 1;
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
return bonus;
|
||||||
|
}
|
||||||
|
|
||||||
|
function buildSeedContext(seeds) {
|
||||||
|
const genreHits = new Set();
|
||||||
|
const mediaTypes = new Set();
|
||||||
|
for (const seed of seeds || []) {
|
||||||
|
for (const genre of seed.genres || []) {
|
||||||
|
genreHits.add(genre);
|
||||||
|
}
|
||||||
|
if (seed.type) mediaTypes.add(seed.type);
|
||||||
|
}
|
||||||
|
return { genres: genreHits, mediaTypes };
|
||||||
|
}
|
||||||
|
|
||||||
|
function scoreSeedAffinity(item, seedContext) {
|
||||||
|
const genres = new Set(item.genres || []);
|
||||||
|
const text = `${item.name || ''} ${item.overview || ''}`.toLowerCase();
|
||||||
|
let score = 0;
|
||||||
|
|
||||||
|
for (const genre of seedContext.genres) {
|
||||||
|
if (genres.has(genre)) score += 8;
|
||||||
|
}
|
||||||
|
|
||||||
|
const warmSeed =
|
||||||
|
seedContext.genres.has('Romance') ||
|
||||||
|
seedContext.genres.has('Drama') ||
|
||||||
|
seedContext.genres.has('Comedy') ||
|
||||||
|
seedContext.genres.has('Family');
|
||||||
|
|
||||||
|
if (warmSeed) {
|
||||||
|
if (genres.has('Romance')) score += 12;
|
||||||
|
if (genres.has('Drama')) score += 10;
|
||||||
|
if (genres.has('Comedy')) score += 8;
|
||||||
|
if (genres.has('Family')) score += 5;
|
||||||
|
if (genres.has('Horror')) score -= 28;
|
||||||
|
if (genres.has('Science Fiction')) score -= 18;
|
||||||
|
if (genres.has('Action')) score -= 16;
|
||||||
|
if (genres.has('Thriller')) score -= 14;
|
||||||
|
if (genres.has('War')) score -= 10;
|
||||||
|
if (genres.has('Crime')) score -= 8;
|
||||||
|
|
||||||
|
for (const term of POSITIVE_TERMS) {
|
||||||
|
if (text.includes(term)) score += 1;
|
||||||
|
}
|
||||||
|
for (const term of NEGATIVE_TERMS) {
|
||||||
|
if (text.includes(term)) score -= 2;
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
return score;
|
||||||
|
}
|
||||||
|
|
||||||
|
function normalizeResultSet(resultSet) {
|
||||||
|
if (Array.isArray(resultSet)) {
|
||||||
|
return {
|
||||||
|
items: resultSet,
|
||||||
|
sourceWeight: 1
|
||||||
|
};
|
||||||
|
}
|
||||||
|
|
||||||
|
return {
|
||||||
|
items: Array.isArray(resultSet?.items) ? resultSet.items : [],
|
||||||
|
sourceWeight: Math.max(1, Number(resultSet?.sourceWeight || 1))
|
||||||
|
};
|
||||||
|
}
|
||||||
|
|
||||||
|
function evaluateClassicComfort(item, matches, seedContext) {
|
||||||
|
const genres = item.genres || [];
|
||||||
|
const year = Number.isFinite(Number(item.year)) ? Number(item.year) : null;
|
||||||
|
const rating = Number.isFinite(Number(item.communityRating)) ? Number(item.communityRating) : null;
|
||||||
|
const hasAnimation = genres.includes('Animation');
|
||||||
|
|
||||||
|
if (hasAnimation) return { keep: false };
|
||||||
|
if (year !== null && (year < 1980 || year > 2012)) return { keep: false };
|
||||||
|
if (rating !== null && rating < 6.3) return { keep: false };
|
||||||
|
|
||||||
|
const bias = genreBiasScore(genres, item.overview);
|
||||||
|
const bonus = classicBonus(year, matches, `${item.name} ${item.overview}`, seedContext.genres);
|
||||||
|
const styleScore = bias + bonus;
|
||||||
|
|
||||||
|
if (styleScore < 16) {
|
||||||
|
return { keep: false };
|
||||||
|
}
|
||||||
|
|
||||||
|
return {
|
||||||
|
keep: true,
|
||||||
|
styleScore,
|
||||||
|
qualityScore: rating !== null ? Math.round(rating * 10) : null
|
||||||
|
};
|
||||||
|
}
|
||||||
|
|
||||||
|
export function rankRecommendationResults(seedIds, resultSets, limitOrOptions = 24) {
|
||||||
|
const options = resolveRankingOptions(limitOrOptions);
|
||||||
|
const excluded = new Set([
|
||||||
|
...(seedIds || []).map((id) => String(id).trim()),
|
||||||
|
...options.excludeIds
|
||||||
|
].filter(Boolean));
|
||||||
|
const allowedTypes = new Set(options.allowedTypes);
|
||||||
|
const scored = new Map();
|
||||||
|
|
||||||
|
for (const resultSet of resultSets || []) {
|
||||||
|
const normalizedSet = normalizeResultSet(resultSet);
|
||||||
|
const seenInSet = new Set();
|
||||||
|
const items = normalizedSet.items;
|
||||||
|
for (let index = 0; index < items.length; index += 1) {
|
||||||
|
const item = normalizeLookupItem(items[index]);
|
||||||
|
if (!item.id || excluded.has(item.id)) continue;
|
||||||
|
if (allowedTypes.size > 0 && !allowedTypes.has(item.type)) continue;
|
||||||
|
if (seenInSet.has(item.id)) continue;
|
||||||
|
seenInSet.add(item.id);
|
||||||
|
|
||||||
|
const weight = Math.max(1, items.length - index) * normalizedSet.sourceWeight;
|
||||||
|
const current = scored.get(item.id) || {
|
||||||
|
item,
|
||||||
|
score: 0,
|
||||||
|
matches: 0,
|
||||||
|
sourceStrength: 0,
|
||||||
|
bestRank: Number.POSITIVE_INFINITY
|
||||||
|
};
|
||||||
|
|
||||||
|
current.item = item;
|
||||||
|
current.score += weight;
|
||||||
|
current.matches += 1;
|
||||||
|
current.sourceStrength += normalizedSet.sourceWeight;
|
||||||
|
current.bestRank = Math.min(current.bestRank, index);
|
||||||
|
scored.set(item.id, current);
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
const seedContext = buildSeedContext(options.seeds);
|
||||||
|
|
||||||
|
return [...scored.values()]
|
||||||
|
.map((entry) => {
|
||||||
|
if (options.profile !== 'classicComfort') {
|
||||||
|
const affinityScore = scoreSeedAffinity(entry.item, seedContext);
|
||||||
|
return {
|
||||||
|
...entry,
|
||||||
|
affinityScore,
|
||||||
|
totalScore:
|
||||||
|
(entry.sourceStrength * 20) +
|
||||||
|
entry.score +
|
||||||
|
affinityScore,
|
||||||
|
styleScore: null,
|
||||||
|
qualityScore: entry.item.communityRating !== null
|
||||||
|
? Math.round(Number(entry.item.communityRating) * 10)
|
||||||
|
: null
|
||||||
|
};
|
||||||
|
}
|
||||||
|
|
||||||
|
const style = evaluateClassicComfort(entry.item, entry.matches, seedContext);
|
||||||
|
if (!style.keep) return null;
|
||||||
|
|
||||||
|
return {
|
||||||
|
...entry,
|
||||||
|
styleScore: style.styleScore,
|
||||||
|
qualityScore: style.qualityScore,
|
||||||
|
totalScore:
|
||||||
|
(entry.sourceStrength * 20) +
|
||||||
|
entry.score +
|
||||||
|
(style.styleScore * 2) +
|
||||||
|
(style.qualityScore || 0)
|
||||||
|
};
|
||||||
|
})
|
||||||
|
.filter(Boolean)
|
||||||
|
.sort((a, b) => {
|
||||||
|
if (b.totalScore !== a.totalScore) return b.totalScore - a.totalScore;
|
||||||
|
if ((b.sourceStrength || 0) !== (a.sourceStrength || 0)) return (b.sourceStrength || 0) - (a.sourceStrength || 0);
|
||||||
|
if (b.matches !== a.matches) return b.matches - a.matches;
|
||||||
|
if ((b.styleScore || 0) !== (a.styleScore || 0)) return (b.styleScore || 0) - (a.styleScore || 0);
|
||||||
|
if ((b.qualityScore || 0) !== (a.qualityScore || 0)) return (b.qualityScore || 0) - (a.qualityScore || 0);
|
||||||
|
if (b.score !== a.score) return b.score - a.score;
|
||||||
|
if (a.bestRank !== b.bestRank) return a.bestRank - b.bestRank;
|
||||||
|
return a.item.name.localeCompare(b.item.name);
|
||||||
|
})
|
||||||
|
.slice(0, options.limit)
|
||||||
|
.map((entry) => ({
|
||||||
|
...entry.item,
|
||||||
|
matchCount: entry.matches,
|
||||||
|
score: entry.score,
|
||||||
|
totalScore: entry.totalScore,
|
||||||
|
sourceStrength: entry.sourceStrength,
|
||||||
|
affinityScore: entry.affinityScore ?? null,
|
||||||
|
styleScore: entry.styleScore,
|
||||||
|
qualityScore: entry.qualityScore
|
||||||
|
}));
|
||||||
|
}
|
||||||
+120
-14
@@ -43,6 +43,7 @@ export const ITEM_TYPES = ['Movie', 'Series', 'Episode', 'BoxSet'];
|
|||||||
export const SORT_OPTIONS = [
|
export const SORT_OPTIONS = [
|
||||||
{ value: '', label: '(none)' },
|
{ value: '', label: '(none)' },
|
||||||
{ value: 'default', label: 'Default (boxset)' },
|
{ value: 'default', label: 'Default (boxset)' },
|
||||||
|
{ value: 'DatePlayed', label: 'Date played' },
|
||||||
{ value: 'DateLastContentAdded,SortName', label: 'Date added' },
|
{ value: 'DateLastContentAdded,SortName', label: 'Date added' },
|
||||||
{ value: 'ProductionYear,PremiereDate,SortName', label: 'Release year' },
|
{ value: 'ProductionYear,PremiereDate,SortName', label: 'Release year' },
|
||||||
{ value: 'CommunityRating', label: 'Community rating' },
|
{ value: 'CommunityRating', label: 'Community rating' },
|
||||||
@@ -58,6 +59,24 @@ export const IMAGE_TYPES = [
|
|||||||
{ value: 'Primary', label: 'Primary / Poster' }
|
{ value: 'Primary', label: 'Primary / Poster' }
|
||||||
];
|
];
|
||||||
|
|
||||||
|
export const PAGE_ICONS = {
|
||||||
|
edit: 'edit',
|
||||||
|
sync: 'sync',
|
||||||
|
collections: 'collections',
|
||||||
|
settings: 'settings'
|
||||||
|
};
|
||||||
|
|
||||||
|
export const SECTION_ICONS = {
|
||||||
|
resume: 'resume',
|
||||||
|
items: 'items',
|
||||||
|
userviews: 'userviews',
|
||||||
|
boxset: 'boxset',
|
||||||
|
collections: 'collections',
|
||||||
|
latestepisodereleases: 'latestepisodereleases',
|
||||||
|
latestmoviereleases: 'latestmoviereleases',
|
||||||
|
latestmediablock: 'latestmediablock'
|
||||||
|
};
|
||||||
|
|
||||||
export function genId() {
|
export function genId() {
|
||||||
return crypto.randomUUID().replace(/-/g, '').slice(0, 32);
|
return crypto.randomUUID().replace(/-/g, '').slice(0, 32);
|
||||||
}
|
}
|
||||||
@@ -88,11 +107,64 @@ export function createEmptySection(userId) {
|
|||||||
};
|
};
|
||||||
}
|
}
|
||||||
|
|
||||||
|
export function createRecentlyWatchedSection(userId, userName = '') {
|
||||||
|
return {
|
||||||
|
UserId: userId,
|
||||||
|
Name: `Recently Watched${userName ? ` - ${userName}` : ''}`,
|
||||||
|
CustomName: `Recently Watched${userName ? ` - ${userName}` : ''}`,
|
||||||
|
Id: genId(),
|
||||||
|
SectionType: 'items',
|
||||||
|
ImageType: 'Thumb',
|
||||||
|
CollectionType: '',
|
||||||
|
SortBy: 'DatePlayed',
|
||||||
|
SortOrder: 'Descending',
|
||||||
|
Monitor: [],
|
||||||
|
ItemTypes: ['Movie', 'Series'],
|
||||||
|
ExcludedFolders: [],
|
||||||
|
CardSizeOffset: 0,
|
||||||
|
IncludeNextUpInResume: true,
|
||||||
|
Query: {
|
||||||
|
StudioIds: [],
|
||||||
|
TagIds: [],
|
||||||
|
GenreIds: [],
|
||||||
|
CollectionTypes: [],
|
||||||
|
IsPlayed: true
|
||||||
|
}
|
||||||
|
};
|
||||||
|
}
|
||||||
|
|
||||||
|
export function createBoxSetSection(userId, collectionName, collectionId) {
|
||||||
|
return {
|
||||||
|
UserId: userId,
|
||||||
|
Name: collectionName || 'New Collection',
|
||||||
|
CustomName: collectionName || 'New Collection',
|
||||||
|
Id: genId(),
|
||||||
|
SectionType: 'boxset',
|
||||||
|
ImageType: 'Thumb',
|
||||||
|
ItemTypes: [],
|
||||||
|
SortBy: 'default',
|
||||||
|
SortOrder: 'Descending',
|
||||||
|
Monitor: [],
|
||||||
|
ExcludedFolders: [],
|
||||||
|
CardSizeOffset: 0,
|
||||||
|
IncludeNextUpInResume: true,
|
||||||
|
ParentItem: {
|
||||||
|
Name: collectionName || 'New Collection',
|
||||||
|
Id: String(collectionId || '')
|
||||||
|
},
|
||||||
|
ParentId: String(collectionId || '')
|
||||||
|
};
|
||||||
|
}
|
||||||
|
|
||||||
export function getSectionTypeLabel(type) {
|
export function getSectionTypeLabel(type) {
|
||||||
const found = SECTION_TYPES.find((t) => t.value === type);
|
const found = SECTION_TYPES.find((t) => t.value === type);
|
||||||
return found ? found.label : type;
|
return found ? found.label : type;
|
||||||
}
|
}
|
||||||
|
|
||||||
|
export function getSectionIconName(type) {
|
||||||
|
return SECTION_ICONS[type] || 'spark';
|
||||||
|
}
|
||||||
|
|
||||||
export function getGenreNames(genreIds) {
|
export function getGenreNames(genreIds) {
|
||||||
if (!genreIds || genreIds.length === 0) return '';
|
if (!genreIds || genreIds.length === 0) return '';
|
||||||
return genreIds.map((id) => GENRES[id] || id).join(', ');
|
return genreIds.map((id) => GENRES[id] || id).join(', ');
|
||||||
@@ -101,14 +173,8 @@ export function getGenreNames(genreIds) {
|
|||||||
export function extractUserName(sections) {
|
export function extractUserName(sections) {
|
||||||
for (const s of sections) {
|
for (const s of sections) {
|
||||||
const name = s.CustomName || s.Name || '';
|
const name = s.CustomName || s.Name || '';
|
||||||
const lower = name.toLowerCase();
|
const extracted = extractWatchlistOwnerName(name);
|
||||||
if (lower.includes('watchlist')) {
|
if (extracted && extracted !== name) return extracted;
|
||||||
const extracted = name
|
|
||||||
.replace(/['']s\s*Watchlist/i, '')
|
|
||||||
.replace(/\s*Watchlist/i, '')
|
|
||||||
.trim();
|
|
||||||
if (extracted && extracted !== name) return extracted;
|
|
||||||
}
|
|
||||||
}
|
}
|
||||||
return null;
|
return null;
|
||||||
}
|
}
|
||||||
@@ -122,11 +188,11 @@ export function isWatchlistSection(section) {
|
|||||||
|
|
||||||
function renameWatchlistLabel(label, targetName) {
|
function renameWatchlistLabel(label, targetName) {
|
||||||
if (!label || !targetName) return label;
|
if (!label || !targetName) return label;
|
||||||
if (/^\s*watch\s*list\s*$/i.test(label)) return 'Watch List';
|
if (/^\s*watch\s+list\s*$/i.test(label)) return 'Watch List';
|
||||||
if (/^\s*watchlist\s*$/i.test(label)) return 'Watchlist';
|
if (/^\s*watchlist\s*$/i.test(label)) return 'Watchlist';
|
||||||
|
|
||||||
if (/watch\s*list/i.test(label)) {
|
if (/watch\s+list/i.test(label)) {
|
||||||
return label.replace(/^.*?watch\s*list/i, `${targetName}'s Watch List`);
|
return label.replace(/^.*?watch\s+list/i, `${targetName}'s Watch List`);
|
||||||
}
|
}
|
||||||
|
|
||||||
if (/watchlist/i.test(label)) {
|
if (/watchlist/i.test(label)) {
|
||||||
@@ -136,17 +202,57 @@ function renameWatchlistLabel(label, targetName) {
|
|||||||
return label;
|
return label;
|
||||||
}
|
}
|
||||||
|
|
||||||
|
function extractWatchlistOwnerName(label) {
|
||||||
|
if (!label) return '';
|
||||||
|
const normalized = String(label).trim();
|
||||||
|
if (!/watchlist|watch\s+list/i.test(normalized)) return '';
|
||||||
|
|
||||||
|
return normalized
|
||||||
|
.replace(/\bwatchlist\b/i, '')
|
||||||
|
.replace(/\bwatch\s+list\b/i, '')
|
||||||
|
.replace(/['’]s$/i, '')
|
||||||
|
.trim();
|
||||||
|
}
|
||||||
|
|
||||||
|
function normalizeNameForCompare(name) {
|
||||||
|
return String(name || '')
|
||||||
|
.toLowerCase()
|
||||||
|
.replace(/['’]/g, '')
|
||||||
|
.replace(/[^a-z0-9]+/g, '');
|
||||||
|
}
|
||||||
|
|
||||||
|
function labelsMatchUser(label, targetName) {
|
||||||
|
const owner = extractWatchlistOwnerName(label);
|
||||||
|
if (!owner || !targetName) return false;
|
||||||
|
return normalizeNameForCompare(owner) === normalizeNameForCompare(targetName);
|
||||||
|
}
|
||||||
|
|
||||||
|
function getPreferredUserName(user) {
|
||||||
|
return user?.embyName || user?.name || '';
|
||||||
|
}
|
||||||
|
|
||||||
export function getWatchlistLabelsForTarget(sourceSection, targetUser) {
|
export function getWatchlistLabelsForTarget(sourceSection, targetUser) {
|
||||||
const existingTargetWatchlist = targetUser?.sections?.find((section) => isWatchlistSection(section));
|
const existingTargetWatchlist = targetUser?.sections?.find((section) => isWatchlistSection(section));
|
||||||
|
const targetName = getPreferredUserName(targetUser);
|
||||||
|
|
||||||
if (existingTargetWatchlist) {
|
if (existingTargetWatchlist) {
|
||||||
|
const existingName = existingTargetWatchlist.CustomName || existingTargetWatchlist.Name || '';
|
||||||
|
if (!targetName || labelsMatchUser(existingName, targetName)) {
|
||||||
|
return {
|
||||||
|
Name: existingTargetWatchlist.Name || sourceSection.Name,
|
||||||
|
CustomName: existingTargetWatchlist.CustomName || existingTargetWatchlist.Name || sourceSection.CustomName || sourceSection.Name
|
||||||
|
};
|
||||||
|
}
|
||||||
|
|
||||||
return {
|
return {
|
||||||
Name: existingTargetWatchlist.Name || sourceSection.Name,
|
Name: renameWatchlistLabel(existingTargetWatchlist.Name || sourceSection.Name, targetName),
|
||||||
CustomName: existingTargetWatchlist.CustomName || existingTargetWatchlist.Name || sourceSection.CustomName || sourceSection.Name
|
CustomName: renameWatchlistLabel(
|
||||||
|
existingTargetWatchlist.CustomName || existingTargetWatchlist.Name || sourceSection.CustomName || sourceSection.Name,
|
||||||
|
targetName
|
||||||
|
)
|
||||||
};
|
};
|
||||||
}
|
}
|
||||||
|
|
||||||
const targetName = targetUser?.name || '';
|
|
||||||
return {
|
return {
|
||||||
Name: renameWatchlistLabel(sourceSection.Name, targetName),
|
Name: renameWatchlistLabel(sourceSection.Name, targetName),
|
||||||
CustomName: renameWatchlistLabel(sourceSection.CustomName, targetName)
|
CustomName: renameWatchlistLabel(sourceSection.CustomName, targetName)
|
||||||
|
|||||||
@@ -0,0 +1,69 @@
|
|||||||
|
import { readFileSync, existsSync } from 'fs';
|
||||||
|
import { resolve } from 'path';
|
||||||
|
|
||||||
|
const CONFIG_PATH = resolve('config.json');
|
||||||
|
|
||||||
|
export function loadEmbyConfig() {
|
||||||
|
if (!existsSync(CONFIG_PATH)) return {};
|
||||||
|
try {
|
||||||
|
return JSON.parse(readFileSync(CONFIG_PATH, 'utf-8'));
|
||||||
|
} catch {
|
||||||
|
return {};
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
export function normalizeEmbyGuid(value) {
|
||||||
|
return typeof value === 'string' ? value.replace(/-/g, '').trim().toLowerCase() : '';
|
||||||
|
}
|
||||||
|
|
||||||
|
export function buildEmbyUrl(pathname, params = {}) {
|
||||||
|
const { embyUrl, apiKey } = loadEmbyConfig();
|
||||||
|
if (!embyUrl || !apiKey) {
|
||||||
|
throw new Error('Emby URL and API key not configured');
|
||||||
|
}
|
||||||
|
|
||||||
|
const base = embyUrl.replace(/\/+$/, '');
|
||||||
|
const path = pathname.startsWith('/') ? pathname : `/${pathname}`;
|
||||||
|
const url = new URL(`${base}${path}`);
|
||||||
|
|
||||||
|
for (const [key, value] of Object.entries({
|
||||||
|
...params,
|
||||||
|
api_key: apiKey
|
||||||
|
})) {
|
||||||
|
if (value === undefined || value === null || value === '') continue;
|
||||||
|
url.searchParams.set(key, String(value));
|
||||||
|
}
|
||||||
|
|
||||||
|
return { url, apiKey };
|
||||||
|
}
|
||||||
|
|
||||||
|
export async function fetchEmby(pathname, options = {}) {
|
||||||
|
const {
|
||||||
|
params = {},
|
||||||
|
method = 'GET',
|
||||||
|
headers = {},
|
||||||
|
body
|
||||||
|
} = options;
|
||||||
|
const { url, apiKey } = buildEmbyUrl(pathname, params);
|
||||||
|
const response = await fetch(url, {
|
||||||
|
method,
|
||||||
|
headers: {
|
||||||
|
Accept: 'application/json',
|
||||||
|
'X-Emby-Token': apiKey,
|
||||||
|
...headers
|
||||||
|
},
|
||||||
|
body
|
||||||
|
});
|
||||||
|
|
||||||
|
if (!response.ok) {
|
||||||
|
const text = await response.text().catch(() => response.statusText);
|
||||||
|
throw new Error(text || `${response.status} ${response.statusText}`);
|
||||||
|
}
|
||||||
|
|
||||||
|
return response;
|
||||||
|
}
|
||||||
|
|
||||||
|
export async function fetchEmbyJson(pathname, options = {}) {
|
||||||
|
const response = await fetchEmby(pathname, options);
|
||||||
|
return response.json();
|
||||||
|
}
|
||||||
@@ -0,0 +1,139 @@
|
|||||||
|
import { existsSync, mkdirSync } from 'fs';
|
||||||
|
import { dirname, resolve } from 'path';
|
||||||
|
import { DatabaseSync } from 'node:sqlite';
|
||||||
|
|
||||||
|
function getCacheDbPath() {
|
||||||
|
return process.env.EMBY_USER_CACHE_DB_PATH || resolve('.cache', 'emby-users.db');
|
||||||
|
}
|
||||||
|
|
||||||
|
function ensureCacheDir() {
|
||||||
|
const dir = dirname(getCacheDbPath());
|
||||||
|
if (!existsSync(dir)) {
|
||||||
|
mkdirSync(dir, { recursive: true });
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
function openCacheDb() {
|
||||||
|
ensureCacheDir();
|
||||||
|
const db = new DatabaseSync(getCacheDbPath());
|
||||||
|
db.exec(`
|
||||||
|
CREATE TABLE IF NOT EXISTS EmbyUsers (
|
||||||
|
embyGuid TEXT PRIMARY KEY,
|
||||||
|
name TEXT NOT NULL,
|
||||||
|
fetchedAt TEXT NOT NULL
|
||||||
|
);
|
||||||
|
CREATE TABLE IF NOT EXISTS CacheMeta (
|
||||||
|
key TEXT PRIMARY KEY,
|
||||||
|
value TEXT NOT NULL
|
||||||
|
);
|
||||||
|
`);
|
||||||
|
return db;
|
||||||
|
}
|
||||||
|
|
||||||
|
function normalizeGuid(value) {
|
||||||
|
return typeof value === 'string' ? value.replace(/-/g, '').trim().toLowerCase() : '';
|
||||||
|
}
|
||||||
|
|
||||||
|
export function normalizeEmbyUsers(users) {
|
||||||
|
if (!Array.isArray(users)) return [];
|
||||||
|
|
||||||
|
return users
|
||||||
|
.map((user) => ({
|
||||||
|
embyGuid: normalizeGuid(user?.embyGuid ?? user?.Id),
|
||||||
|
name: String(user?.name ?? user?.Name ?? '').trim()
|
||||||
|
}))
|
||||||
|
.filter((user) => user.embyGuid && user.name);
|
||||||
|
}
|
||||||
|
|
||||||
|
export function writeCachedEmbyUsers(users) {
|
||||||
|
const normalizedUsers = normalizeEmbyUsers(users);
|
||||||
|
const fetchedAt = new Date().toISOString();
|
||||||
|
const db = openCacheDb();
|
||||||
|
|
||||||
|
try {
|
||||||
|
const clearStmt = db.prepare('DELETE FROM EmbyUsers');
|
||||||
|
const insertStmt = db.prepare(
|
||||||
|
'INSERT INTO EmbyUsers (embyGuid, name, fetchedAt) VALUES (?, ?, ?)'
|
||||||
|
);
|
||||||
|
const metaStmt = db.prepare(
|
||||||
|
'INSERT INTO CacheMeta (key, value) VALUES (?, ?) ON CONFLICT(key) DO UPDATE SET value = excluded.value'
|
||||||
|
);
|
||||||
|
|
||||||
|
db.exec('BEGIN');
|
||||||
|
try {
|
||||||
|
clearStmt.run();
|
||||||
|
for (const user of normalizedUsers) {
|
||||||
|
insertStmt.run(user.embyGuid, user.name, fetchedAt);
|
||||||
|
}
|
||||||
|
metaStmt.run('lastSyncedAt', fetchedAt);
|
||||||
|
db.exec('COMMIT');
|
||||||
|
} catch (error) {
|
||||||
|
db.exec('ROLLBACK');
|
||||||
|
throw error;
|
||||||
|
}
|
||||||
|
} finally {
|
||||||
|
db.close();
|
||||||
|
}
|
||||||
|
|
||||||
|
return { users: normalizedUsers, lastSyncedAt: fetchedAt };
|
||||||
|
}
|
||||||
|
|
||||||
|
export function readCachedEmbyUsers() {
|
||||||
|
if (!existsSync(getCacheDbPath())) {
|
||||||
|
return { users: [], lastSyncedAt: null };
|
||||||
|
}
|
||||||
|
|
||||||
|
const db = openCacheDb();
|
||||||
|
try {
|
||||||
|
const users = db
|
||||||
|
.prepare('SELECT embyGuid, name, fetchedAt FROM EmbyUsers ORDER BY lower(name), embyGuid')
|
||||||
|
.all()
|
||||||
|
.map((row) => ({
|
||||||
|
embyGuid: normalizeGuid(row.embyGuid),
|
||||||
|
name: row.name
|
||||||
|
}));
|
||||||
|
const meta = db.prepare("SELECT value FROM CacheMeta WHERE key = 'lastSyncedAt'").get();
|
||||||
|
|
||||||
|
return {
|
||||||
|
users,
|
||||||
|
lastSyncedAt: meta?.value || users[0]?.fetchedAt || null
|
||||||
|
};
|
||||||
|
} finally {
|
||||||
|
db.close();
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
export function getCachedEmbyUserMap() {
|
||||||
|
return new Map(readCachedEmbyUsers().users.map((user) => [user.embyGuid, user]));
|
||||||
|
}
|
||||||
|
|
||||||
|
export function applyCachedEmbyNames(users) {
|
||||||
|
const cached = readCachedEmbyUsers();
|
||||||
|
const lookup = new Map(cached.users.map((user) => [user.embyGuid, user]));
|
||||||
|
let matchedCount = 0;
|
||||||
|
|
||||||
|
const hydratedUsers = (users || []).map((user) => {
|
||||||
|
const embyGuid = normalizeGuid(user?.embyGuid);
|
||||||
|
const cachedUser = lookup.get(embyGuid);
|
||||||
|
if (!cachedUser) {
|
||||||
|
return user;
|
||||||
|
}
|
||||||
|
|
||||||
|
matchedCount++;
|
||||||
|
return {
|
||||||
|
...user,
|
||||||
|
dbName: user.dbName || user.name,
|
||||||
|
embyName: cachedUser.name,
|
||||||
|
name: cachedUser.name
|
||||||
|
};
|
||||||
|
});
|
||||||
|
|
||||||
|
return {
|
||||||
|
users: hydratedUsers,
|
||||||
|
cache: {
|
||||||
|
matchedCount,
|
||||||
|
totalCachedUsers: lookup.size,
|
||||||
|
lastSyncedAt: cached.lastSyncedAt
|
||||||
|
}
|
||||||
|
};
|
||||||
|
}
|
||||||
@@ -0,0 +1,60 @@
|
|||||||
|
import { existsSync, mkdirSync, readFileSync, writeFileSync } from 'fs';
|
||||||
|
import { dirname, resolve } from 'path';
|
||||||
|
|
||||||
|
const CACHE_PATH = resolve('.cache', 'emby-user-context.json');
|
||||||
|
|
||||||
|
function ensureCacheDir() {
|
||||||
|
const dir = dirname(CACHE_PATH);
|
||||||
|
if (!existsSync(dir)) {
|
||||||
|
mkdirSync(dir, { recursive: true });
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
function normalizeGuid(value) {
|
||||||
|
return typeof value === 'string' ? value.replace(/-/g, '').trim().toLowerCase() : '';
|
||||||
|
}
|
||||||
|
|
||||||
|
function loadCacheFile() {
|
||||||
|
if (!existsSync(CACHE_PATH)) {
|
||||||
|
return { users: {} };
|
||||||
|
}
|
||||||
|
|
||||||
|
try {
|
||||||
|
const parsed = JSON.parse(readFileSync(CACHE_PATH, 'utf8'));
|
||||||
|
return parsed && typeof parsed === 'object' ? parsed : { users: {} };
|
||||||
|
} catch {
|
||||||
|
return { users: {} };
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
function saveCacheFile(cache) {
|
||||||
|
ensureCacheDir();
|
||||||
|
writeFileSync(CACHE_PATH, JSON.stringify(cache, null, 2));
|
||||||
|
}
|
||||||
|
|
||||||
|
export function readCachedEmbyUserContext(embyGuid) {
|
||||||
|
const normalizedGuid = normalizeGuid(embyGuid);
|
||||||
|
if (!normalizedGuid) return null;
|
||||||
|
|
||||||
|
const cache = loadCacheFile();
|
||||||
|
return cache.users?.[normalizedGuid] || null;
|
||||||
|
}
|
||||||
|
|
||||||
|
export function writeCachedEmbyUserContext(embyGuid, context) {
|
||||||
|
const normalizedGuid = normalizeGuid(embyGuid);
|
||||||
|
if (!normalizedGuid) return null;
|
||||||
|
|
||||||
|
const cache = loadCacheFile();
|
||||||
|
const nextContext = {
|
||||||
|
views: Array.isArray(context?.views) ? context.views : [],
|
||||||
|
recentlyPlayed: Array.isArray(context?.recentlyPlayed) ? context.recentlyPlayed : [],
|
||||||
|
excludedFolderLookup: context?.excludedFolderLookup || {},
|
||||||
|
lastSyncedAt: context?.lastSyncedAt || new Date().toISOString()
|
||||||
|
};
|
||||||
|
|
||||||
|
cache.users ||= {};
|
||||||
|
cache.users[normalizedGuid] = nextContext;
|
||||||
|
saveCacheFile(cache);
|
||||||
|
return nextContext;
|
||||||
|
}
|
||||||
|
|
||||||
@@ -168,6 +168,13 @@ export function loadHomeScreenUsers(db) {
|
|||||||
guid: user.guid,
|
guid: user.guid,
|
||||||
embyGuid: user.embyGuid,
|
embyGuid: user.embyGuid,
|
||||||
sections: normalizedSections,
|
sections: normalizedSections,
|
||||||
|
details: {
|
||||||
|
sourceTable: user.sourceTable,
|
||||||
|
lastLoginDate: user.profile?.LastLoginDate || null,
|
||||||
|
lastActivityDate: user.profile?.LastActivityDate || null,
|
||||||
|
usesIdForConfigurationPath: user.profile?.UsesIdForConfigurationPath ?? null,
|
||||||
|
importedCollectionsCount: Array.isArray(user.profile?.ImportedCollections) ? user.profile.ImportedCollections.length : 0
|
||||||
|
},
|
||||||
match: {
|
match: {
|
||||||
sourceTable: user.sourceTable,
|
sourceTable: user.sourceTable,
|
||||||
settingsUserId: user.id,
|
settingsUserId: user.id,
|
||||||
|
|||||||
@@ -0,0 +1,123 @@
|
|||||||
|
import { readFileSync, existsSync } from 'fs';
|
||||||
|
import { resolve } from 'path';
|
||||||
|
|
||||||
|
const CONFIG_PATH = resolve('config.json');
|
||||||
|
const TMDB_BASE = 'https://api.themoviedb.org/3';
|
||||||
|
|
||||||
|
function loadConfig() {
|
||||||
|
if (!existsSync(CONFIG_PATH)) return {};
|
||||||
|
try {
|
||||||
|
return JSON.parse(readFileSync(CONFIG_PATH, 'utf-8'));
|
||||||
|
} catch {
|
||||||
|
return {};
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
export function loadTmdbConfig() {
|
||||||
|
const config = loadConfig();
|
||||||
|
return {
|
||||||
|
tmdbApiKey: String(config?.tmdbApiKey || '').trim()
|
||||||
|
};
|
||||||
|
}
|
||||||
|
|
||||||
|
export function hasTmdbConfig() {
|
||||||
|
return !!loadTmdbConfig().tmdbApiKey;
|
||||||
|
}
|
||||||
|
|
||||||
|
async function fetchTmdb(pathname, params = {}) {
|
||||||
|
const { tmdbApiKey } = loadTmdbConfig();
|
||||||
|
if (!tmdbApiKey) {
|
||||||
|
throw new Error('TMDB API key not configured');
|
||||||
|
}
|
||||||
|
|
||||||
|
const url = new URL(`${TMDB_BASE}${pathname.startsWith('/') ? pathname : `/${pathname}`}`);
|
||||||
|
for (const [key, value] of Object.entries({
|
||||||
|
api_key: tmdbApiKey,
|
||||||
|
language: 'en-US',
|
||||||
|
include_adult: 'false',
|
||||||
|
...params
|
||||||
|
})) {
|
||||||
|
if (value === undefined || value === null || value === '') continue;
|
||||||
|
url.searchParams.set(key, String(value));
|
||||||
|
}
|
||||||
|
|
||||||
|
const response = await fetch(url, {
|
||||||
|
headers: { Accept: 'application/json' }
|
||||||
|
});
|
||||||
|
|
||||||
|
if (!response.ok) {
|
||||||
|
const text = await response.text().catch(() => response.statusText);
|
||||||
|
throw new Error(text || `${response.status} ${response.statusText}`);
|
||||||
|
}
|
||||||
|
|
||||||
|
return response.json();
|
||||||
|
}
|
||||||
|
|
||||||
|
function normalizeTmdbSearchItem(item, mediaType) {
|
||||||
|
return {
|
||||||
|
tmdbId: Number(item?.id || 0) || null,
|
||||||
|
name: item?.title || item?.name || '',
|
||||||
|
year: Number(String(item?.release_date || item?.first_air_date || '').slice(0, 4)) || null,
|
||||||
|
mediaType,
|
||||||
|
overview: item?.overview || '',
|
||||||
|
genreIds: Array.isArray(item?.genre_ids) ? item.genre_ids : [],
|
||||||
|
originalLanguage: item?.original_language || '',
|
||||||
|
popularity: Number(item?.popularity || 0),
|
||||||
|
voteAverage: Number(item?.vote_average || 0),
|
||||||
|
voteCount: Number(item?.vote_count || 0)
|
||||||
|
};
|
||||||
|
}
|
||||||
|
|
||||||
|
export async function searchTmdbByTitle({ mediaType, name, year }) {
|
||||||
|
const searchType = mediaType === 'tv' ? 'tv' : 'movie';
|
||||||
|
const payload = await fetchTmdb(`/search/${searchType}`, {
|
||||||
|
query: name,
|
||||||
|
page: 1,
|
||||||
|
...(year ? (searchType === 'movie' ? { year } : { first_air_date_year: year }) : {})
|
||||||
|
});
|
||||||
|
|
||||||
|
return (payload?.results || []).map((item) => normalizeTmdbSearchItem(item, searchType));
|
||||||
|
}
|
||||||
|
|
||||||
|
export async function fetchTmdbSimilar({ mediaType, tmdbId, page = 1 }) {
|
||||||
|
const pathType = mediaType === 'tv' ? 'tv' : 'movie';
|
||||||
|
const payload = await fetchTmdb(`/${pathType}/${encodeURIComponent(tmdbId)}/similar`, { page });
|
||||||
|
return (payload?.results || []).map((item) => normalizeTmdbSearchItem(item, pathType));
|
||||||
|
}
|
||||||
|
|
||||||
|
export async function fetchTmdbRecommendations({ mediaType, tmdbId, page = 1 }) {
|
||||||
|
const pathType = mediaType === 'tv' ? 'tv' : 'movie';
|
||||||
|
const payload = await fetchTmdb(`/${pathType}/${encodeURIComponent(tmdbId)}/recommendations`, { page });
|
||||||
|
return (payload?.results || []).map((item) => normalizeTmdbSearchItem(item, pathType));
|
||||||
|
}
|
||||||
|
|
||||||
|
export async function fetchTmdbDetails({ mediaType, tmdbId }) {
|
||||||
|
const pathType = mediaType === 'tv' ? 'tv' : 'movie';
|
||||||
|
return fetchTmdb(`/${pathType}/${encodeURIComponent(tmdbId)}`);
|
||||||
|
}
|
||||||
|
|
||||||
|
export async function fetchTmdbKeywords({ mediaType, tmdbId }) {
|
||||||
|
const pathType = mediaType === 'tv' ? 'tv' : 'movie';
|
||||||
|
const payload = await fetchTmdb(`/${pathType}/${encodeURIComponent(tmdbId)}/keywords`);
|
||||||
|
return payload?.keywords || payload?.results || [];
|
||||||
|
}
|
||||||
|
|
||||||
|
export async function fetchTmdbCredits({ mediaType, tmdbId }) {
|
||||||
|
if (mediaType === 'tv') {
|
||||||
|
return fetchTmdb(`/tv/${encodeURIComponent(tmdbId)}/aggregate_credits`);
|
||||||
|
}
|
||||||
|
|
||||||
|
return fetchTmdb(`/movie/${encodeURIComponent(tmdbId)}/credits`);
|
||||||
|
}
|
||||||
|
|
||||||
|
export async function fetchTmdbGenres(mediaType) {
|
||||||
|
const pathType = mediaType === 'tv' ? 'tv' : 'movie';
|
||||||
|
const payload = await fetchTmdb(`/genre/${pathType}/list`, { language: 'en' });
|
||||||
|
return payload?.genres || [];
|
||||||
|
}
|
||||||
|
|
||||||
|
export async function fetchTmdbDiscover({ mediaType, params = {} }) {
|
||||||
|
const pathType = mediaType === 'tv' ? 'tv' : 'movie';
|
||||||
|
const payload = await fetchTmdb(`/discover/${pathType}`, params);
|
||||||
|
return (payload?.results || []).map((item) => normalizeTmdbSearchItem(item, pathType));
|
||||||
|
}
|
||||||
@@ -1,24 +1,30 @@
|
|||||||
import { readFileSync, existsSync } from 'fs';
|
import { readFileSync, existsSync } from 'fs';
|
||||||
import { resolve } from 'path';
|
import { resolve } from 'path';
|
||||||
|
import { applyCachedEmbyNames } from '../lib/server/emby-user-cache.js';
|
||||||
|
|
||||||
/** @type {import('./$types').PageServerLoad} */
|
/** @type {import('./$types').PageServerLoad} */
|
||||||
export async function load() {
|
export async function load() {
|
||||||
const dataPath = resolve('static/db_export.json');
|
const dataPath = resolve('static/db_export.json');
|
||||||
const raw = readFileSync(dataPath, 'utf-8');
|
const raw = readFileSync(dataPath, 'utf-8');
|
||||||
const data = JSON.parse(raw);
|
const data = JSON.parse(raw);
|
||||||
|
const enrichedUsers = applyCachedEmbyNames(data.users);
|
||||||
|
|
||||||
const configPath = resolve('config.json');
|
const configPath = resolve('config.json');
|
||||||
let config = { embyUrl: '', apiKey: '', dbPath: '' };
|
let config = { embyUrl: '', apiKey: '', tmdbApiKey: '', dbPath: '' };
|
||||||
if (existsSync(configPath)) {
|
if (existsSync(configPath)) {
|
||||||
try {
|
try {
|
||||||
config = JSON.parse(readFileSync(configPath, 'utf-8'));
|
config = {
|
||||||
|
...config,
|
||||||
|
...JSON.parse(readFileSync(configPath, 'utf-8'))
|
||||||
|
};
|
||||||
} catch { /* use defaults */ }
|
} catch { /* use defaults */ }
|
||||||
}
|
}
|
||||||
|
|
||||||
return {
|
return {
|
||||||
users: data.users,
|
users: enrichedUsers.users,
|
||||||
genres: data.genres,
|
genres: data.genres,
|
||||||
enums: data.enums,
|
enums: data.enums,
|
||||||
config
|
config,
|
||||||
|
embyCache: enrichedUsers.cache
|
||||||
};
|
};
|
||||||
}
|
}
|
||||||
|
|||||||
+691
-121
File diff suppressed because it is too large
Load Diff
@@ -3,13 +3,17 @@ import { resolve } from 'path';
|
|||||||
import { json } from '@sveltejs/kit';
|
import { json } from '@sveltejs/kit';
|
||||||
|
|
||||||
const CONFIG_PATH = resolve('config.json');
|
const CONFIG_PATH = resolve('config.json');
|
||||||
|
const DEFAULT_CONFIG = { embyUrl: '', apiKey: '', tmdbApiKey: '', dbPath: '' };
|
||||||
|
|
||||||
function loadConfig() {
|
function loadConfig() {
|
||||||
if (!existsSync(CONFIG_PATH)) return { embyUrl: '', apiKey: '', dbPath: '' };
|
if (!existsSync(CONFIG_PATH)) return DEFAULT_CONFIG;
|
||||||
try {
|
try {
|
||||||
return JSON.parse(readFileSync(CONFIG_PATH, 'utf-8'));
|
return {
|
||||||
|
...DEFAULT_CONFIG,
|
||||||
|
...JSON.parse(readFileSync(CONFIG_PATH, 'utf-8'))
|
||||||
|
};
|
||||||
} catch {
|
} catch {
|
||||||
return { embyUrl: '', apiKey: '', dbPath: '' };
|
return DEFAULT_CONFIG;
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
|
||||||
@@ -22,6 +26,7 @@ export async function POST({ request }) {
|
|||||||
const config = {
|
const config = {
|
||||||
embyUrl: String(body.embyUrl || '').trim(),
|
embyUrl: String(body.embyUrl || '').trim(),
|
||||||
apiKey: String(body.apiKey || '').trim(),
|
apiKey: String(body.apiKey || '').trim(),
|
||||||
|
tmdbApiKey: String(body.tmdbApiKey || '').trim(),
|
||||||
dbPath: String(body.dbPath || '').trim()
|
dbPath: String(body.dbPath || '').trim()
|
||||||
};
|
};
|
||||||
writeFileSync(CONFIG_PATH, JSON.stringify(config, null, 2));
|
writeFileSync(CONFIG_PATH, JSON.stringify(config, null, 2));
|
||||||
|
|||||||
@@ -2,6 +2,7 @@ import { json, error } from '@sveltejs/kit';
|
|||||||
import { existsSync } from 'fs';
|
import { existsSync } from 'fs';
|
||||||
import { DatabaseSync } from 'node:sqlite';
|
import { DatabaseSync } from 'node:sqlite';
|
||||||
import { loadHomeScreenUsers } from '../../../lib/server/emby-user-db.js';
|
import { loadHomeScreenUsers } from '../../../lib/server/emby-user-db.js';
|
||||||
|
import { applyCachedEmbyNames } from '../../../lib/server/emby-user-cache.js';
|
||||||
|
|
||||||
export async function POST({ request }) {
|
export async function POST({ request }) {
|
||||||
const { dbPath } = await request.json();
|
const { dbPath } = await request.json();
|
||||||
@@ -12,8 +13,18 @@ export async function POST({ request }) {
|
|||||||
try {
|
try {
|
||||||
db = new DatabaseSync(dbPath, { readOnly: true });
|
db = new DatabaseSync(dbPath, { readOnly: true });
|
||||||
const result = loadHomeScreenUsers(db);
|
const result = loadHomeScreenUsers(db);
|
||||||
|
const enriched = applyCachedEmbyNames(result.users);
|
||||||
db.close();
|
db.close();
|
||||||
return json(result);
|
return json({
|
||||||
|
...result,
|
||||||
|
users: enriched.users,
|
||||||
|
validation: {
|
||||||
|
...result.validation,
|
||||||
|
embyCacheMatchedUsers: enriched.cache.matchedCount,
|
||||||
|
embyCacheUserCount: enriched.cache.totalCachedUsers,
|
||||||
|
embyCacheLastSyncedAt: enriched.cache.lastSyncedAt
|
||||||
|
}
|
||||||
|
});
|
||||||
} catch (err) {
|
} catch (err) {
|
||||||
if (db) try { db.close(); } catch { /* ignore */ }
|
if (db) try { db.close(); } catch { /* ignore */ }
|
||||||
throw error(500, err.message);
|
throw error(500, err.message);
|
||||||
|
|||||||
@@ -0,0 +1,695 @@
|
|||||||
|
import { error, json } from '@sveltejs/kit';
|
||||||
|
import {
|
||||||
|
RECOMMENDATION_PROFILES,
|
||||||
|
rankRecommendationResults,
|
||||||
|
normalizeLookupItem
|
||||||
|
} from '../../../lib/collection-tools.js';
|
||||||
|
import { fetchEmby, fetchEmbyJson, normalizeEmbyGuid } from '../../../lib/server/emby-api.js';
|
||||||
|
import {
|
||||||
|
fetchTmdbCredits,
|
||||||
|
fetchTmdbDetails,
|
||||||
|
fetchTmdbDiscover,
|
||||||
|
fetchTmdbGenres,
|
||||||
|
fetchTmdbKeywords,
|
||||||
|
fetchTmdbRecommendations,
|
||||||
|
fetchTmdbSimilar,
|
||||||
|
searchTmdbByTitle
|
||||||
|
} from '../../../lib/server/tmdb-api.js';
|
||||||
|
|
||||||
|
const EMBY_PAGE_SIZE = 200;
|
||||||
|
|
||||||
|
function normalizeTitle(value) {
|
||||||
|
return String(value || '')
|
||||||
|
.toLowerCase()
|
||||||
|
.replace(/[^a-z0-9]+/g, '');
|
||||||
|
}
|
||||||
|
|
||||||
|
function normalizeYear(value) {
|
||||||
|
const year = Number(value || 0);
|
||||||
|
return Number.isFinite(year) && year > 0 ? year : null;
|
||||||
|
}
|
||||||
|
|
||||||
|
function titleYearKey(name, year) {
|
||||||
|
return `${normalizeTitle(name)}::${normalizeYear(year) || ''}`;
|
||||||
|
}
|
||||||
|
|
||||||
|
function normalizeCollectionName(name) {
|
||||||
|
return String(name || '')
|
||||||
|
.toLowerCase()
|
||||||
|
.replace(/['’]/g, '')
|
||||||
|
.replace(/[^a-z0-9]+/g, '');
|
||||||
|
}
|
||||||
|
|
||||||
|
function getSeedTmdbId(item) {
|
||||||
|
const providerIds = item?.providerIds || {};
|
||||||
|
return String(providerIds.Tmdb || providerIds.TMDB || providerIds.tmdb || '').trim();
|
||||||
|
}
|
||||||
|
|
||||||
|
function getTmdbMediaType(item) {
|
||||||
|
return item?.type === 'Series' ? 'tv' : 'movie';
|
||||||
|
}
|
||||||
|
|
||||||
|
function summarizeItems(items, limit = 5) {
|
||||||
|
return (items || []).slice(0, limit).map((item) => ({
|
||||||
|
name: item?.name || item?.title || item?.Name || '',
|
||||||
|
year: item?.year || item?.ProductionYear || null,
|
||||||
|
id: item?.id || item?.Id || '',
|
||||||
|
type: item?.type || item?.Type || ''
|
||||||
|
}));
|
||||||
|
}
|
||||||
|
|
||||||
|
function filterEnglishCandidates(items) {
|
||||||
|
return (items || []).filter((item) => !item?.originalLanguage || item.originalLanguage === 'en');
|
||||||
|
}
|
||||||
|
|
||||||
|
async function fetchAllUserItems(userId, params) {
|
||||||
|
const allItems = [];
|
||||||
|
let startIndex = 0;
|
||||||
|
|
||||||
|
while (true) {
|
||||||
|
const payload = await fetchEmbyJson(`/Users/${encodeURIComponent(userId)}/Items`, {
|
||||||
|
params: {
|
||||||
|
Recursive: true,
|
||||||
|
GroupItemsIntoCollections: false,
|
||||||
|
Limit: EMBY_PAGE_SIZE,
|
||||||
|
StartIndex: startIndex,
|
||||||
|
...params
|
||||||
|
}
|
||||||
|
});
|
||||||
|
|
||||||
|
const pageItems = (payload.Items || payload || []).map(normalizeLookupItem);
|
||||||
|
allItems.push(...pageItems);
|
||||||
|
|
||||||
|
const total = Number(payload.TotalRecordCount || 0);
|
||||||
|
if (!pageItems.length) break;
|
||||||
|
if (total > 0 && allItems.length >= total) break;
|
||||||
|
if (pageItems.length < EMBY_PAGE_SIZE) break;
|
||||||
|
|
||||||
|
startIndex += pageItems.length;
|
||||||
|
}
|
||||||
|
|
||||||
|
return allItems;
|
||||||
|
}
|
||||||
|
|
||||||
|
async function fetchAllLibraryItems(userId) {
|
||||||
|
return fetchAllUserItems(userId, {
|
||||||
|
IncludeItemTypes: 'Movie,Series',
|
||||||
|
Fields: 'Overview,Genres,CommunityRating,ProductionYear,ProviderIds'
|
||||||
|
});
|
||||||
|
}
|
||||||
|
|
||||||
|
async function fetchAllPlayedItems(userId) {
|
||||||
|
return fetchAllUserItems(userId, {
|
||||||
|
Filters: 'IsPlayed',
|
||||||
|
IncludeItemTypes: 'Movie,Series',
|
||||||
|
SortBy: 'DatePlayed',
|
||||||
|
SortOrder: 'Descending',
|
||||||
|
Fields: 'Overview,Genres,CommunityRating,ProductionYear,ProviderIds,UserData'
|
||||||
|
});
|
||||||
|
}
|
||||||
|
|
||||||
|
function uniqueById(items) {
|
||||||
|
const seen = new Set();
|
||||||
|
const unique = [];
|
||||||
|
for (const item of items || []) {
|
||||||
|
if (!item?.id || seen.has(item.id)) continue;
|
||||||
|
seen.add(item.id);
|
||||||
|
unique.push(item);
|
||||||
|
}
|
||||||
|
return unique;
|
||||||
|
}
|
||||||
|
|
||||||
|
function buildLibraryIndex(items) {
|
||||||
|
const byId = new Map();
|
||||||
|
const byTmdbId = new Map();
|
||||||
|
const byTitleYear = new Map();
|
||||||
|
const byTitle = new Map();
|
||||||
|
|
||||||
|
for (const item of items || []) {
|
||||||
|
if (!item?.id) continue;
|
||||||
|
byId.set(item.id, item);
|
||||||
|
|
||||||
|
const tmdbId = getSeedTmdbId(item);
|
||||||
|
if (tmdbId && !byTmdbId.has(tmdbId)) {
|
||||||
|
byTmdbId.set(tmdbId, item);
|
||||||
|
}
|
||||||
|
|
||||||
|
const titleKey = titleYearKey(item.name, item.year);
|
||||||
|
if (normalizeTitle(item.name) && !byTitleYear.has(titleKey)) {
|
||||||
|
byTitleYear.set(titleKey, item);
|
||||||
|
}
|
||||||
|
|
||||||
|
const normalizedName = normalizeTitle(item.name);
|
||||||
|
if (normalizedName && !byTitle.has(normalizedName)) {
|
||||||
|
byTitle.set(normalizedName, item);
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
return { byId, byTmdbId, byTitleYear, byTitle };
|
||||||
|
}
|
||||||
|
|
||||||
|
async function fetchSeedItems(userId, ids, libraryIndex) {
|
||||||
|
const fromLibrary = ids
|
||||||
|
.map((id) => libraryIndex.byId.get(String(id)))
|
||||||
|
.filter(Boolean);
|
||||||
|
|
||||||
|
if (fromLibrary.length === ids.length) {
|
||||||
|
return fromLibrary;
|
||||||
|
}
|
||||||
|
|
||||||
|
const payload = await fetchEmbyJson(`/Users/${encodeURIComponent(userId)}/Items`, {
|
||||||
|
params: {
|
||||||
|
Ids: ids.join(','),
|
||||||
|
Limit: ids.length,
|
||||||
|
Fields: 'Overview,Genres,CommunityRating,ProductionYear,ProviderIds',
|
||||||
|
GroupItemsIntoCollections: false
|
||||||
|
}
|
||||||
|
});
|
||||||
|
|
||||||
|
return uniqueById((payload.Items || payload || []).map(normalizeLookupItem));
|
||||||
|
}
|
||||||
|
|
||||||
|
async function findExistingCollection(userId, name) {
|
||||||
|
const trimmedName = String(name || '').trim();
|
||||||
|
if (!userId || !trimmedName) return null;
|
||||||
|
|
||||||
|
const payload = await fetchEmbyJson(`/Users/${encodeURIComponent(userId)}/Items`, {
|
||||||
|
params: {
|
||||||
|
Recursive: true,
|
||||||
|
SearchTerm: trimmedName,
|
||||||
|
Limit: 20,
|
||||||
|
SortBy: 'SortName',
|
||||||
|
SortOrder: 'Ascending',
|
||||||
|
Fields: 'Overview',
|
||||||
|
IncludeItemTypes: 'BoxSet',
|
||||||
|
GroupItemsIntoCollections: false
|
||||||
|
}
|
||||||
|
});
|
||||||
|
|
||||||
|
const normalizedTarget = normalizeCollectionName(trimmedName);
|
||||||
|
const exact = (payload.Items || payload || [])
|
||||||
|
.map(normalizeLookupItem)
|
||||||
|
.find((item) => item.type === 'BoxSet' && normalizeCollectionName(item.name) === normalizedTarget);
|
||||||
|
|
||||||
|
return exact || null;
|
||||||
|
}
|
||||||
|
|
||||||
|
async function resolveTmdbMatch(item) {
|
||||||
|
const directTmdbId = getSeedTmdbId(item);
|
||||||
|
if (directTmdbId) {
|
||||||
|
return {
|
||||||
|
tmdbId: Number(directTmdbId),
|
||||||
|
mediaType: getTmdbMediaType(item)
|
||||||
|
};
|
||||||
|
}
|
||||||
|
|
||||||
|
const matches = await searchTmdbByTitle({
|
||||||
|
mediaType: getTmdbMediaType(item),
|
||||||
|
name: item.name,
|
||||||
|
year: item.year
|
||||||
|
});
|
||||||
|
const best = matches[0];
|
||||||
|
if (!best?.tmdbId) return null;
|
||||||
|
|
||||||
|
return {
|
||||||
|
tmdbId: best.tmdbId,
|
||||||
|
mediaType: best.mediaType
|
||||||
|
};
|
||||||
|
}
|
||||||
|
|
||||||
|
function resolveLocalCandidate(index, candidate) {
|
||||||
|
const tmdbId = String(candidate?.tmdbId || '').trim();
|
||||||
|
if (tmdbId && index.byTmdbId.has(tmdbId)) {
|
||||||
|
return index.byTmdbId.get(tmdbId);
|
||||||
|
}
|
||||||
|
|
||||||
|
const exactTitleYear = index.byTitleYear.get(titleYearKey(candidate?.name, candidate?.year));
|
||||||
|
if (exactTitleYear) {
|
||||||
|
return exactTitleYear;
|
||||||
|
}
|
||||||
|
|
||||||
|
return index.byTitle.get(normalizeTitle(candidate?.name)) || null;
|
||||||
|
}
|
||||||
|
|
||||||
|
function addWeight(map, key, amount = 1) {
|
||||||
|
if (key === null || key === undefined || key === '') return;
|
||||||
|
map.set(key, (map.get(key) || 0) + amount);
|
||||||
|
}
|
||||||
|
|
||||||
|
function sortWeightedEntries(map, limit) {
|
||||||
|
return [...map.entries()]
|
||||||
|
.sort((a, b) => b[1] - a[1])
|
||||||
|
.slice(0, limit);
|
||||||
|
}
|
||||||
|
|
||||||
|
function getGenreLookup(genres) {
|
||||||
|
return new Map((genres || []).map((genre) => [normalizeTitle(genre.name), Number(genre.id)]));
|
||||||
|
}
|
||||||
|
|
||||||
|
function resolveGenreIdsFromLocalItems(items, genreLookup, weight, targetMap) {
|
||||||
|
for (const item of items || []) {
|
||||||
|
for (const genreName of item.genres || []) {
|
||||||
|
const id = genreLookup.get(normalizeTitle(genreName));
|
||||||
|
if (id) addWeight(targetMap, id, weight);
|
||||||
|
}
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
function normalizeKeywordIds(keywords) {
|
||||||
|
return (keywords || []).map((keyword) => Number(keyword?.id || 0)).filter(Boolean);
|
||||||
|
}
|
||||||
|
|
||||||
|
function extractPeopleIds(credits, mediaType) {
|
||||||
|
const ids = new Set();
|
||||||
|
|
||||||
|
for (const person of (credits?.cast || []).slice(0, 5)) {
|
||||||
|
if (person?.id) ids.add(Number(person.id));
|
||||||
|
}
|
||||||
|
|
||||||
|
if (mediaType === 'movie') {
|
||||||
|
for (const crew of credits?.crew || []) {
|
||||||
|
if (!crew?.id) continue;
|
||||||
|
if (crew.job === 'Director' || crew.job === 'Writer' || crew.job === 'Screenplay') {
|
||||||
|
ids.add(Number(crew.id));
|
||||||
|
}
|
||||||
|
if (ids.size >= 8) break;
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
return [...ids];
|
||||||
|
}
|
||||||
|
|
||||||
|
function buildDiscoverQueries({ mediaType, genreIds, keywordIds, peopleIds }) {
|
||||||
|
const base = {
|
||||||
|
page: 1,
|
||||||
|
sort_by: 'popularity.desc',
|
||||||
|
'vote_count.gte': mediaType === 'tv' ? 10 : 25,
|
||||||
|
with_original_language: 'en'
|
||||||
|
};
|
||||||
|
const queries = [];
|
||||||
|
|
||||||
|
if (genreIds.length) {
|
||||||
|
queries.push({
|
||||||
|
label: `${mediaType}-genres`,
|
||||||
|
params: {
|
||||||
|
...base,
|
||||||
|
with_genres: genreIds.slice(0, 4).join('|')
|
||||||
|
}
|
||||||
|
});
|
||||||
|
}
|
||||||
|
|
||||||
|
if (keywordIds.length) {
|
||||||
|
queries.push({
|
||||||
|
label: `${mediaType}-keywords`,
|
||||||
|
params: {
|
||||||
|
...base,
|
||||||
|
with_keywords: keywordIds.slice(0, 6).join('|')
|
||||||
|
}
|
||||||
|
});
|
||||||
|
}
|
||||||
|
|
||||||
|
if (genreIds.length && keywordIds.length) {
|
||||||
|
queries.push({
|
||||||
|
label: `${mediaType}-genres-keywords`,
|
||||||
|
params: {
|
||||||
|
...base,
|
||||||
|
with_genres: genreIds.slice(0, 3).join('|'),
|
||||||
|
with_keywords: keywordIds.slice(0, 4).join('|')
|
||||||
|
}
|
||||||
|
});
|
||||||
|
}
|
||||||
|
|
||||||
|
if (mediaType === 'movie' && peopleIds.length) {
|
||||||
|
queries.push({
|
||||||
|
label: 'movie-people',
|
||||||
|
params: {
|
||||||
|
...base,
|
||||||
|
with_people: peopleIds.slice(0, 5).join('|')
|
||||||
|
}
|
||||||
|
});
|
||||||
|
}
|
||||||
|
|
||||||
|
return queries;
|
||||||
|
}
|
||||||
|
|
||||||
|
function inspectCandidateSets(seedIds, resultSets) {
|
||||||
|
const excluded = new Set((seedIds || []).map((id) => String(id)));
|
||||||
|
const uniqueIncluded = new Map();
|
||||||
|
const uniqueExcluded = new Map();
|
||||||
|
|
||||||
|
for (const resultSet of resultSets || []) {
|
||||||
|
const items = Array.isArray(resultSet) ? resultSet : (Array.isArray(resultSet?.items) ? resultSet.items : []);
|
||||||
|
for (const item of items) {
|
||||||
|
const normalized = normalizeLookupItem(item);
|
||||||
|
if (!normalized.id) continue;
|
||||||
|
if (excluded.has(normalized.id)) {
|
||||||
|
if (!uniqueExcluded.has(normalized.id)) uniqueExcluded.set(normalized.id, normalized);
|
||||||
|
continue;
|
||||||
|
}
|
||||||
|
if (!uniqueIncluded.has(normalized.id)) uniqueIncluded.set(normalized.id, normalized);
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
return {
|
||||||
|
uniqueCandidateCount: uniqueIncluded.size,
|
||||||
|
excludedSeedCandidateCount: uniqueExcluded.size,
|
||||||
|
sampleExcludedSeedCandidates: summarizeItems([...uniqueExcluded.values()])
|
||||||
|
};
|
||||||
|
}
|
||||||
|
|
||||||
|
async function buildSeedContext(seed, libraryIndex) {
|
||||||
|
const diagnostic = {
|
||||||
|
seedId: seed.id,
|
||||||
|
seedName: seed.name,
|
||||||
|
seedType: seed.type,
|
||||||
|
seedYear: seed.year,
|
||||||
|
providerTmdbId: getSeedTmdbId(seed) || null,
|
||||||
|
tmdbMatch: null,
|
||||||
|
similarCount: 0,
|
||||||
|
recommendationCount: 0,
|
||||||
|
localSimilarCount: 0,
|
||||||
|
localRecommendationCount: 0,
|
||||||
|
error: null
|
||||||
|
};
|
||||||
|
|
||||||
|
try {
|
||||||
|
const tmdbMatch = await resolveTmdbMatch(seed);
|
||||||
|
if (!tmdbMatch?.tmdbId) {
|
||||||
|
diagnostic.error = 'No TMDB match resolved for seed';
|
||||||
|
return { seed, diagnostic, resultSets: [], details: null, keywords: [], peopleIds: [] };
|
||||||
|
}
|
||||||
|
|
||||||
|
diagnostic.tmdbMatch = tmdbMatch;
|
||||||
|
|
||||||
|
const [details, keywords, credits, similar, recommendations] = await Promise.all([
|
||||||
|
fetchTmdbDetails(tmdbMatch),
|
||||||
|
fetchTmdbKeywords(tmdbMatch),
|
||||||
|
fetchTmdbCredits(tmdbMatch),
|
||||||
|
fetchTmdbSimilar({ ...tmdbMatch, page: 1 }),
|
||||||
|
fetchTmdbRecommendations({ ...tmdbMatch, page: 1 })
|
||||||
|
]);
|
||||||
|
|
||||||
|
const englishSimilar = filterEnglishCandidates(similar);
|
||||||
|
const englishRecommendations = filterEnglishCandidates(recommendations);
|
||||||
|
const localSimilar = uniqueById(englishSimilar.map((candidate) => resolveLocalCandidate(libraryIndex, candidate)).filter(Boolean));
|
||||||
|
const localRecommendations = uniqueById(
|
||||||
|
englishRecommendations.map((candidate) => resolveLocalCandidate(libraryIndex, candidate)).filter(Boolean)
|
||||||
|
);
|
||||||
|
|
||||||
|
diagnostic.similarCount = englishSimilar.length;
|
||||||
|
diagnostic.recommendationCount = englishRecommendations.length;
|
||||||
|
diagnostic.localSimilarCount = localSimilar.length;
|
||||||
|
diagnostic.localRecommendationCount = localRecommendations.length;
|
||||||
|
|
||||||
|
return {
|
||||||
|
seed,
|
||||||
|
details,
|
||||||
|
keywords,
|
||||||
|
peopleIds: extractPeopleIds(credits, tmdbMatch.mediaType),
|
||||||
|
diagnostic,
|
||||||
|
resultSets: [
|
||||||
|
{ items: localSimilar, sourceWeight: 5, label: 'tmdb-similar' },
|
||||||
|
{ items: localRecommendations, sourceWeight: 6, label: 'tmdb-recommendations' }
|
||||||
|
]
|
||||||
|
};
|
||||||
|
} catch (err) {
|
||||||
|
diagnostic.error = err.message;
|
||||||
|
return { seed, diagnostic, resultSets: [], details: null, keywords: [], peopleIds: [] };
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
async function buildPreview(userId, seedIds, limit, profile) {
|
||||||
|
const normalizedSeedIds = [...new Set(seedIds.map((id) => String(id).trim()).filter(Boolean))];
|
||||||
|
const libraryItems = uniqueById(await fetchAllLibraryItems(userId));
|
||||||
|
const playedItems = uniqueById(await fetchAllPlayedItems(userId));
|
||||||
|
const excludedIds = [...new Set([...normalizedSeedIds, ...playedItems.map((item) => item.id).filter(Boolean)])];
|
||||||
|
const libraryIndex = buildLibraryIndex(libraryItems);
|
||||||
|
const seeds = await fetchSeedItems(userId, normalizedSeedIds, libraryIndex);
|
||||||
|
const allowedTypes = [...new Set(seeds.map((item) => item.type).filter(Boolean))];
|
||||||
|
const includeMovies = allowedTypes.includes('Movie');
|
||||||
|
const includeSeries = allowedTypes.includes('Series');
|
||||||
|
|
||||||
|
if (!seeds.length) {
|
||||||
|
return {
|
||||||
|
seeds: [],
|
||||||
|
recommendations: [],
|
||||||
|
diagnostics: {
|
||||||
|
libraryItemCount: libraryItems.length,
|
||||||
|
watchedItemCount: playedItems.length,
|
||||||
|
seedCount: 0,
|
||||||
|
embyCandidates: 0,
|
||||||
|
tmdbEnabled: true,
|
||||||
|
tmdbCandidates: 0,
|
||||||
|
tmdbResolved: 0,
|
||||||
|
uniqueCandidateCount: 0,
|
||||||
|
excludedSeedCandidateCount: 0,
|
||||||
|
sampleExcludedSeedCandidates: [],
|
||||||
|
perSeed: [],
|
||||||
|
discoverQueries: []
|
||||||
|
}
|
||||||
|
};
|
||||||
|
}
|
||||||
|
|
||||||
|
const seedContexts = await Promise.all(seeds.map((seed) => buildSeedContext(seed, libraryIndex)));
|
||||||
|
const movieGenres = await fetchTmdbGenres('movie');
|
||||||
|
const tvGenres = await fetchTmdbGenres('tv');
|
||||||
|
const movieGenreLookup = getGenreLookup(movieGenres);
|
||||||
|
const tvGenreLookup = getGenreLookup(tvGenres);
|
||||||
|
|
||||||
|
const movieGenreWeights = new Map();
|
||||||
|
const tvGenreWeights = new Map();
|
||||||
|
const movieKeywordWeights = new Map();
|
||||||
|
const tvKeywordWeights = new Map();
|
||||||
|
const moviePeopleWeights = new Map();
|
||||||
|
|
||||||
|
for (const context of seedContexts) {
|
||||||
|
const mediaType = getTmdbMediaType(context.seed);
|
||||||
|
const genreTarget = mediaType === 'tv' ? tvGenreWeights : movieGenreWeights;
|
||||||
|
const keywordTarget = mediaType === 'tv' ? tvKeywordWeights : movieKeywordWeights;
|
||||||
|
|
||||||
|
for (const genre of context.details?.genres || []) {
|
||||||
|
addWeight(genreTarget, Number(genre.id), 5);
|
||||||
|
}
|
||||||
|
|
||||||
|
for (const keywordId of normalizeKeywordIds(context.keywords)) {
|
||||||
|
addWeight(keywordTarget, keywordId, 4);
|
||||||
|
}
|
||||||
|
|
||||||
|
if (mediaType === 'movie') {
|
||||||
|
for (const personId of context.peopleIds || []) {
|
||||||
|
addWeight(moviePeopleWeights, personId, 3);
|
||||||
|
}
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
resolveGenreIdsFromLocalItems(seeds.filter((item) => getTmdbMediaType(item) === 'movie'), movieGenreLookup, 3, movieGenreWeights);
|
||||||
|
resolveGenreIdsFromLocalItems(seeds.filter((item) => getTmdbMediaType(item) === 'tv'), tvGenreLookup, 3, tvGenreWeights);
|
||||||
|
resolveGenreIdsFromLocalItems(playedItems.filter((item) => getTmdbMediaType(item) === 'movie'), movieGenreLookup, 1, movieGenreWeights);
|
||||||
|
resolveGenreIdsFromLocalItems(playedItems.filter((item) => getTmdbMediaType(item) === 'tv'), tvGenreLookup, 1, tvGenreWeights);
|
||||||
|
|
||||||
|
const discoverQueries = [
|
||||||
|
...(includeMovies
|
||||||
|
? buildDiscoverQueries({
|
||||||
|
mediaType: 'movie',
|
||||||
|
genreIds: sortWeightedEntries(movieGenreWeights, 6).map(([id]) => id),
|
||||||
|
keywordIds: sortWeightedEntries(movieKeywordWeights, 8).map(([id]) => id),
|
||||||
|
peopleIds: sortWeightedEntries(moviePeopleWeights, 6).map(([id]) => id)
|
||||||
|
})
|
||||||
|
: []),
|
||||||
|
...(includeSeries
|
||||||
|
? buildDiscoverQueries({
|
||||||
|
mediaType: 'tv',
|
||||||
|
genreIds: sortWeightedEntries(tvGenreWeights, 6).map(([id]) => id),
|
||||||
|
keywordIds: sortWeightedEntries(tvKeywordWeights, 8).map(([id]) => id),
|
||||||
|
peopleIds: []
|
||||||
|
})
|
||||||
|
: [])
|
||||||
|
];
|
||||||
|
|
||||||
|
const queryResults = await Promise.all(
|
||||||
|
discoverQueries.map(async (query) => {
|
||||||
|
try {
|
||||||
|
const tmdbResults = filterEnglishCandidates(await fetchTmdbDiscover({
|
||||||
|
mediaType: query.label.startsWith('tv') ? 'tv' : 'movie',
|
||||||
|
params: query.params
|
||||||
|
}));
|
||||||
|
const localMatches = uniqueById(
|
||||||
|
tmdbResults.map((candidate) => resolveLocalCandidate(libraryIndex, candidate)).filter(Boolean)
|
||||||
|
);
|
||||||
|
|
||||||
|
return {
|
||||||
|
label: query.label,
|
||||||
|
params: query.params,
|
||||||
|
tmdbCount: tmdbResults.length,
|
||||||
|
localCount: localMatches.length,
|
||||||
|
localMatches,
|
||||||
|
sourceWeight: 1
|
||||||
|
};
|
||||||
|
} catch (err) {
|
||||||
|
return {
|
||||||
|
label: query.label,
|
||||||
|
params: query.params,
|
||||||
|
tmdbCount: 0,
|
||||||
|
localCount: 0,
|
||||||
|
localMatches: [],
|
||||||
|
error: err.message
|
||||||
|
};
|
||||||
|
}
|
||||||
|
})
|
||||||
|
);
|
||||||
|
|
||||||
|
const resultSets = [
|
||||||
|
...seedContexts.flatMap((context) => context.resultSets),
|
||||||
|
...queryResults.map((query) => ({
|
||||||
|
items: query.localMatches,
|
||||||
|
sourceWeight: query.sourceWeight || 1,
|
||||||
|
label: query.label
|
||||||
|
}))
|
||||||
|
].filter((resultSet) => (Array.isArray(resultSet) ? resultSet.length : resultSet.items.length) > 0);
|
||||||
|
|
||||||
|
const candidateInspection = inspectCandidateSets(excludedIds, resultSets);
|
||||||
|
const recommendations = rankRecommendationResults(normalizedSeedIds, resultSets, {
|
||||||
|
limit,
|
||||||
|
profile,
|
||||||
|
seeds,
|
||||||
|
excludeIds: excludedIds,
|
||||||
|
allowedTypes
|
||||||
|
});
|
||||||
|
|
||||||
|
return {
|
||||||
|
seeds,
|
||||||
|
recommendations,
|
||||||
|
diagnostics: {
|
||||||
|
libraryItemCount: libraryItems.length,
|
||||||
|
watchedItemCount: playedItems.length,
|
||||||
|
seedCount: seeds.length,
|
||||||
|
embyCandidates: seedContexts.reduce(
|
||||||
|
(total, context) => total + context.diagnostic.localSimilarCount + context.diagnostic.localRecommendationCount,
|
||||||
|
0
|
||||||
|
),
|
||||||
|
tmdbEnabled: true,
|
||||||
|
tmdbCandidates:
|
||||||
|
seedContexts.reduce(
|
||||||
|
(total, context) => total + context.diagnostic.similarCount + context.diagnostic.recommendationCount,
|
||||||
|
0
|
||||||
|
) + queryResults.reduce((total, query) => total + query.tmdbCount, 0),
|
||||||
|
tmdbResolved:
|
||||||
|
seedContexts.reduce(
|
||||||
|
(total, context) => total + context.diagnostic.localSimilarCount + context.diagnostic.localRecommendationCount,
|
||||||
|
0
|
||||||
|
) + queryResults.reduce((total, query) => total + query.localCount, 0),
|
||||||
|
uniqueCandidateCount: candidateInspection.uniqueCandidateCount,
|
||||||
|
excludedSeedCandidateCount: candidateInspection.excludedSeedCandidateCount,
|
||||||
|
sampleExcludedSeedCandidates: candidateInspection.sampleExcludedSeedCandidates,
|
||||||
|
perSeed: seedContexts.map((context) => context.diagnostic),
|
||||||
|
discoverQueries: queryResults.map((query) => ({
|
||||||
|
label: query.label,
|
||||||
|
tmdbCount: query.tmdbCount,
|
||||||
|
localCount: query.localCount,
|
||||||
|
error: query.error || null,
|
||||||
|
params: query.params
|
||||||
|
}))
|
||||||
|
}
|
||||||
|
};
|
||||||
|
}
|
||||||
|
|
||||||
|
async function createOrUpdateCollection(userId, name, itemIds) {
|
||||||
|
const existingCollection = await findExistingCollection(userId, name);
|
||||||
|
|
||||||
|
if (existingCollection?.id) {
|
||||||
|
await fetchEmby(`/Collections/${encodeURIComponent(existingCollection.id)}/Items`, {
|
||||||
|
method: 'POST',
|
||||||
|
params: {
|
||||||
|
Ids: itemIds.join(',')
|
||||||
|
}
|
||||||
|
});
|
||||||
|
|
||||||
|
return {
|
||||||
|
id: existingCollection.id,
|
||||||
|
name: existingCollection.name || name,
|
||||||
|
updated: true
|
||||||
|
};
|
||||||
|
}
|
||||||
|
|
||||||
|
const response = await fetchEmby('/Collections', {
|
||||||
|
method: 'POST',
|
||||||
|
params: {
|
||||||
|
Name: name,
|
||||||
|
Ids: itemIds.join(',')
|
||||||
|
}
|
||||||
|
});
|
||||||
|
const created = await response.json();
|
||||||
|
|
||||||
|
return {
|
||||||
|
id: String(created?.Id || ''),
|
||||||
|
name: created?.Name || name,
|
||||||
|
updated: false
|
||||||
|
};
|
||||||
|
}
|
||||||
|
|
||||||
|
export async function POST({ request }) {
|
||||||
|
const body = await request.json();
|
||||||
|
const mode = body?.mode === 'create' ? 'create' : 'preview';
|
||||||
|
const userId = normalizeEmbyGuid(body?.userId);
|
||||||
|
const seedIds = Array.isArray(body?.seedIds) ? body.seedIds : [];
|
||||||
|
const limit = Math.min(Math.max(Number(body?.limit || 18), 1), 48);
|
||||||
|
const includeSeeds = body?.includeSeeds !== false;
|
||||||
|
const name = String(body?.name || '').trim();
|
||||||
|
const profile = RECOMMENDATION_PROFILES[body?.profile] ? body.profile : 'balanced';
|
||||||
|
|
||||||
|
if (!userId) {
|
||||||
|
throw error(400, 'Missing userId');
|
||||||
|
}
|
||||||
|
|
||||||
|
if (seedIds.length === 0) {
|
||||||
|
throw error(400, 'Select at least one seed item');
|
||||||
|
}
|
||||||
|
|
||||||
|
try {
|
||||||
|
const preview = await buildPreview(userId, seedIds, limit, profile);
|
||||||
|
const existingCollection = name ? await findExistingCollection(userId, name) : null;
|
||||||
|
|
||||||
|
if (mode !== 'create') {
|
||||||
|
return json({
|
||||||
|
...preview,
|
||||||
|
profile,
|
||||||
|
collection: existingCollection
|
||||||
|
? {
|
||||||
|
id: existingCollection.id,
|
||||||
|
name: existingCollection.name,
|
||||||
|
updated: true
|
||||||
|
}
|
||||||
|
: null
|
||||||
|
});
|
||||||
|
}
|
||||||
|
|
||||||
|
if (!name) {
|
||||||
|
throw error(400, 'Collection name is required');
|
||||||
|
}
|
||||||
|
|
||||||
|
const collectionIds = [
|
||||||
|
...(includeSeeds ? preview.seeds.map((item) => item.id) : []),
|
||||||
|
...preview.recommendations.map((item) => item.id)
|
||||||
|
];
|
||||||
|
const uniqueIds = [...new Set(collectionIds.filter(Boolean))];
|
||||||
|
|
||||||
|
if (uniqueIds.length === 0) {
|
||||||
|
throw error(400, 'No items were available to add to the collection');
|
||||||
|
}
|
||||||
|
|
||||||
|
const collection = await createOrUpdateCollection(userId, name, uniqueIds);
|
||||||
|
|
||||||
|
return json({
|
||||||
|
seeds: preview.seeds,
|
||||||
|
recommendations: preview.recommendations,
|
||||||
|
profile,
|
||||||
|
collection
|
||||||
|
});
|
||||||
|
} catch (err) {
|
||||||
|
if (err?.status) throw err;
|
||||||
|
if (String(err.message || '').includes('not configured')) {
|
||||||
|
throw error(400, err.message);
|
||||||
|
}
|
||||||
|
throw error(502, `Could not build Emby collection: ${err.message}`);
|
||||||
|
}
|
||||||
|
}
|
||||||
@@ -0,0 +1,45 @@
|
|||||||
|
import { error, json } from '@sveltejs/kit';
|
||||||
|
import { fetchEmbyJson, normalizeEmbyGuid } from '../../../lib/server/emby-api.js';
|
||||||
|
import { normalizeLookupItem } from '../../../lib/collection-tools.js';
|
||||||
|
|
||||||
|
export async function GET({ url }) {
|
||||||
|
const userId = normalizeEmbyGuid(url.searchParams.get('userId'));
|
||||||
|
const term = String(url.searchParams.get('term') || '').trim();
|
||||||
|
const limit = Math.min(Math.max(Number(url.searchParams.get('limit') || 12), 1), 50);
|
||||||
|
const types = (url.searchParams.get('types') || '')
|
||||||
|
.split(',')
|
||||||
|
.map((value) => value.trim())
|
||||||
|
.filter(Boolean);
|
||||||
|
|
||||||
|
if (!userId) {
|
||||||
|
throw error(400, 'Missing userId');
|
||||||
|
}
|
||||||
|
|
||||||
|
if (!term) {
|
||||||
|
return json({ items: [] });
|
||||||
|
}
|
||||||
|
|
||||||
|
try {
|
||||||
|
const payload = await fetchEmbyJson(`/Users/${encodeURIComponent(userId)}/Items`, {
|
||||||
|
params: {
|
||||||
|
Recursive: true,
|
||||||
|
SearchTerm: term,
|
||||||
|
Limit: limit,
|
||||||
|
SortBy: 'SortName',
|
||||||
|
SortOrder: 'Ascending',
|
||||||
|
Fields: 'Overview',
|
||||||
|
IncludeItemTypes: types.join(','),
|
||||||
|
GroupItemsIntoCollections: false
|
||||||
|
}
|
||||||
|
});
|
||||||
|
|
||||||
|
return json({
|
||||||
|
items: (payload.Items || payload || []).map(normalizeLookupItem)
|
||||||
|
});
|
||||||
|
} catch (err) {
|
||||||
|
if (String(err.message || '').includes('not configured')) {
|
||||||
|
throw error(400, err.message);
|
||||||
|
}
|
||||||
|
throw error(502, `Could not search Emby items: ${err.message}`);
|
||||||
|
}
|
||||||
|
}
|
||||||
@@ -0,0 +1,160 @@
|
|||||||
|
import { readFileSync, existsSync } from 'fs';
|
||||||
|
import { resolve } from 'path';
|
||||||
|
import { json, error } from '@sveltejs/kit';
|
||||||
|
import {
|
||||||
|
readCachedEmbyUserContext,
|
||||||
|
writeCachedEmbyUserContext
|
||||||
|
} from '../../../lib/server/emby-user-context-cache.js';
|
||||||
|
|
||||||
|
const CONFIG_PATH = resolve('config.json');
|
||||||
|
const EMBY_PAGE_SIZE = 200;
|
||||||
|
|
||||||
|
function loadConfig() {
|
||||||
|
if (!existsSync(CONFIG_PATH)) return {};
|
||||||
|
try {
|
||||||
|
return JSON.parse(readFileSync(CONFIG_PATH, 'utf-8'));
|
||||||
|
} catch {
|
||||||
|
return {};
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
function normalizeGuid(value) {
|
||||||
|
return typeof value === 'string' ? value.replace(/-/g, '').trim().toLowerCase() : '';
|
||||||
|
}
|
||||||
|
|
||||||
|
async function fetchJson(url) {
|
||||||
|
const response = await fetch(url);
|
||||||
|
if (!response.ok) {
|
||||||
|
throw new Error(`${response.status} ${response.statusText}`);
|
||||||
|
}
|
||||||
|
return response.json();
|
||||||
|
}
|
||||||
|
|
||||||
|
async function fetchAllUserItems(base, embyGuid, apiKey, params = {}) {
|
||||||
|
const allItems = [];
|
||||||
|
let startIndex = 0;
|
||||||
|
|
||||||
|
while (true) {
|
||||||
|
const query = new URLSearchParams({
|
||||||
|
api_key: apiKey,
|
||||||
|
Recursive: 'true',
|
||||||
|
GroupItemsIntoCollections: 'false',
|
||||||
|
Limit: String(EMBY_PAGE_SIZE),
|
||||||
|
StartIndex: String(startIndex),
|
||||||
|
...Object.fromEntries(
|
||||||
|
Object.entries(params).map(([key, value]) => [key, String(value)])
|
||||||
|
)
|
||||||
|
});
|
||||||
|
const payload = await fetchJson(`${base}/Users/${encodeURIComponent(embyGuid)}/Items?${query.toString()}`);
|
||||||
|
const pageItems = payload.Items || payload || [];
|
||||||
|
allItems.push(...pageItems);
|
||||||
|
|
||||||
|
const total = Number(payload.TotalRecordCount || 0);
|
||||||
|
if (!pageItems.length) break;
|
||||||
|
if (total > 0 && allItems.length >= total) break;
|
||||||
|
if (pageItems.length < EMBY_PAGE_SIZE) break;
|
||||||
|
|
||||||
|
startIndex += pageItems.length;
|
||||||
|
}
|
||||||
|
|
||||||
|
return allItems;
|
||||||
|
}
|
||||||
|
|
||||||
|
function normalizeViews(items = []) {
|
||||||
|
return items.map((item) => ({
|
||||||
|
id: String(item.Id || ''),
|
||||||
|
name: item.Name || 'Unnamed view',
|
||||||
|
type: item.CollectionType || item.Type || 'View'
|
||||||
|
}));
|
||||||
|
}
|
||||||
|
|
||||||
|
function normalizeRecentlyPlayed(items = []) {
|
||||||
|
return items.map((item) => ({
|
||||||
|
id: String(item.Id || ''),
|
||||||
|
name: item.Name || item.SeriesName || 'Unknown item',
|
||||||
|
type: item.Type || 'Item',
|
||||||
|
seriesName: item.SeriesName || null,
|
||||||
|
datePlayed: item.UserData?.LastPlayedDate || item.DateLastMediaAdded || null,
|
||||||
|
isPlayed: item.UserData?.Played ?? true
|
||||||
|
}));
|
||||||
|
}
|
||||||
|
|
||||||
|
function buildExcludedFolderLookup(items = []) {
|
||||||
|
return Object.fromEntries(
|
||||||
|
items
|
||||||
|
.filter((item) => item?.Id)
|
||||||
|
.map((item) => [
|
||||||
|
String(item.Id),
|
||||||
|
{
|
||||||
|
name: item.Name || item.Path || `Item ${item.Id}`,
|
||||||
|
type: item.CollectionType || item.Type || 'Item'
|
||||||
|
}
|
||||||
|
])
|
||||||
|
);
|
||||||
|
}
|
||||||
|
|
||||||
|
export async function GET({ url }) {
|
||||||
|
const embyGuid = normalizeGuid(url.searchParams.get('embyGuid'));
|
||||||
|
const excludedIds = (url.searchParams.get('excludedIds') || '')
|
||||||
|
.split(',')
|
||||||
|
.map((id) => id.trim())
|
||||||
|
.filter(Boolean);
|
||||||
|
|
||||||
|
if (!embyGuid) {
|
||||||
|
throw error(400, 'Missing embyGuid');
|
||||||
|
}
|
||||||
|
|
||||||
|
const { embyUrl, apiKey } = loadConfig();
|
||||||
|
const cached = readCachedEmbyUserContext(embyGuid);
|
||||||
|
|
||||||
|
if (!embyUrl || !apiKey) {
|
||||||
|
if (cached) {
|
||||||
|
return json({ ...cached, source: 'cache' });
|
||||||
|
}
|
||||||
|
throw error(400, 'Emby URL and API key not configured');
|
||||||
|
}
|
||||||
|
|
||||||
|
const base = embyUrl.replace(/\/+$/, '');
|
||||||
|
|
||||||
|
try {
|
||||||
|
const [viewsPayload, recentlyPlayedPayload, excludedPayload] = await Promise.all([
|
||||||
|
fetchJson(`${base}/Users/${encodeURIComponent(embyGuid)}/Views?api_key=${encodeURIComponent(apiKey)}`),
|
||||||
|
fetchAllUserItems(base, embyGuid, apiKey, {
|
||||||
|
Filters: 'IsPlayed',
|
||||||
|
IncludeItemTypes: 'Movie,Series',
|
||||||
|
SortBy: 'DatePlayed',
|
||||||
|
SortOrder: 'Descending',
|
||||||
|
Fields: 'UserData'
|
||||||
|
}),
|
||||||
|
excludedIds.length > 0
|
||||||
|
? fetchJson(
|
||||||
|
`${base}/Users/${encodeURIComponent(embyGuid)}/Items?api_key=${encodeURIComponent(apiKey)}&Ids=${encodeURIComponent(excludedIds.join(','))}&Fields=Path`
|
||||||
|
)
|
||||||
|
: Promise.resolve({ Items: [] })
|
||||||
|
]);
|
||||||
|
|
||||||
|
const context = writeCachedEmbyUserContext(embyGuid, {
|
||||||
|
views: normalizeViews(viewsPayload.Items || viewsPayload || []),
|
||||||
|
recentlyPlayed: normalizeRecentlyPlayed(recentlyPlayedPayload),
|
||||||
|
excludedFolderLookup: buildExcludedFolderLookup(excludedPayload.Items || excludedPayload || []),
|
||||||
|
lastSyncedAt: new Date().toISOString()
|
||||||
|
});
|
||||||
|
|
||||||
|
return json({ ...context, source: 'live' });
|
||||||
|
} catch (err) {
|
||||||
|
if (cached) {
|
||||||
|
const filteredLookup = excludedIds.length
|
||||||
|
? Object.fromEntries(
|
||||||
|
Object.entries(cached.excludedFolderLookup || {}).filter(([id]) => excludedIds.includes(id))
|
||||||
|
)
|
||||||
|
: (cached.excludedFolderLookup || {});
|
||||||
|
return json({
|
||||||
|
...cached,
|
||||||
|
excludedFolderLookup: filteredLookup,
|
||||||
|
source: 'cache',
|
||||||
|
message: `Using cached Emby user context because live fetch failed: ${err.message}`
|
||||||
|
});
|
||||||
|
}
|
||||||
|
throw error(502, `Could not load Emby user context: ${err.message}`);
|
||||||
|
}
|
||||||
|
}
|
||||||
@@ -1,6 +1,7 @@
|
|||||||
import { readFileSync, existsSync } from 'fs';
|
import { readFileSync, existsSync } from 'fs';
|
||||||
import { resolve } from 'path';
|
import { resolve } from 'path';
|
||||||
import { json, error } from '@sveltejs/kit';
|
import { json, error } from '@sveltejs/kit';
|
||||||
|
import { readCachedEmbyUsers, writeCachedEmbyUsers } from '../../../lib/server/emby-user-cache.js';
|
||||||
|
|
||||||
const CONFIG_PATH = resolve('config.json');
|
const CONFIG_PATH = resolve('config.json');
|
||||||
|
|
||||||
@@ -24,19 +25,44 @@ export async function GET() {
|
|||||||
try {
|
try {
|
||||||
res = await fetch(`${base}/Users?api_key=${encodeURIComponent(apiKey)}`);
|
res = await fetch(`${base}/Users?api_key=${encodeURIComponent(apiKey)}`);
|
||||||
} catch (e) {
|
} catch (e) {
|
||||||
|
const cached = readCachedEmbyUsers();
|
||||||
|
if (cached.users.length > 0) {
|
||||||
|
return json({
|
||||||
|
users: cached.users,
|
||||||
|
source: 'cache',
|
||||||
|
lastSyncedAt: cached.lastSyncedAt,
|
||||||
|
message: `Using cached Emby users because the server could not be reached: ${e.message}`
|
||||||
|
});
|
||||||
|
}
|
||||||
throw error(502, `Could not reach Emby server: ${e.message}`);
|
throw error(502, `Could not reach Emby server: ${e.message}`);
|
||||||
}
|
}
|
||||||
|
|
||||||
if (!res.ok) {
|
if (!res.ok) {
|
||||||
|
if (res.status >= 500) {
|
||||||
|
const cached = readCachedEmbyUsers();
|
||||||
|
if (cached.users.length > 0) {
|
||||||
|
return json({
|
||||||
|
users: cached.users,
|
||||||
|
source: 'cache',
|
||||||
|
lastSyncedAt: cached.lastSyncedAt,
|
||||||
|
message: `Using cached Emby users because the Emby API returned ${res.status} ${res.statusText}`
|
||||||
|
});
|
||||||
|
}
|
||||||
|
}
|
||||||
throw error(res.status, `Emby API error: ${res.status} ${res.statusText}`);
|
throw error(res.status, `Emby API error: ${res.status} ${res.statusText}`);
|
||||||
}
|
}
|
||||||
|
|
||||||
const users = await res.json();
|
const users = await res.json();
|
||||||
// Return embyGuid (no dashes, lowercase) -> name mapping
|
const cached = writeCachedEmbyUsers(
|
||||||
return json(
|
|
||||||
users.map((u) => ({
|
users.map((u) => ({
|
||||||
embyGuid: u.Id.replace(/-/g, '').toLowerCase(),
|
embyGuid: u.Id,
|
||||||
name: u.Name
|
name: u.Name
|
||||||
}))
|
}))
|
||||||
);
|
);
|
||||||
|
|
||||||
|
return json({
|
||||||
|
users: cached.users,
|
||||||
|
source: 'live',
|
||||||
|
lastSyncedAt: cached.lastSyncedAt
|
||||||
|
});
|
||||||
}
|
}
|
||||||
|
|||||||
@@ -0,0 +1,453 @@
|
|||||||
|
import assert from 'node:assert/strict';
|
||||||
|
import { mkdtempSync, rmSync } from 'fs';
|
||||||
|
import { join } from 'path';
|
||||||
|
import { tmpdir } from 'os';
|
||||||
|
import { createBoxSetSection, createRecentlyWatchedSection, getWatchlistLabelsForTarget } from '../src/lib/constants.js';
|
||||||
|
import { normalizeLookupItem, rankRecommendationResults } from '../src/lib/collection-tools.js';
|
||||||
|
import * as embyUserCache from '../src/lib/server/emby-user-cache.js';
|
||||||
|
|
||||||
|
const tests = [];
|
||||||
|
|
||||||
|
function test(name, fn) {
|
||||||
|
tests.push({ name, fn });
|
||||||
|
}
|
||||||
|
|
||||||
|
function withTempCache(testFn) {
|
||||||
|
const tempDir = mkdtempSync(join(tmpdir(), 'homescreenpal-cache-'));
|
||||||
|
const cachePath = join(tempDir, 'emby-users.db');
|
||||||
|
process.env.EMBY_USER_CACHE_DB_PATH = cachePath;
|
||||||
|
|
||||||
|
try {
|
||||||
|
return testFn(cachePath);
|
||||||
|
} finally {
|
||||||
|
delete process.env.EMBY_USER_CACHE_DB_PATH;
|
||||||
|
rmSync(tempDir, { recursive: true, force: true });
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
test('watchlist labels use the target Emby name for new watchlists', () => {
|
||||||
|
const labels = getWatchlistLabelsForTarget(
|
||||||
|
{
|
||||||
|
Name: "Matt's Watchlist",
|
||||||
|
CustomName: "Matt's Watchlist",
|
||||||
|
Query: { IsFavorite: true }
|
||||||
|
},
|
||||||
|
{
|
||||||
|
name: 'User 11',
|
||||||
|
embyName: 'Bob',
|
||||||
|
sections: []
|
||||||
|
}
|
||||||
|
);
|
||||||
|
|
||||||
|
assert.equal(labels.Name, "Bob's Watchlist");
|
||||||
|
assert.equal(labels.CustomName, "Bob's Watchlist");
|
||||||
|
});
|
||||||
|
|
||||||
|
test('watchlist labels preserve an existing matching target watchlist', () => {
|
||||||
|
const labels = getWatchlistLabelsForTarget(
|
||||||
|
{
|
||||||
|
Name: "Matt's Watchlist",
|
||||||
|
CustomName: "Matt's Watchlist",
|
||||||
|
Query: { IsFavorite: true }
|
||||||
|
},
|
||||||
|
{
|
||||||
|
name: 'User 14',
|
||||||
|
embyName: 'Bob',
|
||||||
|
sections: [
|
||||||
|
{
|
||||||
|
Name: "Bob's Watchlist",
|
||||||
|
CustomName: "Bob's Watchlist",
|
||||||
|
Query: { IsFavorite: true }
|
||||||
|
}
|
||||||
|
]
|
||||||
|
}
|
||||||
|
);
|
||||||
|
|
||||||
|
assert.equal(labels.Name, "Bob's Watchlist");
|
||||||
|
assert.equal(labels.CustomName, "Bob's Watchlist");
|
||||||
|
});
|
||||||
|
|
||||||
|
test('watchlist labels correct an existing mismatched target watchlist', () => {
|
||||||
|
const labels = getWatchlistLabelsForTarget(
|
||||||
|
{
|
||||||
|
Name: "Matt's Watchlist",
|
||||||
|
CustomName: "Matt's Watchlist",
|
||||||
|
Query: { IsFavorite: true }
|
||||||
|
},
|
||||||
|
{
|
||||||
|
name: 'User 14',
|
||||||
|
embyName: 'Bob',
|
||||||
|
sections: [
|
||||||
|
{
|
||||||
|
Name: "Matt's Watchlist",
|
||||||
|
CustomName: "Matt's Watchlist",
|
||||||
|
Query: { IsFavorite: true }
|
||||||
|
}
|
||||||
|
]
|
||||||
|
}
|
||||||
|
);
|
||||||
|
|
||||||
|
assert.equal(labels.Name, "Bob's Watchlist");
|
||||||
|
assert.equal(labels.CustomName, "Bob's Watchlist");
|
||||||
|
});
|
||||||
|
|
||||||
|
test('cached Emby users round-trip through the local SQLite cache', () => {
|
||||||
|
withTempCache(() => {
|
||||||
|
const written = embyUserCache.writeCachedEmbyUsers([
|
||||||
|
{ embyGuid: 'ABC-123', name: 'Matt' },
|
||||||
|
{ embyGuid: 'def456', name: 'Bob' }
|
||||||
|
]);
|
||||||
|
const readBack = embyUserCache.readCachedEmbyUsers();
|
||||||
|
|
||||||
|
assert.equal(written.users.length, 2);
|
||||||
|
assert.deepEqual(readBack.users, [
|
||||||
|
{ embyGuid: 'def456', name: 'Bob' },
|
||||||
|
{ embyGuid: 'abc123', name: 'Matt' }
|
||||||
|
]);
|
||||||
|
assert.ok(readBack.lastSyncedAt);
|
||||||
|
});
|
||||||
|
});
|
||||||
|
|
||||||
|
test('cached Emby users override DB display names without losing the original DB name', () => {
|
||||||
|
withTempCache(() => {
|
||||||
|
embyUserCache.writeCachedEmbyUsers([{ embyGuid: 'abc123', name: 'Matt' }]);
|
||||||
|
|
||||||
|
const enriched = embyUserCache.applyCachedEmbyNames([
|
||||||
|
{ id: 11, name: 'User 11', embyGuid: 'abc123' },
|
||||||
|
{ id: 12, name: 'Already Good', embyGuid: 'zzz999' }
|
||||||
|
]);
|
||||||
|
|
||||||
|
assert.equal(enriched.users[0].name, 'Matt');
|
||||||
|
assert.equal(enriched.users[0].embyName, 'Matt');
|
||||||
|
assert.equal(enriched.users[0].dbName, 'User 11');
|
||||||
|
assert.equal(enriched.users[1].name, 'Already Good');
|
||||||
|
assert.equal(enriched.cache.matchedCount, 1);
|
||||||
|
});
|
||||||
|
});
|
||||||
|
|
||||||
|
test('recently watched section template uses played-content defaults', () => {
|
||||||
|
const section = createRecentlyWatchedSection('abc123', 'Matt');
|
||||||
|
|
||||||
|
assert.equal(section.UserId, 'abc123');
|
||||||
|
assert.equal(section.Name, 'Recently Watched - Matt');
|
||||||
|
assert.equal(section.CustomName, 'Recently Watched - Matt');
|
||||||
|
assert.equal(section.SectionType, 'items');
|
||||||
|
assert.equal(section.SortBy, 'DatePlayed');
|
||||||
|
assert.equal(section.SortOrder, 'Descending');
|
||||||
|
assert.deepEqual(section.ItemTypes, ['Movie', 'Series']);
|
||||||
|
assert.equal(section.Query.IsPlayed, true);
|
||||||
|
});
|
||||||
|
|
||||||
|
test('box set section template links a created collection', () => {
|
||||||
|
const section = createBoxSetSection('abc123', 'Recommended for Matt', 'boxset-55');
|
||||||
|
|
||||||
|
assert.equal(section.UserId, 'abc123');
|
||||||
|
assert.equal(section.SectionType, 'boxset');
|
||||||
|
assert.equal(section.Name, 'Recommended for Matt');
|
||||||
|
assert.equal(section.CustomName, 'Recommended for Matt');
|
||||||
|
assert.equal(section.ParentId, 'boxset-55');
|
||||||
|
assert.deepEqual(section.ParentItem, {
|
||||||
|
Name: 'Recommended for Matt',
|
||||||
|
Id: 'boxset-55'
|
||||||
|
});
|
||||||
|
});
|
||||||
|
|
||||||
|
test('normalizeLookupItem preserves already-normalized recommendation items', () => {
|
||||||
|
const normalized = normalizeLookupItem({
|
||||||
|
id: 'pick-1',
|
||||||
|
name: 'Alpha',
|
||||||
|
type: 'Movie',
|
||||||
|
overview: 'Overview',
|
||||||
|
year: 1998,
|
||||||
|
communityRating: 7.4,
|
||||||
|
providerIds: { Tmdb: '123' },
|
||||||
|
genres: ['Comedy']
|
||||||
|
});
|
||||||
|
|
||||||
|
assert.deepEqual(normalized, {
|
||||||
|
id: 'pick-1',
|
||||||
|
name: 'Alpha',
|
||||||
|
type: 'Movie',
|
||||||
|
overview: 'Overview',
|
||||||
|
year: 1998,
|
||||||
|
communityRating: 7.4,
|
||||||
|
providerIds: { Tmdb: '123' },
|
||||||
|
genres: ['Comedy'],
|
||||||
|
parentId: null
|
||||||
|
});
|
||||||
|
});
|
||||||
|
|
||||||
|
test('recommendation ranking deduplicates and prioritizes overlaps across seeds', () => {
|
||||||
|
const ranked = rankRecommendationResults(
|
||||||
|
['seed-a', 'seed-b'],
|
||||||
|
[
|
||||||
|
[
|
||||||
|
{ Id: 'pick-1', Name: 'Alpha', Type: 'Movie' },
|
||||||
|
{ Id: 'pick-2', Name: 'Bravo', Type: 'Movie' }
|
||||||
|
],
|
||||||
|
[
|
||||||
|
{ Id: 'pick-2', Name: 'Bravo', Type: 'Movie' },
|
||||||
|
{ Id: 'pick-3', Name: 'Charlie', Type: 'Movie' }
|
||||||
|
]
|
||||||
|
],
|
||||||
|
5
|
||||||
|
);
|
||||||
|
|
||||||
|
assert.deepEqual(
|
||||||
|
ranked.map((item) => item.id),
|
||||||
|
['pick-2', 'pick-1', 'pick-3']
|
||||||
|
);
|
||||||
|
assert.equal(ranked[0].matchCount, 2);
|
||||||
|
});
|
||||||
|
|
||||||
|
test('recommendation ranking supports already-normalized result items', () => {
|
||||||
|
const ranked = rankRecommendationResults(
|
||||||
|
['seed-a'],
|
||||||
|
[
|
||||||
|
[
|
||||||
|
{ id: 'pick-1', name: 'Alpha', type: 'Movie' },
|
||||||
|
{ id: 'pick-2', name: 'Bravo', type: 'Movie' }
|
||||||
|
]
|
||||||
|
],
|
||||||
|
5
|
||||||
|
);
|
||||||
|
|
||||||
|
assert.deepEqual(
|
||||||
|
ranked.map((item) => item.id),
|
||||||
|
['pick-1', 'pick-2']
|
||||||
|
);
|
||||||
|
});
|
||||||
|
|
||||||
|
test('recommendation ranking filters results to the selected seed media type', () => {
|
||||||
|
const ranked = rankRecommendationResults(
|
||||||
|
['seed-a'],
|
||||||
|
[
|
||||||
|
[
|
||||||
|
{ Id: 'movie-1', Name: 'Movie Match', Type: 'Movie', Genres: ['Drama'] },
|
||||||
|
{ Id: 'series-1', Name: 'Series Match', Type: 'Series', Genres: ['Drama'] }
|
||||||
|
]
|
||||||
|
],
|
||||||
|
{
|
||||||
|
limit: 5,
|
||||||
|
allowedTypes: ['Movie']
|
||||||
|
}
|
||||||
|
);
|
||||||
|
|
||||||
|
assert.deepEqual(
|
||||||
|
ranked.map((item) => item.id),
|
||||||
|
['movie-1']
|
||||||
|
);
|
||||||
|
});
|
||||||
|
|
||||||
|
test('recommendation ranking heavily favors TMDB similar and recommendation sources over discover', () => {
|
||||||
|
const ranked = rankRecommendationResults(
|
||||||
|
['seed-a'],
|
||||||
|
[
|
||||||
|
{
|
||||||
|
items: [{ Id: 'pick-seed-like', Name: 'Seed Like', Type: 'Movie', Genres: ['Drama', 'Romance'] }],
|
||||||
|
sourceWeight: 6
|
||||||
|
},
|
||||||
|
{
|
||||||
|
items: [{ Id: 'pick-discover', Name: 'Broad Discover', Type: 'Movie', Genres: ['Drama'] }],
|
||||||
|
sourceWeight: 1
|
||||||
|
},
|
||||||
|
{
|
||||||
|
items: [{ Id: 'pick-discover', Name: 'Broad Discover', Type: 'Movie', Genres: ['Drama'] }],
|
||||||
|
sourceWeight: 1
|
||||||
|
},
|
||||||
|
{
|
||||||
|
items: [{ Id: 'pick-discover', Name: 'Broad Discover', Type: 'Movie', Genres: ['Drama'] }],
|
||||||
|
sourceWeight: 1
|
||||||
|
}
|
||||||
|
],
|
||||||
|
5
|
||||||
|
);
|
||||||
|
|
||||||
|
assert.deepEqual(
|
||||||
|
ranked.map((item) => item.id),
|
||||||
|
['pick-seed-like', 'pick-discover']
|
||||||
|
);
|
||||||
|
});
|
||||||
|
|
||||||
|
test('recommendation ranking excludes watched items passed in options', () => {
|
||||||
|
const ranked = rankRecommendationResults(
|
||||||
|
['seed-a'],
|
||||||
|
[
|
||||||
|
[
|
||||||
|
{ Id: 'watched-1', Name: 'Already Seen', Type: 'Movie' },
|
||||||
|
{ Id: 'pick-1', Name: 'Fresh Pick', Type: 'Movie' }
|
||||||
|
],
|
||||||
|
[
|
||||||
|
{ Id: 'pick-1', Name: 'Fresh Pick', Type: 'Movie' },
|
||||||
|
{ Id: 'pick-2', Name: 'Another Pick', Type: 'Movie' }
|
||||||
|
]
|
||||||
|
],
|
||||||
|
{
|
||||||
|
limit: 5,
|
||||||
|
excludeIds: ['watched-1']
|
||||||
|
}
|
||||||
|
);
|
||||||
|
|
||||||
|
assert.deepEqual(
|
||||||
|
ranked.map((item) => item.id),
|
||||||
|
['pick-1', 'pick-2']
|
||||||
|
);
|
||||||
|
});
|
||||||
|
|
||||||
|
test('recommendation ranking penalizes horror and sci-fi mismatches for romance drama seeds', () => {
|
||||||
|
const ranked = rankRecommendationResults(
|
||||||
|
['seed-a'],
|
||||||
|
[
|
||||||
|
[
|
||||||
|
{
|
||||||
|
Id: 'pick-warm',
|
||||||
|
Name: 'Quiet Hearts',
|
||||||
|
Type: 'Movie',
|
||||||
|
Overview: 'A love story about family, friendship, and second chances.',
|
||||||
|
Genres: ['Romance', 'Drama']
|
||||||
|
},
|
||||||
|
{
|
||||||
|
Id: 'pick-mismatch',
|
||||||
|
Name: 'Night Terrors',
|
||||||
|
Type: 'Movie',
|
||||||
|
Overview: 'A horror thriller about murder, zombies, and survival.',
|
||||||
|
Genres: ['Horror', 'Science Fiction', 'Thriller']
|
||||||
|
}
|
||||||
|
]
|
||||||
|
],
|
||||||
|
{
|
||||||
|
limit: 5,
|
||||||
|
seeds: [
|
||||||
|
{
|
||||||
|
Id: 'seed-a',
|
||||||
|
Name: 'The English Patient',
|
||||||
|
Type: 'Movie',
|
||||||
|
Genres: ['Romance', 'Drama']
|
||||||
|
}
|
||||||
|
]
|
||||||
|
}
|
||||||
|
);
|
||||||
|
|
||||||
|
assert.deepEqual(
|
||||||
|
ranked.map((item) => item.id),
|
||||||
|
['pick-warm', 'pick-mismatch']
|
||||||
|
);
|
||||||
|
assert.ok(ranked[0].affinityScore > ranked[1].affinityScore);
|
||||||
|
});
|
||||||
|
|
||||||
|
test('recommendation ranking ignores duplicate items within a single seed result set', () => {
|
||||||
|
const ranked = rankRecommendationResults(
|
||||||
|
['seed-a'],
|
||||||
|
[
|
||||||
|
[
|
||||||
|
{ Id: 'pick-1', Name: 'Alpha', Type: 'Movie' },
|
||||||
|
{ Id: 'pick-1', Name: 'Alpha', Type: 'Movie' },
|
||||||
|
{ Id: 'pick-2', Name: 'Bravo', Type: 'Movie' }
|
||||||
|
]
|
||||||
|
],
|
||||||
|
5
|
||||||
|
);
|
||||||
|
|
||||||
|
assert.deepEqual(
|
||||||
|
ranked.map((item) => item.id),
|
||||||
|
['pick-1', 'pick-2']
|
||||||
|
);
|
||||||
|
assert.equal(ranked[0].matchCount, 1);
|
||||||
|
});
|
||||||
|
|
||||||
|
test('classic comfort profile filters modern and mismatched picks while promoting older warm titles', () => {
|
||||||
|
const ranked = rankRecommendationResults(
|
||||||
|
['seed-a', 'seed-b'],
|
||||||
|
[
|
||||||
|
[
|
||||||
|
{
|
||||||
|
Id: 'pick-classic',
|
||||||
|
Name: 'Warm Hearts',
|
||||||
|
Type: 'Movie',
|
||||||
|
Overview: 'A romantic family comedy about friendship, second chances, and love.',
|
||||||
|
ProductionYear: 1998,
|
||||||
|
CommunityRating: 7.6,
|
||||||
|
Genres: ['Romance', 'Comedy']
|
||||||
|
},
|
||||||
|
{
|
||||||
|
Id: 'pick-modern',
|
||||||
|
Name: 'Future War',
|
||||||
|
Type: 'Movie',
|
||||||
|
Overview: 'A modern action thriller about combat and alien invasion.',
|
||||||
|
ProductionYear: 2021,
|
||||||
|
CommunityRating: 7.8,
|
||||||
|
Genres: ['Action', 'Science Fiction']
|
||||||
|
}
|
||||||
|
],
|
||||||
|
[
|
||||||
|
{
|
||||||
|
Id: 'pick-classic',
|
||||||
|
Name: 'Warm Hearts',
|
||||||
|
Type: 'Movie',
|
||||||
|
Overview: 'A romantic family comedy about friendship, second chances, and love.',
|
||||||
|
ProductionYear: 1998,
|
||||||
|
CommunityRating: 7.6,
|
||||||
|
Genres: ['Romance', 'Comedy']
|
||||||
|
},
|
||||||
|
{
|
||||||
|
Id: 'pick-animation',
|
||||||
|
Name: 'Toon Town',
|
||||||
|
Type: 'Movie',
|
||||||
|
Overview: 'An animated adventure about a magical family.',
|
||||||
|
ProductionYear: 2002,
|
||||||
|
CommunityRating: 8.0,
|
||||||
|
Genres: ['Animation', 'Family']
|
||||||
|
}
|
||||||
|
]
|
||||||
|
],
|
||||||
|
{
|
||||||
|
limit: 5,
|
||||||
|
profile: 'classicComfort',
|
||||||
|
seeds: [
|
||||||
|
{
|
||||||
|
Id: 'seed-a',
|
||||||
|
Name: 'Sleepless Style',
|
||||||
|
Type: 'Movie',
|
||||||
|
Overview: 'A romantic comedy about love and second chances.',
|
||||||
|
ProductionYear: 1993,
|
||||||
|
CommunityRating: 7.1,
|
||||||
|
Genres: ['Romance', 'Comedy']
|
||||||
|
},
|
||||||
|
{
|
||||||
|
Id: 'seed-b',
|
||||||
|
Name: 'Family Ties',
|
||||||
|
Type: 'Movie',
|
||||||
|
Overview: 'A family friendship story with warmth.',
|
||||||
|
ProductionYear: 1995,
|
||||||
|
CommunityRating: 7.0,
|
||||||
|
Genres: ['Comedy', 'Drama']
|
||||||
|
}
|
||||||
|
]
|
||||||
|
}
|
||||||
|
);
|
||||||
|
|
||||||
|
assert.deepEqual(ranked.map((item) => item.id), ['pick-classic']);
|
||||||
|
assert.equal(ranked[0].matchCount, 2);
|
||||||
|
assert.ok(ranked[0].styleScore >= 16);
|
||||||
|
assert.equal(ranked[0].qualityScore, 76);
|
||||||
|
});
|
||||||
|
|
||||||
|
let failures = 0;
|
||||||
|
|
||||||
|
for (const { name, fn } of tests) {
|
||||||
|
try {
|
||||||
|
await fn();
|
||||||
|
console.log(`PASS ${name}`);
|
||||||
|
} catch (error) {
|
||||||
|
failures++;
|
||||||
|
console.error(`FAIL ${name}`);
|
||||||
|
console.error(error);
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
if (failures > 0) {
|
||||||
|
console.error(`\n${failures} test(s) failed.`);
|
||||||
|
process.exit(1);
|
||||||
|
}
|
||||||
|
|
||||||
|
console.log(`\n${tests.length} test(s) passed.`);
|
||||||
Reference in New Issue
Block a user