Compare commits

..

No commits in common. "main" and "v0.1.0" have entirely different histories.
main ... v0.1.0

42 changed files with 319 additions and 873 deletions

1
Cargo.lock generated
View file

@ -1880,6 +1880,7 @@ dependencies = [
"glib", "glib",
"gstreamer-play", "gstreamer-play",
"gtk4", "gtk4",
"lazy_static",
"libadwaita", "libadwaita",
"log", "log",
"mpris-server", "mpris-server",

View file

@ -17,6 +17,7 @@ gettext-rs = { version = "0.7", features = ["gettext-system"] }
glib = { version = "0.20", features = ["v2_84"] } glib = { version = "0.20", features = ["v2_84"] }
gstreamer-play = "0.23" gstreamer-play = "0.23"
gtk = { package = "gtk4", version = "0.9", features = ["v4_18", "blueprint"] } gtk = { package = "gtk4", version = "0.9", features = ["v4_18", "blueprint"] }
lazy_static = "1"
log = "0.4" log = "0.4"
mpris-server = "0.8" mpris-server = "0.8"
once_cell = "1" once_cell = "1"

View file

@ -6,12 +6,15 @@
font-size: smaller; font-size: smaller;
} }
.rounded-entry { .searchbar .searchtag {
border-radius: 999px; background-color: alpha(currentColor, 0.1);
padding-left: 12px; border-radius: 100px;
padding-right: 12px; }
padding-top: 3px;
padding-bottom: 3px; .searchbar .searchtag>button {
min-width: 24px;
min-height: 24px;
margin: 0px;
} }
.tile { .tile {

View file

@ -65,15 +65,9 @@ template $MusicusAlbumEditor: Adw.NavigationPage {
margin-top: 24; margin-top: 24;
styles [ styles [
"boxed-list-separate", "boxed-list",
] ]
Adw.SwitchRow enable_updates_row {
title: _("Enable updates");
subtitle: _("Keep this item up to date with the online metadata library");
active: true;
}
Adw.ButtonRow save_row { Adw.ButtonRow save_row {
title: _("_Create album"); title: _("_Create album");
use-underline: true; use-underline: true;

View file

@ -50,32 +50,23 @@ template $MusicusEmptyPage: Adw.NavigationPage {
} }
menu primary_menu { menu primary_menu {
section { item {
item { label: _("_Import music");
label: _("_Import music"); action: "win.import";
action: "win.import";
}
item {
label: _("_Create album");
action: "win.create-album";
}
item {
label: _("_Library manager");
action: "win.library";
}
} }
section { item {
item { label: _("_Library manager");
label: _("_Preferences"); action: "win.library";
action: "win.preferences"; }
}
item { item {
label: _("_About Musicus"); label: _("_Preferences");
action: "app.about"; action: "win.preferences";
} }
item {
label: _("_About Musicus");
action: "app.about";
} }
} }

View file

@ -78,10 +78,6 @@ template $MusicusSearchPage: Adw.NavigationPage {
placeholder-text: _("Enter composers, performers, works…"); placeholder-text: _("Enter composers, performers, works…");
margin-top: 24; margin-top: 24;
activate => $select() swapped; activate => $select() swapped;
styles [
"rounded-entry",
]
} }
Gtk.Stack stack { Gtk.Stack stack {
@ -264,33 +260,24 @@ template $MusicusSearchPage: Adw.NavigationPage {
} }
menu primary_menu { menu primary_menu {
section { item {
item { label: _("_Import music");
label: _("_Import music"); action: "win.import";
action: "win.import";
}
item {
label: _("_Create album");
action: "win.create-album";
}
item {
label: _("_Library manager");
action: "win.library";
}
} }
section { item {
item { label: _("_Library manager");
label: _("_Preferences"); action: "win.library";
action: "win.preferences"; }
}
item { item {
label: _("_About Musicus"); label: _("_Preferences");
action: "app.about"; action: "win.preferences";
} }
item {
label: _("_About Musicus");
action: "app.about";
} }
} }

22
data/ui/search_tag.blp Normal file
View file

@ -0,0 +1,22 @@
using Gtk 4.0;
template $MusicusSearchTag : Gtk.Box {
styles ["searchtag"]
margin-start: 6;
margin-end: 6;
Gtk.Label label {
styles ["caption-heading"]
margin-start: 12;
margin-end: 6;
max-width-chars: 15;
ellipsize: end;
}
Gtk.Button button {
styles ["flat", "circular"]
icon-name: "window-close-symbolic";
clicked => $remove() swapped;
}
}

View file

@ -5,7 +5,7 @@
"sdk": "org.gnome.Sdk", "sdk": "org.gnome.Sdk",
"sdk-extensions": [ "sdk-extensions": [
"org.freedesktop.Sdk.Extension.rust-stable", "org.freedesktop.Sdk.Extension.rust-stable",
"org.freedesktop.Sdk.Extension.llvm20" "org.freedesktop.Sdk.Extension.llvm18"
], ],
"command": "musicus", "command": "musicus",
"finish-args": [ "finish-args": [
@ -20,7 +20,7 @@
"--env=G_MESSAGES_DEBUG=none" "--env=G_MESSAGES_DEBUG=none"
], ],
"build-options": { "build-options": {
"append-path": "/usr/lib/sdk/rust-stable/bin:/usr/lib/sdk/llvm20/bin", "append-path": "/usr/lib/sdk/rust-stable/bin:/usr/lib/sdk/llvm18/bin",
"build-args": [ "build-args": [
"--share=network" "--share=network"
], ],
@ -31,6 +31,17 @@
} }
}, },
"modules": [ "modules": [
{
"name": "blueprint-compiler",
"buildsystem": "meson",
"sources": [
{
"type": "git",
"url": "https://gitlab.gnome.org/jwestman/blueprint-compiler.git",
"tag": "v0.16.0"
}
]
},
{ {
"name": "musicus", "name": "musicus",
"buildsystem": "meson", "buildsystem": "meson",

View file

@ -5,7 +5,7 @@
"sdk": "org.gnome.Sdk", "sdk": "org.gnome.Sdk",
"sdk-extensions": [ "sdk-extensions": [
"org.freedesktop.Sdk.Extension.rust-stable", "org.freedesktop.Sdk.Extension.rust-stable",
"org.freedesktop.Sdk.Extension.llvm20" "org.freedesktop.Sdk.Extension.llvm18"
], ],
"command": "musicus", "command": "musicus",
"finish-args": [ "finish-args": [
@ -20,7 +20,7 @@
"--env=G_MESSAGES_DEBUG=none" "--env=G_MESSAGES_DEBUG=none"
], ],
"build-options": { "build-options": {
"append-path": "/usr/lib/sdk/rust-stable/bin:/usr/lib/sdk/llvm20/bin", "append-path": "/usr/lib/sdk/rust-stable/bin:/usr/lib/sdk/llvm18/bin",
"build-args": [ "build-args": [
"--share=network" "--share=network"
], ],

View file

@ -1,248 +0,0 @@
CREATE TABLE persons_old (
person_id TEXT NOT NULL PRIMARY KEY,
name TEXT NOT NULL,
created_at TIMESTAMP NOT NULL DEFAULT (DATETIME('now', 'localtime')),
edited_at TIMESTAMP NOT NULL DEFAULT (DATETIME('now', 'localtime')),
last_used_at TIMESTAMP NOT NULL DEFAULT (DATETIME('now', 'localtime')),
last_played_at TIMESTAMP,
enable_updates BOOLEAN NOT NULL DEFAULT TRUE
);
CREATE TABLE roles_old (
role_id TEXT NOT NULL PRIMARY KEY,
name TEXT NOT NULL,
created_at TIMESTAMP NOT NULL DEFAULT (DATETIME('now', 'localtime')),
edited_at TIMESTAMP NOT NULL DEFAULT (DATETIME('now', 'localtime')),
last_used_at TIMESTAMP NOT NULL DEFAULT (DATETIME('now', 'localtime')),
enable_updates BOOLEAN NOT NULL DEFAULT TRUE
);
CREATE TABLE instruments_old (
instrument_id TEXT NOT NULL PRIMARY KEY,
name TEXT NOT NULL,
created_at TIMESTAMP NOT NULL DEFAULT (DATETIME('now', 'localtime')),
edited_at TIMESTAMP NOT NULL DEFAULT (DATETIME('now', 'localtime')),
last_used_at TIMESTAMP NOT NULL DEFAULT (DATETIME('now', 'localtime')),
last_played_at TIMESTAMP,
enable_updates BOOLEAN NOT NULL DEFAULT TRUE
);
CREATE TABLE works_old (
work_id TEXT NOT NULL PRIMARY KEY,
parent_work_id TEXT REFERENCES works(work_id),
sequence_number INTEGER,
name TEXT NOT NULL,
created_at TIMESTAMP NOT NULL DEFAULT (DATETIME('now', 'localtime')),
edited_at TIMESTAMP NOT NULL DEFAULT (DATETIME('now', 'localtime')),
last_used_at TIMESTAMP NOT NULL DEFAULT (DATETIME('now', 'localtime')),
last_played_at TIMESTAMP,
enable_updates BOOLEAN NOT NULL DEFAULT TRUE
);
CREATE TABLE ensembles_old (
ensemble_id TEXT NOT NULL PRIMARY KEY,
name TEXT NOT NULL,
created_at TIMESTAMP NOT NULL DEFAULT (DATETIME('now', 'localtime')),
edited_at TIMESTAMP NOT NULL DEFAULT (DATETIME('now', 'localtime')),
last_used_at TIMESTAMP NOT NULL DEFAULT (DATETIME('now', 'localtime')),
last_played_at TIMESTAMP,
enable_updates BOOLEAN NOT NULL DEFAULT TRUE
);
CREATE TABLE recordings_old (
recording_id TEXT NOT NULL PRIMARY KEY,
work_id TEXT NOT NULL REFERENCES works(work_id),
year INTEGER,
created_at TIMESTAMP NOT NULL DEFAULT (DATETIME('now', 'localtime')),
edited_at TIMESTAMP NOT NULL DEFAULT (DATETIME('now', 'localtime')),
last_used_at TIMESTAMP NOT NULL DEFAULT (DATETIME('now', 'localtime')),
last_played_at TIMESTAMP,
enable_updates BOOLEAN NOT NULL DEFAULT TRUE
);
CREATE TABLE mediums_old (
medium_id TEXT NOT NULL PRIMARY KEY REFERENCES item_state(id),
discid TEXT NOT NULL,
enable_updates BOOLEAN NOT NULL DEFAULT TRUE,
created_at TIMESTAMP NOT NULL DEFAULT (DATETIME('now', 'localtime')),
edited_at TIMESTAMP NOT NULL DEFAULT (DATETIME('now', 'localtime')),
last_used_at TIMESTAMP NOT NULL DEFAULT (DATETIME('now', 'localtime')),
last_played_at TIMESTAMP
);
CREATE TABLE albums_old (
album_id TEXT NOT NULL PRIMARY KEY REFERENCES item_state(id),
name TEXT NOT NULL,
created_at TIMESTAMP NOT NULL DEFAULT (DATETIME('now', 'localtime')),
edited_at TIMESTAMP NOT NULL DEFAULT (DATETIME('now', 'localtime')),
last_used_at TIMESTAMP NOT NULL DEFAULT (DATETIME('now', 'localtime')),
last_played_at TIMESTAMP
);
INSERT INTO persons_old (
person_id,
name,
created_at,
edited_at,
last_used_at,
last_played_at,
enable_updates
)
SELECT person_id,
name,
created_at,
edited_at,
last_used_at,
last_played_at,
enable_updates
FROM persons;
DROP TABLE persons;
ALTER TABLE persons_old
RENAME TO persons;
INSERT INTO roles_old (
role_id,
name,
created_at,
edited_at,
last_used_at,
enable_updates
)
SELECT role_id,
name,
created_at,
edited_at,
last_used_at,
enable_updates
FROM roles;
DROP TABLE roles;
ALTER TABLE roles_old
RENAME TO roles;
INSERT INTO instruments_old (
instrument_id,
name,
created_at,
edited_at,
last_used_at,
last_played_at,
enable_updates
)
SELECT instrument_id,
name,
created_at,
edited_at,
last_used_at,
last_played_at,
enable_updates
FROM instruments;
DROP TABLE instruments;
ALTER TABLE instruments_old
RENAME TO instruments;
INSERT INTO works_old (
work_id,
parent_work_id,
sequence_number,
name,
created_at,
edited_at,
last_used_at,
last_played_at,
enable_updates
)
SELECT work_id,
parent_work_id,
sequence_number,
name,
created_at,
edited_at,
last_used_at,
last_played_at,
enable_updates
FROM works;
DROP TABLE works;
ALTER TABLE works_old
RENAME TO works;
INSERT INTO ensembles_old (
ensemble_id,
name,
created_at,
edited_at,
last_used_at,
last_played_at,
enable_updates
)
SELECT ensemble_id,
name,
created_at,
edited_at,
last_used_at,
last_played_at,
enable_updates
FROM ensembles;
DROP TABLE ensembles;
ALTER TABLE ensembles_old
RENAME TO ensembles;
INSERT INTO recordings_old (
recording_id,
work_id,
year,
created_at,
edited_at,
last_used_at,
last_played_at,
enable_updates
)
SELECT recording_id,
work_id,
year,
created_at,
edited_at,
last_used_at,
last_played_at,
enable_updates
FROM recordings;
DROP TABLE recordings;
ALTER TABLE recordings_old
RENAME TO recordings;
INSERT INTO mediums_old (
medium_id,
discid,
created_at,
edited_at,
last_used_at,
last_played_at
)
SELECT medium_id,
discid,
created_at,
edited_at,
last_used_at,
last_played_at
FROM mediums;
DROP TABLE mediums;
ALTER TABLE mediums_old
RENAME TO mediums;
INSERT INTO albums_old (
album_id,
name,
created_at,
edited_at,
last_used_at,
last_played_at
)
SELECT album_id,
name,
created_at,
edited_at,
last_used_at,
last_played_at
FROM albums;
DROP TABLE albums;
ALTER TABLE albums_old
RENAME TO albums;

View file

@ -1,245 +0,0 @@
CREATE TABLE persons_new (
person_id TEXT NOT NULL PRIMARY KEY REFERENCES item_state(id),
name TEXT NOT NULL,
source TEXT NOT NULL DEFAULT 'user',
enable_updates BOOLEAN NOT NULL DEFAULT TRUE,
created_at TIMESTAMP NOT NULL DEFAULT (DATETIME('now', 'localtime')),
edited_at TIMESTAMP NOT NULL DEFAULT (DATETIME('now', 'localtime')),
last_used_at TIMESTAMP NOT NULL DEFAULT (DATETIME('now', 'localtime')),
last_played_at TIMESTAMP
);
CREATE TABLE roles_new (
role_id TEXT NOT NULL PRIMARY KEY REFERENCES item_state(id),
name TEXT NOT NULL,
source TEXT NOT NULL DEFAULT 'user',
enable_updates BOOLEAN NOT NULL DEFAULT TRUE,
created_at TIMESTAMP NOT NULL DEFAULT (DATETIME('now', 'localtime')),
edited_at TIMESTAMP NOT NULL DEFAULT (DATETIME('now', 'localtime')),
last_used_at TIMESTAMP NOT NULL DEFAULT (DATETIME('now', 'localtime'))
);
CREATE TABLE instruments_new (
instrument_id TEXT NOT NULL PRIMARY KEY REFERENCES item_state(id),
name TEXT NOT NULL,
source TEXT NOT NULL DEFAULT 'user',
enable_updates BOOLEAN NOT NULL DEFAULT TRUE,
created_at TIMESTAMP NOT NULL DEFAULT (DATETIME('now', 'localtime')),
edited_at TIMESTAMP NOT NULL DEFAULT (DATETIME('now', 'localtime')),
last_used_at TIMESTAMP NOT NULL DEFAULT (DATETIME('now', 'localtime')),
last_played_at TIMESTAMP
);
CREATE TABLE works_new (
work_id TEXT NOT NULL PRIMARY KEY REFERENCES item_state(id),
parent_work_id TEXT REFERENCES works(work_id),
sequence_number INTEGER,
name TEXT NOT NULL,
source TEXT NOT NULL DEFAULT 'user',
enable_updates BOOLEAN NOT NULL DEFAULT TRUE,
created_at TIMESTAMP NOT NULL DEFAULT (DATETIME('now', 'localtime')),
edited_at TIMESTAMP NOT NULL DEFAULT (DATETIME('now', 'localtime')),
last_used_at TIMESTAMP NOT NULL DEFAULT (DATETIME('now', 'localtime')),
last_played_at TIMESTAMP
);
CREATE TABLE ensembles_new (
ensemble_id TEXT NOT NULL PRIMARY KEY REFERENCES item_state(id),
name TEXT NOT NULL,
source TEXT NOT NULL DEFAULT 'user',
enable_updates BOOLEAN NOT NULL DEFAULT TRUE,
created_at TIMESTAMP NOT NULL DEFAULT (DATETIME('now', 'localtime')),
edited_at TIMESTAMP NOT NULL DEFAULT (DATETIME('now', 'localtime')),
last_used_at TIMESTAMP NOT NULL DEFAULT (DATETIME('now', 'localtime')),
last_played_at TIMESTAMP
);
CREATE TABLE recordings_new (
recording_id TEXT NOT NULL PRIMARY KEY REFERENCES item_state(id),
work_id TEXT NOT NULL REFERENCES works(work_id),
year INTEGER,
source TEXT NOT NULL DEFAULT 'user',
enable_updates BOOLEAN NOT NULL DEFAULT TRUE,
created_at TIMESTAMP NOT NULL DEFAULT (DATETIME('now', 'localtime')),
edited_at TIMESTAMP NOT NULL DEFAULT (DATETIME('now', 'localtime')),
last_used_at TIMESTAMP NOT NULL DEFAULT (DATETIME('now', 'localtime')),
last_played_at TIMESTAMP
);
CREATE TABLE mediums_new (
medium_id TEXT NOT NULL PRIMARY KEY REFERENCES item_state(id),
discid TEXT NOT NULL,
source TEXT NOT NULL DEFAULT 'user',
enable_updates BOOLEAN NOT NULL DEFAULT TRUE,
created_at TIMESTAMP NOT NULL DEFAULT (DATETIME('now', 'localtime')),
edited_at TIMESTAMP NOT NULL DEFAULT (DATETIME('now', 'localtime')),
last_used_at TIMESTAMP NOT NULL DEFAULT (DATETIME('now', 'localtime')),
last_played_at TIMESTAMP
);
CREATE TABLE albums_new (
album_id TEXT NOT NULL PRIMARY KEY REFERENCES item_state(id),
name TEXT NOT NULL,
source TEXT NOT NULL DEFAULT 'user',
enable_updates BOOLEAN NOT NULL DEFAULT TRUE,
created_at TIMESTAMP NOT NULL DEFAULT (DATETIME('now', 'localtime')),
edited_at TIMESTAMP NOT NULL DEFAULT (DATETIME('now', 'localtime')),
last_used_at TIMESTAMP NOT NULL DEFAULT (DATETIME('now', 'localtime')),
last_played_at TIMESTAMP
);
INSERT INTO persons_new (
person_id,
name,
created_at,
edited_at,
last_used_at,
last_played_at
)
SELECT person_id,
name,
created_at,
edited_at,
last_used_at,
last_played_at
FROM persons;
DROP TABLE persons;
ALTER TABLE persons_new
RENAME TO persons;
INSERT INTO roles_new (
role_id,
name,
created_at,
edited_at,
last_used_at
)
SELECT role_id,
name,
created_at,
edited_at,
last_used_at
FROM roles;
DROP TABLE roles;
ALTER TABLE roles_new
RENAME TO roles;
INSERT INTO instruments_new (
instrument_id,
name,
created_at,
edited_at,
last_used_at,
last_played_at
)
SELECT instrument_id,
name,
created_at,
edited_at,
last_used_at,
last_played_at
FROM instruments;
DROP TABLE instruments;
ALTER TABLE instruments_new
RENAME TO instruments;
INSERT INTO works_new (
work_id,
parent_work_id,
sequence_number,
name,
created_at,
edited_at,
last_used_at,
last_played_at
)
SELECT work_id,
parent_work_id,
sequence_number,
name,
created_at,
edited_at,
last_used_at,
last_played_at
FROM works;
DROP TABLE works;
ALTER TABLE works_new
RENAME TO works;
INSERT INTO ensembles_new (
ensemble_id,
name,
created_at,
edited_at,
last_used_at,
last_played_at
)
SELECT ensemble_id,
name,
created_at,
edited_at,
last_used_at,
last_played_at
FROM ensembles;
DROP TABLE ensembles;
ALTER TABLE ensembles_new
RENAME TO ensembles;
INSERT INTO recordings_new (
recording_id,
work_id,
year,
created_at,
edited_at,
last_used_at,
last_played_at
)
SELECT recording_id,
work_id,
year,
created_at,
edited_at,
last_used_at,
last_played_at
FROM recordings;
DROP TABLE recordings;
ALTER TABLE recordings_new
RENAME TO recordings;
INSERT INTO mediums_new (
medium_id,
discid,
created_at,
edited_at,
last_used_at,
last_played_at
)
SELECT medium_id,
discid,
created_at,
edited_at,
last_used_at,
last_played_at
FROM mediums;
DROP TABLE mediums;
ALTER TABLE mediums_new
RENAME TO mediums;
INSERT INTO albums_new (
album_id,
name,
created_at,
edited_at,
last_used_at,
last_played_at
)
SELECT album_id,
name,
created_at,
edited_at,
last_used_at,
last_played_at
FROM albums;
DROP TABLE albums;
ALTER TABLE albums_new
RENAME TO albums;

View file

@ -8,7 +8,7 @@ msgid ""
msgstr "" msgstr ""
"Project-Id-Version: \n" "Project-Id-Version: \n"
"Report-Msgid-Bugs-To: \n" "Report-Msgid-Bugs-To: \n"
"POT-Creation-Date: 2025-05-30 15:27+0200\n" "POT-Creation-Date: 2025-04-27 17:54+0200\n"
"PO-Revision-Date: 2025-04-27 18:23+0200\n" "PO-Revision-Date: 2025-04-27 18:23+0200\n"
"Last-Translator: elias@johrpan.de\n" "Last-Translator: elias@johrpan.de\n"
"Language-Team: German <translation-team-de@lists.sourceforge.net>\n" "Language-Team: German <translation-team-de@lists.sourceforge.net>\n"
@ -818,7 +818,7 @@ msgstr "Bibliothek exportieren"
msgid "Exporting music library to {}" msgid "Exporting music library to {}"
msgstr "Bibliothek wird nach {} exportiert" msgstr "Bibliothek wird nach {} exportiert"
#: src/library_manager.rs:234 src/window.rs:305 #: src/library_manager.rs:234 src/window.rs:282
msgid "Updating metadata" msgid "Updating metadata"
msgstr "Metadaten werden aktualisiert" msgstr "Metadaten werden aktualisiert"
@ -826,23 +826,19 @@ msgstr "Metadaten werden aktualisiert"
msgid "Updating music library" msgid "Updating music library"
msgstr "Musikbibliothek wird aktualisiert" msgstr "Musikbibliothek wird aktualisiert"
#: src/window.rs:166 #: src/window.rs:167
msgid "Currently playing music"
msgstr "Musik wird abgespielt"
#: src/window.rs:190
msgid "Close window?" msgid "Close window?"
msgstr "Fenster schließen?" msgstr "Fenster schließen?"
#: src/window.rs:192 #: src/window.rs:169
msgid "There are ongoing processes that will be canceled." msgid "There are ongoing processes that will be canceled."
msgstr "Es gibt laufende Prozesse, die abgebrochen werden." msgstr "Es gibt laufende Prozesse, die abgebrochen werden."
#: src/window.rs:197 #: src/window.rs:174
msgid "Keep open" msgid "Keep open"
msgstr "Nicht schließen" msgstr "Nicht schließen"
#: src/window.rs:198 #: src/window.rs:175
msgid "Close window" msgid "Close window"
msgstr "Fenster schließen" msgstr "Fenster schließen"

View file

@ -8,7 +8,7 @@ msgid ""
msgstr "" msgstr ""
"Project-Id-Version: PACKAGE VERSION\n" "Project-Id-Version: PACKAGE VERSION\n"
"Report-Msgid-Bugs-To: \n" "Report-Msgid-Bugs-To: \n"
"POT-Creation-Date: 2025-05-30 15:27+0200\n" "POT-Creation-Date: 2025-04-27 17:54+0200\n"
"PO-Revision-Date: YEAR-MO-DA HO:MI+ZONE\n" "PO-Revision-Date: YEAR-MO-DA HO:MI+ZONE\n"
"Last-Translator: FULL NAME <EMAIL@ADDRESS>\n" "Last-Translator: FULL NAME <EMAIL@ADDRESS>\n"
"Language-Team: LANGUAGE <LL@li.org>\n" "Language-Team: LANGUAGE <LL@li.org>\n"
@ -784,7 +784,7 @@ msgstr ""
msgid "Exporting music library to {}" msgid "Exporting music library to {}"
msgstr "" msgstr ""
#: src/library_manager.rs:234 src/window.rs:305 #: src/library_manager.rs:234 src/window.rs:282
msgid "Updating metadata" msgid "Updating metadata"
msgstr "" msgstr ""
@ -792,22 +792,18 @@ msgstr ""
msgid "Updating music library" msgid "Updating music library"
msgstr "" msgstr ""
#: src/window.rs:166 #: src/window.rs:167
msgid "Currently playing music"
msgstr ""
#: src/window.rs:190
msgid "Close window?" msgid "Close window?"
msgstr "" msgstr ""
#: src/window.rs:192 #: src/window.rs:169
msgid "There are ongoing processes that will be canceled." msgid "There are ongoing processes that will be canceled."
msgstr "" msgstr ""
#: src/window.rs:197 #: src/window.rs:174
msgid "Keep open" msgid "Keep open"
msgstr "" msgstr ""
#: src/window.rs:198 #: src/window.rs:175
msgid "Close window" msgid "Close window"
msgstr "" msgstr ""

View file

@ -72,7 +72,8 @@ mod imp {
.unwrap() .unwrap()
.recordings .recordings
.iter() .iter()
.flat_map(|r| obj.player().recording_to_playlist(r)) .map(|r| obj.player().recording_to_playlist(r))
.flatten()
.collect::<Vec<PlaylistItem>>(); .collect::<Vec<PlaylistItem>>();
if let Err(err) = obj.player().append(playlist) { if let Err(err) = obj.player().append(playlist) {
@ -164,7 +165,8 @@ impl AlbumPage {
.unwrap() .unwrap()
.recordings .recordings
.iter() .iter()
.flat_map(|r| self.player().recording_to_playlist(r)) .map(|r| self.player().recording_to_playlist(r))
.flatten()
.collect::<Vec<PlaylistItem>>(); .collect::<Vec<PlaylistItem>>();
self.player().append_and_play(playlist); self.player().append_and_play(playlist);

View file

@ -45,7 +45,7 @@ impl AlbumTile {
pub fn new(album: &Album) -> Self { pub fn new(album: &Album) -> Self {
let obj: Self = glib::Object::new(); let obj: Self = glib::Object::new();
obj.imp().title_label.set_label(album.name.get()); obj.imp().title_label.set_label(&album.name.get());
obj.imp().album.set(album.clone()).unwrap(); obj.imp().album.set(album.clone()).unwrap();
obj obj

View file

@ -74,7 +74,6 @@ pub struct Album {
pub album_id: String, pub album_id: String,
pub name: TranslatedString, pub name: TranslatedString,
pub recordings: Vec<Recording>, pub recordings: Vec<Recording>,
pub enable_updates: bool,
} }
impl Eq for Person {} impl Eq for Person {}
@ -434,7 +433,6 @@ impl Album {
album_id: data.album_id, album_id: data.album_id,
name: data.name, name: data.name,
recordings, recordings,
enable_updates: data.enable_updates,
}) })
} }

View file

@ -20,8 +20,6 @@ diesel::table! {
albums (album_id) { albums (album_id) {
album_id -> Text, album_id -> Text,
name -> Text, name -> Text,
source -> Text,
enable_updates -> Bool,
created_at -> Timestamp, created_at -> Timestamp,
edited_at -> Timestamp, edited_at -> Timestamp,
last_used_at -> Timestamp, last_used_at -> Timestamp,
@ -42,12 +40,11 @@ diesel::table! {
ensembles (ensemble_id) { ensembles (ensemble_id) {
ensemble_id -> Text, ensemble_id -> Text,
name -> Text, name -> Text,
source -> Text,
enable_updates -> Bool,
created_at -> Timestamp, created_at -> Timestamp,
edited_at -> Timestamp, edited_at -> Timestamp,
last_used_at -> Timestamp, last_used_at -> Timestamp,
last_played_at -> Nullable<Timestamp>, last_played_at -> Nullable<Timestamp>,
enable_updates -> Bool,
} }
} }
@ -55,12 +52,11 @@ diesel::table! {
instruments (instrument_id) { instruments (instrument_id) {
instrument_id -> Text, instrument_id -> Text,
name -> Text, name -> Text,
source -> Text,
enable_updates -> Bool,
created_at -> Timestamp, created_at -> Timestamp,
edited_at -> Timestamp, edited_at -> Timestamp,
last_used_at -> Timestamp, last_used_at -> Timestamp,
last_played_at -> Nullable<Timestamp>, last_played_at -> Nullable<Timestamp>,
enable_updates -> Bool,
} }
} }
@ -68,8 +64,6 @@ diesel::table! {
mediums (medium_id) { mediums (medium_id) {
medium_id -> Text, medium_id -> Text,
discid -> Text, discid -> Text,
source -> Text,
enable_updates -> Bool,
created_at -> Timestamp, created_at -> Timestamp,
edited_at -> Timestamp, edited_at -> Timestamp,
last_used_at -> Timestamp, last_used_at -> Timestamp,
@ -81,17 +75,16 @@ diesel::table! {
persons (person_id) { persons (person_id) {
person_id -> Text, person_id -> Text,
name -> Text, name -> Text,
source -> Text,
enable_updates -> Bool,
created_at -> Timestamp, created_at -> Timestamp,
edited_at -> Timestamp, edited_at -> Timestamp,
last_used_at -> Timestamp, last_used_at -> Timestamp,
last_played_at -> Nullable<Timestamp>, last_played_at -> Nullable<Timestamp>,
enable_updates -> Bool,
} }
} }
diesel::table! { diesel::table! {
recording_ensembles (recording_id, ensemble_id, sequence_number) { recording_ensembles (recording_id, ensemble_id) {
recording_id -> Text, recording_id -> Text,
ensemble_id -> Text, ensemble_id -> Text,
role_id -> Nullable<Text>, role_id -> Nullable<Text>,
@ -100,7 +93,7 @@ diesel::table! {
} }
diesel::table! { diesel::table! {
recording_persons (recording_id, person_id, sequence_number) { recording_persons (recording_id, person_id) {
recording_id -> Text, recording_id -> Text,
person_id -> Text, person_id -> Text,
role_id -> Nullable<Text>, role_id -> Nullable<Text>,
@ -114,12 +107,11 @@ diesel::table! {
recording_id -> Text, recording_id -> Text,
work_id -> Text, work_id -> Text,
year -> Nullable<Integer>, year -> Nullable<Integer>,
source -> Text,
enable_updates -> Bool,
created_at -> Timestamp, created_at -> Timestamp,
edited_at -> Timestamp, edited_at -> Timestamp,
last_used_at -> Timestamp, last_used_at -> Timestamp,
last_played_at -> Nullable<Timestamp>, last_played_at -> Nullable<Timestamp>,
enable_updates -> Bool,
} }
} }
@ -127,11 +119,10 @@ diesel::table! {
roles (role_id) { roles (role_id) {
role_id -> Text, role_id -> Text,
name -> Text, name -> Text,
source -> Text,
enable_updates -> Bool,
created_at -> Timestamp, created_at -> Timestamp,
edited_at -> Timestamp, edited_at -> Timestamp,
last_used_at -> Timestamp, last_used_at -> Timestamp,
enable_updates -> Bool,
} }
} }
@ -167,7 +158,7 @@ diesel::table! {
} }
diesel::table! { diesel::table! {
work_persons (work_id, person_id, sequence_number) { work_persons (work_id, person_id) {
work_id -> Text, work_id -> Text,
person_id -> Text, person_id -> Text,
role_id -> Nullable<Text>, role_id -> Nullable<Text>,
@ -181,12 +172,11 @@ diesel::table! {
parent_work_id -> Nullable<Text>, parent_work_id -> Nullable<Text>,
sequence_number -> Nullable<Integer>, sequence_number -> Nullable<Integer>,
name -> Text, name -> Text,
source -> Text,
enable_updates -> Bool,
created_at -> Timestamp, created_at -> Timestamp,
edited_at -> Timestamp, edited_at -> Timestamp,
last_used_at -> Timestamp, last_used_at -> Timestamp,
last_played_at -> Nullable<Timestamp>, last_played_at -> Nullable<Timestamp>,
enable_updates -> Bool,
} }
} }

View file

@ -24,12 +24,11 @@ use super::{schema::*, TranslatedString};
pub struct Person { pub struct Person {
pub person_id: String, pub person_id: String,
pub name: TranslatedString, pub name: TranslatedString,
pub source: Source,
pub enable_updates: bool,
pub created_at: NaiveDateTime, pub created_at: NaiveDateTime,
pub edited_at: NaiveDateTime, pub edited_at: NaiveDateTime,
pub last_used_at: NaiveDateTime, pub last_used_at: NaiveDateTime,
pub last_played_at: Option<NaiveDateTime>, pub last_played_at: Option<NaiveDateTime>,
pub enable_updates: bool,
} }
#[derive(Boxed, Insertable, Queryable, Selectable, Clone, Debug)] #[derive(Boxed, Insertable, Queryable, Selectable, Clone, Debug)]
@ -38,11 +37,10 @@ pub struct Person {
pub struct Role { pub struct Role {
pub role_id: String, pub role_id: String,
pub name: TranslatedString, pub name: TranslatedString,
pub source: Source,
pub enable_updates: bool,
pub created_at: NaiveDateTime, pub created_at: NaiveDateTime,
pub edited_at: NaiveDateTime, pub edited_at: NaiveDateTime,
pub last_used_at: NaiveDateTime, pub last_used_at: NaiveDateTime,
pub enable_updates: bool,
} }
#[derive(Boxed, Insertable, Queryable, Selectable, Clone, Debug)] #[derive(Boxed, Insertable, Queryable, Selectable, Clone, Debug)]
@ -51,12 +49,11 @@ pub struct Role {
pub struct Instrument { pub struct Instrument {
pub instrument_id: String, pub instrument_id: String,
pub name: TranslatedString, pub name: TranslatedString,
pub source: Source,
pub enable_updates: bool,
pub created_at: NaiveDateTime, pub created_at: NaiveDateTime,
pub edited_at: NaiveDateTime, pub edited_at: NaiveDateTime,
pub last_used_at: NaiveDateTime, pub last_used_at: NaiveDateTime,
pub last_played_at: Option<NaiveDateTime>, pub last_played_at: Option<NaiveDateTime>,
pub enable_updates: bool,
} }
#[derive(Insertable, Queryable, Selectable, Clone, Debug)] #[derive(Insertable, Queryable, Selectable, Clone, Debug)]
@ -66,12 +63,11 @@ pub struct Work {
pub parent_work_id: Option<String>, pub parent_work_id: Option<String>,
pub sequence_number: Option<i32>, pub sequence_number: Option<i32>,
pub name: TranslatedString, pub name: TranslatedString,
pub source: Source,
pub enable_updates: bool,
pub created_at: NaiveDateTime, pub created_at: NaiveDateTime,
pub edited_at: NaiveDateTime, pub edited_at: NaiveDateTime,
pub last_used_at: NaiveDateTime, pub last_used_at: NaiveDateTime,
pub last_played_at: Option<NaiveDateTime>, pub last_played_at: Option<NaiveDateTime>,
pub enable_updates: bool,
} }
#[derive(Insertable, Queryable, Selectable, Clone, Debug)] #[derive(Insertable, Queryable, Selectable, Clone, Debug)]
@ -96,12 +92,11 @@ pub struct WorkInstrument {
pub struct Ensemble { pub struct Ensemble {
pub ensemble_id: String, pub ensemble_id: String,
pub name: TranslatedString, pub name: TranslatedString,
pub source: Source,
pub enable_updates: bool,
pub created_at: NaiveDateTime, pub created_at: NaiveDateTime,
pub edited_at: NaiveDateTime, pub edited_at: NaiveDateTime,
pub last_used_at: NaiveDateTime, pub last_used_at: NaiveDateTime,
pub last_played_at: Option<NaiveDateTime>, pub last_played_at: Option<NaiveDateTime>,
pub enable_updates: bool,
} }
#[derive(Insertable, Queryable, Selectable, Clone, Debug)] #[derive(Insertable, Queryable, Selectable, Clone, Debug)]
@ -119,12 +114,11 @@ pub struct Recording {
pub recording_id: String, pub recording_id: String,
pub work_id: String, pub work_id: String,
pub year: Option<i32>, pub year: Option<i32>,
pub source: Source,
pub enable_updates: bool,
pub created_at: NaiveDateTime, pub created_at: NaiveDateTime,
pub edited_at: NaiveDateTime, pub edited_at: NaiveDateTime,
pub last_used_at: NaiveDateTime, pub last_used_at: NaiveDateTime,
pub last_played_at: Option<NaiveDateTime>, pub last_played_at: Option<NaiveDateTime>,
pub enable_updates: bool,
} }
#[derive(Insertable, Queryable, Selectable, Clone, Debug)] #[derive(Insertable, Queryable, Selectable, Clone, Debug)]
@ -174,8 +168,6 @@ pub struct TrackWork {
pub struct Medium { pub struct Medium {
pub medium_id: String, pub medium_id: String,
pub discid: String, pub discid: String,
pub source: Source,
pub enable_updates: bool,
pub created_at: NaiveDateTime, pub created_at: NaiveDateTime,
pub edited_at: NaiveDateTime, pub edited_at: NaiveDateTime,
pub last_used_at: NaiveDateTime, pub last_used_at: NaiveDateTime,
@ -187,8 +179,6 @@ pub struct Medium {
pub struct Album { pub struct Album {
pub album_id: String, pub album_id: String,
pub name: TranslatedString, pub name: TranslatedString,
pub source: Source,
pub enable_updates: bool,
pub created_at: NaiveDateTime, pub created_at: NaiveDateTime,
pub edited_at: NaiveDateTime, pub edited_at: NaiveDateTime,
pub last_used_at: NaiveDateTime, pub last_used_at: NaiveDateTime,
@ -266,40 +256,3 @@ impl AsRef<Path> for PathBufWrapper {
self.0.as_ref() self.0.as_ref()
} }
} }
#[derive(AsExpression, FromSqlRow, Copy, Clone, Debug)]
#[diesel(sql_type = Text)]
pub enum Source {
Metadata,
User,
Import,
Unknown,
}
impl ToSql<Text, Sqlite> for Source {
fn to_sql<'b>(&'b self, out: &mut Output<'b, '_, Sqlite>) -> diesel::serialize::Result {
out.set_value(match self {
Source::Metadata => "metadata",
Source::User => "user",
Source::Import => "import",
Source::Unknown => "unknown",
});
Ok(IsNull::No)
}
}
impl<DB> FromSql<Text, DB> for Source
where
DB: Backend,
String: FromSql<Text, DB>,
{
fn from_sql(bytes: DB::RawValue<'_>) -> diesel::deserialize::Result<Self> {
Ok(match String::from_sql(bytes)?.as_str() {
"metadata" => Source::Metadata,
"user" => Source::User,
"import" => Source::Import,
_ => Source::Unknown,
})
}
}

View file

@ -39,8 +39,6 @@ mod imp {
#[template_child] #[template_child]
pub recordings_list: TemplateChild<gtk::ListBox>, pub recordings_list: TemplateChild<gtk::ListBox>,
#[template_child] #[template_child]
pub enable_updates_row: TemplateChild<adw::SwitchRow>,
#[template_child]
pub save_row: TemplateChild<adw::ButtonRow>, pub save_row: TemplateChild<adw::ButtonRow>,
} }
@ -128,10 +126,6 @@ impl AlbumEditor {
for recording in &album.recordings { for recording in &album.recordings {
obj.add_recording(recording.to_owned()); obj.add_recording(recording.to_owned());
} }
obj.imp()
.enable_updates_row
.set_active(album.enable_updates);
} }
obj obj
@ -197,16 +191,10 @@ impl AlbumEditor {
.map(|r| r.recording()) .map(|r| r.recording())
.collect::<Vec<Recording>>(); .collect::<Vec<Recording>>();
let enable_updates = self.imp().enable_updates_row.is_active();
if let Some(album_id) = self.imp().album_id.get() { if let Some(album_id) = self.imp().album_id.get() {
library library.update_album(album_id, name, recordings).unwrap();
.update_album(album_id, name, recordings, enable_updates)
.unwrap();
} else { } else {
let album = library let album = library.create_album(name, recordings).unwrap();
.create_album(name, recordings, enable_updates)
.unwrap();
self.emit_by_name::<()>("created", &[&album]); self.emit_by_name::<()>("created", &[&album]);
} }

View file

@ -69,7 +69,7 @@ mod imp {
self.parent_constructed(); self.parent_constructed();
let set_design_action = gio::ActionEntry::builder("set-design") let set_design_action = gio::ActionEntry::builder("set-design")
.parameter_type(Some(glib::VariantTy::STRING)) .parameter_type(Some(&glib::VariantTy::STRING))
.state(glib::Variant::from("program-1")) .state(glib::Variant::from("program-1"))
.build(); .build();

View file

@ -246,7 +246,7 @@ impl RecordingEditor {
} }
fn set_work(&self, work: Work) { fn set_work(&self, work: Work) {
self.imp().work_row.set_title(work.name.get()); self.imp().work_row.set_title(&work.name.get());
self.imp().work_row.set_subtitle( self.imp().work_row.set_subtitle(
&work &work
.composers_string() .composers_string()

View file

@ -245,7 +245,8 @@ impl TracksEditor {
.track_rows .track_rows
.borrow() .borrow()
.iter() .iter()
.flat_map(|t| t.track_data().parts.clone()) .map(|t| t.track_data().parts.clone())
.flatten()
.collect::<Vec<Work>>() .collect::<Vec<Work>>()
}; };

View file

@ -390,15 +390,17 @@ impl WorkEditor {
}; };
self.emit_by_name::<()>("created", &[&part]); self.emit_by_name::<()>("created", &[&part]);
} else if let Some(work_id) = self.imp().work_id.get() {
library
.update_work(work_id, name, parts, composers, instruments, enable_updates)
.unwrap();
} else { } else {
let work = library if let Some(work_id) = self.imp().work_id.get() {
.create_work(name, parts, composers, instruments, enable_updates) library
.unwrap(); .update_work(work_id, name, parts, composers, instruments, enable_updates)
self.emit_by_name::<()>("created", &[&work]); .unwrap();
} else {
let work = library
.create_work(name, parts, composers, instruments, enable_updates)
.unwrap();
self.emit_by_name::<()>("created", &[&work]);
}
} }
self.imp().navigation.get().unwrap().pop(); self.imp().navigation.get().unwrap().pop();

View file

@ -145,7 +145,7 @@ impl WorkEditorPartRow {
} }
fn set_part(&self, part: Work) { fn set_part(&self, part: Work) {
self.set_title(part.name.get()); self.set_title(&part.name.get());
if !part.parts.is_empty() { if !part.parts.is_empty() {
self.set_subtitle( self.set_subtitle(

View file

@ -10,8 +10,8 @@ use gtk::{gio, glib, glib::subclass::Signal};
use once_cell::sync::Lazy; use once_cell::sync::Lazy;
use crate::{ use crate::{
config, db::tables::Source, library::Library, process::Process, config, library::Library, process::Process, process_manager::ProcessManager,
process_manager::ProcessManager, process_row::ProcessRow, process_row::ProcessRow,
}; };
mod imp { mod imp {
@ -92,8 +92,8 @@ impl EmptyPage {
#[template_callback] #[template_callback]
async fn download_library(&self) { async fn download_library(&self) {
let dialog = adw::AlertDialog::builder() let dialog = adw::AlertDialog::builder()
.heading(gettext("Disclaimer")) .heading(&gettext("Disclaimer"))
.body(gettext("You are about to download a library of audio files. These are from recordings that are in the public domain under EU law and are hosted on a server within the EU. Please ensure that you comply with the copyright laws of you country.")) .body(&gettext("You are about to download a library of audio files. These are from recordings that are in the public domain under EU law and are hosted on a server within the EU. Please ensure that you comply with the copyright laws of you country."))
.build(); .build();
dialog.add_response("continue", &gettext("Continue")); dialog.add_response("continue", &gettext("Continue"));
@ -119,7 +119,7 @@ impl EmptyPage {
.library .library
.get() .get()
.unwrap() .unwrap()
.import_library_from_url(&url, Source::Metadata) .import_library_from_url(&url)
{ {
Ok(receiver) => { Ok(receiver) => {
let process = Process::new(&gettext("Downloading music library"), receiver); let process = Process::new(&gettext("Downloading music library"), receiver);

View file

@ -10,13 +10,7 @@ use chrono::prelude::*;
use diesel::{prelude::*, QueryDsl, SqliteConnection}; use diesel::{prelude::*, QueryDsl, SqliteConnection};
use super::Library; use super::Library;
use crate::db::{ use crate::db::{self, models::*, schema::*, tables, TranslatedString};
self,
models::*,
schema::*,
tables::{self, Source},
TranslatedString,
};
impl Library { impl Library {
pub fn create_person(&self, name: TranslatedString, enable_updates: bool) -> Result<Person> { pub fn create_person(&self, name: TranslatedString, enable_updates: bool) -> Result<Person> {
@ -27,7 +21,6 @@ impl Library {
let person = Person { let person = Person {
person_id: db::generate_id(), person_id: db::generate_id(),
name, name,
source: Source::User,
created_at: now, created_at: now,
edited_at: now, edited_at: now,
last_used_at: now, last_used_at: now,
@ -93,7 +86,6 @@ impl Library {
let instrument = Instrument { let instrument = Instrument {
instrument_id: db::generate_id(), instrument_id: db::generate_id(),
name, name,
source: Source::User,
created_at: now, created_at: now,
edited_at: now, edited_at: now,
last_used_at: now, last_used_at: now,
@ -155,7 +147,6 @@ impl Library {
let role = Role { let role = Role {
role_id: db::generate_id(), role_id: db::generate_id(),
name, name,
source: Source::User,
created_at: now, created_at: now,
edited_at: now, edited_at: now,
last_used_at: now, last_used_at: now,
@ -218,7 +209,7 @@ impl Library {
) -> Result<Work> { ) -> Result<Work> {
let connection = &mut *self.imp().connection.get().unwrap().lock().unwrap(); let connection = &mut *self.imp().connection.get().unwrap().lock().unwrap();
let work = Self::create_work_priv( let work = self.create_work_priv(
connection, connection,
name, name,
parts, parts,
@ -235,6 +226,7 @@ impl Library {
} }
fn create_work_priv( fn create_work_priv(
&self,
connection: &mut SqliteConnection, connection: &mut SqliteConnection,
name: TranslatedString, name: TranslatedString,
parts: Vec<Work>, parts: Vec<Work>,
@ -250,9 +242,8 @@ impl Library {
let work_data = tables::Work { let work_data = tables::Work {
work_id: work_id.clone(), work_id: work_id.clone(),
parent_work_id: parent_work_id.map(|w| w.to_string()), parent_work_id: parent_work_id.map(|w| w.to_string()),
sequence_number, sequence_number: sequence_number,
name, name,
source: Source::User,
created_at: now, created_at: now,
edited_at: now, edited_at: now,
last_used_at: now, last_used_at: now,
@ -265,7 +256,7 @@ impl Library {
.execute(connection)?; .execute(connection)?;
for (index, part) in parts.into_iter().enumerate() { for (index, part) in parts.into_iter().enumerate() {
Self::create_work_priv( self.create_work_priv(
connection, connection,
part.name, part.name,
part.parts, part.parts,
@ -318,7 +309,7 @@ impl Library {
) -> Result<()> { ) -> Result<()> {
let connection = &mut *self.imp().connection.get().unwrap().lock().unwrap(); let connection = &mut *self.imp().connection.get().unwrap().lock().unwrap();
Self::update_work_priv( self.update_work_priv(
connection, connection,
work_id, work_id,
name, name,
@ -336,6 +327,7 @@ impl Library {
} }
fn update_work_priv( fn update_work_priv(
&self,
connection: &mut SqliteConnection, connection: &mut SqliteConnection,
work_id: &str, work_id: &str,
name: TranslatedString, name: TranslatedString,
@ -375,7 +367,7 @@ impl Library {
.optional()? .optional()?
.is_some() .is_some()
{ {
Self::update_work_priv( self.update_work_priv(
connection, connection,
&part.work_id, &part.work_id,
part.name, part.name,
@ -389,7 +381,7 @@ impl Library {
} else { } else {
// Note: The previously used ID is discarded. This should be OK, because // Note: The previously used ID is discarded. This should be OK, because
// at this point, the part ID should not have been used anywhere. // at this point, the part ID should not have been used anywhere.
Self::create_work_priv( self.create_work_priv(
connection, connection,
part.name, part.name,
part.parts, part.parts,
@ -462,7 +454,6 @@ impl Library {
let ensemble_data = tables::Ensemble { let ensemble_data = tables::Ensemble {
ensemble_id: db::generate_id(), ensemble_id: db::generate_id(),
name, name,
source: Source::User,
created_at: now, created_at: now,
edited_at: now, edited_at: now,
last_used_at: now, last_used_at: now,
@ -539,7 +530,6 @@ impl Library {
recording_id: recording_id.clone(), recording_id: recording_id.clone(),
work_id: work.work_id.clone(), work_id: work.work_id.clone(),
year, year,
source: Source::User,
created_at: now, created_at: now,
edited_at: now, edited_at: now,
last_used_at: now, last_used_at: now,
@ -703,7 +693,6 @@ impl Library {
&self, &self,
name: TranslatedString, name: TranslatedString,
recordings: Vec<Recording>, recordings: Vec<Recording>,
enable_updates: bool,
) -> Result<Album> { ) -> Result<Album> {
let connection = &mut *self.imp().connection.get().unwrap().lock().unwrap(); let connection = &mut *self.imp().connection.get().unwrap().lock().unwrap();
@ -713,8 +702,6 @@ impl Library {
let album_data = tables::Album { let album_data = tables::Album {
album_id: album_id.clone(), album_id: album_id.clone(),
name, name,
source: Source::User,
enable_updates,
created_at: now, created_at: now,
edited_at: now, edited_at: now,
last_used_at: now, last_used_at: now,
@ -749,7 +736,6 @@ impl Library {
album_id: &str, album_id: &str,
name: TranslatedString, name: TranslatedString,
recordings: Vec<Recording>, recordings: Vec<Recording>,
enable_updates: bool,
) -> Result<()> { ) -> Result<()> {
let connection = &mut *self.imp().connection.get().unwrap().lock().unwrap(); let connection = &mut *self.imp().connection.get().unwrap().lock().unwrap();
@ -759,7 +745,6 @@ impl Library {
.filter(albums::album_id.eq(album_id)) .filter(albums::album_id.eq(album_id))
.set(( .set((
albums::name.eq(name), albums::name.eq(name),
albums::enable_updates.eq(enable_updates),
albums::edited_at.eq(now), albums::edited_at.eq(now),
albums::last_used_at.eq(now), albums::last_used_at.eq(now),
)) ))

View file

@ -19,11 +19,7 @@ use zip::{write::SimpleFileOptions, ZipWriter};
use super::Library; use super::Library;
use crate::{ use crate::{
db::{ db::{self, schema::*, tables},
self,
schema::*,
tables::{self, Source},
},
process::ProcessMsg, process::ProcessMsg,
}; };
@ -32,27 +28,17 @@ impl Library {
pub fn import_library_from_zip( pub fn import_library_from_zip(
&self, &self,
path: impl AsRef<Path>, path: impl AsRef<Path>,
source: Source,
) -> Result<async_channel::Receiver<ProcessMsg>> { ) -> Result<async_channel::Receiver<ProcessMsg>> {
log::info!( log::info!("Importing library from ZIP at {}", path.as_ref().to_string_lossy());
"Importing library from ZIP at {}",
path.as_ref().to_string_lossy()
);
let path = path.as_ref().to_owned(); let path = path.as_ref().to_owned();
let library_folder = PathBuf::from(&self.folder()); let library_folder = PathBuf::from(&self.folder());
let this_connection = self.imp().connection.get().unwrap().clone(); let this_connection = self.imp().connection.get().unwrap().clone();
let (sender, receiver) = async_channel::unbounded::<ProcessMsg>(); let (sender, receiver) = async_channel::unbounded::<ProcessMsg>();
thread::spawn(move || { thread::spawn(move || {
if let Err(err) = if let Err(err) = sender.send_blocking(ProcessMsg::Result(
sender.send_blocking(ProcessMsg::Result(import_library_from_zip_priv( import_library_from_zip_priv(path, library_folder, this_connection, &sender),
path, )) {
library_folder,
source,
this_connection,
&sender,
)))
{
log::error!("Failed to send library action result: {err:?}"); log::error!("Failed to send library action result: {err:?}");
} }
}); });
@ -66,10 +52,7 @@ impl Library {
&self, &self,
path: impl AsRef<Path>, path: impl AsRef<Path>,
) -> Result<async_channel::Receiver<ProcessMsg>> { ) -> Result<async_channel::Receiver<ProcessMsg>> {
log::info!( log::info!("Exporting library to ZIP at {}", path.as_ref().to_string_lossy());
"Exporting library to ZIP at {}",
path.as_ref().to_string_lossy()
);
let connection = &mut *self.imp().connection.get().unwrap().lock().unwrap(); let connection = &mut *self.imp().connection.get().unwrap().lock().unwrap();
let path = path.as_ref().to_owned(); let path = path.as_ref().to_owned();
@ -95,7 +78,6 @@ impl Library {
pub fn import_library_from_url( pub fn import_library_from_url(
&self, &self,
url: &str, url: &str,
source: Source,
) -> Result<async_channel::Receiver<ProcessMsg>> { ) -> Result<async_channel::Receiver<ProcessMsg>> {
log::info!("Importing library from URL {url}"); log::info!("Importing library from URL {url}");
let url = url.to_owned(); let url = url.to_owned();
@ -106,7 +88,7 @@ impl Library {
thread::spawn(move || { thread::spawn(move || {
if let Err(err) = sender.send_blocking(ProcessMsg::Result( if let Err(err) = sender.send_blocking(ProcessMsg::Result(
import_library_from_url_priv(url, library_folder, source, this_connection, &sender), import_library_from_url_priv(url, library_folder, this_connection, &sender),
)) { )) {
log::error!("Failed to send library action result: {err:?}"); log::error!("Failed to send library action result: {err:?}");
} }
@ -119,7 +101,6 @@ impl Library {
pub fn import_metadata_from_url( pub fn import_metadata_from_url(
&self, &self,
url: &str, url: &str,
source: Source,
) -> Result<async_channel::Receiver<ProcessMsg>> { ) -> Result<async_channel::Receiver<ProcessMsg>> {
log::info!("Importing metadata from URL {url}"); log::info!("Importing metadata from URL {url}");
@ -130,7 +111,7 @@ impl Library {
thread::spawn(move || { thread::spawn(move || {
if let Err(err) = sender.send_blocking(ProcessMsg::Result( if let Err(err) = sender.send_blocking(ProcessMsg::Result(
import_metadata_from_url_priv(url, source, this_connection, &sender), import_metadata_from_url_priv(url, this_connection, &sender),
)) { )) {
log::error!("Failed to send library action result: {err:?}"); log::error!("Failed to send library action result: {err:?}");
} }
@ -144,7 +125,6 @@ impl Library {
fn import_library_from_zip_priv( fn import_library_from_zip_priv(
zip_path: impl AsRef<Path>, zip_path: impl AsRef<Path>,
library_folder: impl AsRef<Path>, library_folder: impl AsRef<Path>,
source: Source,
this_connection: Arc<Mutex<SqliteConnection>>, this_connection: Arc<Mutex<SqliteConnection>>,
sender: &async_channel::Sender<ProcessMsg>, sender: &async_channel::Sender<ProcessMsg>,
) -> Result<()> { ) -> Result<()> {
@ -158,7 +138,7 @@ fn import_library_from_zip_priv(
)?; )?;
// Import metadata. // Import metadata.
let tracks = import_metadata_from_file(tmp_db_file.path(), source, this_connection, false)?; let tracks = import_metadata_from_file(tmp_db_file.path(), this_connection, false)?;
// Import audio files. // Import audio files.
let n_tracks = tracks.len(); let n_tracks = tracks.len();
@ -232,7 +212,6 @@ fn add_file_to_zip(
fn import_metadata_from_url_priv( fn import_metadata_from_url_priv(
url: String, url: String,
source: Source,
this_connection: Arc<Mutex<SqliteConnection>>, this_connection: Arc<Mutex<SqliteConnection>>,
sender: &async_channel::Sender<ProcessMsg>, sender: &async_channel::Sender<ProcessMsg>,
) -> Result<()> { ) -> Result<()> {
@ -244,18 +223,20 @@ fn import_metadata_from_url_priv(
formatx!(gettext("Downloading {}"), &url).unwrap(), formatx!(gettext("Downloading {}"), &url).unwrap(),
)); ));
match runtime.block_on(download_tmp_file(&url, sender)) { match runtime.block_on(download_tmp_file(&url, &sender)) {
Ok(db_file) => { Ok(db_file) => {
let _ = sender.send_blocking(ProcessMsg::Message( let _ = sender.send_blocking(ProcessMsg::Message(
formatx!(gettext("Importing downloaded library"), &url).unwrap(), formatx!(gettext("Importing downloaded library"), &url).unwrap(),
)); ));
let _ = sender.send_blocking(ProcessMsg::Result( let _ = sender.send_blocking(ProcessMsg::Result(
import_metadata_from_file(db_file.path(), source, this_connection, true).map( import_metadata_from_file(db_file.path(), this_connection, true).and_then(
|tracks| { |tracks| {
if !tracks.is_empty() { if !tracks.is_empty() {
log::warn!("The metadata file at {url} contains tracks."); log::warn!("The metadata file at {url} contains tracks.");
} }
Ok(())
}, },
), ),
)); ));
@ -271,7 +252,6 @@ fn import_metadata_from_url_priv(
fn import_library_from_url_priv( fn import_library_from_url_priv(
url: String, url: String,
library_folder: impl AsRef<Path>, library_folder: impl AsRef<Path>,
source: Source,
this_connection: Arc<Mutex<SqliteConnection>>, this_connection: Arc<Mutex<SqliteConnection>>,
sender: &async_channel::Sender<ProcessMsg>, sender: &async_channel::Sender<ProcessMsg>,
) -> Result<()> { ) -> Result<()> {
@ -283,7 +263,7 @@ fn import_library_from_url_priv(
formatx!(gettext("Downloading {}"), &url).unwrap(), formatx!(gettext("Downloading {}"), &url).unwrap(),
)); ));
let archive_file = runtime.block_on(download_tmp_file(&url, sender)); let archive_file = runtime.block_on(download_tmp_file(&url, &sender));
match archive_file { match archive_file {
Ok(archive_file) => { Ok(archive_file) => {
@ -294,9 +274,8 @@ fn import_library_from_url_priv(
let _ = sender.send_blocking(ProcessMsg::Result(import_library_from_zip_priv( let _ = sender.send_blocking(ProcessMsg::Result(import_library_from_zip_priv(
archive_file.path(), archive_file.path(),
library_folder, library_folder,
source,
this_connection, this_connection,
sender, &sender,
))); )));
} }
Err(err) => { Err(err) => {
@ -314,7 +293,6 @@ fn import_library_from_url_priv(
/// In any case, tracks are returned. /// In any case, tracks are returned.
fn import_metadata_from_file( fn import_metadata_from_file(
path: impl AsRef<Path>, path: impl AsRef<Path>,
source: Source,
this_connection: Arc<Mutex<SqliteConnection>>, this_connection: Arc<Mutex<SqliteConnection>>,
ignore_tracks: bool, ignore_tracks: bool,
) -> Result<Vec<tables::Track>> { ) -> Result<Vec<tables::Track>> {
@ -349,7 +327,6 @@ fn import_metadata_from_file(
// Import metadata that is not already present. // Import metadata that is not already present.
for mut person in persons { for mut person in persons {
person.source = source;
person.created_at = now; person.created_at = now;
person.edited_at = now; person.edited_at = now;
person.last_used_at = now; person.last_used_at = now;
@ -362,7 +339,6 @@ fn import_metadata_from_file(
} }
for mut role in roles { for mut role in roles {
role.source = source;
role.created_at = now; role.created_at = now;
role.edited_at = now; role.edited_at = now;
role.last_used_at = now; role.last_used_at = now;
@ -374,7 +350,6 @@ fn import_metadata_from_file(
} }
for mut instrument in instruments { for mut instrument in instruments {
instrument.source = source;
instrument.created_at = now; instrument.created_at = now;
instrument.edited_at = now; instrument.edited_at = now;
instrument.last_used_at = now; instrument.last_used_at = now;
@ -387,7 +362,6 @@ fn import_metadata_from_file(
} }
for mut work in works { for mut work in works {
work.source = source;
work.created_at = now; work.created_at = now;
work.edited_at = now; work.edited_at = now;
work.last_used_at = now; work.last_used_at = now;
@ -414,7 +388,6 @@ fn import_metadata_from_file(
} }
for mut ensemble in ensembles { for mut ensemble in ensembles {
ensemble.source = source;
ensemble.created_at = now; ensemble.created_at = now;
ensemble.edited_at = now; ensemble.edited_at = now;
ensemble.last_used_at = now; ensemble.last_used_at = now;
@ -434,7 +407,6 @@ fn import_metadata_from_file(
} }
for mut recording in recordings { for mut recording in recordings {
recording.source = source;
recording.created_at = now; recording.created_at = now;
recording.edited_at = now; recording.edited_at = now;
recording.last_used_at = now; recording.last_used_at = now;
@ -494,7 +466,6 @@ fn import_metadata_from_file(
} }
for mut album in albums { for mut album in albums {
album.source = source;
album.created_at = now; album.created_at = now;
album.edited_at = now; album.edited_at = now;
album.last_used_at = now; album.last_used_at = now;

View file

@ -414,6 +414,7 @@ impl Library {
works, works,
recordings, recordings,
albums, albums,
..Default::default()
} }
} }
LibraryQuery { LibraryQuery {

View file

@ -9,8 +9,8 @@ use gtk::{
}; };
use crate::{ use crate::{
config, db::tables::Source, library::Library, process::Process, config, library::Library, process::Process, process_manager::ProcessManager,
process_manager::ProcessManager, process_row::ProcessRow, window::Window, process_row::ProcessRow, window::Window,
}; };
mod imp { mod imp {
@ -128,13 +128,7 @@ impl LibraryManager {
} }
Ok(path) => { Ok(path) => {
if let Some(path) = path.path() { if let Some(path) = path.path() {
match self match self.imp().library.get().unwrap().import_library_from_zip(&path) {
.imp()
.library
.get()
.unwrap()
.import_library_from_zip(&path, Source::Import)
{
Ok(receiver) => { Ok(receiver) => {
let process = Process::new( let process = Process::new(
&formatx!( &formatx!(
@ -192,13 +186,7 @@ impl LibraryManager {
} }
Ok(path) => { Ok(path) => {
if let Some(path) = path.path() { if let Some(path) = path.path() {
match self match self.imp().library.get().unwrap().export_library_to_zip(&path) {
.imp()
.library
.get()
.unwrap()
.export_library_to_zip(&path)
{
Ok(receiver) => { Ok(receiver) => {
let process = Process::new( let process = Process::new(
&formatx!( &formatx!(
@ -240,7 +228,7 @@ impl LibraryManager {
.library .library
.get() .get()
.unwrap() .unwrap()
.import_metadata_from_url(&url, Source::Metadata) .import_metadata_from_url(&url)
{ {
Ok(receiver) => { Ok(receiver) => {
let process = Process::new(&gettext("Updating metadata"), receiver); let process = Process::new(&gettext("Updating metadata"), receiver);
@ -271,7 +259,7 @@ impl LibraryManager {
.library .library
.get() .get()
.unwrap() .unwrap()
.import_library_from_url(&url, Source::Metadata) .import_library_from_url(&url)
{ {
Ok(receiver) => { Ok(receiver) => {
let process = Process::new(&gettext("Updating music library"), receiver); let process = Process::new(&gettext("Updating music library"), receiver);

View file

@ -20,6 +20,7 @@ mod program;
mod program_tile; mod program_tile;
mod recording_tile; mod recording_tile;
mod search_page; mod search_page;
mod search_tag;
mod selector; mod selector;
mod slider_row; mod slider_row;
mod tag_tile; mod tag_tile;
@ -46,7 +47,7 @@ fn main() -> glib::ExitCode {
gettextrs::textdomain(config::PKGNAME).unwrap(); gettextrs::textdomain(config::PKGNAME).unwrap();
gio::resources_register( gio::resources_register(
&gio::Resource::load(format!( &gio::Resource::load(&format!(
"{}/{}/{}.gresource", "{}/{}/{}.gresource",
config::DATADIR, config::DATADIR,
config::PKGNAME, config::PKGNAME,

View file

@ -221,14 +221,14 @@ impl Player {
items.push(PlaylistItem::new( items.push(PlaylistItem::new(
true, true,
recording.work.composers_string(), recording.work.composers_string(),
recording.work.name.get(), &recording.work.name.get(),
Some(&performances), Some(&performances),
None, None,
self.library_path_to_file_path(&tracks[0].path), &self.library_path_to_file_path(&tracks[0].path),
&tracks[0].track_id, &tracks[0].track_id,
)); ));
} else { } else {
let mut tracks = tracks.iter(); let mut tracks = tracks.into_iter();
let first_track = tracks.next().unwrap(); let first_track = tracks.next().unwrap();
let track_title = |track: &Track, number: usize| -> String { let track_title = |track: &Track, number: usize| -> String {
@ -249,10 +249,10 @@ impl Player {
items.push(PlaylistItem::new( items.push(PlaylistItem::new(
true, true,
recording.work.composers_string(), recording.work.composers_string(),
recording.work.name.get(), &recording.work.name.get(),
Some(&performances), Some(&performances),
Some(&track_title(first_track, 1)), Some(&track_title(&first_track, 1)),
self.library_path_to_file_path(&first_track.path), &self.library_path_to_file_path(&first_track.path),
&first_track.track_id, &first_track.track_id,
)); ));
@ -260,11 +260,11 @@ impl Player {
items.push(PlaylistItem::new( items.push(PlaylistItem::new(
false, false,
recording.work.composers_string(), recording.work.composers_string(),
recording.work.name.get(), &recording.work.name.get(),
Some(&performances), Some(&performances),
// track number = track index + 1 (first track) + 1 (zero based) // track number = track index + 1 (first track) + 1 (zero based)
Some(&track_title(track, index + 2)), Some(&track_title(&track, index + 2)),
self.library_path_to_file_path(&track.path), &self.library_path_to_file_path(&track.path),
&track.track_id, &track.track_id,
)); ));
} }

View file

@ -83,7 +83,7 @@ impl Program {
} }
pub fn from_query(query: LibraryQuery) -> Self { pub fn from_query(query: LibraryQuery) -> Self {
let settings = gio::Settings::new(config::APP_ID); let settings = gio::Settings::new(&config::APP_ID);
glib::Object::builder() glib::Object::builder()
.property( .property(

View file

@ -56,7 +56,7 @@ mod imp {
self.set_program_from_settings(&settings); self.set_program_from_settings(&settings);
let obj = self.obj().to_owned(); let obj = self.obj().to_owned();
settings.connect_changed(Some(self.key.get().unwrap()), move |settings, _| { settings.connect_changed(Some(&self.key.get().unwrap()), move |settings, _| {
obj.imp().set_program_from_settings(settings); obj.imp().set_program_from_settings(settings);
}); });
} }

View file

@ -69,7 +69,7 @@ mod imp {
.push(&RecordingEditor::new( .push(&RecordingEditor::new(
obj.imp().navigation.get().unwrap(), obj.imp().navigation.get().unwrap(),
obj.imp().library.get().unwrap(), obj.imp().library.get().unwrap(),
Some(obj.imp().recording.get().unwrap()), Some(&obj.imp().recording.get().unwrap()),
)); ));
}) })
.build(); .build();
@ -90,8 +90,8 @@ mod imp {
let delete_action = gio::ActionEntry::builder("delete") let delete_action = gio::ActionEntry::builder("delete")
.activate(move |_, _, _| { .activate(move |_, _, _| {
let dialog = adw::AlertDialog::builder() let dialog = adw::AlertDialog::builder()
.heading(gettext("Delete recording?")) .heading(&gettext("Delete recording?"))
.body(gettext("The recording will be removed from your music library and the corresponding audio files will be deleted. This action cannot be undone.")) .body(&gettext("The recording will be removed from your music library and the corresponding audio files will be deleted. This action cannot be undone."))
.build(); .build();
dialog.add_response("delete", &gettext("Delete")); dialog.add_response("delete", &gettext("Delete"));
@ -142,7 +142,7 @@ impl RecordingTile {
let obj: Self = glib::Object::new(); let obj: Self = glib::Object::new();
let imp = obj.imp(); let imp = obj.imp();
imp.work_label.set_label(recording.work.name.get()); imp.work_label.set_label(&recording.work.name.get());
imp.composer_label.set_label( imp.composer_label.set_label(
&recording &recording
.work .work

View file

@ -22,7 +22,8 @@ use crate::{
program::Program, program::Program,
program_tile::ProgramTile, program_tile::ProgramTile,
recording_tile::RecordingTile, recording_tile::RecordingTile,
tag_tile::{Tag, TagTile}, search_tag::Tag,
tag_tile::TagTile,
util, util,
}; };
@ -391,7 +392,7 @@ impl SearchPage {
imp.header_box.set_visible(!query.is_empty()); imp.header_box.set_visible(!query.is_empty());
let highlight = if let Some(work) = &query.work { let highlight = if let Some(work) = &query.work {
imp.title_label.set_text(work.name.get()); imp.title_label.set_text(&work.name.get());
if let Some(composers) = work.composers_string() { if let Some(composers) = work.composers_string() {
imp.subtitle_label.set_text(&composers); imp.subtitle_label.set_text(&composers);
imp.subtitle_label.set_visible(true); imp.subtitle_label.set_visible(true);
@ -400,15 +401,15 @@ impl SearchPage {
} }
Some(Tag::Work(work.to_owned())) Some(Tag::Work(work.to_owned()))
} else if let Some(person) = &query.composer { } else if let Some(person) = &query.composer {
imp.title_label.set_text(person.name.get()); imp.title_label.set_text(&person.name.get());
imp.subtitle_label.set_visible(false); imp.subtitle_label.set_visible(false);
Some(Tag::Composer(person.to_owned())) Some(Tag::Composer(person.to_owned()))
} else if let Some(person) = &query.performer { } else if let Some(person) = &query.performer {
imp.title_label.set_text(person.name.get()); imp.title_label.set_text(&person.name.get());
imp.subtitle_label.set_visible(false); imp.subtitle_label.set_visible(false);
Some(Tag::Performer(person.to_owned())) Some(Tag::Performer(person.to_owned()))
} else if let Some(ensemble) = &query.ensemble { } else if let Some(ensemble) = &query.ensemble {
imp.title_label.set_text(ensemble.name.get()); imp.title_label.set_text(&ensemble.name.get());
imp.subtitle_label.set_visible(false); imp.subtitle_label.set_visible(false);
Some(Tag::Ensemble(ensemble.to_owned())) Some(Tag::Ensemble(ensemble.to_owned()))
} else if let Some(instrument) = &query.instrument { } else if let Some(instrument) = &query.instrument {

98
src/search_tag.rs Normal file
View file

@ -0,0 +1,98 @@
use std::cell::OnceCell;
use adw::{glib, glib::subclass::Signal, prelude::*, subclass::prelude::*};
use once_cell::sync::Lazy;
use crate::db::models::{Ensemble, Instrument, Person, Work};
mod imp {
use super::*;
#[derive(Debug, Default, gtk::CompositeTemplate)]
#[template(file = "data/ui/search_tag.blp")]
pub struct SearchTag {
#[template_child]
pub label: TemplateChild<gtk::Label>,
pub tag: OnceCell<Tag>,
}
#[glib::object_subclass]
impl ObjectSubclass for SearchTag {
const NAME: &'static str = "MusicusSearchTag";
type Type = super::SearchTag;
type ParentType = gtk::Box;
fn class_init(klass: &mut Self::Class) {
klass.bind_template();
klass.bind_template_instance_callbacks();
}
fn instance_init(obj: &glib::subclass::InitializingObject<Self>) {
obj.init_template();
}
}
impl ObjectImpl for SearchTag {
fn signals() -> &'static [Signal] {
static SIGNALS: Lazy<Vec<Signal>> =
Lazy::new(|| vec![Signal::builder("remove").build()]);
SIGNALS.as_ref()
}
}
impl WidgetImpl for SearchTag {}
impl BoxImpl for SearchTag {}
}
glib::wrapper! {
pub struct SearchTag(ObjectSubclass<imp::SearchTag>)
@extends gtk::Widget;
}
#[gtk::template_callbacks]
impl SearchTag {
pub fn new(tag: Tag) -> Self {
let obj: SearchTag = glib::Object::new();
let label = match &tag {
Tag::Composer(person) => person.name.get(),
Tag::Performer(person) => person.name.get(),
Tag::Ensemble(ensemble) => ensemble.name.get(),
Tag::Instrument(instrument) => instrument.name.get(),
Tag::Work(work) => work.name.get(),
};
obj.imp().label.set_label(label);
obj.set_tooltip_text(Some(label));
obj.imp().tag.set(tag).unwrap();
obj
}
pub fn connect_remove<F: Fn(&Self) + 'static>(&self, f: F) -> glib::SignalHandlerId {
self.connect_local("remove", true, move |values| {
let obj = values[0].get::<Self>().unwrap();
f(&obj);
None
})
}
pub fn tag(&self) -> &Tag {
self.imp().tag.get().unwrap()
}
#[template_callback]
fn remove(&self) {
self.emit_by_name::<()>("remove", &[]);
}
}
#[derive(Debug, Clone, PartialEq, Eq)]
pub enum Tag {
Composer(Person),
Performer(Person),
Ensemble(Ensemble),
Instrument(Instrument),
Work(Work),
}

View file

@ -294,7 +294,7 @@ impl RecordingSelectorPopover {
.build(), .build(),
); );
row.set_tooltip_text(Some(work.name.get())); row.set_tooltip_text(Some(&work.name.get()));
let work = work.clone(); let work = work.clone();
let obj = self.clone(); let obj = self.clone();

View file

@ -256,7 +256,7 @@ impl WorkSelectorPopover {
.build(), .build(),
); );
row.set_tooltip_text(Some(work.name.get())); row.set_tooltip_text(Some(&work.name.get()));
let work = work.clone(); let work = work.clone();
let obj = self.clone(); let obj = self.clone();

View file

@ -2,7 +2,7 @@ use std::cell::OnceCell;
use gtk::{glib, prelude::*, subclass::prelude::*}; use gtk::{glib, prelude::*, subclass::prelude::*};
use crate::db::models::{Ensemble, Instrument, Person, Work}; use crate::search_tag::Tag;
mod imp { mod imp {
use super::*; use super::*;
@ -78,12 +78,3 @@ impl TagTile {
self.imp().tag.get().unwrap() self.imp().tag.get().unwrap()
} }
} }
#[derive(Debug, Clone, PartialEq, Eq)]
pub enum Tag {
Composer(Person),
Performer(Person),
Ensemble(Ensemble),
Instrument(Instrument),
Work(Work),
}

View file

@ -2,33 +2,34 @@ pub mod activatable_row;
pub mod drag_widget; pub mod drag_widget;
pub mod error_dialog; pub mod error_dialog;
use std::sync::LazyLock;
use gettextrs::gettext; use gettextrs::gettext;
use gtk::glib::{self, clone}; use gtk::glib::{self, clone};
use lazy_static::lazy_static;
use error_dialog::ErrorDialog; use error_dialog::ErrorDialog;
/// The user's language code. lazy_static! {
pub static LANG: LazyLock<String> = LazyLock::new(|| { /// The user's language code.
let lang = match glib::language_names().first() { pub static ref LANG: String = {
Some(language_name) => match language_name.split('_').next() { let lang = match glib::language_names().first() {
Some(lang) => lang.to_string(), Some(language_name) => match language_name.split('_').next() {
Some(lang) => lang.to_string(),
None => "generic".to_string(),
},
None => "generic".to_string(), None => "generic".to_string(),
}, };
None => "generic".to_string(),
};
log::info!("Intialized user language to '{lang}'."); log::info!("Intialized user language to '{lang}'.");
lang lang
}); };
}
/// Create and show an error toast. This will also log the error to the console. /// Create and show an error toast. This will also log the error to the console.
pub fn error_toast(msgid: &str, err: anyhow::Error, toast_overlay: &adw::ToastOverlay) { pub fn error_toast(msgid: &str, err: anyhow::Error, toast_overlay: &adw::ToastOverlay) {
log::error!("{msgid}: {err:?}"); log::error!("{msgid}: {err:?}");
let toast = adw::Toast::builder() let toast = adw::Toast::builder()
.title(gettext(msgid)) .title(&gettext(msgid))
.button_label("Details") .button_label("Details")
.build(); .build();

View file

@ -61,7 +61,7 @@ glib::wrapper! {
impl ErrorDialog { impl ErrorDialog {
pub fn present(err: &anyhow::Error, parent: &impl IsA<gtk::Widget>) { pub fn present(err: &anyhow::Error, parent: &impl IsA<gtk::Widget>) {
let obj: Self = glib::Object::builder() let obj: Self = glib::Object::builder()
.property("error-text", format!("{err:?}")) .property("error-text", &format!("{err:?}"))
.build(); .build();
obj.present(Some(parent)); obj.present(Some(parent));

View file

@ -1,7 +1,4 @@
use std::{ use std::{cell::RefCell, path::Path};
cell::{Cell, RefCell},
path::Path,
};
use adw::{prelude::*, subclass::prelude::*}; use adw::{prelude::*, subclass::prelude::*};
use anyhow::{anyhow, Result}; use anyhow::{anyhow, Result};
@ -9,10 +6,8 @@ use gettextrs::gettext;
use gtk::{gio, glib, glib::clone}; use gtk::{gio, glib, glib::clone};
use crate::{ use crate::{
album_page::AlbumPage,
config, config,
db::tables::Source, editor::tracks::TracksEditor,
editor::{album::AlbumEditor, tracks::TracksEditor},
empty_page::EmptyPage, empty_page::EmptyPage,
library::{Library, LibraryQuery}, library::{Library, LibraryQuery},
library_manager::LibraryManager, library_manager::LibraryManager,
@ -36,7 +31,6 @@ mod imp {
pub library: RefCell<Option<Library>>, pub library: RefCell<Option<Library>>,
pub player: Player, pub player: Player,
pub process_manager: ProcessManager, pub process_manager: ProcessManager,
pub inhibitor_cookie: Cell<Option<u32>>,
#[template_child] #[template_child]
pub toast_overlay: TemplateChild<adw::ToastOverlay>, pub toast_overlay: TemplateChild<adw::ToastOverlay>,
@ -89,16 +83,6 @@ mod imp {
}) })
.build(); .build();
let obj = self.obj().to_owned();
let create_album_action = gio::ActionEntry::builder("create-album")
.activate(move |_, _, _| {
if let Some(library) = &*obj.imp().library.borrow() {
let editor = AlbumEditor::new(&obj.imp().navigation_view, library, None);
obj.imp().navigation_view.push(&editor);
}
})
.build();
let obj = self.obj().to_owned(); let obj = self.obj().to_owned();
let library_action = gio::ActionEntry::builder("library") let library_action = gio::ActionEntry::builder("library")
.activate(move |_, _, _| { .activate(move |_, _, _| {
@ -120,12 +104,8 @@ mod imp {
}) })
.build(); .build();
self.obj().add_action_entries([ self.obj()
import_action, .add_action_entries([import_action, library_action, preferences_action]);
create_album_action,
library_action,
preferences_action,
]);
let player_bar = PlayerBar::new(&self.player); let player_bar = PlayerBar::new(&self.player);
self.player_bar_revealer.set_child(Some(&player_bar)); self.player_bar_revealer.set_child(Some(&player_bar));
@ -168,25 +148,6 @@ mod imp {
let obj = self.obj().to_owned(); let obj = self.obj().to_owned();
self.player.connect_raise(move |_| obj.present()); self.player.connect_raise(move |_| obj.present());
let obj = self.obj().to_owned();
self.player.connect_playing_notify(move |player| {
if let Some(app) = obj.application() {
if let Some(cookie) = obj.imp().inhibitor_cookie.take() {
app.uninhibit(cookie);
};
if player.playing() {
let cookie = app.inhibit(
Some(&obj),
gtk::ApplicationInhibitFlags::SUSPEND,
Some(&gettext("Currently playing music")),
);
obj.imp().inhibitor_cookie.set(Some(cookie));
}
}
});
let settings = gio::Settings::new(config::APP_ID); let settings = gio::Settings::new(config::APP_ID);
let library_path = settings.string("library-path").to_string(); let library_path = settings.string("library-path").to_string();
if !library_path.is_empty() { if !library_path.is_empty() {
@ -203,8 +164,8 @@ mod imp {
fn close_request(&self) -> glib::signal::Propagation { fn close_request(&self) -> glib::signal::Propagation {
if self.process_manager.any_ongoing() { if self.process_manager.any_ongoing() {
let dialog = adw::AlertDialog::builder() let dialog = adw::AlertDialog::builder()
.heading(gettext("Close window?")) .heading(&gettext("Close window?"))
.body(gettext( .body(&gettext(
"There are ongoing processes that will be canceled.", "There are ongoing processes that will be canceled.",
)) ))
.build(); .build();
@ -316,7 +277,7 @@ impl Window {
config::METADATA_URL.to_string() config::METADATA_URL.to_string()
}; };
match library.import_metadata_from_url(&url, Source::Metadata) { match library.import_metadata_from_url(&url) {
Ok(receiver) => { Ok(receiver) => {
let process = Process::new(&gettext("Updating metadata"), receiver); let process = Process::new(&gettext("Updating metadata"), receiver);
self.imp().process_manager.add_process(&process); self.imp().process_manager.add_process(&process);
@ -378,12 +339,17 @@ impl Window {
fn reset_view(&self) { fn reset_view(&self) {
let navigation = self.imp().navigation_view.get(); let navigation = self.imp().navigation_view.get();
// Get all pages that are not instances of SearchPage or AlbumPage. // Get all pages that are not instances of SearchPage.
let mut navigation_stack = navigation let mut navigation_stack = navigation
.navigation_stack() .navigation_stack()
.iter::<adw::NavigationPage>() .iter::<adw::NavigationPage>()
.filter_map(|page| page.ok()) .filter_map(|page| match page {
.filter(|page| !page.is::<SearchPage>() && !page.is::<AlbumPage>()) Ok(page) => match page.downcast_ref::<SearchPage>() {
Some(_) => None,
None => Some(page),
},
Err(_) => None,
})
.collect::<Vec<adw::NavigationPage>>(); .collect::<Vec<adw::NavigationPage>>();
navigation_stack.insert( navigation_stack.insert(