ADMIN: Added track management
Build and Publish / Build and Publish Docker Image (push) Successful in 2m56s

This commit is contained in:
2026-05-29 00:43:32 +03:00
parent 8073ac9a97
commit 1bb5a2f973
4 changed files with 392 additions and 26 deletions
Generated
+1 -1
View File
@@ -1418,7 +1418,7 @@ checksum = "e6d5a32815ae3f33302d95fdcb2ce17862f8c65363dcfd29360480ba1001fc9c"
[[package]]
name = "furumusic"
version = "0.2.4"
version = "0.2.5"
dependencies = [
"anyhow",
"async-trait",
+1 -1
View File
@@ -1,6 +1,6 @@
[package]
name = "furumusic"
version = "0.2.5"
version = "0.2.6"
edition = "2024"
description = "Reusable web-app boilerplate: auth, OIDC/SSO, admin panel, user management, i18n, PostgreSQL"
+260 -1
View File
@@ -69,6 +69,9 @@ pub(super) struct UpdateLibraryItemRequest {
hidden: bool,
release_type: Option<String>,
year: Option<String>,
release_id: Option<i64>,
track_number: Option<String>,
disc_number: Option<String>,
artist_ids: Option<Vec<i64>>,
}
@@ -385,9 +388,13 @@ struct LibraryItemDetailDto {
hidden: bool,
release_type: Option<String>,
year: Option<i32>,
release_id: Option<i64>,
track_number: Option<i32>,
disc_number: Option<i32>,
current_image_url: Option<String>,
selected_artist_ids: Vec<i64>,
artists: Vec<ArtistOptionDto>,
releases: Vec<ReleaseOptionDto>,
available_covers: Vec<AvailableCoverDto>,
}
@@ -397,6 +404,13 @@ struct ArtistOptionDto {
name: String,
}
#[derive(Debug, Serialize, JsonSchema)]
struct ReleaseOptionDto {
id: i64,
title: String,
subtitle: String,
}
#[derive(Debug, Serialize, JsonSchema)]
struct AvailableCoverDto {
media_file_id: i64,
@@ -1050,6 +1064,40 @@ pub async fn update_library_item(
.execute(pool)
.await
}
"tracks" => {
let release_id = body.release_id.unwrap_or(0);
if release_id <= 0 {
return Ok(json_error(StatusCode::BAD_REQUEST, "release is required"));
}
let release_exists: Option<i64> =
sqlx::query_scalar("SELECT id FROM furumusic__release WHERE id = $1")
.bind(release_id)
.fetch_optional(pool)
.await
.map_err(|e| cot::Error::internal(e.to_string()))?;
if release_exists.is_none() {
return Ok(json_error(StatusCode::NOT_FOUND, "release not found"));
}
let year = parse_optional_admin_i32(body.year.as_deref(), 0, 3000);
let track_number = parse_optional_admin_i32(body.track_number.as_deref(), 1, 9999);
let disc_number = parse_optional_admin_i32(body.disc_number.as_deref(), 1, 999);
sqlx::query(
"UPDATE furumusic__track \
SET title = $1, title_sort = $2, release_id = $3, track_number = $4, disc_number = $5, year = $6, is_hidden = $7, updated_at = $8 \
WHERE id = $9",
)
.bind(title)
.bind(normalize_name(title))
.bind(release_id)
.bind(track_number)
.bind(disc_number)
.bind(year)
.bind(body.hidden)
.bind(&now)
.bind(body.id)
.execute(pool)
.await
}
"playlists" => {
sqlx::query(
"UPDATE furumusic__playlist \
@@ -1071,10 +1119,11 @@ pub async fn update_library_item(
if affected == 0 {
return Ok(json_error(StatusCode::NOT_FOUND, "library item not found"));
}
if kind == "releases" {
if kind == "releases" || kind == "tracks" {
if let Some(mut artist_ids) = body.artist_ids {
let mut seen_artist_ids = HashSet::new();
artist_ids.retain(|id| *id > 0 && seen_artist_ids.insert(*id));
if kind == "releases" {
sqlx::query("DELETE FROM furumusic__release_artist WHERE release_id = $1")
.bind(body.id)
.execute(pool)
@@ -1091,6 +1140,24 @@ pub async fn update_library_item(
.await
.map_err(|e| cot::Error::internal(e.to_string()))?;
}
} else {
sqlx::query("DELETE FROM furumusic__track_artist WHERE track_id = $1 AND role = 'main'")
.bind(body.id)
.execute(pool)
.await
.map_err(|e| cot::Error::internal(e.to_string()))?;
for (position, artist_id) in artist_ids.iter().enumerate() {
sqlx::query(
"INSERT INTO furumusic__track_artist (track_id, artist_id, role, position) VALUES ($1, $2, 'main', $3)",
)
.bind(body.id)
.bind(*artist_id)
.bind(position as i32)
.execute(pool)
.await
.map_err(|e| cot::Error::internal(e.to_string()))?;
}
}
}
}
@@ -1781,6 +1848,7 @@ async fn load_library_page(pool: &PgPool, query: LibraryQuery) -> anyhow::Result
let total = count_library(pool, &kind, search_pattern.clone()).await?;
let rows = match kind.as_str() {
"releases" => load_release_items(pool, search_pattern.clone(), limit, offset).await?,
"tracks" => load_track_items(pool, search_pattern.clone(), limit, offset).await?,
"playlists" => load_playlist_items(pool, search_pattern.clone(), limit, offset).await?,
_ => load_artist_items(pool, search_pattern.clone(), limit, offset).await?,
};
@@ -1841,6 +1909,26 @@ async fn fetch_library_item(
.fetch_optional(pool)
.await?
}
"tracks" => {
sqlx::query_as::<_, LibraryItemRow>(
"SELECT t.id, t.title::text AS title, \
CONCAT(r.title::text, COALESCE(' / #' || t.track_number::text, '')) AS subtitle, \
t.is_hidden, COUNT(DISTINCT ta.artist_id)::bigint AS primary_count, \
COUNT(DISTINCT ph.id)::bigint AS secondary_count, \
COUNT(DISTINCT pt.playlist_id)::bigint AS tertiary_count, \
t.updated_at::text AS updated_at \
FROM furumusic__track t \
JOIN furumusic__release r ON r.id = t.release_id \
LEFT JOIN furumusic__track_artist ta ON ta.track_id = t.id \
LEFT JOIN furumusic__play_history ph ON ph.track_id = t.id \
LEFT JOIN furumusic__playlist_track pt ON pt.track_id = t.id \
WHERE t.id = $1 \
GROUP BY t.id, r.title",
)
.bind(id)
.fetch_optional(pool)
.await?
}
_ => {
sqlx::query_as::<_, LibraryItemRow>(
"SELECT a.id, a.name::text AS title, NULL::text AS subtitle, a.is_hidden, \
@@ -1874,9 +1962,13 @@ async fn load_library_item_detail(
hidden: item.is_hidden.unwrap_or(false),
release_type: None,
year: None,
release_id: None,
track_number: None,
disc_number: None,
current_image_url: None,
selected_artist_ids: Vec::new(),
artists: Vec::new(),
releases: Vec::new(),
available_covers: Vec::new(),
item,
};
@@ -1917,6 +2009,31 @@ async fn load_library_item_detail(
.collect();
detail.artists = load_artist_options(pool).await?;
}
"tracks" => {
let row: Option<(i64, Option<i32>, Option<i32>, Option<i32>)> = sqlx::query_as(
"SELECT release_id, track_number, disc_number, year FROM furumusic__track WHERE id = $1",
)
.bind(detail.item.id)
.fetch_optional(pool)
.await?;
if let Some((release_id, track_number, disc_number, year)) = row {
detail.release_id = Some(release_id);
detail.track_number = track_number;
detail.disc_number = disc_number;
detail.year = year;
}
detail.selected_artist_ids = sqlx::query_as::<_, IdRow>(
"SELECT artist_id AS id FROM furumusic__track_artist WHERE track_id = $1 AND role = 'main' ORDER BY position, artist_id",
)
.bind(detail.item.id)
.fetch_all(pool)
.await?
.into_iter()
.map(|row| row.id)
.collect();
detail.artists = load_artist_options(pool).await?;
detail.releases = load_release_options(pool).await?;
}
_ => {}
}
@@ -1935,6 +2052,25 @@ async fn load_artist_options(pool: &PgPool) -> anyhow::Result<Vec<ArtistOptionDt
.collect())
}
async fn load_release_options(pool: &PgPool) -> anyhow::Result<Vec<ReleaseOptionDto>> {
let rows = sqlx::query_as::<_, (i64, String, Option<String>)>(
"SELECT r.id, r.title::text AS title, \
CONCAT(r.release_type::text, COALESCE(' / ' || r.year::text, '')) AS subtitle \
FROM furumusic__release r \
ORDER BY r.title_sort ASC, r.year NULLS LAST, r.id ASC",
)
.fetch_all(pool)
.await?;
Ok(rows
.into_iter()
.map(|(id, title, subtitle)| ReleaseOptionDto {
id,
title,
subtitle: subtitle.unwrap_or_default(),
})
.collect())
}
async fn artist_available_covers(
pool: &PgPool,
artist_id: i64,
@@ -1974,6 +2110,13 @@ async fn library_ids_by_filter(
LEFT JOIN furumusic__release_artist ra ON ra.release_id = r.id \
LEFT JOIN furumusic__artist a ON a.id = ra.artist_id WHERE 1=1",
),
"tracks" => QueryBuilder::<Postgres>::new(
"SELECT DISTINCT t.id \
FROM furumusic__track t \
JOIN furumusic__release r ON r.id = t.release_id \
LEFT JOIN furumusic__track_artist ta ON ta.track_id = t.id \
LEFT JOIN furumusic__artist a ON a.id = ta.artist_id WHERE 1=1",
),
"playlists" => QueryBuilder::<Postgres>::new(
"SELECT DISTINCT p.id \
FROM furumusic__playlist p \
@@ -2030,6 +2173,14 @@ async fn set_library_visibility(
.bind(ids)
.execute(pool)
.await,
"tracks" => sqlx::query(
"UPDATE furumusic__track SET is_hidden = $1, updated_at = $2 WHERE id = ANY($3)",
)
.bind(hidden)
.bind(&now)
.bind(ids)
.execute(pool)
.await,
_ => sqlx::query(
"UPDATE furumusic__artist SET is_hidden = $1, updated_at = $2 WHERE id = ANY($3)",
)
@@ -2046,6 +2197,7 @@ async fn set_library_visibility(
async fn delete_library_items(pool: &PgPool, kind: &str, ids: &[i64]) -> cot::Result<u64> {
match kind {
"releases" => delete_releases(pool, ids).await,
"tracks" => delete_tracks(pool, ids).await,
"playlists" => delete_playlists(pool, ids).await,
_ => delete_artists(pool, ids).await,
}
@@ -2132,6 +2284,40 @@ async fn delete_releases(pool: &PgPool, ids: &[i64]) -> cot::Result<u64> {
Ok(result.rows_affected())
}
async fn delete_tracks(pool: &PgPool, ids: &[i64]) -> cot::Result<u64> {
sqlx::query("DELETE FROM furumusic__playlist_track WHERE track_id = ANY($1)")
.bind(ids)
.execute(pool)
.await
.map_err(|e| cot::Error::internal(e.to_string()))?;
sqlx::query("DELETE FROM furumusic__user_liked_track WHERE track_id = ANY($1)")
.bind(ids)
.execute(pool)
.await
.map_err(|e| cot::Error::internal(e.to_string()))?;
sqlx::query("DELETE FROM furumusic__play_history WHERE track_id = ANY($1)")
.bind(ids)
.execute(pool)
.await
.map_err(|e| cot::Error::internal(e.to_string()))?;
sqlx::query("DELETE FROM furumusic__track_genre WHERE track_id = ANY($1)")
.bind(ids)
.execute(pool)
.await
.map_err(|e| cot::Error::internal(e.to_string()))?;
sqlx::query("DELETE FROM furumusic__track_artist WHERE track_id = ANY($1)")
.bind(ids)
.execute(pool)
.await
.map_err(|e| cot::Error::internal(e.to_string()))?;
let result = sqlx::query("DELETE FROM furumusic__track WHERE id = ANY($1)")
.bind(ids)
.execute(pool)
.await
.map_err(|e| cot::Error::internal(e.to_string()))?;
Ok(result.rows_affected())
}
async fn delete_playlists(pool: &PgPool, ids: &[i64]) -> cot::Result<u64> {
sqlx::query("DELETE FROM furumusic__playlist_track WHERE playlist_id = ANY($1)")
.bind(ids)
@@ -2163,6 +2349,13 @@ async fn count_library(
LEFT JOIN furumusic__release_artist ra ON ra.release_id = r.id \
LEFT JOIN furumusic__artist a ON a.id = ra.artist_id WHERE 1=1",
),
"tracks" => QueryBuilder::<Postgres>::new(
"SELECT COUNT(DISTINCT t.id) AS count \
FROM furumusic__track t \
JOIN furumusic__release r ON r.id = t.release_id \
LEFT JOIN furumusic__track_artist ta ON ta.track_id = t.id \
LEFT JOIN furumusic__artist a ON a.id = ta.artist_id WHERE 1=1",
),
"playlists" => QueryBuilder::<Postgres>::new(
"SELECT COUNT(DISTINCT p.id) AS count \
FROM furumusic__playlist p \
@@ -2203,6 +2396,15 @@ fn push_library_search_filter(
qb.push_bind(pattern);
qb.push(")");
}
"tracks" => {
qb.push(" AND (t.title ILIKE ");
qb.push_bind(pattern.clone());
qb.push(" OR r.title ILIKE ");
qb.push_bind(pattern.clone());
qb.push(" OR a.name ILIKE ");
qb.push_bind(pattern);
qb.push(")");
}
_ => {
qb.push(" AND a.name ILIKE ");
qb.push_bind(pattern);
@@ -2279,6 +2481,46 @@ async fn load_release_items(
.await?)
}
async fn load_track_items(
pool: &PgPool,
search_pattern: Option<String>,
limit: i64,
offset: i64,
) -> anyhow::Result<Vec<LibraryItemRow>> {
let mut qb = QueryBuilder::<Postgres>::new(
"SELECT t.id, t.title::text AS title, \
CONCAT(r.title::text, COALESCE(' / #' || t.track_number::text, '')) AS subtitle, \
t.is_hidden, COUNT(DISTINCT ta.artist_id)::bigint AS primary_count, \
COUNT(DISTINCT ph.id)::bigint AS secondary_count, \
COUNT(DISTINCT pt.playlist_id)::bigint AS tertiary_count, \
t.updated_at::text AS updated_at \
FROM furumusic__track t \
JOIN furumusic__release r ON r.id = t.release_id \
LEFT JOIN furumusic__track_artist ta ON ta.track_id = t.id \
LEFT JOIN furumusic__artist a ON a.id = ta.artist_id \
LEFT JOIN furumusic__play_history ph ON ph.track_id = t.id \
LEFT JOIN furumusic__playlist_track pt ON pt.track_id = t.id \
WHERE 1=1",
);
if let Some(pattern) = search_pattern {
qb.push(" AND (t.title ILIKE ");
qb.push_bind(pattern.clone());
qb.push(" OR r.title ILIKE ");
qb.push_bind(pattern.clone());
qb.push(" OR a.name ILIKE ");
qb.push_bind(pattern);
qb.push(")");
}
qb.push(" GROUP BY t.id, r.title ORDER BY r.title ASC, t.disc_number NULLS FIRST, t.track_number NULLS FIRST, t.title ASC LIMIT ");
qb.push_bind(limit);
qb.push(" OFFSET ");
qb.push_bind(offset);
Ok(qb
.build_query_as::<LibraryItemRow>()
.fetch_all(pool)
.await?)
}
async fn load_playlist_items(
pool: &PgPool,
search_pattern: Option<String>,
@@ -2324,6 +2566,11 @@ fn library_item_dto(kind: &str, row: LibraryItemRow) -> LibraryItemDto {
tag(format!("{} artists", row.secondary_count), "relation"),
tag(format!("{} plays", row.tertiary_count), "plays"),
],
"tracks" => vec![
tag(format!("{} artists", row.primary_count), "relation"),
tag(format!("{} plays", row.secondary_count), "plays"),
tag(format!("{} playlists", row.tertiary_count), "count"),
],
"playlists" => vec![
tag(format!("{} tracks", row.primary_count), "count"),
tag(
@@ -2412,6 +2659,7 @@ fn optional_job_time(value: &str) -> Option<String> {
fn normalize_library_kind(kind: Option<&str>) -> String {
match kind {
Some("releases") => "releases",
Some("tracks") => "tracks",
Some("playlists") => "playlists",
_ => "artists",
}
@@ -2437,6 +2685,17 @@ fn normalize_name(value: &str) -> String {
value.trim().to_lowercase()
}
fn parse_optional_admin_i32(value: Option<&str>, min: i32, max: i32) -> Option<i32> {
let value = value?.trim();
if value.is_empty() {
return None;
}
value
.parse::<i32>()
.ok()
.map(|parsed| parsed.clamp(min, max))
}
fn now_string() -> String {
chrono::Utc::now().format("%Y-%m-%dT%H:%M:%SZ").to_string()
}
+115 -8
View File
@@ -975,6 +975,16 @@ tbody tr:hover {
cursor: pointer;
}
.artist-result small {
display: block;
margin-top: 2px;
color: var(--text-subdued);
font-size: 11px;
overflow: hidden;
text-overflow: ellipsis;
white-space: nowrap;
}
.artist-result:hover,
.artist-result:focus {
background: var(--bg-hover);
@@ -1452,6 +1462,7 @@ tbody tr:hover {
<div class="segmented">
<button class="seg-btn" :class="{active: libraryKind === 'artists'}" @click="openLibrary('artists')">Artists</button>
<button class="seg-btn" :class="{active: libraryKind === 'releases'}" @click="openLibrary('releases')">Releases</button>
<button class="seg-btn" :class="{active: libraryKind === 'tracks'}" @click="openLibrary('tracks')">Tracks</button>
<button class="seg-btn" :class="{active: libraryKind === 'playlists'}" @click="openLibrary('playlists')">Playlists</button>
</div>
<input class="search" placeholder="Search library" x-model="librarySearch" @input.debounce.350ms="loadLibrary()" />
@@ -1947,8 +1958,43 @@ tbody tr:hover {
</div>
</div>
<div class="field" x-show="isReleaseEditor()">
<label>Release artists</label>
<div class="editor-grid" x-show="isTrackEditor()">
<div class="field">
<label>Track #</label>
<input type="number" min="1" max="9999" x-model="editorDraft.track_number" />
</div>
<div class="field">
<label>Disc #</label>
<input type="number" min="1" max="999" x-model="editorDraft.disc_number" />
</div>
<div class="field">
<label>Year</label>
<input type="number" min="0" max="3000" x-model="editorDraft.year" />
</div>
</div>
<div class="field" x-show="isTrackEditor()">
<label>Release</label>
<div class="artist-tags">
<span class="tag relation" x-show="selectedEditorRelease()" x-text="selectedEditorReleaseLabel()"></span>
<span class="muted" x-show="!selectedEditorRelease()">No release selected</span>
</div>
<div class="artist-picker">
<input class="search" placeholder="Search release to move track" x-model="editorReleaseToAdd" @keydown.enter.prevent="selectEditorRelease()" @keydown.escape="editorReleaseToAdd = ''" />
<div class="artist-results" x-show="editorReleaseSearchOpen()" x-transition>
<template x-for="release in filteredEditorReleases()" :key="release.id">
<button class="artist-result" type="button" @click="selectEditorRelease(release)">
<span x-text="release.title"></span>
<small x-text="release.subtitle"></small>
</button>
</template>
<div class="artist-result muted" x-show="filteredEditorReleases().length === 0">No matching releases</div>
</div>
</div>
</div>
<div class="field" x-show="isReleaseEditor() || isTrackEditor()">
<label x-text="isTrackEditor() ? 'Track artists' : 'Release artists'"></label>
<div class="artist-tags">
<template x-for="artist in selectedEditorArtists()" :key="artist.id">
<span class="tag relation">
@@ -2099,8 +2145,9 @@ function adminV2() {
editorImageUploading: false,
editorImageFile: null,
editorArtistToAdd: '',
editorReleaseToAdd: '',
editorDetail: null,
editorDraft: { title: '', hidden: 'false', release_type: 'album', year: '', artist_ids: [] },
editorDraft: { title: '', hidden: 'false', release_type: 'album', year: '', release_id: null, track_number: '', disc_number: '', artist_ids: [] },
settings: { values: {}, sources: {}, lastfm_api_key_configured: false, lastfm_shared_secret_configured: false, lastfm_scrobbling_configured: false },
settingsDraft: {
auth_password_enabled: false,
@@ -2196,7 +2243,7 @@ function adminV2() {
this.activeView = 'jobs';
if (parts[1]) this.activeJobName = decodeURIComponent(parts[1]);
} else if (view === 'library') {
const nextKind = ['artists', 'releases', 'playlists'].includes(parts[1]) ? parts[1] : 'artists';
const nextKind = ['artists', 'releases', 'tracks', 'playlists'].includes(parts[1]) ? parts[1] : 'artists';
if (this.libraryKind !== nextKind) this.clearLibrarySelection();
this.activeView = 'library';
this.libraryKind = nextKind;
@@ -2256,7 +2303,7 @@ function adminV2() {
openLibrary(kind = this.libraryKind) {
this.activeView = 'library';
const nextKind = ['artists', 'releases', 'playlists'].includes(kind) ? kind : 'artists';
const nextKind = ['artists', 'releases', 'tracks', 'playlists'].includes(kind) ? kind : 'artists';
if (this.libraryKind !== nextKind) this.clearLibrarySelection();
this.libraryKind = nextKind;
this.setRoute(`#library/${this.libraryKind}`);
@@ -2618,11 +2665,15 @@ function adminV2() {
hidden: item.is_hidden ? 'true' : 'false',
release_type: 'album',
year: '',
release_id: null,
track_number: '',
disc_number: '',
artist_ids: []
};
this.editorDetail = null;
this.editorImageFile = null;
this.editorArtistToAdd = '';
this.editorReleaseToAdd = '';
this.editorOpen = true;
this.loadEditorDetail(item);
},
@@ -2640,10 +2691,14 @@ function adminV2() {
hidden: detail.hidden ? 'true' : 'false',
release_type: detail.release_type || 'album',
year: detail.year || '',
release_id: detail.release_id || null,
track_number: detail.track_number || '',
disc_number: detail.disc_number || '',
artist_ids: Array.isArray(detail.selected_artist_ids) ? detail.selected_artist_ids.slice() : []
};
this.editorImageFile = null;
this.editorArtistToAdd = '';
this.editorReleaseToAdd = '';
} catch (error) {
this.showToast(error.message);
} finally {
@@ -2662,12 +2717,18 @@ function adminV2() {
return this.activeLibraryItem && this.activeLibraryItem.kind === 'releases';
},
isTrackEditor() {
return this.activeLibraryItem && this.activeLibraryItem.kind === 'tracks';
},
canEditLibraryImage() {
return this.isArtistEditor() || this.isReleaseEditor();
},
editorCanSave() {
return Boolean(this.activeLibraryItem && this.editorDetail && !this.editorLoading && !this.editorSaving);
if (!this.activeLibraryItem || !this.editorDetail || this.editorLoading || this.editorSaving) return false;
if (this.isTrackEditor() && !this.editorDraft.release_id) return false;
return true;
},
selectedEditorArtists() {
@@ -2710,7 +2771,7 @@ function adminV2() {
},
editorArtistSearchOpen() {
return this.isReleaseEditor() && String(this.editorArtistToAdd || '').trim().length > 0;
return (this.isReleaseEditor() || this.isTrackEditor()) && String(this.editorArtistToAdd || '').trim().length > 0;
},
addEditorArtist(artist = null) {
@@ -2729,6 +2790,49 @@ function adminV2() {
this.editorDraft.artist_ids = (this.editorDraft.artist_ids || []).filter(value => Number(value) !== Number(id));
},
selectedEditorRelease() {
const releases = (this.editorDetail && this.editorDetail.releases) || [];
return releases.find(row => Number(row.id) === Number(this.editorDraft.release_id));
},
selectedEditorReleaseLabel() {
const release = this.selectedEditorRelease();
if (!release) return '';
return release.subtitle ? `${release.title} / ${release.subtitle}` : release.title;
},
filteredEditorReleases() {
const releases = (this.editorDetail && this.editorDetail.releases) || [];
const query = String(this.editorReleaseToAdd || '').trim().toLowerCase();
const currentId = Number(this.editorDraft.release_id || 0);
const candidates = releases.filter(release => Number(release.id) !== currentId);
if (!query) return candidates.slice(0, 12);
return candidates
.map(release => {
const haystack = `${release.title || ''} ${release.subtitle || ''}`.toLowerCase();
let score = 3;
if (String(release.title || '').toLowerCase() === query) score = 0;
else if (String(release.title || '').toLowerCase().startsWith(query)) score = 1;
else if (haystack.includes(query)) score = 2;
return { release, score };
})
.filter(row => row.score < 3)
.sort((a, b) => a.score - b.score || a.release.title.localeCompare(b.release.title))
.slice(0, 12)
.map(row => row.release);
},
editorReleaseSearchOpen() {
return this.isTrackEditor() && String(this.editorReleaseToAdd || '').trim().length > 0;
},
selectEditorRelease(release = null) {
const candidates = this.filteredEditorReleases();
release = release || candidates[0];
if (release) this.editorDraft.release_id = Number(release.id);
this.editorReleaseToAdd = '';
},
setEditorImageFile(event) {
this.editorImageFile = event.target.files && event.target.files.length ? event.target.files[0] : null;
},
@@ -2856,6 +2960,9 @@ function adminV2() {
hidden: this.editorDraft.hidden === 'true',
release_type: this.editorDraft.release_type || null,
year: this.editorDraft.year || '',
release_id: this.editorDraft.release_id ? Number(this.editorDraft.release_id) : null,
track_number: this.editorDraft.track_number || '',
disc_number: this.editorDraft.disc_number || '',
artist_ids: this.editorDraft.artist_ids || []
})
});
@@ -2931,7 +3038,7 @@ function adminV2() {
},
pageSubtitle() {
if (this.activeView === 'library') return 'Fast entity control surface for artists, releases, and playlists';
if (this.activeView === 'library') return 'Fast entity control surface for artists, releases, tracks, and playlists';
if (this.activeView === 'jobs') return 'Scheduler state, recent runs, and manual controls in one place';
if (this.activeView === 'tools') return 'Reserved space for merge, split, enrichment, and destructive workflows';
if (this.activeView === 'settings') return 'Application configuration and external API credentials';